| /******************************************************************************* |
| * Copyright (C) 2018 Cadence Design Systems, Inc. |
| * |
| * Permission is hereby granted, free of charge, to any person obtaining |
| * a copy of this software and associated documentation files (the |
| * "Software"), to use this Software with Cadence processor cores only and |
| * not with any other processors and platforms, subject to |
| * the following conditions: |
| * |
| * The above copyright notice and this permission notice shall be included |
| * in all copies or substantial portions of the Software. |
| * |
| * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, |
| * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF |
| * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. |
| * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY |
| * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, |
| * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE |
| * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. |
| |
| ******************************************************************************/ |
| |
| /******************************************************************************* |
| * rbtree.h |
| * |
| * Generic implmentation of red-black trees |
| * |
| *******************************************************************************/ |
| |
| #ifndef __RBTREE_H |
| #define __RBTREE_H |
| |
| /******************************************************************************* |
| * Red-black tree node definition |
| ******************************************************************************/ |
| |
| /* ...reference to rb-tree node */ |
| typedef struct rb_node *rb_idx_t; |
| |
| /* ...rb-tree node */ |
| typedef struct rb_node |
| { |
| /* ...pointers to parent and two children */ |
| rb_idx_t parent, left, right; |
| |
| /* ...node color (least-significant-bit only) */ |
| u32 color; |
| |
| } rb_node_t; |
| |
| /* ...rb-tree data */ |
| typedef struct rb_tree_t |
| { |
| /* ...tree sentinel node */ |
| rb_node_t root; |
| |
| } rb_tree_t; |
| |
| /******************************************************************************* |
| * Helpers |
| ******************************************************************************/ |
| |
| /* ...left child accessor */ |
| static inline rb_idx_t rb_left(rb_tree_t *tree, rb_idx_t n_idx) |
| { |
| return n_idx->left; |
| } |
| |
| /* ...right child accessor */ |
| static inline rb_idx_t rb_right(rb_tree_t *tree, rb_idx_t n_idx) |
| { |
| return n_idx->right; |
| } |
| |
| /* ...parent node accessor */ |
| static inline rb_idx_t rb_parent(rb_tree_t *tree, rb_idx_t n_idx) |
| { |
| return n_idx->parent; |
| } |
| |
| /* ...tree root node accessor */ |
| static inline rb_idx_t rb_root(rb_tree_t *tree) |
| { |
| return rb_left(tree, &tree->root); |
| } |
| |
| /* ...tree data pointer accessor */ |
| static inline rb_idx_t rb_cache(rb_tree_t *tree) |
| { |
| return rb_right(tree, &tree->root); |
| } |
| |
| /* ...tree null node */ |
| static inline rb_idx_t rb_null(rb_tree_t *tree) |
| { |
| return &tree->root; |
| } |
| |
| /* ...get user-bits stored in node color */ |
| static inline u32 rb_node_data(rb_tree_t *tree, rb_idx_t n_idx) |
| { |
| return (n_idx->color >> 1); |
| } |
| |
| /* ...left child assignment */ |
| static inline void rb_set_left(rb_tree_t *tree, rb_idx_t n_idx, rb_node_t *child) |
| { |
| n_idx->left = child; |
| } |
| |
| /* ...right child assignment */ |
| static inline void rb_set_right(rb_tree_t *tree, rb_idx_t n_idx, rb_node_t *child) |
| { |
| n_idx->right = child; |
| } |
| |
| /* ...cache tree client index */ |
| static inline void rb_set_cache(rb_tree_t *tree, rb_idx_t c_idx) |
| { |
| tree->root.right = c_idx; |
| } |
| |
| /* ...get user-bits stored in node color */ |
| static inline void rb_set_node_data(rb_tree_t *tree, rb_idx_t n_idx, u32 data) |
| { |
| n_idx->color = (n_idx->color & 0x1) | (data << 1); |
| } |
| |
| /******************************************************************************* |
| * API functions |
| ******************************************************************************/ |
| |
| /* ...initialize rb-tree */ |
| extern void rb_init(rb_tree_t *tree); |
| |
| /* ...insert node into tree as a child of p */ |
| extern void rb_insert(rb_tree_t *tree, rb_idx_t n_idx, rb_idx_t p_idx); |
| |
| /* ...replace the node with same-key value and fixup tree pointers */ |
| extern void rb_replace(rb_tree_t *tree, rb_idx_t n_idx, rb_idx_t t_idx); |
| |
| /* ...delete node from the tree and return its in-order predecessor/successor */ |
| extern rb_idx_t rb_delete(rb_tree_t *tree, rb_idx_t n_idx); |
| |
| /* ...first in-order item in the tree */ |
| extern rb_idx_t rb_first(rb_tree_t *tree); |
| |
| /* ...last in-order item in the tree */ |
| extern rb_idx_t rb_last(rb_tree_t *tree); |
| |
| /* ...forward tree iterator */ |
| extern rb_idx_t rb_next(rb_tree_t *tree, rb_idx_t n_idx); |
| |
| /* ...backward tree iterator */ |
| extern rb_idx_t rb_prev(rb_tree_t *tree, rb_idx_t n_idx); |
| |
| #endif /* __RBTREE_H */ |