/* $NetBSD: linux_radixtree.c,v 1.4 2021/12/19 12:07:29 riastradh Exp $ */ /*- * Copyright (c) 2021 The NetBSD Foundation, Inc. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include __KERNEL_RCSID(0, "$NetBSD: linux_radixtree.c,v 1.4 2021/12/19 12:07:29 riastradh Exp $"); #include #include #include #include #include #include #include #include /* XXX mega-kludgerific */ #define __UNVOLANST(p) ((void *)(unsigned long)(const volatile void *)(p)) struct kludge { uint64_t k_key; void *k_datum; struct rcu_head k_rcu; }; void INIT_RADIX_TREE(struct radix_tree_root *root, gfp_t gfp) { radix_tree_init_tree(&root->rtr_tree); __cpu_simple_lock_init(&root->rtr_lock); } int radix_tree_insert(struct radix_tree_root *root, unsigned long key, void *datum) { struct kludge *kludge; int error; /* XXX No way to know whether the caller can sleep or not... */ if ((kludge = kzalloc(sizeof(*kludge), GFP_ATOMIC)) == NULL) return -ENOMEM; kludge->k_key = key; kludge->k_datum = datum; __cpu_simple_lock(&root->rtr_lock); error = radix_tree_insert_node(&root->rtr_tree, key, kludge); __cpu_simple_unlock(&root->rtr_lock); /* XXX errno NetBSD->Linux */ return -error; } void * radix_tree_delete(struct radix_tree_root *root, unsigned long key) { struct kludge *kludge; void *datum = NULL; __cpu_simple_lock(&root->rtr_lock); kludge = radix_tree_remove_node(&root->rtr_tree, key); __cpu_simple_unlock(&root->rtr_lock); if (kludge == NULL) return NULL; datum = kludge->k_datum; kfree_rcu(kludge, k_rcu); return datum; } bool radix_tree_empty(struct radix_tree_root *root) { bool empty; __cpu_simple_lock(&root->rtr_lock); empty = radix_tree_empty_tree_p(&root->rtr_tree); __cpu_simple_unlock(&root->rtr_lock); return empty; } void * radix_tree_lookup(const struct radix_tree_root *root, unsigned long key) { struct kludge *kludge; __cpu_simple_lock(__UNVOLANST(&root->rtr_lock)); kludge = radix_tree_lookup_node(__UNVOLANST(&root->rtr_tree), key); __cpu_simple_unlock(__UNVOLANST(&root->rtr_lock)); if (kludge == NULL) return NULL; return kludge->k_datum; } void * radix_tree_deref_slot(void **slot) { return atomic_load_consume(slot); } void ** radix_tree_iter_init(struct radix_tree_iter *I, unsigned long start) { I->index = start; I->rti_tree = NULL; return NULL; } void ** radix_tree_next_chunk(const struct radix_tree_root *root, struct radix_tree_iter *I, unsigned flags) { void *result; struct kludge *kludge; unsigned nresults; KASSERT(flags == 0); __cpu_simple_lock(__UNVOLANST(&root->rtr_lock)); nresults = radix_tree_gang_lookup_node(__UNVOLANST(&root->rtr_tree), I->index, &result, /*maxresults*/1, /*dense*/false); __cpu_simple_unlock(__UNVOLANST(&root->rtr_lock)); if (nresults == 0) return NULL; KASSERT(nresults == 1); kludge = result; I->index = kludge->k_key; I->rti_tree = root; return &kludge->k_datum; } void ** radix_tree_next_slot(void **slot, struct radix_tree_iter *I, unsigned flags) { struct kludge *kludge; void *result; unsigned nresults; KASSERT(flags == 0); kludge = container_of(slot, struct kludge, k_datum); __cpu_simple_lock(__UNVOLANST(&I->rti_tree->rtr_lock)); nresults = radix_tree_gang_lookup_node( __UNVOLANST(&I->rti_tree->rtr_tree), kludge->k_key, &result, /*maxresults*/1, /*dense*/true); __cpu_simple_unlock(__UNVOLANST(&I->rti_tree->rtr_lock)); if (nresults == 0) return NULL; KASSERT(nresults == 1); kludge = result; I->index = kludge->k_key; /* XXX Hope the tree hasn't changed! */ return &kludge->k_datum; } void radix_tree_iter_delete(struct radix_tree_root *root, struct radix_tree_iter *I, void **slot) { struct kludge *kludge = container_of(slot, struct kludge, k_datum); struct kludge *kludge0 __diagused; __cpu_simple_lock(&root->rtr_lock); kludge0 = radix_tree_remove_node(&root->rtr_tree, kludge->k_key); __cpu_simple_unlock(&root->rtr_lock); KASSERT(kludge0 == kludge); }