comparison ext/sord/src/zix/btree.c @ 226:c5cdc9e6a4bf

Add these external library files
author Chris Cannam <cannam@all-day-breakfast.com>
date Fri, 09 Jun 2017 16:41:31 +0100
parents
children b32c68f08ec0
comparison
equal deleted inserted replaced
225:025b3e2f7c17 226:c5cdc9e6a4bf
1 /*
2 Copyright 2011-2014 David Robillard <http://drobilla.net>
3
4 Permission to use, copy, modify, and/or distribute this software for any
5 purpose with or without fee is hereby granted, provided that the above
6 copyright notice and this permission notice appear in all copies.
7
8 THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 */
16
17 #include <assert.h>
18 #include <stdint.h>
19 #include <stdio.h>
20 #include <stdlib.h>
21 #include <string.h>
22
23 #include "zix/btree.h"
24
25 // #define ZIX_BTREE_DEBUG 1
26
27 #define ZIX_BTREE_PAGE_SIZE 4096
28 #define ZIX_BTREE_NODE_SPACE (ZIX_BTREE_PAGE_SIZE - 2 * sizeof(uint16_t))
29 #define ZIX_BTREE_LEAF_VALS ((ZIX_BTREE_NODE_SPACE / sizeof(void*)) - 1)
30 #define ZIX_BTREE_INODE_VALS (ZIX_BTREE_LEAF_VALS / 2)
31
32 struct ZixBTreeImpl {
33 ZixBTreeNode* root;
34 ZixDestroyFunc destroy;
35 ZixComparator cmp;
36 void* cmp_data;
37 size_t size;
38 unsigned height; ///< Number of levels, i.e. root only has height 1
39 };
40
41 struct ZixBTreeNodeImpl {
42 uint16_t is_leaf;
43 uint16_t n_vals;
44 // On 64-bit we rely on some padding here to get page-sized nodes
45 void* vals[ZIX_BTREE_INODE_VALS]; // ZIX_BTREE_LEAF_VALS for leaves
46 ZixBTreeNode* children[ZIX_BTREE_INODE_VALS + 1]; // Nonexistent for leaves
47 };
48
49 typedef struct {
50 ZixBTreeNode* node;
51 unsigned index;
52 } ZixBTreeIterFrame;
53
54 struct ZixBTreeIterImpl {
55 unsigned level; ///< Current level in stack
56 ZixBTreeIterFrame stack[]; ///< Position stack
57 };
58
59 #ifdef ZIX_BTREE_DEBUG
60
61 ZIX_PRIVATE void
62 print_node(const ZixBTreeNode* n, const char* prefix)
63 {
64 printf("%s[", prefix);
65 for (uint16_t v = 0; v < n->n_vals; ++v) {
66 printf(" %lu", (uintptr_t)n->vals[v]);
67 }
68 printf(" ]\n");
69 }
70
71 ZIX_PRIVATE void
72 print_tree(const ZixBTreeNode* parent, const ZixBTreeNode* node, int level)
73 {
74 if (node) {
75 if (!parent) {
76 printf("TREE {\n");
77 }
78 for (int i = 0; i < level + 1; ++i) {
79 printf(" ");
80 }
81 print_node(node, "");
82 if (!node->is_leaf) {
83 for (uint16_t i = 0; i < node->n_vals + 1; ++i) {
84 print_tree(node, node->children[i], level + 1);
85 }
86 }
87 if (!parent) {
88 printf("}\n");
89 }
90 }
91 }
92
93 #endif // ZIX_BTREE_DEBUG
94
95 ZIX_PRIVATE ZixBTreeNode*
96 zix_btree_node_new(const bool leaf)
97 {
98 assert(sizeof(ZixBTreeNode) == ZIX_BTREE_PAGE_SIZE);
99 ZixBTreeNode* node = (ZixBTreeNode*)malloc(sizeof(ZixBTreeNode));
100 if (node) {
101 node->is_leaf = leaf;
102 node->n_vals = 0;
103 }
104 return node;
105 }
106
107 ZIX_API ZixBTree*
108 zix_btree_new(const ZixComparator cmp,
109 void* const cmp_data,
110 const ZixDestroyFunc destroy)
111 {
112 ZixBTree* t = (ZixBTree*)malloc(sizeof(ZixBTree));
113 if (t) {
114 t->root = zix_btree_node_new(true);
115 t->destroy = destroy;
116 t->cmp = cmp;
117 t->cmp_data = cmp_data;
118 t->size = 0;
119 t->height = 1;
120 if (!t->root) {
121 free(t);
122 return NULL;
123 }
124 }
125 return t;
126 }
127
128 ZIX_PRIVATE void
129 zix_btree_free_rec(ZixBTree* const t, ZixBTreeNode* const n)
130 {
131 if (n) {
132 if (t->destroy) {
133 for (uint16_t i = 0; i < n->n_vals; ++i) {
134 t->destroy(n->vals[i]);
135 }
136 }
137 if (!n->is_leaf) {
138 for (uint16_t i = 0; i < n->n_vals + 1; ++i) {
139 zix_btree_free_rec(t, n->children[i]);
140 }
141 }
142 free(n);
143 }
144 }
145
146 ZIX_API void
147 zix_btree_free(ZixBTree* const t)
148 {
149 if (t) {
150 zix_btree_free_rec(t, t->root);
151 free(t);
152 }
153 }
154
155 ZIX_API size_t
156 zix_btree_size(const ZixBTree* const t)
157 {
158 return t->size;
159 }
160
161 ZIX_PRIVATE uint16_t
162 zix_btree_max_vals(const ZixBTreeNode* const node)
163 {
164 return node->is_leaf ? ZIX_BTREE_LEAF_VALS : ZIX_BTREE_INODE_VALS;
165 }
166
167 ZIX_PRIVATE uint16_t
168 zix_btree_min_vals(const ZixBTreeNode* const node)
169 {
170 return ((zix_btree_max_vals(node) + 1) / 2) - 1;
171 }
172
173 /** Shift pointers in `array` of length `n` right starting at `i`. */
174 ZIX_PRIVATE void
175 zix_btree_ainsert(void** const array,
176 const uint16_t n,
177 const uint16_t i,
178 void* const e)
179 {
180 memmove(array + i + 1, array + i, (n - i) * sizeof(e));
181 array[i] = e;
182 }
183
184 /** Erase element `i` in `array` of length `n` and return erased element. */
185 ZIX_PRIVATE void*
186 zix_btree_aerase(void** const array, const uint16_t n, const uint16_t i)
187 {
188 void* const ret = array[i];
189 memmove(array + i, array + i + 1, (n - i) * sizeof(ret));
190 return ret;
191 }
192
193 /** Split lhs, the i'th child of `n`, into two nodes. */
194 ZIX_PRIVATE ZixBTreeNode*
195 zix_btree_split_child(ZixBTreeNode* const n,
196 const uint16_t i,
197 ZixBTreeNode* const lhs)
198 {
199 assert(lhs->n_vals == zix_btree_max_vals(lhs));
200 assert(n->n_vals < ZIX_BTREE_INODE_VALS);
201 assert(i < n->n_vals + 1);
202 assert(n->children[i] == lhs);
203
204 const uint16_t max_n_vals = zix_btree_max_vals(lhs);
205 ZixBTreeNode* rhs = zix_btree_node_new(lhs->is_leaf);
206 if (!rhs) {
207 return NULL;
208 }
209
210 // LHS and RHS get roughly half, less the middle value which moves up
211 lhs->n_vals = max_n_vals / 2;
212 rhs->n_vals = max_n_vals - lhs->n_vals - 1;
213
214 // Copy large half of values from LHS to new RHS node
215 memcpy(rhs->vals,
216 lhs->vals + lhs->n_vals + 1,
217 rhs->n_vals * sizeof(void*));
218
219 // Copy large half of children from LHS to new RHS node
220 if (!lhs->is_leaf) {
221 memcpy(rhs->children,
222 lhs->children + lhs->n_vals + 1,
223 (rhs->n_vals + 1) * sizeof(ZixBTreeNode*));
224 }
225
226 // Move middle value up to parent
227 zix_btree_ainsert(n->vals, n->n_vals, i, lhs->vals[lhs->n_vals]);
228
229 // Insert new RHS node in parent at position i
230 zix_btree_ainsert((void**)n->children, ++n->n_vals, i + 1, rhs);
231
232 return rhs;
233 }
234
235 /** Find the first value in `n` that is not less than `e` (lower bound). */
236 ZIX_PRIVATE uint16_t
237 zix_btree_node_find(const ZixBTree* const t,
238 const ZixBTreeNode* const n,
239 const void* const e,
240 bool* const equal)
241 {
242 uint16_t first = 0;
243 uint16_t len = n->n_vals;
244 while (len > 0) {
245 const uint16_t half = len >> 1;
246 const uint16_t i = first + half;
247 const int cmp = t->cmp(n->vals[i], e, t->cmp_data);
248 if (cmp == 0) {
249 *equal = true;
250 len = half; // Keep searching for wildcard matches
251 } else if (cmp < 0) {
252 const uint16_t chop = half + 1;
253 first += chop;
254 len -= chop;
255 } else {
256 len = half;
257 }
258 }
259 assert(!*equal || t->cmp(n->vals[first], e, t->cmp_data) == 0);
260 return first;
261 }
262
263 ZIX_API ZixStatus
264 zix_btree_insert(ZixBTree* const t, void* const e)
265 {
266 ZixBTreeNode* parent = NULL; // Parent of n
267 ZixBTreeNode* n = t->root; // Current node
268 uint16_t i = 0; // Index of n in parent
269 while (n) {
270 if (n->n_vals == zix_btree_max_vals(n)) {
271 // Node is full, split to ensure there is space for a leaf split
272 if (!parent) {
273 // Root is full, grow tree upwards
274 if (!(parent = zix_btree_node_new(false))) {
275 return ZIX_STATUS_NO_MEM;
276 }
277 t->root = parent;
278 parent->children[0] = n;
279 ++t->height;
280 }
281
282 ZixBTreeNode* const rhs = zix_btree_split_child(parent, i, n);
283 if (!rhs) {
284 return ZIX_STATUS_NO_MEM;
285 }
286
287 const int cmp = t->cmp(parent->vals[i], e, t->cmp_data);
288 if (cmp == 0) {
289 return ZIX_STATUS_EXISTS;
290 } else if (cmp < 0) {
291 // Move to new RHS
292 n = rhs;
293 ++i;
294 }
295 }
296
297 assert(!parent || parent->children[i] == n);
298
299 bool equal = false;
300 i = zix_btree_node_find(t, n, e, &equal);
301 if (equal) {
302 return ZIX_STATUS_EXISTS;
303 } else if (!n->is_leaf) {
304 // Descend to child node left of value
305 parent = n;
306 n = n->children[i];
307 } else {
308 // Insert into internal node
309 zix_btree_ainsert(n->vals, n->n_vals++, i, e);
310 break;
311 }
312 }
313
314 ++t->size;
315
316 return ZIX_STATUS_SUCCESS;
317 }
318
319 ZIX_PRIVATE ZixBTreeIter*
320 zix_btree_iter_new(const ZixBTree* const t)
321 {
322 const size_t s = t->height * sizeof(ZixBTreeIterFrame);
323 ZixBTreeIter* const i = (ZixBTreeIter*)malloc(sizeof(ZixBTreeIter) + s);
324 if (i) {
325 i->level = 0;
326 }
327 return i;
328 }
329
330 ZIX_PRIVATE void
331 zix_btree_iter_set_frame(ZixBTreeIter* const ti,
332 ZixBTreeNode* const n,
333 const uint16_t i)
334 {
335 if (ti) {
336 ti->stack[ti->level].node = n;
337 ti->stack[ti->level].index = i;
338 }
339 }
340
341 ZIX_PRIVATE bool
342 zix_btree_node_is_minimal(ZixBTreeNode* const n)
343 {
344 assert(n->n_vals >= zix_btree_min_vals(n));
345 return n->n_vals == zix_btree_min_vals(n);
346 }
347
348 /** Enlarge left child by stealing a value from its right sibling. */
349 ZIX_PRIVATE ZixBTreeNode*
350 zix_btree_rotate_left(ZixBTreeNode* const parent, const uint16_t i)
351 {
352 ZixBTreeNode* const lhs = parent->children[i];
353 ZixBTreeNode* const rhs = parent->children[i + 1];
354
355 // Move parent value to end of LHS
356 lhs->vals[lhs->n_vals++] = parent->vals[i];
357
358 // Move first child pointer from RHS to end of LHS
359 if (!lhs->is_leaf) {
360 lhs->children[lhs->n_vals] = (ZixBTreeNode*)zix_btree_aerase(
361 (void**)rhs->children, rhs->n_vals, 0);
362 }
363
364 // Move first value in RHS to parent
365 parent->vals[i] = zix_btree_aerase(rhs->vals, --rhs->n_vals, 0);
366
367 return lhs;
368 }
369
370 /** Enlarge right child by stealing a value from its left sibling. */
371 ZIX_PRIVATE ZixBTreeNode*
372 zix_btree_rotate_right(ZixBTreeNode* const parent, const uint16_t i)
373 {
374 ZixBTreeNode* const lhs = parent->children[i - 1];
375 ZixBTreeNode* const rhs = parent->children[i];
376
377 // Prepend parent value to RHS
378 zix_btree_ainsert(rhs->vals, rhs->n_vals++, 0, parent->vals[i - 1]);
379
380 // Move last child pointer from LHS and prepend to RHS
381 if (!lhs->is_leaf) {
382 zix_btree_ainsert((void**)rhs->children,
383 rhs->n_vals,
384 0,
385 lhs->children[lhs->n_vals]);
386 }
387
388 // Move last value from LHS to parent
389 parent->vals[i - 1] = lhs->vals[--lhs->n_vals];
390
391 return rhs;
392 }
393
394 /** Move n[i] down, merge the left and right child, return the merged node. */
395 ZIX_PRIVATE ZixBTreeNode*
396 zix_btree_merge(ZixBTree* const t, ZixBTreeNode* const n, const uint16_t i)
397 {
398 ZixBTreeNode* const lhs = n->children[i];
399 ZixBTreeNode* const rhs = n->children[i + 1];
400
401 assert(zix_btree_node_is_minimal(n->children[i]));
402 assert(lhs->n_vals + rhs->n_vals < zix_btree_max_vals(lhs));
403
404 // Move parent value to end of LHS
405 lhs->vals[lhs->n_vals++] = zix_btree_aerase(n->vals, n->n_vals, i);
406
407 // Erase corresponding child pointer (to RHS) in parent
408 zix_btree_aerase((void**)n->children, n->n_vals, i + 1);
409
410 // Add everything from RHS to end of LHS
411 memcpy(lhs->vals + lhs->n_vals, rhs->vals, rhs->n_vals * sizeof(void*));
412 if (!lhs->is_leaf) {
413 memcpy(lhs->children + lhs->n_vals,
414 rhs->children,
415 (rhs->n_vals + 1) * sizeof(void*));
416 }
417 lhs->n_vals += rhs->n_vals;
418
419 if (--n->n_vals == 0) {
420 // Root is now empty, replace it with its only child
421 assert(n == t->root);
422 t->root = lhs;
423 free(n);
424 }
425
426 free(rhs);
427 return lhs;
428 }
429
430 /** Remove and return the min value from the subtree rooted at `n`. */
431 ZIX_PRIVATE void*
432 zix_btree_remove_min(ZixBTree* const t, ZixBTreeNode* n)
433 {
434 while (!n->is_leaf) {
435 if (zix_btree_node_is_minimal(n->children[0])) {
436 // Leftmost child is minimal, must expand
437 if (!zix_btree_node_is_minimal(n->children[1])) {
438 // Child's right sibling has at least one key to steal
439 n = zix_btree_rotate_left(n, 0);
440 } else {
441 // Both child and right sibling are minimal, merge
442 n = zix_btree_merge(t, n, 0);
443 }
444 } else {
445 n = n->children[0];
446 }
447 }
448
449 return zix_btree_aerase(n->vals, --n->n_vals, 0);
450 }
451
452 /** Remove and return the max value from the subtree rooted at `n`. */
453 ZIX_PRIVATE void*
454 zix_btree_remove_max(ZixBTree* const t, ZixBTreeNode* n)
455 {
456 while (!n->is_leaf) {
457 if (zix_btree_node_is_minimal(n->children[n->n_vals])) {
458 // Leftmost child is minimal, must expand
459 if (!zix_btree_node_is_minimal(n->children[n->n_vals - 1])) {
460 // Child's left sibling has at least one key to steal
461 n = zix_btree_rotate_right(n, n->n_vals);
462 } else {
463 // Both child and left sibling are minimal, merge
464 n = zix_btree_merge(t, n, n->n_vals - 1);
465 }
466 } else {
467 n = n->children[n->n_vals];
468 }
469 }
470
471 return n->vals[--n->n_vals];
472 }
473
474 ZIX_API ZixStatus
475 zix_btree_remove(ZixBTree* const t,
476 const void* const e,
477 void** const out,
478 ZixBTreeIter** const next)
479 {
480 ZixBTreeNode* n = t->root;
481 ZixBTreeIter* ti = NULL;
482 const bool user_iter = next && *next;
483 if (next) {
484 if (!*next && !(*next = zix_btree_iter_new(t))) {
485 return ZIX_STATUS_NO_MEM;
486 }
487 ti = *next;
488 ti->level = 0;
489 }
490
491 while (true) {
492 /* To remove in a single walk down, the tree is adjusted along the way
493 so that the current node always has at least one more value than the
494 minimum required in general. Thus, there is always room to remove
495 without adjusting on the way back up. */
496 assert(n == t->root || !zix_btree_node_is_minimal(n));
497
498 bool equal = false;
499 const uint16_t i = zix_btree_node_find(t, n, e, &equal);
500 zix_btree_iter_set_frame(ti, n, i);
501 if (n->is_leaf) {
502 if (equal) {
503 // Found in leaf node
504 *out = zix_btree_aerase(n->vals, --n->n_vals, i);
505 if (ti && i == n->n_vals) {
506 if (i == 0) {
507 ti->stack[ti->level = 0].node = NULL;
508 } else {
509 --ti->stack[ti->level].index;
510 zix_btree_iter_increment(ti);
511 }
512 }
513 --t->size;
514 return ZIX_STATUS_SUCCESS;
515 } else {
516 // Not found in leaf node, or tree
517 if (ti && !user_iter) {
518 zix_btree_iter_free(ti);
519 *next = NULL;
520 }
521 return ZIX_STATUS_NOT_FOUND;
522 }
523 } else if (equal) {
524 // Found in internal node
525 if (!zix_btree_node_is_minimal(n->children[i])) {
526 // Left child can remove without merge
527 *out = n->vals[i];
528 n->vals[i] = zix_btree_remove_max(t, n->children[i]);
529 --t->size;
530 return ZIX_STATUS_SUCCESS;
531 } else if (!zix_btree_node_is_minimal(n->children[i + 1])) {
532 // Right child can remove without merge
533 *out = n->vals[i];
534 n->vals[i] = zix_btree_remove_min(t, n->children[i + 1]);
535 --t->size;
536 return ZIX_STATUS_SUCCESS;
537 } else {
538 // Both preceding and succeeding child are minimal
539 n = zix_btree_merge(t, n, i);
540 }
541 } else {
542 // Not found in internal node, key is in/under children[i]
543 if (zix_btree_node_is_minimal(n->children[i])) {
544 if (i > 0 && !zix_btree_node_is_minimal(n->children[i - 1])) {
545 // Steal a key from child's left sibling
546 n = zix_btree_rotate_right(n, i);
547 } else if (i < n->n_vals &&
548 !zix_btree_node_is_minimal(n->children[i + 1])) {
549 // Steal a key from child's right sibling
550 n = zix_btree_rotate_left(n, i);
551 } else {
552 // Both child's siblings are minimal, merge them
553 if (i < n->n_vals) {
554 n = zix_btree_merge(t, n, i);
555 } else {
556 n = zix_btree_merge(t, n, i - 1);
557 if (ti) {
558 --ti->stack[ti->level].index;
559 }
560 }
561 }
562 } else {
563 n = n->children[i];
564 }
565 }
566 if (ti) {
567 ++ti->level;
568 }
569 }
570
571 assert(false); // Not reached
572 return ZIX_STATUS_ERROR;
573 }
574
575 ZIX_API ZixStatus
576 zix_btree_find(const ZixBTree* const t,
577 const void* const e,
578 ZixBTreeIter** const ti)
579 {
580 ZixBTreeNode* n = t->root;
581 if (!(*ti = zix_btree_iter_new(t))) {
582 return ZIX_STATUS_NO_MEM;
583 }
584
585 while (n) {
586 bool equal = false;
587 const uint16_t i = zix_btree_node_find(t, n, e, &equal);
588
589 zix_btree_iter_set_frame(*ti, n, i);
590
591 if (equal) {
592 return ZIX_STATUS_SUCCESS;
593 } else if (n->is_leaf) {
594 break;
595 } else {
596 ++(*ti)->level;
597 n = n->children[i];
598 }
599 }
600
601 zix_btree_iter_free(*ti);
602 *ti = NULL;
603 return ZIX_STATUS_NOT_FOUND;
604 }
605
606 ZIX_API ZixStatus
607 zix_btree_lower_bound(const ZixBTree* const t,
608 const void* const e,
609 ZixBTreeIter** const ti)
610 {
611 if (!t) {
612 *ti = NULL;
613 return ZIX_STATUS_BAD_ARG;
614 }
615
616 ZixBTreeNode* n = t->root;
617 bool found = false;
618 unsigned found_level = 0;
619 if (!(*ti = zix_btree_iter_new(t))) {
620 return ZIX_STATUS_NO_MEM;
621 }
622
623 while (n) {
624 bool equal = false;
625 const uint16_t i = zix_btree_node_find(t, n, e, &equal);
626
627 zix_btree_iter_set_frame(*ti, n, i);
628
629 if (equal) {
630 found_level = (*ti)->level;
631 found = true;
632 }
633
634 if (n->is_leaf) {
635 break;
636 } else {
637 ++(*ti)->level;
638 n = n->children[i];
639 assert(n);
640 }
641 }
642
643 const ZixBTreeIterFrame* const frame = &(*ti)->stack[(*ti)->level];
644 if (frame->index == frame->node->n_vals) {
645 if (found) {
646 // Found on a previous level but went too far
647 (*ti)->level = found_level;
648 } else {
649 // Reached end (key is greater than everything in tree)
650 (*ti)->stack[0].node = NULL;
651 }
652 }
653
654 return ZIX_STATUS_SUCCESS;
655 }
656
657 ZIX_API void*
658 zix_btree_get(const ZixBTreeIter* const ti)
659 {
660 const ZixBTreeIterFrame* const frame = &ti->stack[ti->level];
661 assert(frame->index < frame->node->n_vals);
662 return frame->node->vals[frame->index];
663 }
664
665 ZIX_API ZixBTreeIter*
666 zix_btree_begin(const ZixBTree* const t)
667 {
668 ZixBTreeIter* const i = zix_btree_iter_new(t);
669 if (!i) {
670 return NULL;
671 } else if (t->size == 0) {
672 i->stack[0].node = NULL;
673 } else {
674 ZixBTreeNode* n = t->root;
675 i->stack[0].node = n;
676 i->stack[0].index = 0;
677 while (!n->is_leaf) {
678 n = n->children[0];
679 ++i->level;
680 i->stack[i->level].node = n;
681 i->stack[i->level].index = 0;
682 }
683 }
684 return i;
685 }
686
687 ZIX_API bool
688 zix_btree_iter_is_end(const ZixBTreeIter* const i)
689 {
690 return !i || i->stack[0].node == NULL;
691 }
692
693 ZIX_API void
694 zix_btree_iter_increment(ZixBTreeIter* const i)
695 {
696 ZixBTreeIterFrame* f = &i->stack[i->level];
697 if (f->node->is_leaf) {
698 // Leaf, move right
699 assert(f->index < f->node->n_vals);
700 if (++f->index == f->node->n_vals) {
701 // Reached end of leaf, move up
702 f = &i->stack[i->level];
703 while (i->level > 0 && f->index == f->node->n_vals) {
704 f = &i->stack[--i->level];
705 assert(f->index <= f->node->n_vals);
706 }
707
708 if (f->index == f->node->n_vals) {
709 // Reached end of tree
710 assert(i->level == 0);
711 f->node = NULL;
712 f->index = 0;
713 }
714 }
715 } else {
716 // Internal node, move down to next child
717 assert(f->index < f->node->n_vals);
718 ZixBTreeNode* child = f->node->children[++f->index];
719
720 f = &i->stack[++i->level];
721 f->node = child;
722 f->index = 0;
723
724 // Move down and left until we hit a leaf
725 while (!f->node->is_leaf) {
726 child = f->node->children[0];
727 f = &i->stack[++i->level];
728 f->node = child;
729 f->index = 0;
730 }
731 }
732 }
733
734 ZIX_API void
735 zix_btree_iter_free(ZixBTreeIter* const i)
736 {
737 free(i);
738 }