Line data Source code
1 : /*--------------------------------------------------------------------------
2 : *
3 : * test_rbtree.c
4 : * Test correctness of red-black tree operations.
5 : *
6 : * Copyright (c) 2009-2025, PostgreSQL Global Development Group
7 : *
8 : * IDENTIFICATION
9 : * src/test/modules/test_rbtree/test_rbtree.c
10 : *
11 : * -------------------------------------------------------------------------
12 : */
13 :
14 : #include "postgres.h"
15 :
16 : #include "common/pg_prng.h"
17 : #include "fmgr.h"
18 : #include "lib/rbtree.h"
19 : #include "utils/memutils.h"
20 :
21 2 : PG_MODULE_MAGIC;
22 :
23 :
24 : /*
25 : * Our test trees store an integer key, and nothing else.
26 : */
27 : typedef struct IntRBTreeNode
28 : {
29 : RBTNode rbtnode;
30 : int key;
31 : } IntRBTreeNode;
32 :
33 :
34 : /*
35 : * Node comparator. We don't worry about overflow in the subtraction,
36 : * since none of our test keys are negative.
37 : */
38 : static int
39 2675160 : irbt_cmp(const RBTNode *a, const RBTNode *b, void *arg)
40 : {
41 2675160 : const IntRBTreeNode *ea = (const IntRBTreeNode *) a;
42 2675160 : const IntRBTreeNode *eb = (const IntRBTreeNode *) b;
43 :
44 2675160 : return ea->key - eb->key;
45 : }
46 :
47 : /*
48 : * Node combiner. For testing purposes, just check that library doesn't
49 : * try to combine unequal keys.
50 : */
51 : static void
52 12 : irbt_combine(RBTNode *existing, const RBTNode *newdata, void *arg)
53 : {
54 12 : const IntRBTreeNode *eexist = (const IntRBTreeNode *) existing;
55 12 : const IntRBTreeNode *enew = (const IntRBTreeNode *) newdata;
56 :
57 12 : if (eexist->key != enew->key)
58 0 : elog(ERROR, "red-black tree combines %d into %d",
59 : enew->key, eexist->key);
60 12 : }
61 :
62 : /* Node allocator */
63 : static RBTNode *
64 120000 : irbt_alloc(void *arg)
65 : {
66 120000 : return (RBTNode *) palloc(sizeof(IntRBTreeNode));
67 : }
68 :
69 : /* Node freer */
70 : static void
71 30182 : irbt_free(RBTNode *node, void *arg)
72 : {
73 30182 : pfree(node);
74 30182 : }
75 :
76 : /*
77 : * Create a red-black tree using our support functions
78 : */
79 : static RBTree *
80 12 : create_int_rbtree(void)
81 : {
82 12 : return rbt_create(sizeof(IntRBTreeNode),
83 : irbt_cmp,
84 : irbt_combine,
85 : irbt_alloc,
86 : irbt_free,
87 : NULL);
88 : }
89 :
90 : /*
91 : * Generate a random permutation of the integers 0..size-1
92 : */
93 : static int *
94 12 : GetPermutation(int size)
95 : {
96 : int *permutation;
97 : int i;
98 :
99 12 : permutation = (int *) palloc(size * sizeof(int));
100 :
101 12 : permutation[0] = 0;
102 :
103 : /*
104 : * This is the "inside-out" variant of the Fisher-Yates shuffle algorithm.
105 : * Notionally, we append each new value to the array and then swap it with
106 : * a randomly-chosen array element (possibly including itself, else we
107 : * fail to generate permutations with the last integer last). The swap
108 : * step can be optimized by combining it with the insertion.
109 : */
110 120000 : for (i = 1; i < size; i++)
111 : {
112 119988 : int j = pg_prng_uint64_range(&pg_global_prng_state, 0, i);
113 :
114 119988 : if (j < i) /* avoid fetching undefined data if j=i */
115 119870 : permutation[i] = permutation[j];
116 119988 : permutation[j] = i;
117 : }
118 :
119 12 : return permutation;
120 : }
121 :
122 : /*
123 : * Populate an empty RBTree with "size" integers having the values
124 : * 0, step, 2*step, 3*step, ..., inserting them in random order
125 : */
126 : static void
127 12 : rbt_populate(RBTree *tree, int size, int step)
128 : {
129 12 : int *permutation = GetPermutation(size);
130 : IntRBTreeNode node;
131 : bool isNew;
132 : int i;
133 :
134 : /* Insert values. We don't expect any collisions. */
135 120012 : for (i = 0; i < size; i++)
136 : {
137 120000 : node.key = step * permutation[i];
138 120000 : rbt_insert(tree, (RBTNode *) &node, &isNew);
139 120000 : if (!isNew)
140 0 : elog(ERROR, "unexpected !isNew result from rbt_insert");
141 : }
142 :
143 : /*
144 : * Re-insert the first value to make sure collisions work right. It's
145 : * probably not useful to test that case over again for all the values.
146 : */
147 12 : if (size > 0)
148 : {
149 12 : node.key = step * permutation[0];
150 12 : rbt_insert(tree, (RBTNode *) &node, &isNew);
151 12 : if (isNew)
152 0 : elog(ERROR, "unexpected isNew result from rbt_insert");
153 : }
154 :
155 12 : pfree(permutation);
156 12 : }
157 :
158 : /*
159 : * Check the correctness of left-right traversal.
160 : * Left-right traversal is correct if all elements are
161 : * visited in increasing order.
162 : */
163 : static void
164 2 : testleftright(int size)
165 : {
166 2 : RBTree *tree = create_int_rbtree();
167 : IntRBTreeNode *node;
168 : RBTreeIterator iter;
169 2 : int lastKey = -1;
170 2 : int count = 0;
171 :
172 : /* check iteration over empty tree */
173 2 : rbt_begin_iterate(tree, LeftRightWalk, &iter);
174 2 : if (rbt_iterate(&iter) != NULL)
175 0 : elog(ERROR, "left-right walk over empty tree produced an element");
176 :
177 : /* fill tree with consecutive natural numbers */
178 2 : rbt_populate(tree, size, 1);
179 :
180 : /* iterate over the tree */
181 2 : rbt_begin_iterate(tree, LeftRightWalk, &iter);
182 :
183 20002 : while ((node = (IntRBTreeNode *) rbt_iterate(&iter)) != NULL)
184 : {
185 : /* check that order is increasing */
186 20000 : if (node->key <= lastKey)
187 0 : elog(ERROR, "left-right walk gives elements not in sorted order");
188 20000 : lastKey = node->key;
189 20000 : count++;
190 : }
191 :
192 2 : if (lastKey != size - 1)
193 0 : elog(ERROR, "left-right walk did not reach end");
194 2 : if (count != size)
195 0 : elog(ERROR, "left-right walk missed some elements");
196 2 : }
197 :
198 : /*
199 : * Check the correctness of right-left traversal.
200 : * Right-left traversal is correct if all elements are
201 : * visited in decreasing order.
202 : */
203 : static void
204 2 : testrightleft(int size)
205 : {
206 2 : RBTree *tree = create_int_rbtree();
207 : IntRBTreeNode *node;
208 : RBTreeIterator iter;
209 2 : int lastKey = size;
210 2 : int count = 0;
211 :
212 : /* check iteration over empty tree */
213 2 : rbt_begin_iterate(tree, RightLeftWalk, &iter);
214 2 : if (rbt_iterate(&iter) != NULL)
215 0 : elog(ERROR, "right-left walk over empty tree produced an element");
216 :
217 : /* fill tree with consecutive natural numbers */
218 2 : rbt_populate(tree, size, 1);
219 :
220 : /* iterate over the tree */
221 2 : rbt_begin_iterate(tree, RightLeftWalk, &iter);
222 :
223 20002 : while ((node = (IntRBTreeNode *) rbt_iterate(&iter)) != NULL)
224 : {
225 : /* check that order is decreasing */
226 20000 : if (node->key >= lastKey)
227 0 : elog(ERROR, "right-left walk gives elements not in sorted order");
228 20000 : lastKey = node->key;
229 20000 : count++;
230 : }
231 :
232 2 : if (lastKey != 0)
233 0 : elog(ERROR, "right-left walk did not reach end");
234 2 : if (count != size)
235 0 : elog(ERROR, "right-left walk missed some elements");
236 2 : }
237 :
238 : /*
239 : * Check the correctness of the rbt_find operation by searching for
240 : * both elements we inserted and elements we didn't.
241 : */
242 : static void
243 2 : testfind(int size)
244 : {
245 2 : RBTree *tree = create_int_rbtree();
246 : int i;
247 :
248 : /* Insert even integers from 0 to 2 * (size-1) */
249 2 : rbt_populate(tree, size, 2);
250 :
251 : /* Check that all inserted elements can be found */
252 20002 : for (i = 0; i < size; i++)
253 : {
254 : IntRBTreeNode node;
255 : IntRBTreeNode *resultNode;
256 :
257 20000 : node.key = 2 * i;
258 20000 : resultNode = (IntRBTreeNode *) rbt_find(tree, (RBTNode *) &node);
259 20000 : if (resultNode == NULL)
260 0 : elog(ERROR, "inserted element was not found");
261 20000 : if (node.key != resultNode->key)
262 0 : elog(ERROR, "find operation in rbtree gave wrong result");
263 : }
264 :
265 : /*
266 : * Check that not-inserted elements can not be found, being sure to try
267 : * values before the first and after the last element.
268 : */
269 20004 : for (i = -1; i <= 2 * size; i += 2)
270 : {
271 : IntRBTreeNode node;
272 : IntRBTreeNode *resultNode;
273 :
274 20002 : node.key = i;
275 20002 : resultNode = (IntRBTreeNode *) rbt_find(tree, (RBTNode *) &node);
276 20002 : if (resultNode != NULL)
277 0 : elog(ERROR, "not-inserted element was found");
278 : }
279 2 : }
280 :
281 : /*
282 : * Check the correctness of the rbt_find_less() and rbt_find_great() functions
283 : * by searching for an equal key and iterating the lesser keys then the greater
284 : * keys.
285 : */
286 : static void
287 2 : testfindltgt(int size)
288 : {
289 2 : RBTree *tree = create_int_rbtree();
290 :
291 : /*
292 : * Using the size as the random key to search wouldn't allow us to get at
293 : * least one greater match, so we do size - 1
294 : */
295 2 : int randomKey = pg_prng_uint64_range(&pg_global_prng_state, 0, size - 1);
296 : bool keyDeleted;
297 2 : IntRBTreeNode searchNode = {.key = randomKey};
298 : IntRBTreeNode *lteNode;
299 : IntRBTreeNode *gteNode;
300 : IntRBTreeNode *node;
301 :
302 : /* Insert natural numbers */
303 2 : rbt_populate(tree, size, 1);
304 :
305 : /*
306 : * Since the search key is included in the naturals of the tree, we're
307 : * sure to find an equal match
308 : */
309 2 : lteNode = (IntRBTreeNode *) rbt_find_less(tree, (RBTNode *) &searchNode, true);
310 2 : gteNode = (IntRBTreeNode *) rbt_find_great(tree, (RBTNode *) &searchNode, true);
311 :
312 2 : if (lteNode == NULL || lteNode->key != searchNode.key)
313 0 : elog(ERROR, "rbt_find_less() didn't find the equal key");
314 :
315 2 : if (gteNode == NULL || gteNode->key != searchNode.key)
316 0 : elog(ERROR, "rbt_find_great() didn't find the equal key");
317 :
318 2 : if (lteNode != gteNode)
319 0 : elog(ERROR, "rbt_find_less() and rbt_find_great() found different equal keys");
320 :
321 : /* Find the rest of the naturals lesser than the search key */
322 2 : keyDeleted = false;
323 3076 : for (; searchNode.key > 0; searchNode.key--)
324 : {
325 : /*
326 : * Find the next key. If the current key is deleted, we can pass
327 : * equal_match == true and still find the next one.
328 : */
329 3074 : node = (IntRBTreeNode *) rbt_find_less(tree, (RBTNode *) &searchNode,
330 : keyDeleted);
331 :
332 : /* ensure we find a lesser match */
333 3074 : if (!node || !(node->key < searchNode.key))
334 0 : elog(ERROR, "rbt_find_less() didn't find a lesser key");
335 :
336 : /* randomly delete the found key or leave it */
337 3074 : keyDeleted = (pg_prng_uint64_range(&pg_global_prng_state, 0, 1) == 1);
338 3074 : if (keyDeleted)
339 1574 : rbt_delete(tree, (RBTNode *) node);
340 : }
341 :
342 : /* Find the rest of the naturals greater than the search key */
343 2 : keyDeleted = false;
344 16926 : for (searchNode.key = randomKey; searchNode.key < size - 1; searchNode.key++)
345 : {
346 : /*
347 : * Find the next key. If the current key is deleted, we can pass
348 : * equal_match == true and still find the next one.
349 : */
350 16924 : node = (IntRBTreeNode *) rbt_find_great(tree, (RBTNode *) &searchNode,
351 : keyDeleted);
352 :
353 : /* ensure we find a greater match */
354 16924 : if (!node || !(node->key > searchNode.key))
355 0 : elog(ERROR, "rbt_find_great() didn't find a greater key");
356 :
357 : /* randomly delete the found key or leave it */
358 16924 : keyDeleted = (pg_prng_uint64_range(&pg_global_prng_state, 0, 1) == 1);
359 16924 : if (keyDeleted)
360 8608 : rbt_delete(tree, (RBTNode *) node);
361 : }
362 :
363 : /* Check out of bounds searches find nothing */
364 2 : searchNode.key = -1;
365 2 : node = (IntRBTreeNode *) rbt_find_less(tree, (RBTNode *) &searchNode, true);
366 2 : if (node != NULL)
367 0 : elog(ERROR, "rbt_find_less() found non-inserted element");
368 2 : searchNode.key = 0;
369 2 : node = (IntRBTreeNode *) rbt_find_less(tree, (RBTNode *) &searchNode, false);
370 2 : if (node != NULL)
371 0 : elog(ERROR, "rbt_find_less() found non-inserted element");
372 2 : searchNode.key = size;
373 2 : node = (IntRBTreeNode *) rbt_find_great(tree, (RBTNode *) &searchNode, true);
374 2 : if (node != NULL)
375 0 : elog(ERROR, "rbt_find_great() found non-inserted element");
376 2 : searchNode.key = size - 1;
377 2 : node = (IntRBTreeNode *) rbt_find_great(tree, (RBTNode *) &searchNode, false);
378 2 : if (node != NULL)
379 0 : elog(ERROR, "rbt_find_great() found non-inserted element");
380 2 : }
381 :
382 : /*
383 : * Check the correctness of the rbt_leftmost operation.
384 : * This operation should always return the smallest element of the tree.
385 : */
386 : static void
387 2 : testleftmost(int size)
388 : {
389 2 : RBTree *tree = create_int_rbtree();
390 : IntRBTreeNode *result;
391 :
392 : /* Check that empty tree has no leftmost element */
393 2 : if (rbt_leftmost(tree) != NULL)
394 0 : elog(ERROR, "leftmost node of empty tree is not NULL");
395 :
396 : /* fill tree with consecutive natural numbers */
397 2 : rbt_populate(tree, size, 1);
398 :
399 : /* Check that leftmost element is the smallest one */
400 2 : result = (IntRBTreeNode *) rbt_leftmost(tree);
401 2 : if (result == NULL || result->key != 0)
402 0 : elog(ERROR, "rbt_leftmost gave wrong result");
403 2 : }
404 :
405 : /*
406 : * Check the correctness of the rbt_delete operation.
407 : */
408 : static void
409 2 : testdelete(int size, int delsize)
410 : {
411 2 : RBTree *tree = create_int_rbtree();
412 : int *deleteIds;
413 : bool *chosen;
414 : int i;
415 :
416 : /* fill tree with consecutive natural numbers */
417 2 : rbt_populate(tree, size, 1);
418 :
419 : /* Choose unique ids to delete */
420 2 : deleteIds = (int *) palloc(delsize * sizeof(int));
421 2 : chosen = (bool *) palloc0(size * sizeof(bool));
422 :
423 2002 : for (i = 0; i < delsize; i++)
424 : {
425 2000 : int k = pg_prng_uint64_range(&pg_global_prng_state, 0, size - 1);
426 :
427 2148 : while (chosen[k])
428 148 : k = (k + 1) % size;
429 2000 : deleteIds[i] = k;
430 2000 : chosen[k] = true;
431 : }
432 :
433 : /* Delete elements */
434 2002 : for (i = 0; i < delsize; i++)
435 : {
436 : IntRBTreeNode find;
437 : IntRBTreeNode *node;
438 :
439 2000 : find.key = deleteIds[i];
440 : /* Locate the node to be deleted */
441 2000 : node = (IntRBTreeNode *) rbt_find(tree, (RBTNode *) &find);
442 2000 : if (node == NULL || node->key != deleteIds[i])
443 0 : elog(ERROR, "expected element was not found during deleting");
444 : /* Delete it */
445 2000 : rbt_delete(tree, (RBTNode *) node);
446 : }
447 :
448 : /* Check that deleted elements are deleted */
449 20002 : for (i = 0; i < size; i++)
450 : {
451 : IntRBTreeNode node;
452 : IntRBTreeNode *result;
453 :
454 20000 : node.key = i;
455 20000 : result = (IntRBTreeNode *) rbt_find(tree, (RBTNode *) &node);
456 20000 : if (chosen[i])
457 : {
458 : /* Deleted element should be absent */
459 2000 : if (result != NULL)
460 0 : elog(ERROR, "deleted element still present in the rbtree");
461 : }
462 : else
463 : {
464 : /* Else it should be present */
465 18000 : if (result == NULL || result->key != i)
466 0 : elog(ERROR, "delete operation removed wrong rbtree value");
467 : }
468 : }
469 :
470 : /* Delete remaining elements, so as to exercise reducing tree to empty */
471 20002 : for (i = 0; i < size; i++)
472 : {
473 : IntRBTreeNode find;
474 : IntRBTreeNode *node;
475 :
476 20000 : if (chosen[i])
477 2000 : continue;
478 18000 : find.key = i;
479 : /* Locate the node to be deleted */
480 18000 : node = (IntRBTreeNode *) rbt_find(tree, (RBTNode *) &find);
481 18000 : if (node == NULL || node->key != i)
482 0 : elog(ERROR, "expected element was not found during deleting");
483 : /* Delete it */
484 18000 : rbt_delete(tree, (RBTNode *) node);
485 : }
486 :
487 : /* Tree should now be empty */
488 2 : if (rbt_leftmost(tree) != NULL)
489 0 : elog(ERROR, "deleting all elements failed");
490 :
491 2 : pfree(deleteIds);
492 2 : pfree(chosen);
493 2 : }
494 :
495 : /*
496 : * SQL-callable entry point to perform all tests
497 : *
498 : * Argument is the number of entries to put in the trees
499 : */
500 4 : PG_FUNCTION_INFO_V1(test_rb_tree);
501 :
502 : Datum
503 2 : test_rb_tree(PG_FUNCTION_ARGS)
504 : {
505 2 : int size = PG_GETARG_INT32(0);
506 :
507 2 : if (size <= 0 || size > MaxAllocSize / sizeof(int))
508 0 : elog(ERROR, "invalid size for test_rb_tree: %d", size);
509 2 : testleftright(size);
510 2 : testrightleft(size);
511 2 : testfind(size);
512 2 : testfindltgt(size);
513 2 : testleftmost(size);
514 2 : testdelete(size, Max(size / 10, 1));
515 2 : PG_RETURN_VOID();
516 : }
|