-
Notifications
You must be signed in to change notification settings - Fork 38
/
Copy pathbtree.h
206 lines (172 loc) · 8.16 KB
/
btree.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
// Copyright 2020 Joshua J Baker. All rights reserved.
// Use of this source code is governed by an MIT-style
// license that can be found in the LICENSE file.
#ifndef BTREE_H
#define BTREE_H
#include <stdbool.h>
#include <stddef.h>
#include <stdint.h>
struct btree;
// btree_new returns a new B-tree.
//
// Param elsize is the size of each element in the tree. Every element that
// is inserted, deleted, or searched will be this size.
// Param max_items is the maximum number of items per node. Setting this to
// zero will default to 256.
// Param compare is a function that compares items in the tree. See the
// qsort stdlib function for an example of how this function works.
// Param udata is user-defined data that is passed to the compare callback
// and the item callbacks defined in btree_set_item_callbacks.
//
// The btree must be freed with btree_free().
struct btree *btree_new(size_t elsize, size_t max_items,
int (*compare)(const void *a, const void *b, void *udata),
void *udata);
// btree_new_with_allocator returns a new btree using a custom allocator.
//
// See btree_new for more information
struct btree *btree_new_with_allocator(
void *(*malloc)(size_t),
void *(*realloc)(void *, size_t),
void (*free)(void*),
size_t elsize,
size_t max_items,
int (*compare)(const void *a, const void *b, void *udata),
void *udata);
// btree_set_item_callbacks sets the item clone and free callbacks that will be
// called internally by the btree when items are inserted and removed.
//
// These callbacks are optional but may be needed by programs that require
// copy-on-write support by using the btree_clone function.
//
// The clone function should return true if the clone succeeded or false if the
// system is out of memory.
void btree_set_item_callbacks(struct btree *btree,
bool (*clone)(const void *item, void *into, void *udata),
void (*free)(const void *item, void *udata));
// btree_free removes all items from the btree and frees any allocated memory.
void btree_free(struct btree *btree);
// btree_free removes all items from the btree.
void btree_clear(struct btree *btree);
// btree_oom returns true if the last write operation failed because the system
// has no more memory available.
//
// Functions that have the first param being a non-const btree receiver are
// candidates for possible out-of-memory conditions, such as btree_set,
// btree_delete, btree_load, etc.
bool btree_oom(const struct btree *btree);
// btree_height returns the height of the btree from root to leaf or zero if
// the btree is empty.
size_t btree_height(const struct btree *btree);
// btree_count returns the number of items in the btree.
size_t btree_count(const struct btree *btree);
// btree_clone makes an instant copy of the btree.
// This operation uses shadowing / copy-on-write.
struct btree *btree_clone(struct btree *btree);
// btree_set inserts or replaces an item in the btree. If an item is replaced
// then it is returned otherwise NULL is returned.
//
// If the system fails allocate the memory needed then NULL is returned
// and btree_oom() returns true.
const void *btree_set(struct btree *btree, const void *item);
// btree_delete removes an item from the B-tree and returns it.
//
// Returns NULL if item not found.
// This operation may trigger node copies if the btree was cloned using
// btree_clone.
// If the system fails allocate the memory needed then NULL is returned
// and btree_oom() returns true.
const void *btree_delete(struct btree *btree, const void *key);
// btree_load is the same as btree_set but is optimized for sequential bulk
// loading. It can be up to 10x faster than btree_set when the items are
// in exact order, but up to 25% slower when not in exact order.
//
// If the system fails allocate the memory needed then NULL is returned
// and btree_oom() returns true.
const void *btree_load(struct btree *btree, const void *item);
// btree_pop_min removes the first item in the btree and returns it.
//
// Returns NULL if btree is empty.
// This operation may trigger node copies if the btree was cloned using
// btree_clone.
// If the system fails allocate the memory needed then NULL is returned
// and btree_oom() returns true.
const void *btree_pop_min(struct btree *btree);
// btree_pop_min removes the last item in the btree and returns it.
//
// Returns NULL if btree is empty.
// This operation may trigger node copies if the btree was cloned using
// btree_clone.
// If the system fails allocate the memory needed then NULL is returned
// and btree_oom() returns true.
const void *btree_pop_max(struct btree *btree);
// btree_pop_min returns the first item in the btree or NULL if btree is empty.
const void *btree_min(const struct btree *btree);
// btree_pop_min returns the last item in the btree or NULL if btree is empty.
const void *btree_max(const struct btree *btree);
// btree_get returns the item based on the provided key.
//
// Returns NULL if item is not found.
const void *btree_get(const struct btree *btree, const void *key);
// btree_ascend scans the tree within the range [pivot, last].
//
// In other words btree_ascend iterates over all items that are
// greater-than-or-equal-to pivot in ascending order.
//
// Param pivot can be NULL, which means all items are iterated over.
// Param iter can return false to stop iteration early.
// Returns false if the iteration has been stopped early.
bool btree_ascend(const struct btree *btree, const void *pivot,
bool (*iter)(const void *item, void *udata), void *udata);
// btree_descend scans the tree within the range [pivot, first].
// In other words btree_descend() iterates over all items that are
// less-than-or-equal-to pivot in descending order.
//
// Param pivot can be NULL, which means all items are iterated over.
// Param iter can return false to stop iteration early.
// Returns false if the iteration has been stopped early.
bool btree_descend(const struct btree *btree, const void *pivot,
bool (*iter)(const void *item, void *udata), void *udata);
// btree_set_hint is the same as btree_set except that an optional "hint" can
// be provided which may make the operation quicker when done as a batch or
// in a userspace context.
const void *btree_set_hint(struct btree *btree, const void *item,
uint64_t *hint);
// btree_get_hint is the same as btree_get except that an optional "hint" can
// be provided which may make the operation quicker when done as a batch or
// in a userspace context.
const void *btree_get_hint(const struct btree *btree, const void *key,
uint64_t *hint);
// btree_delete_hint is the same as btree_delete except that an optional "hint"
// can be provided which may make the operation quicker when done as a batch or
// in a userspace context.
const void *btree_delete_hint(struct btree *btree, const void *key,
uint64_t *hint);
// btree_ascend_hint is the same as btree_ascend except that an optional
// "hint" can be provided which may make the operation quicker when done as a
// batch or in a userspace context.
bool btree_ascend_hint(const struct btree *btree, const void *pivot,
bool (*iter)(const void *item, void *udata),
void *udata, uint64_t *hint);
// btree_descend_hint is the same as btree_descend except that an optional
// "hint" can be provided which may make the operation quicker when done as a
// batch or in a userspace context.
bool btree_descend_hint(const struct btree *btree, const void *pivot,
bool (*iter)(const void *item, void *udata),
void *udata, uint64_t *hint);
// btree_set_searcher allows for setting a custom search function.
void btree_set_searcher(struct btree *btree,
int (*searcher)(const void *items, size_t nitems, const void *key,
bool *found, void *udata));
// Loop-based iterator
struct btree_iter *btree_iter_new(const struct btree *btree);
void btree_iter_free(struct btree_iter *iter);
bool btree_iter_first(struct btree_iter *iter);
bool btree_iter_last(struct btree_iter *iter);
bool btree_iter_next(struct btree_iter *iter);
bool btree_iter_prev(struct btree_iter *iter);
bool btree_iter_seek(struct btree_iter *iter, const void *key);
const void *btree_iter_item(struct btree_iter *iter);
// DEPRECATED: use `btree_new_with_allocator`
void btree_set_allocator(void *(malloc)(size_t), void (*free)(void*));
#endif