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 : : (C) 2002 David Woodhouse <dwmw2@infradead.org>
6 : : (C) 2012 Michel Lespinasse <walken@google.com>
7 : :
8 : :
9 : : linux/lib/rbtree.c
10 : : */
11 : :
12 : : #include <linux/rbtree_augmented.h>
13 : : #include <linux/export.h>
14 : :
15 : : /*
16 : : * red-black trees properties: http://en.wikipedia.org/wiki/Rbtree
17 : : *
18 : : * 1) A node is either red or black
19 : : * 2) The root is black
20 : : * 3) All leaves (NULL) are black
21 : : * 4) Both children of every red node are black
22 : : * 5) Every simple path from root to leaves contains the same number
23 : : * of black nodes.
24 : : *
25 : : * 4 and 5 give the O(log n) guarantee, since 4 implies you cannot have two
26 : : * consecutive red nodes in a path and every red node is therefore followed by
27 : : * a black. So if B is the number of black nodes on every simple path (as per
28 : : * 5), then the longest possible path due to 4 is 2B.
29 : : *
30 : : * We shall indicate color with case, where black nodes are uppercase and red
31 : : * nodes will be lowercase. Unknown color nodes shall be drawn as red within
32 : : * parentheses and have some accompanying text comment.
33 : : */
34 : :
35 : : /*
36 : : * Notes on lockless lookups:
37 : : *
38 : : * All stores to the tree structure (rb_left and rb_right) must be done using
39 : : * WRITE_ONCE(). And we must not inadvertently cause (temporary) loops in the
40 : : * tree structure as seen in program order.
41 : : *
42 : : * These two requirements will allow lockless iteration of the tree -- not
43 : : * correct iteration mind you, tree rotations are not atomic so a lookup might
44 : : * miss entire subtrees.
45 : : *
46 : : * But they do guarantee that any such traversal will only see valid elements
47 : : * and that it will indeed complete -- does not get stuck in a loop.
48 : : *
49 : : * It also guarantees that if the lookup returns an element it is the 'correct'
50 : : * one. But not returning an element does _NOT_ mean it's not present.
51 : : *
52 : : * NOTE:
53 : : *
54 : : * Stores to __rb_parent_color are not important for simple lookups so those
55 : : * are left undone as of now. Nor did I check for loops involving parent
56 : : * pointers.
57 : : */
58 : :
59 : 176038 : static inline void rb_set_black(struct rb_node *rb)
60 : : {
61 : 176038 : rb->__rb_parent_color |= RB_BLACK;
62 : 176038 : }
63 : :
64 : 6181260 : static inline struct rb_node *rb_red_parent(struct rb_node *red)
65 : : {
66 : 6181260 : return (struct rb_node *)red->__rb_parent_color;
67 : : }
68 : :
69 : : /*
70 : : * Helper function for rotations:
71 : : * - old's parent and color get assigned to new
72 : : * - old gets assigned new as a parent and 'color' as a color.
73 : : */
74 : : static inline void
75 : 1455568 : __rb_rotate_set_parents(struct rb_node *old, struct rb_node *new,
76 : : struct rb_root *root, int color)
77 : : {
78 : 1455568 : struct rb_node *parent = rb_parent(old);
79 : 1455568 : new->__rb_parent_color = old->__rb_parent_color;
80 : 1455568 : rb_set_parent_color(old, new, color);
81 [ + + + + : 1397727 : __rb_change_child(old, new, parent, root);
+ + + + +
+ + + + +
+ + ]
82 : : }
83 : :
84 : : static __always_inline void
85 : 3628920 : __rb_insert(struct rb_node *node, struct rb_root *root,
86 : : void (*augment_rotate)(struct rb_node *old, struct rb_node *new))
87 : : {
88 : 3628920 : struct rb_node *parent = rb_red_parent(node), *gparent, *tmp;
89 : :
90 : 4835380 : while (true) {
91 : : /*
92 : : * Loop invariant: node is red.
93 : : */
94 [ + + + + ]: 4835380 : if (unlikely(!parent)) {
95 : : /*
96 : : * The inserted node is root. Either this is the
97 : : * first node, or we recursed at Case 1 below and
98 : : * are no longer violating 4).
99 : : */
100 : 689755 : rb_set_parent_color(node, NULL, RB_BLACK);
101 : : break;
102 : : }
103 : :
104 : : /*
105 : : * If there is a black parent, we are done.
106 : : * Otherwise, take some corrective action as,
107 : : * per 4), we don't want a red root or two
108 : : * consecutive red nodes.
109 : : */
110 [ + + + + ]: 4145620 : if(rb_is_black(parent))
111 : : break;
112 : :
113 : 2552328 : gparent = rb_red_parent(parent);
114 : :
115 : 2552328 : tmp = gparent->rb_right;
116 [ + + + + ]: 2552328 : if (parent != tmp) { /* parent == gparent->rb_left */
117 [ + + + + : 582091 : if (tmp && rb_is_red(tmp)) {
+ + + + ]
118 : : /*
119 : : * Case 1 - node's uncle is red (color flips).
120 : : *
121 : : * G g
122 : : * / \ / \
123 : : * p u --> P U
124 : : * / /
125 : : * n n
126 : : *
127 : : * However, since g's parent might be red, and
128 : : * 4) does not allow this, we need to recurse
129 : : * at g.
130 : : */
131 : 186917 : rb_set_parent_color(tmp, gparent, RB_BLACK);
132 : 186917 : rb_set_parent_color(parent, gparent, RB_BLACK);
133 : 186917 : node = gparent;
134 : 186917 : parent = rb_parent(node);
135 : 186917 : rb_set_parent_color(node, parent, RB_RED);
136 : 186917 : continue;
137 : : }
138 : :
139 : 395174 : tmp = parent->rb_right;
140 [ + + + + ]: 395174 : if (node == tmp) {
141 : : /*
142 : : * Case 2 - node's uncle is black and node is
143 : : * the parent's right child (left rotate at parent).
144 : : *
145 : : * G G
146 : : * / \ / \
147 : : * p U --> n U
148 : : * \ /
149 : : * n p
150 : : *
151 : : * This still leaves us in violation of 4), the
152 : : * continuation into Case 3 will fix that.
153 : : */
154 : 295111 : tmp = node->rb_left;
155 [ + + + + ]: 295111 : WRITE_ONCE(parent->rb_right, tmp);
156 : 295111 : WRITE_ONCE(node->rb_left, parent);
157 [ + + + + ]: 295111 : if (tmp)
158 : 63384 : rb_set_parent_color(tmp, parent,
159 : : RB_BLACK);
160 : 295111 : rb_set_parent_color(parent, node, RB_RED);
161 : 248809 : augment_rotate(parent, node);
162 : 295111 : parent = node;
163 : 248809 : tmp = node->rb_right;
164 : : }
165 : :
166 : : /*
167 : : * Case 3 - node's uncle is black and node is
168 : : * the parent's left child (right rotate at gparent).
169 : : *
170 : : * G P
171 : : * / \ / \
172 : : * p U --> n g
173 : : * / \
174 : : * n U
175 : : */
176 [ + + + + ]: 395174 : WRITE_ONCE(gparent->rb_left, tmp); /* == parent->rb_right */
177 : 395174 : WRITE_ONCE(parent->rb_right, gparent);
178 [ + + + + ]: 395174 : if (tmp)
179 : 85290 : rb_set_parent_color(tmp, gparent, RB_BLACK);
180 [ + + + + ]: 395174 : __rb_rotate_set_parents(gparent, parent, root, RB_RED);
181 : 343930 : augment_rotate(gparent, parent);
182 : 343930 : break;
183 : : } else {
184 : 1970237 : tmp = gparent->rb_left;
185 [ + + + + : 1970237 : if (tmp && rb_is_red(tmp)) {
+ + + + ]
186 : : /* Case 1 - color flips */
187 : 1019528 : rb_set_parent_color(tmp, gparent, RB_BLACK);
188 : 1019528 : rb_set_parent_color(parent, gparent, RB_BLACK);
189 : 1019528 : node = gparent;
190 : 1019528 : parent = rb_parent(node);
191 : 1019528 : rb_set_parent_color(node, parent, RB_RED);
192 : 1019528 : continue;
193 : : }
194 : :
195 : 950709 : tmp = parent->rb_left;
196 [ + + + + ]: 950709 : if (node == tmp) {
197 : : /* Case 2 - right rotate at parent */
198 : 97900 : tmp = node->rb_right;
199 [ + + + + ]: 97900 : WRITE_ONCE(parent->rb_left, tmp);
200 : 97900 : WRITE_ONCE(node->rb_right, parent);
201 [ + + + + ]: 97900 : if (tmp)
202 : 40587 : rb_set_parent_color(tmp, parent,
203 : : RB_BLACK);
204 : 97900 : rb_set_parent_color(parent, node, RB_RED);
205 : 89798 : augment_rotate(parent, node);
206 : 97900 : parent = node;
207 : 89798 : tmp = node->rb_left;
208 : : }
209 : :
210 : : /* Case 3 - left rotate at gparent */
211 [ + + + + ]: 950709 : WRITE_ONCE(gparent->rb_right, tmp); /* == parent->rb_left */
212 : 950709 : WRITE_ONCE(parent->rb_left, gparent);
213 [ + + + + ]: 950709 : if (tmp)
214 : 339181 : rb_set_parent_color(tmp, gparent, RB_BLACK);
215 [ + + + + ]: 950709 : __rb_rotate_set_parents(gparent, parent, root, RB_RED);
216 : 785248 : augment_rotate(gparent, parent);
217 : 785248 : break;
218 : : }
219 : : }
220 : : }
221 : :
222 : : /*
223 : : * Inline version for rb_erase() use - we want to be able to inline
224 : : * and eliminate the dummy_rotate callback there
225 : : */
226 : : static __always_inline void
227 : 217921 : ____rb_erase_color(struct rb_node *parent, struct rb_root *root,
228 : : void (*augment_rotate)(struct rb_node *old, struct rb_node *new))
229 : : {
230 : 217921 : struct rb_node *node = NULL, *sibling, *tmp1, *tmp2;
231 : :
232 : 459347 : while (true) {
233 : : /*
234 : : * Loop invariants:
235 : : * - node is black (or NULL on first iteration)
236 : : * - node is not the root (parent is not NULL)
237 : : * - All leaf paths going through parent and node have a
238 : : * black node count that is 1 lower than other leaf paths.
239 : : */
240 : 459347 : sibling = parent->rb_right;
241 [ + + + + ]: 459347 : if (node != sibling) { /* node == parent->rb_left */
242 [ + + + + ]: 226399 : if (rb_is_red(sibling)) {
243 : : /*
244 : : * Case 1 - left rotate at parent
245 : : *
246 : : * P S
247 : : * / \ / \
248 : : * N s --> p Sr
249 : : * / \ / \
250 : : * Sl Sr N Sl
251 : : */
252 : 43892 : tmp1 = sibling->rb_left;
253 [ + + + + ]: 43892 : WRITE_ONCE(parent->rb_right, tmp1);
254 : 43892 : WRITE_ONCE(sibling->rb_left, parent);
255 [ + + + + ]: 43892 : rb_set_parent_color(tmp1, parent, RB_BLACK);
256 [ + + + + ]: 43892 : __rb_rotate_set_parents(parent, sibling, root,
257 : : RB_RED);
258 : 865 : augment_rotate(parent, sibling);
259 : 865 : sibling = tmp1;
260 : : }
261 : 226399 : tmp1 = sibling->rb_right;
262 [ + + + + : 226399 : if (!tmp1 || rb_is_black(tmp1)) {
+ + + + ]
263 : 197174 : tmp2 = sibling->rb_left;
264 [ + + + + : 197174 : if (!tmp2 || rb_is_black(tmp2)) {
+ + + + ]
265 : : /*
266 : : * Case 2 - sibling color flip
267 : : * (p could be either color here)
268 : : *
269 : : * (p) (p)
270 : : * / \ / \
271 : : * N S --> N s
272 : : * / \ / \
273 : : * Sl Sr Sl Sr
274 : : *
275 : : * This leaves us violating 5) which
276 : : * can be fixed by flipping p to black
277 : : * if it was red, or by recursing at p.
278 : : * p is red when coming from Case 1.
279 : : */
280 [ + + + + ]: 189897 : rb_set_parent_color(sibling, parent,
281 : : RB_RED);
282 [ + + + + ]: 189897 : if (rb_is_red(parent))
283 : 92264 : rb_set_black(parent);
284 : : else {
285 : 97633 : node = parent;
286 : 97633 : parent = rb_parent(node);
287 [ + + + + ]: 97633 : if (parent)
288 : 87188 : continue;
289 : : }
290 : : break;
291 : : }
292 : : /*
293 : : * Case 3 - right rotate at sibling
294 : : * (p could be either color here)
295 : : *
296 : : * (p) (p)
297 : : * / \ / \
298 : : * N S --> N sl
299 : : * / \ \
300 : : * sl Sr S
301 : : * \
302 : : * Sr
303 : : *
304 : : * Note: p might be red, and then both
305 : : * p and sl are red after rotation(which
306 : : * breaks property 4). This is fixed in
307 : : * Case 4 (in __rb_rotate_set_parents()
308 : : * which set sl the color of p
309 : : * and set p RB_BLACK)
310 : : *
311 : : * (p) (sl)
312 : : * / \ / \
313 : : * N sl --> P S
314 : : * \ / \
315 : : * S N Sr
316 : : * \
317 : : * Sr
318 : : */
319 : 7277 : tmp1 = tmp2->rb_right;
320 [ + + + + ]: 7277 : WRITE_ONCE(sibling->rb_left, tmp1);
321 : 7277 : WRITE_ONCE(tmp2->rb_right, sibling);
322 : 7277 : WRITE_ONCE(parent->rb_right, tmp2);
323 [ + + + + ]: 7277 : if (tmp1)
324 : 3360 : rb_set_parent_color(tmp1, sibling,
325 : : RB_BLACK);
326 : 775 : augment_rotate(sibling, tmp2);
327 : 775 : tmp1 = sibling;
328 : 775 : sibling = tmp2;
329 : : }
330 : : /*
331 : : * Case 4 - left rotate at parent + color flips
332 : : * (p and sl could be either color here.
333 : : * After rotation, p becomes black, s acquires
334 : : * p's color, and sl keeps its color)
335 : : *
336 : : * (p) (s)
337 : : * / \ / \
338 : : * N S --> P Sr
339 : : * / \ / \
340 : : * (sl) sr N (sl)
341 : : */
342 : 36502 : tmp2 = sibling->rb_left;
343 [ + + + + ]: 36502 : WRITE_ONCE(parent->rb_right, tmp2);
344 : 36502 : WRITE_ONCE(sibling->rb_left, parent);
345 [ + + + + ]: 36502 : rb_set_parent_color(tmp1, sibling, RB_BLACK);
346 [ + + + + ]: 36502 : if (tmp2)
347 : 29267 : rb_set_parent(tmp2, parent);
348 [ + + + + ]: 36502 : __rb_rotate_set_parents(parent, sibling, root,
349 : : RB_BLACK);
350 : 19533 : augment_rotate(parent, sibling);
351 : 19533 : break;
352 : : } else {
353 : 232948 : sibling = parent->rb_left;
354 [ + + + + ]: 232948 : if (rb_is_red(sibling)) {
355 : : /* Case 1 - right rotate at parent */
356 : 13949 : tmp1 = sibling->rb_right;
357 [ + + + + ]: 13949 : WRITE_ONCE(parent->rb_left, tmp1);
358 : 13949 : WRITE_ONCE(sibling->rb_right, parent);
359 [ + + + + ]: 13949 : rb_set_parent_color(tmp1, parent, RB_BLACK);
360 [ + + + + ]: 13949 : __rb_rotate_set_parents(parent, sibling, root,
361 : : RB_RED);
362 : 1713 : augment_rotate(parent, sibling);
363 : 1713 : sibling = tmp1;
364 : : }
365 : 232948 : tmp1 = sibling->rb_left;
366 [ + + + + : 232948 : if (!tmp1 || rb_is_black(tmp1)) {
+ + + + ]
367 : 219383 : tmp2 = sibling->rb_right;
368 [ + + + + : 219383 : if (!tmp2 || rb_is_black(tmp2)) {
+ + + + ]
369 : : /* Case 2 - sibling color flip */
370 [ + + + + ]: 217606 : rb_set_parent_color(sibling, parent,
371 : : RB_RED);
372 [ + + + + ]: 217606 : if (rb_is_red(parent))
373 : 83774 : rb_set_black(parent);
374 : : else {
375 : 133832 : node = parent;
376 : 133832 : parent = rb_parent(node);
377 [ + + + + ]: 133832 : if (parent)
378 : 42350 : continue;
379 : : }
380 : : break;
381 : : }
382 : : /* Case 3 - left rotate at sibling */
383 : 1777 : tmp1 = tmp2->rb_left;
384 [ + + + + ]: 1777 : WRITE_ONCE(sibling->rb_right, tmp1);
385 : 1777 : WRITE_ONCE(tmp2->rb_left, sibling);
386 : 1777 : WRITE_ONCE(parent->rb_left, tmp2);
387 [ + + + + ]: 1777 : if (tmp1)
388 : 181 : rb_set_parent_color(tmp1, sibling,
389 : : RB_BLACK);
390 : 1603 : augment_rotate(sibling, tmp2);
391 : 1603 : tmp1 = sibling;
392 : 1603 : sibling = tmp2;
393 : : }
394 : : /* Case 4 - right rotate at parent + color flips */
395 : 15342 : tmp2 = sibling->rb_right;
396 [ + + + + ]: 15342 : WRITE_ONCE(parent->rb_left, tmp2);
397 : 15342 : WRITE_ONCE(sibling->rb_right, parent);
398 [ + + + + ]: 15342 : rb_set_parent_color(tmp1, sibling, RB_BLACK);
399 [ + + + + ]: 15342 : if (tmp2)
400 : 4588 : rb_set_parent(tmp2, parent);
401 [ + + + + ]: 15342 : __rb_rotate_set_parents(parent, sibling, root,
402 : : RB_BLACK);
403 : 12154 : augment_rotate(parent, sibling);
404 : 12154 : break;
405 : : }
406 : : }
407 : : }
408 : :
409 : : /* Non-inline version for rb_erase_augmented() use */
410 : 217921 : void __rb_erase_color(struct rb_node *parent, struct rb_root *root,
411 : : void (*augment_rotate)(struct rb_node *old, struct rb_node *new))
412 : : {
413 : 217921 : ____rb_erase_color(parent, root, augment_rotate);
414 : 217921 : }
415 : : EXPORT_SYMBOL(__rb_erase_color);
416 : :
417 : : /*
418 : : * Non-augmented rbtree manipulation functions.
419 : : *
420 : : * We use dummy augmented callbacks here, and have the compiler optimize them
421 : : * out of the rb_insert_color() and rb_erase() function definitions.
422 : : */
423 : :
424 : : static inline void dummy_propagate(struct rb_node *node, struct rb_node *stop) {}
425 : : static inline void dummy_copy(struct rb_node *old, struct rb_node *new) {}
426 : 54404 : static inline void dummy_rotate(struct rb_node *old, struct rb_node *new) {}
427 : :
428 : : static const struct rb_augment_callbacks dummy_callbacks = {
429 : : .propagate = dummy_propagate,
430 : : .copy = dummy_copy,
431 : : .rotate = dummy_rotate
432 : : };
433 : :
434 : 391382 : void rb_insert_color(struct rb_node *node, struct rb_root *root)
435 : : {
436 : 391382 : __rb_insert(node, root, dummy_rotate);
437 : 391382 : }
438 : : EXPORT_SYMBOL(rb_insert_color);
439 : :
440 : 321983 : void rb_erase(struct rb_node *node, struct rb_root *root)
441 : : {
442 : 321983 : struct rb_node *rebalance;
443 [ + + ]: 321983 : rebalance = __rb_erase_augmented(node, root, &dummy_callbacks);
444 [ + + ]: 321983 : if (rebalance)
445 : : ____rb_erase_color(rebalance, root, dummy_rotate);
446 : 321983 : }
447 : : EXPORT_SYMBOL(rb_erase);
448 : :
449 : : /*
450 : : * Augmented rbtree manipulation functions.
451 : : *
452 : : * This instantiates the same __always_inline functions as in the non-augmented
453 : : * case, but this time with user-defined callbacks.
454 : : */
455 : :
456 : 3237550 : void __rb_insert_augmented(struct rb_node *node, struct rb_root *root,
457 : : void (*augment_rotate)(struct rb_node *old, struct rb_node *new))
458 : : {
459 : 3237550 : __rb_insert(node, root, augment_rotate);
460 : 3237550 : }
461 : : EXPORT_SYMBOL(__rb_insert_augmented);
462 : :
463 : : /*
464 : : * This function returns the first node (in sort order) of the tree.
465 : : */
466 : 8838 : struct rb_node *rb_first(const struct rb_root *root)
467 : : {
468 : 8838 : struct rb_node *n;
469 : :
470 : 8838 : n = root->rb_node;
471 [ + + ]: 8838 : if (!n)
472 : : return NULL;
473 [ + + ]: 2443 : while (n->rb_left)
474 : : n = n->rb_left;
475 : : return n;
476 : : }
477 : : EXPORT_SYMBOL(rb_first);
478 : :
479 : 0 : struct rb_node *rb_last(const struct rb_root *root)
480 : : {
481 : 0 : struct rb_node *n;
482 : :
483 : 0 : n = root->rb_node;
484 [ # # ]: 0 : if (!n)
485 : : return NULL;
486 [ # # ]: 0 : while (n->rb_right)
487 : : n = n->rb_right;
488 : : return n;
489 : : }
490 : : EXPORT_SYMBOL(rb_last);
491 : :
492 : 655202 : struct rb_node *rb_next(const struct rb_node *node)
493 : : {
494 : 655202 : struct rb_node *parent;
495 : :
496 [ + - ]: 655202 : if (RB_EMPTY_NODE(node))
497 : : return NULL;
498 : :
499 : : /*
500 : : * If we have a right-hand child, go down and then left as far
501 : : * as we can.
502 : : */
503 [ + + ]: 655202 : if (node->rb_right) {
504 : : node = node->rb_right;
505 [ + + ]: 99317 : while (node->rb_left)
506 : : node=node->rb_left;
507 : 96114 : return (struct rb_node *)node;
508 : : }
509 : :
510 : : /*
511 : : * No right-hand children. Everything down and left is smaller than us,
512 : : * so any 'next' node must be in the general direction of our parent.
513 : : * Go up the tree; any time the ancestor is a right-hand child of its
514 : : * parent, keep going up. First time it's a left-hand child of its
515 : : * parent, said parent is our 'next' node.
516 : : */
517 [ + + + + ]: 565115 : while ((parent = rb_parent(node)) && node == parent->rb_right)
518 : : node = parent;
519 : :
520 : : return parent;
521 : : }
522 : : EXPORT_SYMBOL(rb_next);
523 : :
524 : 1269 : struct rb_node *rb_prev(const struct rb_node *node)
525 : : {
526 : 1269 : struct rb_node *parent;
527 : :
528 [ + - ]: 1269 : if (RB_EMPTY_NODE(node))
529 : : return NULL;
530 : :
531 : : /*
532 : : * If we have a left-hand child, go down and then right as far
533 : : * as we can.
534 : : */
535 [ + + ]: 1269 : if (node->rb_left) {
536 : : node = node->rb_left;
537 [ - + ]: 72 : while (node->rb_right)
538 : : node=node->rb_right;
539 : 72 : return (struct rb_node *)node;
540 : : }
541 : :
542 : : /*
543 : : * No left-hand children. Go up till we find an ancestor which
544 : : * is a right-hand child of its parent.
545 : : */
546 [ + + + + ]: 1803 : while ((parent = rb_parent(node)) && node == parent->rb_left)
547 : : node = parent;
548 : :
549 : : return parent;
550 : : }
551 : : EXPORT_SYMBOL(rb_prev);
552 : :
553 : 0 : void rb_replace_node(struct rb_node *victim, struct rb_node *new,
554 : : struct rb_root *root)
555 : : {
556 : 0 : struct rb_node *parent = rb_parent(victim);
557 : :
558 : : /* Copy the pointers/colour from the victim to the replacement */
559 : 0 : *new = *victim;
560 : :
561 : : /* Set the surrounding nodes to point to the replacement */
562 [ # # ]: 0 : if (victim->rb_left)
563 : 0 : rb_set_parent(victim->rb_left, new);
564 [ # # ]: 0 : if (victim->rb_right)
565 : 0 : rb_set_parent(victim->rb_right, new);
566 [ # # ]: 0 : __rb_change_child(victim, new, parent, root);
567 : 0 : }
568 : : EXPORT_SYMBOL(rb_replace_node);
569 : :
570 : 0 : void rb_replace_node_rcu(struct rb_node *victim, struct rb_node *new,
571 : : struct rb_root *root)
572 : : {
573 : 0 : struct rb_node *parent = rb_parent(victim);
574 : :
575 : : /* Copy the pointers/colour from the victim to the replacement */
576 : 0 : *new = *victim;
577 : :
578 : : /* Set the surrounding nodes to point to the replacement */
579 [ # # ]: 0 : if (victim->rb_left)
580 : 0 : rb_set_parent(victim->rb_left, new);
581 [ # # ]: 0 : if (victim->rb_right)
582 : 0 : rb_set_parent(victim->rb_right, new);
583 : :
584 : : /* Set the parent's pointer to the new node last after an RCU barrier
585 : : * so that the pointers onwards are seen to be set correctly when doing
586 : : * an RCU walk over the tree.
587 : : */
588 [ # # ]: 0 : __rb_change_child_rcu(victim, new, parent, root);
589 : 0 : }
590 : : EXPORT_SYMBOL(rb_replace_node_rcu);
591 : :
592 : : static struct rb_node *rb_left_deepest_node(const struct rb_node *node)
593 : : {
594 : 1995 : for (;;) {
595 [ + + + + ]: 1995 : if (node->rb_left)
596 : : node = node->rb_left;
597 [ + + + + ]: 1107 : else if (node->rb_right)
598 : : node = node->rb_right;
599 : : else
600 : 687 : return (struct rb_node *)node;
601 : : }
602 : : }
603 : :
604 : 1995 : struct rb_node *rb_next_postorder(const struct rb_node *node)
605 : : {
606 : 1995 : const struct rb_node *parent;
607 [ + - ]: 1995 : if (!node)
608 : : return NULL;
609 : 1995 : parent = rb_parent(node);
610 : :
611 : : /* If we're sitting on node, we've already seen our children */
612 [ + + + + : 1995 : if (parent && node == parent->rb_left && parent->rb_right) {
+ + ]
613 : : /* If we are the parent's left node, go to the parent's right
614 : : * node then all the way down to the left */
615 : 687 : return rb_left_deepest_node(parent->rb_right);
616 : : } else
617 : : /* Otherwise we are the parent's right node, and the parent
618 : : * should be next */
619 : : return (struct rb_node *)parent;
620 : : }
621 : : EXPORT_SYMBOL(rb_next_postorder);
622 : :
623 : 573 : struct rb_node *rb_first_postorder(const struct rb_root *root)
624 : : {
625 [ + + ]: 573 : if (!root->rb_node)
626 : : return NULL;
627 : :
628 : : return rb_left_deepest_node(root->rb_node);
629 : : }
630 : : EXPORT_SYMBOL(rb_first_postorder);
|