LCOV - code coverage report
Current view: top level - include/linux - rbtree.h (source / functions) Hit Total Coverage
Test: combined.info Lines: 17 23 73.9 %
Date: 2022-04-01 14:58:12 Functions: 1 1 100.0 %
Branches: 8 14 57.1 %

           Branch data     Line data    Source code
       1                 :            : /* SPDX-License-Identifier: GPL-2.0-or-later */
       2                 :            : /*
       3                 :            :   Red Black Trees
       4                 :            :   (C) 1999  Andrea Arcangeli <andrea@suse.de>
       5                 :            :   
       6                 :            : 
       7                 :            :   linux/include/linux/rbtree.h
       8                 :            : 
       9                 :            :   To use rbtrees you'll have to implement your own insert and search cores.
      10                 :            :   This will avoid us to use callbacks and to drop drammatically performances.
      11                 :            :   I know it's not the cleaner way,  but in C (not in C++) to get
      12                 :            :   performances and genericity...
      13                 :            : 
      14                 :            :   See Documentation/rbtree.txt for documentation and samples.
      15                 :            : */
      16                 :            : 
      17                 :            : #ifndef _LINUX_RBTREE_H
      18                 :            : #define _LINUX_RBTREE_H
      19                 :            : 
      20                 :            : #include <linux/kernel.h>
      21                 :            : #include <linux/stddef.h>
      22                 :            : #include <linux/rcupdate.h>
      23                 :            : 
      24                 :            : struct rb_node {
      25                 :            :         unsigned long  __rb_parent_color;
      26                 :            :         struct rb_node *rb_right;
      27                 :            :         struct rb_node *rb_left;
      28                 :            : } __attribute__((aligned(sizeof(long))));
      29                 :            :     /* The alignment might seem pointless, but allegedly CRIS needs it */
      30                 :            : 
      31                 :            : struct rb_root {
      32                 :            :         struct rb_node *rb_node;
      33                 :            : };
      34                 :            : 
      35                 :            : #define rb_parent(r)   ((struct rb_node *)((r)->__rb_parent_color & ~3))
      36                 :            : 
      37                 :            : #define RB_ROOT (struct rb_root) { NULL, }
      38                 :            : #define rb_entry(ptr, type, member) container_of(ptr, type, member)
      39                 :            : 
      40                 :            : #define RB_EMPTY_ROOT(root)  (READ_ONCE((root)->rb_node) == NULL)
      41                 :            : 
      42                 :            : /* 'empty' nodes are nodes that are known not to be inserted in an rbtree */
      43                 :            : #define RB_EMPTY_NODE(node)  \
      44                 :            :         ((node)->__rb_parent_color == (unsigned long)(node))
      45                 :            : #define RB_CLEAR_NODE(node)  \
      46                 :            :         ((node)->__rb_parent_color = (unsigned long)(node))
      47                 :            : 
      48                 :            : 
      49                 :            : extern void rb_insert_color(struct rb_node *, struct rb_root *);
      50                 :            : extern void rb_erase(struct rb_node *, struct rb_root *);
      51                 :            : 
      52                 :            : 
      53                 :            : /* Find logical next and previous nodes in a tree */
      54                 :            : extern struct rb_node *rb_next(const struct rb_node *);
      55                 :            : extern struct rb_node *rb_prev(const struct rb_node *);
      56                 :            : extern struct rb_node *rb_first(const struct rb_root *);
      57                 :            : extern struct rb_node *rb_last(const struct rb_root *);
      58                 :            : 
      59                 :            : /* Postorder iteration - always visit the parent after its children */
      60                 :            : extern struct rb_node *rb_first_postorder(const struct rb_root *);
      61                 :            : extern struct rb_node *rb_next_postorder(const struct rb_node *);
      62                 :            : 
      63                 :            : /* Fast replacement of a single node without remove/rebalance/add/rebalance */
      64                 :            : extern void rb_replace_node(struct rb_node *victim, struct rb_node *new,
      65                 :            :                             struct rb_root *root);
      66                 :            : extern void rb_replace_node_rcu(struct rb_node *victim, struct rb_node *new,
      67                 :            :                                 struct rb_root *root);
      68                 :            : 
      69                 :     904598 : static inline void rb_link_node(struct rb_node *node, struct rb_node *parent,
      70                 :            :                                 struct rb_node **rb_link)
      71                 :            : {
      72                 :     904598 :         node->__rb_parent_color = (unsigned long)parent;
      73                 :     904598 :         node->rb_left = node->rb_right = NULL;
      74                 :            : 
      75   [ +  +  +  + ]:     904598 :         *rb_link = node;
      76                 :          0 : }
      77                 :            : 
      78                 :         36 : static inline void rb_link_node_rcu(struct rb_node *node, struct rb_node *parent,
      79                 :            :                                     struct rb_node **rb_link)
      80                 :            : {
      81                 :         36 :         node->__rb_parent_color = (unsigned long)parent;
      82                 :         36 :         node->rb_left = node->rb_right = NULL;
      83                 :            : 
      84                 :         36 :         rcu_assign_pointer(*rb_link, node);
      85                 :            : }
      86                 :            : 
      87                 :            : #define rb_entry_safe(ptr, type, member) \
      88                 :            :         ({ typeof(ptr) ____ptr = (ptr); \
      89                 :            :            ____ptr ? rb_entry(____ptr, type, member) : NULL; \
      90                 :            :         })
      91                 :            : 
      92                 :            : /**
      93                 :            :  * rbtree_postorder_for_each_entry_safe - iterate in post-order over rb_root of
      94                 :            :  * given type allowing the backing memory of @pos to be invalidated
      95                 :            :  *
      96                 :            :  * @pos:        the 'type *' to use as a loop cursor.
      97                 :            :  * @n:          another 'type *' to use as temporary storage
      98                 :            :  * @root:       'rb_root *' of the rbtree.
      99                 :            :  * @field:      the name of the rb_node field within 'type'.
     100                 :            :  *
     101                 :            :  * rbtree_postorder_for_each_entry_safe() provides a similar guarantee as
     102                 :            :  * list_for_each_entry_safe() and allows the iteration to continue independent
     103                 :            :  * of changes to @pos by the body of the loop.
     104                 :            :  *
     105                 :            :  * Note, however, that it cannot handle other modifications that re-order the
     106                 :            :  * rbtree it is iterating over. This includes calling rb_erase() on @pos, as
     107                 :            :  * rb_erase() may rebalance the tree, causing us to miss some nodes.
     108                 :            :  */
     109                 :            : #define rbtree_postorder_for_each_entry_safe(pos, n, root, field) \
     110                 :            :         for (pos = rb_entry_safe(rb_first_postorder(root), typeof(*pos), field); \
     111                 :            :              pos && ({ n = rb_entry_safe(rb_next_postorder(&pos->field), \
     112                 :            :                         typeof(*pos), field); 1; }); \
     113                 :            :              pos = n)
     114                 :            : 
     115                 :            : /*
     116                 :            :  * Leftmost-cached rbtrees.
     117                 :            :  *
     118                 :            :  * We do not cache the rightmost node based on footprint
     119                 :            :  * size vs number of potential users that could benefit
     120                 :            :  * from O(1) rb_last(). Just not worth it, users that want
     121                 :            :  * this feature can always implement the logic explicitly.
     122                 :            :  * Furthermore, users that want to cache both pointers may
     123                 :            :  * find it a bit asymmetric, but that's ok.
     124                 :            :  */
     125                 :            : struct rb_root_cached {
     126                 :            :         struct rb_root rb_root;
     127                 :            :         struct rb_node *rb_leftmost;
     128                 :            : };
     129                 :            : 
     130                 :            : #define RB_ROOT_CACHED (struct rb_root_cached) { {NULL, }, NULL }
     131                 :            : 
     132                 :            : /* Same as rb_first(), but O(1) */
     133                 :            : #define rb_first_cached(root) (root)->rb_leftmost
     134                 :            : 
     135                 :      44131 : static inline void rb_insert_color_cached(struct rb_node *node,
     136                 :            :                                           struct rb_root_cached *root,
     137                 :            :                                           bool leftmost)
     138                 :            : {
     139   [ +  +  #  # ]:      44131 :         if (leftmost)
     140                 :      31565 :                 root->rb_leftmost = node;
     141                 :      44131 :         rb_insert_color(node, &root->rb_root);
     142                 :          0 : }
     143                 :            : 
     144                 :      43873 : static inline void rb_erase_cached(struct rb_node *node,
     145                 :            :                                    struct rb_root_cached *root)
     146                 :            : {
     147         [ +  + ]:      43873 :         if (root->rb_leftmost == node)
     148                 :      42429 :                 root->rb_leftmost = rb_next(node);
     149                 :      43873 :         rb_erase(node, &root->rb_root);
     150                 :      43873 : }
     151                 :            : 
     152                 :          0 : static inline void rb_replace_node_cached(struct rb_node *victim,
     153                 :            :                                           struct rb_node *new,
     154                 :            :                                           struct rb_root_cached *root)
     155                 :            : {
     156   [ #  #  #  # ]:          0 :         if (root->rb_leftmost == victim)
     157                 :          0 :                 root->rb_leftmost = new;
     158                 :          0 :         rb_replace_node(victim, new, &root->rb_root);
     159                 :            : }
     160                 :            : 
     161                 :            : #endif  /* _LINUX_RBTREE_H */

Generated by: LCOV version 1.14