Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 | /* SPDX-License-Identifier: GPL-2.0-only */ /* * Copyright (C) 2012 Red Hat, Inc. * * This file is released under the GPL. */ #ifndef _LINUX_DM_ARRAY_H #define _LINUX_DM_ARRAY_H #include "dm-btree.h" /*----------------------------------------------------------------*/ /* * The dm-array is a persistent version of an array. It packs the data * more efficiently than a btree which will result in less disk space use, * and a performance boost. The element get and set operations are still * O(ln(n)), but with a much smaller constant. * * The value type structure is reused from the btree type to support proper * reference counting of values. * * The arrays implicitly know their length, and bounds are checked for * lookups and updated. It doesn't store this in an accessible place * because it would waste a whole metadata block. Make sure you store the * size along with the array root in your encompassing data. * * Array entries are indexed via an unsigned integer starting from zero. * Arrays are not sparse; if you resize an array to have 'n' entries then * 'n - 1' will be the last valid index. * * Typical use: * * a) initialise a dm_array_info structure. This describes the array * values and ties it into a specific transaction manager. It holds no * instance data; the same info can be used for many similar arrays if * you wish. * * b) Get yourself a root. The root is the index of a block of data on the * disk that holds a particular instance of an array. You may have a * pre existing root in your metadata that you wish to use, or you may * want to create a brand new, empty array with dm_array_empty(). * * Like the other data structures in this library, dm_array objects are * immutable between transactions. Update functions will return you the * root for a _new_ array. If you've incremented the old root, via * dm_tm_inc(), before calling the update function you may continue to use * it in parallel with the new root. * * c) resize an array with dm_array_resize(). * * d) Get a value from the array with dm_array_get_value(). * * e) Set a value in the array with dm_array_set_value(). * * f) Walk an array of values in index order with dm_array_walk(). More * efficient than making many calls to dm_array_get_value(). * * g) Destroy the array with dm_array_del(). This tells the transaction * manager that you're no longer using this data structure so it can * recycle it's blocks. (dm_array_dec() would be a better name for it, * but del is in keeping with dm_btree_del()). */ /* * Describes an array. Don't initialise this structure yourself, use the * init function below. */ struct dm_array_info { struct dm_transaction_manager *tm; struct dm_btree_value_type value_type; struct dm_btree_info btree_info; }; /* * Sets up a dm_array_info structure. You don't need to do anything with * this structure when you finish using it. * * info - the structure being filled in. * tm - the transaction manager that should supervise this structure. * vt - describes the leaf values. */ void dm_array_info_init(struct dm_array_info *info, struct dm_transaction_manager *tm, struct dm_btree_value_type *vt); /* * Create an empty, zero length array. * * info - describes the array * root - on success this will be filled out with the root block */ int dm_array_empty(struct dm_array_info *info, dm_block_t *root); /* * Resizes the array. * * info - describes the array * root - the root block of the array on disk * old_size - the caller is responsible for remembering the size of * the array * new_size - can be bigger or smaller than old_size * value - if we're growing the array the new entries will have this value * new_root - on success, points to the new root block * * If growing the inc function for 'value' will be called the appropriate * number of times. So if the caller is holding a reference they may want * to drop it. */ int dm_array_resize(struct dm_array_info *info, dm_block_t root, uint32_t old_size, uint32_t new_size, const void *value, dm_block_t *new_root) __dm_written_to_disk(value); /* * Creates a new array populated with values provided by a callback * function. This is more efficient than creating an empty array, * resizing, and then setting values since that process incurs a lot of * copying. * * Assumes 32bit values for now since it's only used by the cache hint * array. * * info - describes the array * root - the root block of the array on disk * size - the number of entries in the array * fn - the callback * context - passed to the callback */ typedef int (*value_fn)(uint32_t index, void *value_le, void *context); int dm_array_new(struct dm_array_info *info, dm_block_t *root, uint32_t size, value_fn fn, void *context); /* * Frees a whole array. The value_type's decrement operation will be called * for all values in the array */ int dm_array_del(struct dm_array_info *info, dm_block_t root); /* * Lookup a value in the array * * info - describes the array * root - root block of the array * index - array index * value - the value to be read. Will be in on-disk format of course. * * -ENODATA will be returned if the index is out of bounds. */ int dm_array_get_value(struct dm_array_info *info, dm_block_t root, uint32_t index, void *value); /* * Set an entry in the array. * * info - describes the array * root - root block of the array * index - array index * value - value to be written to disk. Make sure you confirm the value is * in on-disk format with__dm_bless_for_disk() before calling. * new_root - the new root block * * The old value being overwritten will be decremented, the new value * incremented. * * -ENODATA will be returned if the index is out of bounds. */ int dm_array_set_value(struct dm_array_info *info, dm_block_t root, uint32_t index, const void *value, dm_block_t *new_root) __dm_written_to_disk(value); /* * Walk through all the entries in an array. * * info - describes the array * root - root block of the array * fn - called back for every element * context - passed to the callback */ int dm_array_walk(struct dm_array_info *info, dm_block_t root, int (*fn)(void *context, uint64_t key, void *leaf), void *context); /*----------------------------------------------------------------*/ /* * Cursor api. * * This lets you iterate through all the entries in an array efficiently * (it will preload metadata). * * I'm using a cursor, rather than a walk function with a callback because * the cache target needs to iterate both the mapping and hint arrays in * unison. */ struct dm_array_cursor { struct dm_array_info *info; struct dm_btree_cursor cursor; struct dm_block *block; struct array_block *ab; unsigned int index; }; int dm_array_cursor_begin(struct dm_array_info *info, dm_block_t root, struct dm_array_cursor *c); void dm_array_cursor_end(struct dm_array_cursor *c); uint32_t dm_array_cursor_index(struct dm_array_cursor *c); int dm_array_cursor_next(struct dm_array_cursor *c); int dm_array_cursor_skip(struct dm_array_cursor *c, uint32_t count); /* * value_le is only valid while the cursor points at the current value. */ void dm_array_cursor_get_value(struct dm_array_cursor *c, void **value_le); /*----------------------------------------------------------------*/ #endif /* _LINUX_DM_ARRAY_H */ |