Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 1 | .. SPDX-License-Identifier: GPL-2.0+ |
| 2 | |
| 3 | ====== |
| 4 | XArray |
| 5 | ====== |
| 6 | |
| 7 | :Author: Matthew Wilcox |
| 8 | |
| 9 | Overview |
| 10 | ======== |
| 11 | |
| 12 | The XArray is an abstract data type which behaves like a very large array |
| 13 | of pointers. It meets many of the same needs as a hash or a conventional |
| 14 | resizable array. Unlike a hash, it allows you to sensibly go to the |
| 15 | next or previous entry in a cache-efficient manner. In contrast to a |
| 16 | resizable array, there is no need to copy data or change MMU mappings in |
| 17 | order to grow the array. It is more memory-efficient, parallelisable |
| 18 | and cache friendly than a doubly-linked list. It takes advantage of |
| 19 | RCU to perform lookups without locking. |
| 20 | |
| 21 | The XArray implementation is efficient when the indices used are densely |
| 22 | clustered; hashing the object and using the hash as the index will not |
| 23 | perform well. The XArray is optimised for small indices, but still has |
| 24 | good performance with large indices. If your index can be larger than |
| 25 | ``ULONG_MAX`` then the XArray is not the data type for you. The most |
| 26 | important user of the XArray is the page cache. |
| 27 | |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 28 | Normal pointers may be stored in the XArray directly. They must be 4-byte |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 29 | aligned, which is true for any pointer returned from kmalloc() and |
| 30 | alloc_page(). It isn't true for arbitrary user-space pointers, |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 31 | nor for function pointers. You can store pointers to statically allocated |
| 32 | objects, as long as those objects have an alignment of at least 4. |
| 33 | |
| 34 | You can also store integers between 0 and ``LONG_MAX`` in the XArray. |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 35 | You must first convert it into an entry using xa_mk_value(). |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 36 | When you retrieve an entry from the XArray, you can check whether it is |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 37 | a value entry by calling xa_is_value(), and convert it back to |
| 38 | an integer by calling xa_to_value(). |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 39 | |
Matthew Wilcox (Oracle) | 6b81141 | 2019-11-08 23:45:56 -0500 | [diff] [blame] | 40 | Some users want to tag the pointers they store in the XArray. You can |
| 41 | call xa_tag_pointer() to create an entry with a tag, xa_untag_pointer() |
| 42 | to turn a tagged entry back into an untagged pointer and xa_pointer_tag() |
| 43 | to retrieve the tag of an entry. Tagged pointers use the same bits that |
| 44 | are used to distinguish value entries from normal pointers, so you must |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 45 | decide whether they want to store value entries or tagged pointers in |
| 46 | any particular XArray. |
| 47 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 48 | The XArray does not support storing IS_ERR() pointers as some |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 49 | conflict with value entries or internal entries. |
| 50 | |
| 51 | An unusual feature of the XArray is the ability to create entries which |
| 52 | occupy a range of indices. Once stored to, looking up any index in |
| 53 | the range will return the same entry as looking up any other index in |
Matthew Wilcox (Oracle) | 6b81141 | 2019-11-08 23:45:56 -0500 | [diff] [blame] | 54 | the range. Storing to any index will store to all of them. Multi-index |
| 55 | entries can be explicitly split into smaller entries, or storing ``NULL`` |
| 56 | into any entry will cause the XArray to forget about the range. |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 57 | |
| 58 | Normal API |
| 59 | ========== |
| 60 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 61 | Start by initialising an XArray, either with DEFINE_XARRAY() |
| 62 | for statically allocated XArrays or xa_init() for dynamically |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 63 | allocated ones. A freshly-initialised XArray contains a ``NULL`` |
| 64 | pointer at every index. |
| 65 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 66 | You can then set entries using xa_store() and get entries |
| 67 | using xa_load(). xa_store will overwrite any entry with the |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 68 | new entry and return the previous entry stored at that index. You can |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 69 | use xa_erase() instead of calling xa_store() with a |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 70 | ``NULL`` entry. There is no difference between an entry that has never |
Matthew Wilcox | 804dfaf | 2018-11-05 16:37:15 -0500 | [diff] [blame] | 71 | been stored to, one that has been erased and one that has most recently |
| 72 | had ``NULL`` stored to it. |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 73 | |
| 74 | You can conditionally replace an entry at an index by using |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 75 | xa_cmpxchg(). Like cmpxchg(), it will only succeed if |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 76 | the entry at that index has the 'old' value. It also returns the entry |
| 77 | which was at that index; if it returns the same entry which was passed as |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 78 | 'old', then xa_cmpxchg() succeeded. |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 79 | |
| 80 | If you want to only store a new entry to an index if the current entry |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 81 | at that index is ``NULL``, you can use xa_insert() which |
Matthew Wilcox | fd9dc93 | 2019-02-06 13:07:11 -0500 | [diff] [blame] | 82 | returns ``-EBUSY`` if the entry is not empty. |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 83 | |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 84 | You can copy entries out of the XArray into a plain array by calling |
Matthew Wilcox (Oracle) | 00ed452 | 2020-01-12 15:54:10 -0500 | [diff] [blame] | 85 | xa_extract(). Or you can iterate over the present entries in the XArray |
| 86 | by calling xa_for_each(), xa_for_each_start() or xa_for_each_range(). |
| 87 | You may prefer to use xa_find() or xa_find_after() to move to the next |
| 88 | present entry in the XArray. |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 89 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 90 | Calling xa_store_range() stores the same entry in a range |
Matthew Wilcox | 0e9446c | 2018-08-15 14:13:29 -0400 | [diff] [blame] | 91 | of indices. If you do this, some of the other operations will behave |
| 92 | in a slightly odd way. For example, marking the entry at one index |
| 93 | may result in the entry being marked at some, but not all of the other |
| 94 | indices. Storing into one index may result in the entry retrieved by |
| 95 | some, but not all of the other indices changing. |
| 96 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 97 | Sometimes you need to ensure that a subsequent call to xa_store() |
| 98 | will not need to allocate memory. The xa_reserve() function |
Matthew Wilcox | b0606fe | 2019-01-02 13:57:03 -0500 | [diff] [blame] | 99 | will store a reserved entry at the indicated index. Users of the |
| 100 | normal API will see this entry as containing ``NULL``. If you do |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 101 | not need to use the reserved entry, you can call xa_release() |
Matthew Wilcox | b0606fe | 2019-01-02 13:57:03 -0500 | [diff] [blame] | 102 | to remove the unused entry. If another user has stored to the entry |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 103 | in the meantime, xa_release() will do nothing; if instead you |
| 104 | want the entry to become ``NULL``, you should use xa_erase(). |
| 105 | Using xa_insert() on a reserved entry will fail. |
Matthew Wilcox | 4c0608f | 2018-10-30 09:45:55 -0400 | [diff] [blame] | 106 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 107 | If all entries in the array are ``NULL``, the xa_empty() function |
Matthew Wilcox | 804dfaf | 2018-11-05 16:37:15 -0500 | [diff] [blame] | 108 | will return ``true``. |
| 109 | |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 110 | Finally, you can remove all entries from an XArray by calling |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 111 | xa_destroy(). If the XArray entries are pointers, you may wish |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 112 | to free the entries first. You can do this by iterating over all present |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 113 | entries in the XArray using the xa_for_each() iterator. |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 114 | |
Matthew Wilcox (Oracle) | 6b81141 | 2019-11-08 23:45:56 -0500 | [diff] [blame] | 115 | Search Marks |
| 116 | ------------ |
| 117 | |
| 118 | Each entry in the array has three bits associated with it called marks. |
| 119 | Each mark may be set or cleared independently of the others. You can |
| 120 | iterate over marked entries by using the xa_for_each_marked() iterator. |
| 121 | |
| 122 | You can enquire whether a mark is set on an entry by using |
| 123 | xa_get_mark(). If the entry is not ``NULL``, you can set a mark on it |
| 124 | by using xa_set_mark() and remove the mark from an entry by calling |
| 125 | xa_clear_mark(). You can ask whether any entry in the XArray has a |
| 126 | particular mark set by calling xa_marked(). Erasing an entry from the |
| 127 | XArray causes all marks associated with that entry to be cleared. |
| 128 | |
| 129 | Setting or clearing a mark on any index of a multi-index entry will |
| 130 | affect all indices covered by that entry. Querying the mark on any |
| 131 | index will return the same result. |
| 132 | |
| 133 | There is no way to iterate over entries which are not marked; the data |
| 134 | structure does not allow this to be implemented efficiently. There are |
| 135 | not currently iterators to search for logical combinations of bits (eg |
| 136 | iterate over all entries which have both ``XA_MARK_1`` and ``XA_MARK_2`` |
| 137 | set, or iterate over all entries which have ``XA_MARK_0`` or ``XA_MARK_2`` |
| 138 | set). It would be possible to add these if a user arises. |
| 139 | |
Matthew Wilcox | d9c4804 | 2018-11-05 16:15:56 -0500 | [diff] [blame] | 140 | Allocating XArrays |
| 141 | ------------------ |
| 142 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 143 | If you use DEFINE_XARRAY_ALLOC() to define the XArray, or |
| 144 | initialise it by passing ``XA_FLAGS_ALLOC`` to xa_init_flags(), |
Matthew Wilcox | d9c4804 | 2018-11-05 16:15:56 -0500 | [diff] [blame] | 145 | the XArray changes to track whether entries are in use or not. |
Matthew Wilcox | 371c752 | 2018-07-04 10:50:12 -0400 | [diff] [blame] | 146 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 147 | You can call xa_alloc() to store the entry at an unused index |
Matthew Wilcox | 371c752 | 2018-07-04 10:50:12 -0400 | [diff] [blame] | 148 | in the XArray. If you need to modify the array from interrupt context, |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 149 | you can use xa_alloc_bh() or xa_alloc_irq() to disable |
Matthew Wilcox | d9c4804 | 2018-11-05 16:15:56 -0500 | [diff] [blame] | 150 | interrupts while allocating the ID. |
| 151 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 152 | Using xa_store(), xa_cmpxchg() or xa_insert() will |
Matthew Wilcox | 3ccaf57 | 2018-10-26 14:43:22 -0400 | [diff] [blame] | 153 | also mark the entry as being allocated. Unlike a normal XArray, storing |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 154 | ``NULL`` will mark the entry as being in use, like xa_reserve(). |
| 155 | To free an entry, use xa_erase() (or xa_release() if |
Matthew Wilcox | d9c4804 | 2018-11-05 16:15:56 -0500 | [diff] [blame] | 156 | you only want to free the entry if it's ``NULL``). |
| 157 | |
Matthew Wilcox | 3ccaf57 | 2018-10-26 14:43:22 -0400 | [diff] [blame] | 158 | By default, the lowest free entry is allocated starting from 0. If you |
| 159 | want to allocate entries starting at 1, it is more efficient to use |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 160 | DEFINE_XARRAY_ALLOC1() or ``XA_FLAGS_ALLOC1``. If you want to |
Matthew Wilcox | 2fa044e | 2018-11-06 14:13:35 -0500 | [diff] [blame] | 161 | allocate IDs up to a maximum, then wrap back around to the lowest free |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 162 | ID, you can use xa_alloc_cyclic(). |
Matthew Wilcox | 3ccaf57 | 2018-10-26 14:43:22 -0400 | [diff] [blame] | 163 | |
Matthew Wilcox | d9c4804 | 2018-11-05 16:15:56 -0500 | [diff] [blame] | 164 | You cannot use ``XA_MARK_0`` with an allocating XArray as this mark |
| 165 | is used to track whether an entry is free or not. The other marks are |
| 166 | available for your use. |
Matthew Wilcox | 371c752 | 2018-07-04 10:50:12 -0400 | [diff] [blame] | 167 | |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 168 | Memory allocation |
| 169 | ----------------- |
| 170 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 171 | The xa_store(), xa_cmpxchg(), xa_alloc(), |
| 172 | xa_reserve() and xa_insert() functions take a gfp_t |
Matthew Wilcox | 371c752 | 2018-07-04 10:50:12 -0400 | [diff] [blame] | 173 | parameter in case the XArray needs to allocate memory to store this entry. |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 174 | If the entry is being deleted, no memory allocation needs to be performed, |
| 175 | and the GFP flags specified will be ignored. |
| 176 | |
| 177 | It is possible for no memory to be allocatable, particularly if you pass |
| 178 | a restrictive set of GFP flags. In that case, the functions return a |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 179 | special value which can be turned into an errno using xa_err(). |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 180 | If you don't need to know exactly which error occurred, using |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 181 | xa_is_err() is slightly more efficient. |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 182 | |
| 183 | Locking |
| 184 | ------- |
| 185 | |
| 186 | When using the Normal API, you do not have to worry about locking. |
| 187 | The XArray uses RCU and an internal spinlock to synchronise access: |
| 188 | |
| 189 | No lock needed: |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 190 | * xa_empty() |
| 191 | * xa_marked() |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 192 | |
| 193 | Takes RCU read lock: |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 194 | * xa_load() |
| 195 | * xa_for_each() |
Matthew Wilcox (Oracle) | 00ed452 | 2020-01-12 15:54:10 -0500 | [diff] [blame] | 196 | * xa_for_each_start() |
| 197 | * xa_for_each_range() |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 198 | * xa_find() |
| 199 | * xa_find_after() |
| 200 | * xa_extract() |
| 201 | * xa_get_mark() |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 202 | |
| 203 | Takes xa_lock internally: |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 204 | * xa_store() |
| 205 | * xa_store_bh() |
| 206 | * xa_store_irq() |
| 207 | * xa_insert() |
| 208 | * xa_insert_bh() |
| 209 | * xa_insert_irq() |
| 210 | * xa_erase() |
| 211 | * xa_erase_bh() |
| 212 | * xa_erase_irq() |
| 213 | * xa_cmpxchg() |
| 214 | * xa_cmpxchg_bh() |
| 215 | * xa_cmpxchg_irq() |
| 216 | * xa_store_range() |
| 217 | * xa_alloc() |
| 218 | * xa_alloc_bh() |
| 219 | * xa_alloc_irq() |
| 220 | * xa_reserve() |
| 221 | * xa_reserve_bh() |
| 222 | * xa_reserve_irq() |
| 223 | * xa_destroy() |
| 224 | * xa_set_mark() |
| 225 | * xa_clear_mark() |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 226 | |
| 227 | Assumes xa_lock held on entry: |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 228 | * __xa_store() |
| 229 | * __xa_insert() |
| 230 | * __xa_erase() |
| 231 | * __xa_cmpxchg() |
| 232 | * __xa_alloc() |
| 233 | * __xa_set_mark() |
| 234 | * __xa_clear_mark() |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 235 | |
| 236 | If you want to take advantage of the lock to protect the data structures |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 237 | that you are storing in the XArray, you can call xa_lock() |
| 238 | before calling xa_load(), then take a reference count on the |
| 239 | object you have found before calling xa_unlock(). This will |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 240 | prevent stores from removing the object from the array between looking |
| 241 | up the object and incrementing the refcount. You can also use RCU to |
| 242 | avoid dereferencing freed memory, but an explanation of that is beyond |
| 243 | the scope of this document. |
| 244 | |
| 245 | The XArray does not disable interrupts or softirqs while modifying |
| 246 | the array. It is safe to read the XArray from interrupt or softirq |
| 247 | context as the RCU lock provides enough protection. |
| 248 | |
| 249 | If, for example, you want to store entries in the XArray in process |
| 250 | context and then erase them in softirq context, you can do that this way:: |
| 251 | |
| 252 | void foo_init(struct foo *foo) |
| 253 | { |
| 254 | xa_init_flags(&foo->array, XA_FLAGS_LOCK_BH); |
| 255 | } |
| 256 | |
| 257 | int foo_store(struct foo *foo, unsigned long index, void *entry) |
| 258 | { |
| 259 | int err; |
| 260 | |
| 261 | xa_lock_bh(&foo->array); |
| 262 | err = xa_err(__xa_store(&foo->array, index, entry, GFP_KERNEL)); |
| 263 | if (!err) |
| 264 | foo->count++; |
| 265 | xa_unlock_bh(&foo->array); |
| 266 | return err; |
| 267 | } |
| 268 | |
| 269 | /* foo_erase() is only called from softirq context */ |
| 270 | void foo_erase(struct foo *foo, unsigned long index) |
| 271 | { |
| 272 | xa_lock(&foo->array); |
| 273 | __xa_erase(&foo->array, index); |
| 274 | foo->count--; |
| 275 | xa_unlock(&foo->array); |
| 276 | } |
| 277 | |
| 278 | If you are going to modify the XArray from interrupt or softirq context, |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 279 | you need to initialise the array using xa_init_flags(), passing |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 280 | ``XA_FLAGS_LOCK_IRQ`` or ``XA_FLAGS_LOCK_BH``. |
| 281 | |
| 282 | The above example also shows a common pattern of wanting to extend the |
| 283 | coverage of the xa_lock on the store side to protect some statistics |
| 284 | associated with the array. |
| 285 | |
| 286 | Sharing the XArray with interrupt context is also possible, either |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 287 | using xa_lock_irqsave() in both the interrupt handler and process |
| 288 | context, or xa_lock_irq() in process context and xa_lock() |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 289 | in the interrupt handler. Some of the more common patterns have helper |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 290 | functions such as xa_store_bh(), xa_store_irq(), |
| 291 | xa_erase_bh(), xa_erase_irq(), xa_cmpxchg_bh() |
| 292 | and xa_cmpxchg_irq(). |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 293 | |
| 294 | Sometimes you need to protect access to the XArray with a mutex because |
| 295 | that lock sits above another mutex in the locking hierarchy. That does |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 296 | not entitle you to use functions like __xa_erase() without taking |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 297 | the xa_lock; the xa_lock is used for lockdep validation and will be used |
| 298 | for other purposes in the future. |
| 299 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 300 | The __xa_set_mark() and __xa_clear_mark() functions are also |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 301 | available for situations where you look up an entry and want to atomically |
| 302 | set or clear a mark. It may be more efficient to use the advanced API |
| 303 | in this case, as it will save you from walking the tree twice. |
| 304 | |
| 305 | Advanced API |
| 306 | ============ |
| 307 | |
| 308 | The advanced API offers more flexibility and better performance at the |
| 309 | cost of an interface which can be harder to use and has fewer safeguards. |
| 310 | No locking is done for you by the advanced API, and you are required |
| 311 | to use the xa_lock while modifying the array. You can choose whether |
| 312 | to use the xa_lock or the RCU lock while doing read-only operations on |
| 313 | the array. You can mix advanced and normal operations on the same array; |
| 314 | indeed the normal API is implemented in terms of the advanced API. The |
| 315 | advanced API is only available to modules with a GPL-compatible license. |
| 316 | |
| 317 | The advanced API is based around the xa_state. This is an opaque data |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 318 | structure which you declare on the stack using the XA_STATE() |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 319 | macro. This macro initialises the xa_state ready to start walking |
| 320 | around the XArray. It is used as a cursor to maintain the position |
| 321 | in the XArray and let you compose various operations together without |
| 322 | having to restart from the top every time. |
| 323 | |
| 324 | The xa_state is also used to store errors. You can call |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 325 | xas_error() to retrieve the error. All operations check whether |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 326 | the xa_state is in an error state before proceeding, so there's no need |
| 327 | for you to check for an error after each call; you can make multiple |
| 328 | calls in succession and only check at a convenient point. The only |
| 329 | errors currently generated by the XArray code itself are ``ENOMEM`` and |
| 330 | ``EINVAL``, but it supports arbitrary errors in case you want to call |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 331 | xas_set_err() yourself. |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 332 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 333 | If the xa_state is holding an ``ENOMEM`` error, calling xas_nomem() |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 334 | will attempt to allocate more memory using the specified gfp flags and |
| 335 | cache it in the xa_state for the next attempt. The idea is that you take |
| 336 | the xa_lock, attempt the operation and drop the lock. The operation |
| 337 | attempts to allocate memory while holding the lock, but it is more |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 338 | likely to fail. Once you have dropped the lock, xas_nomem() |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 339 | can try harder to allocate more memory. It will return ``true`` if it |
| 340 | is worth retrying the operation (i.e. that there was a memory error *and* |
| 341 | more memory was allocated). If it has previously allocated memory, and |
| 342 | that memory wasn't used, and there is no error (or some error that isn't |
| 343 | ``ENOMEM``), then it will free the memory previously allocated. |
| 344 | |
| 345 | Internal Entries |
| 346 | ---------------- |
| 347 | |
| 348 | The XArray reserves some entries for its own purposes. These are never |
| 349 | exposed through the normal API, but when using the advanced API, it's |
| 350 | possible to see them. Usually the best way to handle them is to pass them |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 351 | to xas_retry(), and retry the operation if it returns ``true``. |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 352 | |
| 353 | .. flat-table:: |
| 354 | :widths: 1 1 6 |
| 355 | |
| 356 | * - Name |
| 357 | - Test |
| 358 | - Usage |
| 359 | |
| 360 | * - Node |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 361 | - xa_is_node() |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 362 | - An XArray node. May be visible when using a multi-index xa_state. |
| 363 | |
| 364 | * - Sibling |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 365 | - xa_is_sibling() |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 366 | - A non-canonical entry for a multi-index entry. The value indicates |
| 367 | which slot in this node has the canonical entry. |
| 368 | |
| 369 | * - Retry |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 370 | - xa_is_retry() |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 371 | - This entry is currently being modified by a thread which has the |
| 372 | xa_lock. The node containing this entry may be freed at the end |
| 373 | of this RCU period. You should restart the lookup from the head |
| 374 | of the array. |
| 375 | |
Matthew Wilcox | 9f14d4f | 2018-10-01 14:54:59 -0400 | [diff] [blame] | 376 | * - Zero |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 377 | - xa_is_zero() |
Matthew Wilcox | 9f14d4f | 2018-10-01 14:54:59 -0400 | [diff] [blame] | 378 | - Zero entries appear as ``NULL`` through the Normal API, but occupy |
| 379 | an entry in the XArray which can be used to reserve the index for |
Matthew Wilcox | d9c4804 | 2018-11-05 16:15:56 -0500 | [diff] [blame] | 380 | future use. This is used by allocating XArrays for allocated entries |
| 381 | which are ``NULL``. |
Matthew Wilcox | 9f14d4f | 2018-10-01 14:54:59 -0400 | [diff] [blame] | 382 | |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 383 | Other internal entries may be added in the future. As far as possible, they |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 384 | will be handled by xas_retry(). |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 385 | |
| 386 | Additional functionality |
| 387 | ------------------------ |
| 388 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 389 | The xas_create_range() function allocates all the necessary memory |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 390 | to store every entry in a range. It will set ENOMEM in the xa_state if |
| 391 | it cannot allocate memory. |
| 392 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 393 | You can use xas_init_marks() to reset the marks on an entry |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 394 | to their default state. This is usually all marks clear, unless the |
| 395 | XArray is marked with ``XA_FLAGS_TRACK_FREE``, in which case mark 0 is set |
| 396 | and all other marks are clear. Replacing one entry with another using |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 397 | xas_store() will not reset the marks on that entry; if you want |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 398 | the marks reset, you should do that explicitly. |
| 399 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 400 | The xas_load() will walk the xa_state as close to the entry |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 401 | as it can. If you know the xa_state has already been walked to the |
| 402 | entry and need to check that the entry hasn't changed, you can use |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 403 | xas_reload() to save a function call. |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 404 | |
| 405 | If you need to move to a different index in the XArray, call |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 406 | xas_set(). This resets the cursor to the top of the tree, which |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 407 | will generally make the next operation walk the cursor to the desired |
| 408 | spot in the tree. If you want to move to the next or previous index, |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 409 | call xas_next() or xas_prev(). Setting the index does |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 410 | not walk the cursor around the array so does not require a lock to be |
| 411 | held, while moving to the next or previous index does. |
| 412 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 413 | You can search for the next present entry using xas_find(). This |
| 414 | is the equivalent of both xa_find() and xa_find_after(); |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 415 | if the cursor has been walked to an entry, then it will find the next |
| 416 | entry after the one currently referenced. If not, it will return the |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 417 | entry at the index of the xa_state. Using xas_next_entry() to |
| 418 | move to the next present entry instead of xas_find() will save |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 419 | a function call in the majority of cases at the expense of emitting more |
| 420 | inline code. |
| 421 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 422 | The xas_find_marked() function is similar. If the xa_state has |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 423 | not been walked, it will return the entry at the index of the xa_state, |
| 424 | if it is marked. Otherwise, it will return the first marked entry after |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 425 | the entry referenced by the xa_state. The xas_next_marked() |
| 426 | function is the equivalent of xas_next_entry(). |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 427 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 428 | When iterating over a range of the XArray using xas_for_each() |
| 429 | or xas_for_each_marked(), it may be necessary to temporarily stop |
| 430 | the iteration. The xas_pause() function exists for this purpose. |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 431 | After you have done the necessary work and wish to resume, the xa_state |
| 432 | is in an appropriate state to continue the iteration after the entry |
| 433 | you last processed. If you have interrupts disabled while iterating, |
| 434 | then it is good manners to pause the iteration and reenable interrupts |
| 435 | every ``XA_CHECK_SCHED`` entries. |
| 436 | |
Matthew Wilcox (Oracle) | 6b81141 | 2019-11-08 23:45:56 -0500 | [diff] [blame] | 437 | The xas_get_mark(), xas_set_mark() and xas_clear_mark() functions require |
| 438 | the xa_state cursor to have been moved to the appropriate location in the |
| 439 | XArray; they will do nothing if you have called xas_pause() or xas_set() |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 440 | immediately before. |
| 441 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 442 | You can call xas_set_update() to have a callback function |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 443 | called each time the XArray updates a node. This is used by the page |
| 444 | cache workingset code to maintain its list of nodes which contain only |
| 445 | shadow entries. |
| 446 | |
| 447 | Multi-Index Entries |
| 448 | ------------------- |
| 449 | |
| 450 | The XArray has the ability to tie multiple indices together so that |
| 451 | operations on one index affect all indices. For example, storing into |
| 452 | any index will change the value of the entry retrieved from any index. |
| 453 | Setting or clearing a mark on any index will set or clear the mark |
| 454 | on every index that is tied together. The current implementation |
| 455 | only allows tying ranges which are aligned powers of two together; |
| 456 | eg indices 64-127 may be tied together, but 2-6 may not be. This may |
| 457 | save substantial quantities of memory; for example tying 512 entries |
| 458 | together will save over 4kB. |
| 459 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 460 | You can create a multi-index entry by using XA_STATE_ORDER() |
| 461 | or xas_set_order() followed by a call to xas_store(). |
| 462 | Calling xas_load() with a multi-index xa_state will walk the |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 463 | xa_state to the right location in the tree, but the return value is not |
| 464 | meaningful, potentially being an internal entry or ``NULL`` even when there |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 465 | is an entry stored within the range. Calling xas_find_conflict() |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 466 | will return the first entry within the range or ``NULL`` if there are no |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 467 | entries in the range. The xas_for_each_conflict() iterator will |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 468 | iterate over every entry which overlaps the specified range. |
| 469 | |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 470 | If xas_load() encounters a multi-index entry, the xa_index |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 471 | in the xa_state will not be changed. When iterating over an XArray |
Jonathan Corbet | 9c79df7 | 2019-04-25 13:48:13 -0600 | [diff] [blame] | 472 | or calling xas_find(), if the initial index is in the middle |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 473 | of a multi-index entry, it will not be altered. Subsequent calls |
| 474 | or iterations will move the index to the first index in the range. |
| 475 | Each entry will only be returned once, no matter how many indices it |
| 476 | occupies. |
| 477 | |
Matthew Wilcox (Oracle) | 8fc7564 | 2020-10-15 20:05:16 -0700 | [diff] [blame] | 478 | Using xas_next() or xas_prev() with a multi-index xa_state is not |
| 479 | supported. Using either of these functions on a multi-index entry will |
| 480 | reveal sibling entries; these should be skipped over by the caller. |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 481 | |
Matthew Wilcox (Oracle) | 8fc7564 | 2020-10-15 20:05:16 -0700 | [diff] [blame] | 482 | Storing ``NULL`` into any index of a multi-index entry will set the |
| 483 | entry at every index to ``NULL`` and dissolve the tie. A multi-index |
| 484 | entry can be split into entries occupying smaller ranges by calling |
| 485 | xas_split_alloc() without the xa_lock held, followed by taking the lock |
| 486 | and calling xas_split(). |
Matthew Wilcox | 992a8e6 | 2017-11-23 22:57:20 -0500 | [diff] [blame] | 487 | |
| 488 | Functions and structures |
| 489 | ======================== |
| 490 | |
| 491 | .. kernel-doc:: include/linux/xarray.h |
| 492 | .. kernel-doc:: lib/xarray.c |