1d96e5996SHans Petter Selasky /*- 2d96e5996SHans Petter Selasky * Copyright (c) 2020 Mellanox Technologies, Ltd. 3d96e5996SHans Petter Selasky * All rights reserved. 4d96e5996SHans Petter Selasky * 5d96e5996SHans Petter Selasky * Redistribution and use in source and binary forms, with or without 6d96e5996SHans Petter Selasky * modification, are permitted provided that the following conditions 7d96e5996SHans Petter Selasky * are met: 8d96e5996SHans Petter Selasky * 1. Redistributions of source code must retain the above copyright 9d96e5996SHans Petter Selasky * notice unmodified, this list of conditions, and the following 10d96e5996SHans Petter Selasky * disclaimer. 11d96e5996SHans Petter Selasky * 2. Redistributions in binary form must reproduce the above copyright 12d96e5996SHans Petter Selasky * notice, this list of conditions and the following disclaimer in the 13d96e5996SHans Petter Selasky * documentation and/or other materials provided with the distribution. 14d96e5996SHans Petter Selasky * 15d96e5996SHans Petter Selasky * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 16d96e5996SHans Petter Selasky * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 17d96e5996SHans Petter Selasky * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 18d96e5996SHans Petter Selasky * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 19d96e5996SHans Petter Selasky * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 20d96e5996SHans Petter Selasky * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 21d96e5996SHans Petter Selasky * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 22d96e5996SHans Petter Selasky * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 23d96e5996SHans Petter Selasky * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 24d96e5996SHans Petter Selasky * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 25d96e5996SHans Petter Selasky */ 26d96e5996SHans Petter Selasky 27d96e5996SHans Petter Selasky #include <sys/cdefs.h> 28d96e5996SHans Petter Selasky #include <linux/xarray.h> 29d96e5996SHans Petter Selasky 30d96e5996SHans Petter Selasky #include <vm/vm_pageout.h> 31d96e5996SHans Petter Selasky 32d96e5996SHans Petter Selasky /* 333102ea3bSJean-Sébastien Pédron * Linux' XArray allows to store a NULL pointer as a value. xa_load() would 343102ea3bSJean-Sébastien Pédron * return NULL for both an unused index and an index set to NULL. But it 353102ea3bSJean-Sébastien Pédron * impacts xa_alloc() which needs to find the next available index. 363102ea3bSJean-Sébastien Pédron * 373102ea3bSJean-Sébastien Pédron * However, our implementation relies on a radix tree (see `linux_radix.c`) 383102ea3bSJean-Sébastien Pédron * which does not accept NULL pointers as values. I'm not sure this is a 393102ea3bSJean-Sébastien Pédron * limitation or a feature, so to work around this, a NULL value is replaced by 403102ea3bSJean-Sébastien Pédron * `NULL_VALUE`, an unlikely address, when we pass it to linux_radix. 413102ea3bSJean-Sébastien Pédron */ 423102ea3bSJean-Sébastien Pédron #define NULL_VALUE (void *)0x1 433102ea3bSJean-Sébastien Pédron 443102ea3bSJean-Sébastien Pédron /* 45d96e5996SHans Petter Selasky * This function removes the element at the given index and returns 46d96e5996SHans Petter Selasky * the pointer to the removed element, if any. 47d96e5996SHans Petter Selasky */ 48d96e5996SHans Petter Selasky void * 49d96e5996SHans Petter Selasky __xa_erase(struct xarray *xa, uint32_t index) 50d96e5996SHans Petter Selasky { 513102ea3bSJean-Sébastien Pédron void *retval; 523102ea3bSJean-Sébastien Pédron 53d96e5996SHans Petter Selasky XA_ASSERT_LOCKED(xa); 54d96e5996SHans Petter Selasky 55*be818f26SJean-Sébastien Pédron retval = radix_tree_delete(&xa->xa_head, index); 563102ea3bSJean-Sébastien Pédron if (retval == NULL_VALUE) 573102ea3bSJean-Sébastien Pédron retval = NULL; 583102ea3bSJean-Sébastien Pédron 593102ea3bSJean-Sébastien Pédron return (retval); 60d96e5996SHans Petter Selasky } 61d96e5996SHans Petter Selasky 62d96e5996SHans Petter Selasky void * 63d96e5996SHans Petter Selasky xa_erase(struct xarray *xa, uint32_t index) 64d96e5996SHans Petter Selasky { 65d96e5996SHans Petter Selasky void *retval; 66d96e5996SHans Petter Selasky 67d96e5996SHans Petter Selasky xa_lock(xa); 68d96e5996SHans Petter Selasky retval = __xa_erase(xa, index); 69d96e5996SHans Petter Selasky xa_unlock(xa); 70d96e5996SHans Petter Selasky 71d96e5996SHans Petter Selasky return (retval); 72d96e5996SHans Petter Selasky } 73d96e5996SHans Petter Selasky 74d96e5996SHans Petter Selasky /* 75d96e5996SHans Petter Selasky * This function returns the element pointer at the given index. A 76d96e5996SHans Petter Selasky * value of NULL is returned if the element does not exist. 77d96e5996SHans Petter Selasky */ 78d96e5996SHans Petter Selasky void * 79d96e5996SHans Petter Selasky xa_load(struct xarray *xa, uint32_t index) 80d96e5996SHans Petter Selasky { 81d96e5996SHans Petter Selasky void *retval; 82d96e5996SHans Petter Selasky 83d96e5996SHans Petter Selasky xa_lock(xa); 84*be818f26SJean-Sébastien Pédron retval = radix_tree_lookup(&xa->xa_head, index); 85d96e5996SHans Petter Selasky xa_unlock(xa); 86d96e5996SHans Petter Selasky 873102ea3bSJean-Sébastien Pédron if (retval == NULL_VALUE) 883102ea3bSJean-Sébastien Pédron retval = NULL; 893102ea3bSJean-Sébastien Pédron 90d96e5996SHans Petter Selasky return (retval); 91d96e5996SHans Petter Selasky } 92d96e5996SHans Petter Selasky 93d96e5996SHans Petter Selasky /* 94d96e5996SHans Petter Selasky * This is an internal function used to sleep until more memory 95d96e5996SHans Petter Selasky * becomes available. 96d96e5996SHans Petter Selasky */ 97d96e5996SHans Petter Selasky static void 98d96e5996SHans Petter Selasky xa_vm_wait_locked(struct xarray *xa) 99d96e5996SHans Petter Selasky { 100d96e5996SHans Petter Selasky xa_unlock(xa); 101d96e5996SHans Petter Selasky vm_wait(NULL); 102d96e5996SHans Petter Selasky xa_lock(xa); 103d96e5996SHans Petter Selasky } 104d96e5996SHans Petter Selasky 105d96e5996SHans Petter Selasky /* 106d96e5996SHans Petter Selasky * This function iterates the xarray until it finds a free slot where 107d96e5996SHans Petter Selasky * it can insert the element pointer to by "ptr". It starts at the 108d96e5996SHans Petter Selasky * index pointed to by "pindex" and updates this value at return. The 109d96e5996SHans Petter Selasky * "mask" argument defines the maximum index allowed, inclusivly, and 110d96e5996SHans Petter Selasky * must be a power of two minus one value. The "gfp" argument 111d96e5996SHans Petter Selasky * basically tells if we can wait for more memory to become available 112d96e5996SHans Petter Selasky * or not. This function returns zero upon success or a negative error 113d96e5996SHans Petter Selasky * code on failure. A typical error code is -ENOMEM which means either 114d96e5996SHans Petter Selasky * the xarray is full, or there was not enough internal memory 115d96e5996SHans Petter Selasky * available to complete the radix tree insertion. 116d96e5996SHans Petter Selasky */ 117d96e5996SHans Petter Selasky int 118d96e5996SHans Petter Selasky __xa_alloc(struct xarray *xa, uint32_t *pindex, void *ptr, uint32_t mask, gfp_t gfp) 119d96e5996SHans Petter Selasky { 120d96e5996SHans Petter Selasky int retval; 121d96e5996SHans Petter Selasky 122d96e5996SHans Petter Selasky XA_ASSERT_LOCKED(xa); 123d96e5996SHans Petter Selasky 124e705066cSVladimir Kondratyev /* mask should allow to allocate at least one item */ 125*be818f26SJean-Sébastien Pédron MPASS(mask > ((xa->xa_flags & XA_FLAGS_ALLOC1) != 0 ? 1 : 0)); 126d96e5996SHans Petter Selasky 127d96e5996SHans Petter Selasky /* mask can be any power of two value minus one */ 128d96e5996SHans Petter Selasky MPASS((mask & (mask + 1)) == 0); 129d96e5996SHans Petter Selasky 130*be818f26SJean-Sébastien Pédron *pindex = (xa->xa_flags & XA_FLAGS_ALLOC1) != 0 ? 1 : 0; 1313102ea3bSJean-Sébastien Pédron if (ptr == NULL) 1323102ea3bSJean-Sébastien Pédron ptr = NULL_VALUE; 133d96e5996SHans Petter Selasky retry: 134*be818f26SJean-Sébastien Pédron retval = radix_tree_insert(&xa->xa_head, *pindex, ptr); 135d96e5996SHans Petter Selasky 136d96e5996SHans Petter Selasky switch (retval) { 137d96e5996SHans Petter Selasky case -EEXIST: 138d96e5996SHans Petter Selasky if (likely(*pindex != mask)) { 139d96e5996SHans Petter Selasky (*pindex)++; 140d96e5996SHans Petter Selasky goto retry; 141d96e5996SHans Petter Selasky } 142d96e5996SHans Petter Selasky retval = -ENOMEM; 143d96e5996SHans Petter Selasky break; 144d96e5996SHans Petter Selasky case -ENOMEM: 145d96e5996SHans Petter Selasky if (likely(gfp & M_WAITOK)) { 146d96e5996SHans Petter Selasky xa_vm_wait_locked(xa); 147d96e5996SHans Petter Selasky goto retry; 148d96e5996SHans Petter Selasky } 149d96e5996SHans Petter Selasky break; 150d96e5996SHans Petter Selasky default: 151d96e5996SHans Petter Selasky break; 152d96e5996SHans Petter Selasky } 153d96e5996SHans Petter Selasky return (retval); 154d96e5996SHans Petter Selasky } 155d96e5996SHans Petter Selasky 156d96e5996SHans Petter Selasky int 157d96e5996SHans Petter Selasky xa_alloc(struct xarray *xa, uint32_t *pindex, void *ptr, uint32_t mask, gfp_t gfp) 158d96e5996SHans Petter Selasky { 159d96e5996SHans Petter Selasky int retval; 160d96e5996SHans Petter Selasky 1613102ea3bSJean-Sébastien Pédron if (ptr == NULL) 1623102ea3bSJean-Sébastien Pédron ptr = NULL_VALUE; 1633102ea3bSJean-Sébastien Pédron 164d96e5996SHans Petter Selasky xa_lock(xa); 165d96e5996SHans Petter Selasky retval = __xa_alloc(xa, pindex, ptr, mask, gfp); 166d96e5996SHans Petter Selasky xa_unlock(xa); 167d96e5996SHans Petter Selasky 168d96e5996SHans Petter Selasky return (retval); 169d96e5996SHans Petter Selasky } 170d96e5996SHans Petter Selasky 171d96e5996SHans Petter Selasky /* 172d96e5996SHans Petter Selasky * This function works the same like the "xa_alloc" function, except 173d96e5996SHans Petter Selasky * it wraps the next index value to zero when there are no entries 174d96e5996SHans Petter Selasky * left at the end of the xarray searching for a free slot from the 175d96e5996SHans Petter Selasky * beginning of the array. If the xarray is full -ENOMEM is returned. 176d96e5996SHans Petter Selasky */ 177d96e5996SHans Petter Selasky int 178d96e5996SHans Petter Selasky __xa_alloc_cyclic(struct xarray *xa, uint32_t *pindex, void *ptr, uint32_t mask, 179d96e5996SHans Petter Selasky uint32_t *pnext_index, gfp_t gfp) 180d96e5996SHans Petter Selasky { 181d96e5996SHans Petter Selasky int retval; 182d96e5996SHans Petter Selasky int timeout = 1; 183d96e5996SHans Petter Selasky 184d96e5996SHans Petter Selasky XA_ASSERT_LOCKED(xa); 185d96e5996SHans Petter Selasky 186e705066cSVladimir Kondratyev /* mask should allow to allocate at least one item */ 187*be818f26SJean-Sébastien Pédron MPASS(mask > ((xa->xa_flags & XA_FLAGS_ALLOC1) != 0 ? 1 : 0)); 188d96e5996SHans Petter Selasky 189d96e5996SHans Petter Selasky /* mask can be any power of two value minus one */ 190d96e5996SHans Petter Selasky MPASS((mask & (mask + 1)) == 0); 191d96e5996SHans Petter Selasky 192*be818f26SJean-Sébastien Pédron *pnext_index = (xa->xa_flags & XA_FLAGS_ALLOC1) != 0 ? 1 : 0; 1933102ea3bSJean-Sébastien Pédron if (ptr == NULL) 1943102ea3bSJean-Sébastien Pédron ptr = NULL_VALUE; 195d96e5996SHans Petter Selasky retry: 196*be818f26SJean-Sébastien Pédron retval = radix_tree_insert(&xa->xa_head, *pnext_index, ptr); 197d96e5996SHans Petter Selasky 198d96e5996SHans Petter Selasky switch (retval) { 199d96e5996SHans Petter Selasky case -EEXIST: 200d96e5996SHans Petter Selasky if (unlikely(*pnext_index == mask) && !timeout--) { 201d96e5996SHans Petter Selasky retval = -ENOMEM; 202d96e5996SHans Petter Selasky break; 203d96e5996SHans Petter Selasky } 204d96e5996SHans Petter Selasky (*pnext_index)++; 205d96e5996SHans Petter Selasky (*pnext_index) &= mask; 206*be818f26SJean-Sébastien Pédron if (*pnext_index == 0 && (xa->xa_flags & XA_FLAGS_ALLOC1) != 0) 207e705066cSVladimir Kondratyev (*pnext_index)++; 208d96e5996SHans Petter Selasky goto retry; 209d96e5996SHans Petter Selasky case -ENOMEM: 210d96e5996SHans Petter Selasky if (likely(gfp & M_WAITOK)) { 211d96e5996SHans Petter Selasky xa_vm_wait_locked(xa); 212d96e5996SHans Petter Selasky goto retry; 213d96e5996SHans Petter Selasky } 214d96e5996SHans Petter Selasky break; 215d96e5996SHans Petter Selasky default: 216d96e5996SHans Petter Selasky break; 217d96e5996SHans Petter Selasky } 218d96e5996SHans Petter Selasky *pindex = *pnext_index; 219d96e5996SHans Petter Selasky 220d96e5996SHans Petter Selasky return (retval); 221d96e5996SHans Petter Selasky } 222d96e5996SHans Petter Selasky 223d96e5996SHans Petter Selasky int 224d96e5996SHans Petter Selasky xa_alloc_cyclic(struct xarray *xa, uint32_t *pindex, void *ptr, uint32_t mask, 225d96e5996SHans Petter Selasky uint32_t *pnext_index, gfp_t gfp) 226d96e5996SHans Petter Selasky { 227d96e5996SHans Petter Selasky int retval; 228d96e5996SHans Petter Selasky 229d96e5996SHans Petter Selasky xa_lock(xa); 230d96e5996SHans Petter Selasky retval = __xa_alloc_cyclic(xa, pindex, ptr, mask, pnext_index, gfp); 231d96e5996SHans Petter Selasky xa_unlock(xa); 232d96e5996SHans Petter Selasky 233d96e5996SHans Petter Selasky return (retval); 234d96e5996SHans Petter Selasky } 235d96e5996SHans Petter Selasky 236668fe90dSJean-Sébastien Pédron int 237668fe90dSJean-Sébastien Pédron xa_alloc_cyclic_irq(struct xarray *xa, uint32_t *pindex, void *ptr, 238668fe90dSJean-Sébastien Pédron uint32_t mask, uint32_t *pnext_index, gfp_t gfp) 239668fe90dSJean-Sébastien Pédron { 240668fe90dSJean-Sébastien Pédron int retval; 241668fe90dSJean-Sébastien Pédron 242668fe90dSJean-Sébastien Pédron xa_lock_irq(xa); 243668fe90dSJean-Sébastien Pédron retval = __xa_alloc_cyclic(xa, pindex, ptr, mask, pnext_index, gfp); 244668fe90dSJean-Sébastien Pédron xa_unlock_irq(xa); 245668fe90dSJean-Sébastien Pédron 246668fe90dSJean-Sébastien Pédron return (retval); 247668fe90dSJean-Sébastien Pédron } 248668fe90dSJean-Sébastien Pédron 249d96e5996SHans Petter Selasky /* 250d96e5996SHans Petter Selasky * This function tries to insert an element at the given index. The 251d96e5996SHans Petter Selasky * "gfp" argument basically decides of this function can sleep or not 252d96e5996SHans Petter Selasky * trying to allocate internal memory for its radix tree. The 253d96e5996SHans Petter Selasky * function returns an error code upon failure. Typical error codes 254d96e5996SHans Petter Selasky * are element exists (-EEXIST) or out of memory (-ENOMEM). 255d96e5996SHans Petter Selasky */ 256d96e5996SHans Petter Selasky int 257d96e5996SHans Petter Selasky __xa_insert(struct xarray *xa, uint32_t index, void *ptr, gfp_t gfp) 258d96e5996SHans Petter Selasky { 259d96e5996SHans Petter Selasky int retval; 260d96e5996SHans Petter Selasky 261d96e5996SHans Petter Selasky XA_ASSERT_LOCKED(xa); 2623102ea3bSJean-Sébastien Pédron if (ptr == NULL) 2633102ea3bSJean-Sébastien Pédron ptr = NULL_VALUE; 264d96e5996SHans Petter Selasky retry: 265*be818f26SJean-Sébastien Pédron retval = radix_tree_insert(&xa->xa_head, index, ptr); 266d96e5996SHans Petter Selasky 267d96e5996SHans Petter Selasky switch (retval) { 268d96e5996SHans Petter Selasky case -ENOMEM: 269d96e5996SHans Petter Selasky if (likely(gfp & M_WAITOK)) { 270d96e5996SHans Petter Selasky xa_vm_wait_locked(xa); 271d96e5996SHans Petter Selasky goto retry; 272d96e5996SHans Petter Selasky } 273d96e5996SHans Petter Selasky break; 274d96e5996SHans Petter Selasky default: 275d96e5996SHans Petter Selasky break; 276d96e5996SHans Petter Selasky } 277d96e5996SHans Petter Selasky return (retval); 278d96e5996SHans Petter Selasky } 279d96e5996SHans Petter Selasky 280d96e5996SHans Petter Selasky int 281d96e5996SHans Petter Selasky xa_insert(struct xarray *xa, uint32_t index, void *ptr, gfp_t gfp) 282d96e5996SHans Petter Selasky { 283d96e5996SHans Petter Selasky int retval; 284d96e5996SHans Petter Selasky 285d96e5996SHans Petter Selasky xa_lock(xa); 286d96e5996SHans Petter Selasky retval = __xa_insert(xa, index, ptr, gfp); 287d96e5996SHans Petter Selasky xa_unlock(xa); 288d96e5996SHans Petter Selasky 289d96e5996SHans Petter Selasky return (retval); 290d96e5996SHans Petter Selasky } 291d96e5996SHans Petter Selasky 292d96e5996SHans Petter Selasky /* 293d96e5996SHans Petter Selasky * This function updates the element at the given index and returns a 294d96e5996SHans Petter Selasky * pointer to the old element. The "gfp" argument basically decides of 295d96e5996SHans Petter Selasky * this function can sleep or not trying to allocate internal memory 296d96e5996SHans Petter Selasky * for its radix tree. The function returns an XA_ERROR() pointer code 297d96e5996SHans Petter Selasky * upon failure. Code using this function must always check if the 298d96e5996SHans Petter Selasky * return value is an XA_ERROR() code before using the returned value. 299d96e5996SHans Petter Selasky */ 300d96e5996SHans Petter Selasky void * 301d96e5996SHans Petter Selasky __xa_store(struct xarray *xa, uint32_t index, void *ptr, gfp_t gfp) 302d96e5996SHans Petter Selasky { 303d96e5996SHans Petter Selasky int retval; 304d96e5996SHans Petter Selasky 305d96e5996SHans Petter Selasky XA_ASSERT_LOCKED(xa); 3063102ea3bSJean-Sébastien Pédron if (ptr == NULL) 3073102ea3bSJean-Sébastien Pédron ptr = NULL_VALUE; 308d96e5996SHans Petter Selasky retry: 309*be818f26SJean-Sébastien Pédron retval = radix_tree_store(&xa->xa_head, index, &ptr); 310d96e5996SHans Petter Selasky 311d96e5996SHans Petter Selasky switch (retval) { 312d96e5996SHans Petter Selasky case 0: 3133102ea3bSJean-Sébastien Pédron if (ptr == NULL_VALUE) 3143102ea3bSJean-Sébastien Pédron ptr = NULL; 315d96e5996SHans Petter Selasky break; 316d96e5996SHans Petter Selasky case -ENOMEM: 317d96e5996SHans Petter Selasky if (likely(gfp & M_WAITOK)) { 318d96e5996SHans Petter Selasky xa_vm_wait_locked(xa); 319d96e5996SHans Petter Selasky goto retry; 320d96e5996SHans Petter Selasky } 321d96e5996SHans Petter Selasky ptr = XA_ERROR(retval); 322d96e5996SHans Petter Selasky break; 323d96e5996SHans Petter Selasky default: 324d96e5996SHans Petter Selasky ptr = XA_ERROR(retval); 325d96e5996SHans Petter Selasky break; 326d96e5996SHans Petter Selasky } 327d96e5996SHans Petter Selasky return (ptr); 328d96e5996SHans Petter Selasky } 329d96e5996SHans Petter Selasky 330d96e5996SHans Petter Selasky void * 331d96e5996SHans Petter Selasky xa_store(struct xarray *xa, uint32_t index, void *ptr, gfp_t gfp) 332d96e5996SHans Petter Selasky { 333d96e5996SHans Petter Selasky void *retval; 334d96e5996SHans Petter Selasky 335d96e5996SHans Petter Selasky xa_lock(xa); 336d96e5996SHans Petter Selasky retval = __xa_store(xa, index, ptr, gfp); 337d96e5996SHans Petter Selasky xa_unlock(xa); 338d96e5996SHans Petter Selasky 339d96e5996SHans Petter Selasky return (retval); 340d96e5996SHans Petter Selasky } 341d96e5996SHans Petter Selasky 342d96e5996SHans Petter Selasky /* 343d96e5996SHans Petter Selasky * This function initialize an xarray structure. 344d96e5996SHans Petter Selasky */ 345d96e5996SHans Petter Selasky void 346d96e5996SHans Petter Selasky xa_init_flags(struct xarray *xa, uint32_t flags) 347d96e5996SHans Petter Selasky { 348d96e5996SHans Petter Selasky memset(xa, 0, sizeof(*xa)); 349d96e5996SHans Petter Selasky 350*be818f26SJean-Sébastien Pédron mtx_init(&xa->xa_lock, "lkpi-xarray", NULL, MTX_DEF | MTX_RECURSE); 351*be818f26SJean-Sébastien Pédron xa->xa_head.gfp_mask = GFP_NOWAIT; 352*be818f26SJean-Sébastien Pédron xa->xa_flags = flags; 353d96e5996SHans Petter Selasky } 354d96e5996SHans Petter Selasky 355d96e5996SHans Petter Selasky /* 356d96e5996SHans Petter Selasky * This function destroys an xarray structure and all its internal 357d96e5996SHans Petter Selasky * memory and locks. 358d96e5996SHans Petter Selasky */ 359d96e5996SHans Petter Selasky void 360d96e5996SHans Petter Selasky xa_destroy(struct xarray *xa) 361d96e5996SHans Petter Selasky { 362d96e5996SHans Petter Selasky struct radix_tree_iter iter; 363d96e5996SHans Petter Selasky void **ppslot; 364d96e5996SHans Petter Selasky 365*be818f26SJean-Sébastien Pédron radix_tree_for_each_slot(ppslot, &xa->xa_head, &iter, 0) 366*be818f26SJean-Sébastien Pédron radix_tree_iter_delete(&xa->xa_head, &iter, ppslot); 367*be818f26SJean-Sébastien Pédron mtx_destroy(&xa->xa_lock); 368d96e5996SHans Petter Selasky } 369d96e5996SHans Petter Selasky 370d96e5996SHans Petter Selasky /* 371d96e5996SHans Petter Selasky * This function checks if an xarray is empty or not. 372d96e5996SHans Petter Selasky * It returns true if empty, else false. 373d96e5996SHans Petter Selasky */ 374d96e5996SHans Petter Selasky bool 375d96e5996SHans Petter Selasky __xa_empty(struct xarray *xa) 376d96e5996SHans Petter Selasky { 377d96e5996SHans Petter Selasky struct radix_tree_iter iter = {}; 378d96e5996SHans Petter Selasky void **temp; 379d96e5996SHans Petter Selasky 380d96e5996SHans Petter Selasky XA_ASSERT_LOCKED(xa); 381d96e5996SHans Petter Selasky 382*be818f26SJean-Sébastien Pédron return (!radix_tree_iter_find(&xa->xa_head, &iter, &temp)); 383d96e5996SHans Petter Selasky } 384d96e5996SHans Petter Selasky 385d96e5996SHans Petter Selasky bool 386d96e5996SHans Petter Selasky xa_empty(struct xarray *xa) 387d96e5996SHans Petter Selasky { 388d96e5996SHans Petter Selasky bool retval; 389d96e5996SHans Petter Selasky 390d96e5996SHans Petter Selasky xa_lock(xa); 391d96e5996SHans Petter Selasky retval = __xa_empty(xa); 392d96e5996SHans Petter Selasky xa_unlock(xa); 393d96e5996SHans Petter Selasky 394d96e5996SHans Petter Selasky return (retval); 395d96e5996SHans Petter Selasky } 396d96e5996SHans Petter Selasky 397d96e5996SHans Petter Selasky /* 398d96e5996SHans Petter Selasky * This function returns the next valid xarray entry based on the 399d96e5996SHans Petter Selasky * index given by "pindex". The valued pointed to by "pindex" is 400d96e5996SHans Petter Selasky * updated before return. 401d96e5996SHans Petter Selasky */ 402d96e5996SHans Petter Selasky void * 403d96e5996SHans Petter Selasky __xa_next(struct xarray *xa, unsigned long *pindex, bool not_first) 404d96e5996SHans Petter Selasky { 405d96e5996SHans Petter Selasky struct radix_tree_iter iter = { .index = *pindex }; 406d96e5996SHans Petter Selasky void **ppslot; 407d96e5996SHans Petter Selasky void *retval; 408d96e5996SHans Petter Selasky bool found; 409d96e5996SHans Petter Selasky 410d96e5996SHans Petter Selasky XA_ASSERT_LOCKED(xa); 411d96e5996SHans Petter Selasky 412d96e5996SHans Petter Selasky if (not_first) { 413d96e5996SHans Petter Selasky /* advance to next index, if any */ 414d96e5996SHans Petter Selasky iter.index++; 415d96e5996SHans Petter Selasky if (iter.index == 0) 416d96e5996SHans Petter Selasky return (NULL); 417d96e5996SHans Petter Selasky } 418d96e5996SHans Petter Selasky 419*be818f26SJean-Sébastien Pédron found = radix_tree_iter_find(&xa->xa_head, &iter, &ppslot); 420d96e5996SHans Petter Selasky if (likely(found)) { 421d96e5996SHans Petter Selasky retval = *ppslot; 4223102ea3bSJean-Sébastien Pédron if (retval == NULL_VALUE) 4233102ea3bSJean-Sébastien Pédron retval = NULL; 424d96e5996SHans Petter Selasky *pindex = iter.index; 425d96e5996SHans Petter Selasky } else { 426d96e5996SHans Petter Selasky retval = NULL; 427d96e5996SHans Petter Selasky } 428d96e5996SHans Petter Selasky return (retval); 429d96e5996SHans Petter Selasky } 430d96e5996SHans Petter Selasky 431d96e5996SHans Petter Selasky void * 432d96e5996SHans Petter Selasky xa_next(struct xarray *xa, unsigned long *pindex, bool not_first) 433d96e5996SHans Petter Selasky { 434d96e5996SHans Petter Selasky void *retval; 435d96e5996SHans Petter Selasky 436d96e5996SHans Petter Selasky xa_lock(xa); 437d96e5996SHans Petter Selasky retval = __xa_next(xa, pindex, not_first); 438d96e5996SHans Petter Selasky xa_unlock(xa); 439d96e5996SHans Petter Selasky 440d96e5996SHans Petter Selasky return (retval); 441d96e5996SHans Petter Selasky } 442