xref: /freebsd/share/man/man3/tree.3 (revision 503b7f94d831642308585c980a7cabe4470960d0)
1 87f5f0ecSDag-Erling Smørgrav.\"	$OpenBSD: tree.3,v 1.7 2002/06/12 01:09:20 provos Exp $
2 480565d5SRuslan Ermilov.\"
3 480565d5SRuslan Ermilov.\" Copyright 2002 Niels Provos <provos@citi.umich.edu>
4 480565d5SRuslan Ermilov.\" All rights reserved.
5 480565d5SRuslan Ermilov.\"
6 480565d5SRuslan Ermilov.\" Redistribution and use in source and binary forms, with or without
7 480565d5SRuslan Ermilov.\" modification, are permitted provided that the following conditions
8 480565d5SRuslan Ermilov.\" are met:
9 480565d5SRuslan Ermilov.\" 1. Redistributions of source code must retain the above copyright
10 480565d5SRuslan Ermilov.\"    notice, this list of conditions and the following disclaimer.
11 480565d5SRuslan Ermilov.\" 2. Redistributions in binary form must reproduce the above copyright
12 480565d5SRuslan Ermilov.\"    notice, this list of conditions and the following disclaimer in the
13 480565d5SRuslan Ermilov.\"    documentation and/or other materials provided with the distribution.
14 480565d5SRuslan Ermilov.\" 3. All advertising materials mentioning features or use of this software
15 480565d5SRuslan Ermilov.\"    must display the following acknowledgement:
16 480565d5SRuslan Ermilov.\"      This product includes software developed by Niels Provos.
17 480565d5SRuslan Ermilov.\" 4. The name of the author may not be used to endorse or promote products
18 480565d5SRuslan Ermilov.\"    derived from this software without specific prior written permission.
19 480565d5SRuslan Ermilov.\"
20 480565d5SRuslan Ermilov.\" THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
21 480565d5SRuslan Ermilov.\" IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
22 480565d5SRuslan Ermilov.\" OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
23 480565d5SRuslan Ermilov.\" IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
24 480565d5SRuslan Ermilov.\" INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
25 480565d5SRuslan Ermilov.\" NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26 480565d5SRuslan Ermilov.\" DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27 480565d5SRuslan Ermilov.\" THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28 480565d5SRuslan Ermilov.\" (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
29 480565d5SRuslan Ermilov.\" THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30 480565d5SRuslan Ermilov.\"
31 *503b7f94SMaxim Konovalov.Dd August 2, 2024
32 87f5f0ecSDag-Erling Smørgrav.Dt TREE 3
33 87f5f0ecSDag-Erling Smørgrav.Os
34 87f5f0ecSDag-Erling Smørgrav.Sh NAME
35 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_PROTOTYPE ,
36 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_GENERATE ,
37 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_ENTRY ,
38 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_HEAD ,
39 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_INITIALIZER ,
40 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_ROOT ,
41 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_EMPTY ,
42 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_NEXT ,
43 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_MIN ,
44 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_MAX ,
45 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_FIND ,
46 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_LEFT ,
47 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_RIGHT ,
48 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_FOREACH ,
49 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_INIT ,
50 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_INSERT ,
51 87f5f0ecSDag-Erling Smørgrav.Nm SPLAY_REMOVE ,
52 87f5f0ecSDag-Erling Smørgrav.Nm RB_PROTOTYPE ,
53 d72cd779SJason Evans.Nm RB_PROTOTYPE_STATIC ,
54 51782e3aSKonstantin Belousov.Nm RB_PROTOTYPE_INSERT ,
55 51782e3aSKonstantin Belousov.Nm RB_PROTOTYPE_INSERT_COLOR ,
56 51782e3aSKonstantin Belousov.Nm RB_PROTOTYPE_REMOVE ,
57 51782e3aSKonstantin Belousov.Nm RB_PROTOTYPE_REMOVE_COLOR ,
58 51782e3aSKonstantin Belousov.Nm RB_PROTOTYPE_FIND ,
59 51782e3aSKonstantin Belousov.Nm RB_PROTOTYPE_NFIND ,
60 51782e3aSKonstantin Belousov.Nm RB_PROTOTYPE_NEXT ,
61 51782e3aSKonstantin Belousov.Nm RB_PROTOTYPE_PREV ,
62 51782e3aSKonstantin Belousov.Nm RB_PROTOTYPE_MINMAX ,
63 22823764SEdward Tomasz Napierala.Nm RB_PROTOTYPE_REINSERT ,
64 87f5f0ecSDag-Erling Smørgrav.Nm RB_GENERATE ,
65 d72cd779SJason Evans.Nm RB_GENERATE_STATIC ,
66 51782e3aSKonstantin Belousov.Nm RB_GENERATE_INSERT ,
67 51782e3aSKonstantin Belousov.Nm RB_GENERATE_INSERT_COLOR ,
68 51782e3aSKonstantin Belousov.Nm RB_GENERATE_REMOVE ,
69 51782e3aSKonstantin Belousov.Nm RB_GENERATE_REMOVE_COLOR ,
70 51782e3aSKonstantin Belousov.Nm RB_GENERATE_FIND ,
71 51782e3aSKonstantin Belousov.Nm RB_GENERATE_NFIND ,
72 51782e3aSKonstantin Belousov.Nm RB_GENERATE_NEXT ,
73 51782e3aSKonstantin Belousov.Nm RB_GENERATE_PREV ,
74 51782e3aSKonstantin Belousov.Nm RB_GENERATE_MINMAX ,
75 22823764SEdward Tomasz Napierala.Nm RB_GENERATE_REINSERT ,
76 87f5f0ecSDag-Erling Smørgrav.Nm RB_ENTRY ,
77 87f5f0ecSDag-Erling Smørgrav.Nm RB_HEAD ,
78 87f5f0ecSDag-Erling Smørgrav.Nm RB_INITIALIZER ,
79 87f5f0ecSDag-Erling Smørgrav.Nm RB_ROOT ,
80 87f5f0ecSDag-Erling Smørgrav.Nm RB_EMPTY ,
81 87f5f0ecSDag-Erling Smørgrav.Nm RB_NEXT ,
82 8e4fd0a1SJason Evans.Nm RB_PREV ,
83 87f5f0ecSDag-Erling Smørgrav.Nm RB_MIN ,
84 87f5f0ecSDag-Erling Smørgrav.Nm RB_MAX ,
85 87f5f0ecSDag-Erling Smørgrav.Nm RB_FIND ,
86 06115e08SJason Evans.Nm RB_NFIND ,
87 87f5f0ecSDag-Erling Smørgrav.Nm RB_LEFT ,
88 87f5f0ecSDag-Erling Smørgrav.Nm RB_RIGHT ,
89 87f5f0ecSDag-Erling Smørgrav.Nm RB_PARENT ,
90 87f5f0ecSDag-Erling Smørgrav.Nm RB_FOREACH ,
91 bff27689SBruce M Simpson.Nm RB_FOREACH_FROM ,
92 9dbae282SGleb Smirnoff.Nm RB_FOREACH_SAFE ,
93 8e4fd0a1SJason Evans.Nm RB_FOREACH_REVERSE ,
94 bff27689SBruce M Simpson.Nm RB_FOREACH_REVERSE_FROM ,
95 9dbae282SGleb Smirnoff.Nm RB_FOREACH_REVERSE_SAFE ,
96 87f5f0ecSDag-Erling Smørgrav.Nm RB_INIT ,
97 87f5f0ecSDag-Erling Smørgrav.Nm RB_INSERT ,
98 368ee2f8SDoug Moore.Nm RB_INSERT_NEXT ,
99 368ee2f8SDoug Moore.Nm RB_INSERT_PREV ,
100 22823764SEdward Tomasz Napierala.Nm RB_REMOVE ,
101 a8380d27SDoug Moore.Nm RB_REINSERT ,
102 a8380d27SDoug Moore.Nm RB_AUGMENT
103 b16f993eSDoug Moore.Nm RB_AUGMENT_CHECK,
104 35557a0dSDoug Moore.Nm RB_UPDATE_AUGMENT
105 e605dcc9SDoug Moore.Nd "implementations of splay and rank-balanced (wavl) trees"
106 87f5f0ecSDag-Erling Smørgrav.Sh SYNOPSIS
107 480565d5SRuslan Ermilov.In sys/tree.h
108 480565d5SRuslan Ermilov.Fn SPLAY_PROTOTYPE NAME TYPE FIELD CMP
109 480565d5SRuslan Ermilov.Fn SPLAY_GENERATE NAME TYPE FIELD CMP
110 480565d5SRuslan Ermilov.Fn SPLAY_ENTRY TYPE
111 480565d5SRuslan Ermilov.Fn SPLAY_HEAD HEADNAME TYPE
112 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
113 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_INITIALIZER "SPLAY_HEAD *head"
114 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_ROOT "SPLAY_HEAD *head"
115 480565d5SRuslan Ermilov.Ft bool
116 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_EMPTY "SPLAY_HEAD *head"
117 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
118 480565d5SRuslan Ermilov.Fn SPLAY_NEXT NAME "SPLAY_HEAD *head" "struct TYPE *elm"
119 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
120 480565d5SRuslan Ermilov.Fn SPLAY_MIN NAME "SPLAY_HEAD *head"
121 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
122 480565d5SRuslan Ermilov.Fn SPLAY_MAX NAME "SPLAY_HEAD *head"
123 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
124 480565d5SRuslan Ermilov.Fn SPLAY_FIND NAME "SPLAY_HEAD *head" "struct TYPE *elm"
125 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
126 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_LEFT "struct TYPE *elm" "SPLAY_ENTRY NAME"
127 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
128 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_RIGHT "struct TYPE *elm" "SPLAY_ENTRY NAME"
129 480565d5SRuslan Ermilov.Fn SPLAY_FOREACH VARNAME NAME "SPLAY_HEAD *head"
130 87f5f0ecSDag-Erling Smørgrav.Ft void
131 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_INIT "SPLAY_HEAD *head"
132 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
133 480565d5SRuslan Ermilov.Fn SPLAY_INSERT NAME "SPLAY_HEAD *head" "struct TYPE *elm"
134 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
135 480565d5SRuslan Ermilov.Fn SPLAY_REMOVE NAME "SPLAY_HEAD *head" "struct TYPE *elm"
136 480565d5SRuslan Ermilov.Fn RB_PROTOTYPE NAME TYPE FIELD CMP
137 d72cd779SJason Evans.Fn RB_PROTOTYPE_STATIC NAME TYPE FIELD CMP
138 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_INSERT NAME TYPE ATTR
139 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_INSERT_COLOR NAME TYPE ATTR
140 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_REMOVE NAME TYPE ATTR
141 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_REMOVE_COLOR NAME TYPE ATTR
142 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_FIND NAME TYPE ATTR
143 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_NFIND NAME TYPE ATTR
144 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_NEXT NAME TYPE ATTR
145 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_PREV NAME TYPE ATTR
146 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_MINMAX NAME TYPE ATTR
147 22823764SEdward Tomasz Napierala.Fn RB_PROTOTYPE_REINSERT NAME TYPE ATTR
148 480565d5SRuslan Ermilov.Fn RB_GENERATE NAME TYPE FIELD CMP
149 d72cd779SJason Evans.Fn RB_GENERATE_STATIC NAME TYPE FIELD CMP
150 51782e3aSKonstantin Belousov.Fn RB_GENERATE_INSERT NAME TYPE FIELD CMP ATTR
151 51782e3aSKonstantin Belousov.Fn RB_GENERATE_INSERT_COLOR NAME TYPE FIELD ATTR
152 51782e3aSKonstantin Belousov.Fn RB_GENERATE_REMOVE NAME TYPE FIELD ATTR
153 51782e3aSKonstantin Belousov.Fn RB_GENERATE_REMOVE_COLOR NAME TYPE FIELD ATTR
154 51782e3aSKonstantin Belousov.Fn RB_GENERATE_FIND NAME TYPE FIELD CMP ATTR
155 51782e3aSKonstantin Belousov.Fn RB_GENERATE_NFIND NAME TYPE FIELD CMP ATTR
156 51782e3aSKonstantin Belousov.Fn RB_GENERATE_NEXT NAME TYPE FIELD ATTR
157 51782e3aSKonstantin Belousov.Fn RB_GENERATE_PREV NAME TYPE FIELD ATTR
158 51782e3aSKonstantin Belousov.Fn RB_GENERATE_MINMAX NAME TYPE FIELD ATTR
159 22823764SEdward Tomasz Napierala.Fn RB_GENERATE_REINSERT NAME TYPE FIELD CMP ATTR
160 480565d5SRuslan Ermilov.Fn RB_ENTRY TYPE
161 480565d5SRuslan Ermilov.Fn RB_HEAD HEADNAME TYPE
162 87f5f0ecSDag-Erling Smørgrav.Fn RB_INITIALIZER "RB_HEAD *head"
163 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
164 87f5f0ecSDag-Erling Smørgrav.Fn RB_ROOT "RB_HEAD *head"
165 87f5f0ecSDag-Erling Smørgrav.Ft "bool"
166 87f5f0ecSDag-Erling Smørgrav.Fn RB_EMPTY "RB_HEAD *head"
167 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
168 480565d5SRuslan Ermilov.Fn RB_NEXT NAME "RB_HEAD *head" "struct TYPE *elm"
169 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
170 8e4fd0a1SJason Evans.Fn RB_PREV NAME "RB_HEAD *head" "struct TYPE *elm"
171 8e4fd0a1SJason Evans.Ft "struct TYPE *"
172 480565d5SRuslan Ermilov.Fn RB_MIN NAME "RB_HEAD *head"
173 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
174 480565d5SRuslan Ermilov.Fn RB_MAX NAME "RB_HEAD *head"
175 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
176 480565d5SRuslan Ermilov.Fn RB_FIND NAME "RB_HEAD *head" "struct TYPE *elm"
177 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
178 06115e08SJason Evans.Fn RB_NFIND NAME "RB_HEAD *head" "struct TYPE *elm"
179 06115e08SJason Evans.Ft "struct TYPE *"
180 87f5f0ecSDag-Erling Smørgrav.Fn RB_LEFT "struct TYPE *elm" "RB_ENTRY NAME"
181 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
182 87f5f0ecSDag-Erling Smørgrav.Fn RB_RIGHT "struct TYPE *elm" "RB_ENTRY NAME"
183 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
184 87f5f0ecSDag-Erling Smørgrav.Fn RB_PARENT "struct TYPE *elm" "RB_ENTRY NAME"
185 480565d5SRuslan Ermilov.Fn RB_FOREACH VARNAME NAME "RB_HEAD *head"
186 639bf7bdSBruce M Simpson.Fn RB_FOREACH_FROM "VARNAME" "NAME" "POS_VARNAME"
187 9dbae282SGleb Smirnoff.Fn RB_FOREACH_SAFE "VARNAME" "NAME" "RB_HEAD *head" "TEMP_VARNAME"
188 8e4fd0a1SJason Evans.Fn RB_FOREACH_REVERSE VARNAME NAME "RB_HEAD *head"
189 639bf7bdSBruce M Simpson.Fn RB_FOREACH_REVERSE_FROM "VARNAME" "NAME" "POS_VARNAME"
190 9dbae282SGleb Smirnoff.Fn RB_FOREACH_REVERSE_SAFE "VARNAME" "NAME" "RB_HEAD *head" "TEMP_VARNAME"
191 87f5f0ecSDag-Erling Smørgrav.Ft void
192 87f5f0ecSDag-Erling Smørgrav.Fn RB_INIT "RB_HEAD *head"
193 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
194 480565d5SRuslan Ermilov.Fn RB_INSERT NAME "RB_HEAD *head" "struct TYPE *elm"
195 87f5f0ecSDag-Erling Smørgrav.Ft "struct TYPE *"
196 368ee2f8SDoug Moore.Fn RB_INSERT_NEXT NAME "RB_HEAD *head" "struct TYPE *elm" "struct TYPE *next"
197 368ee2f8SDoug Moore.Ft "struct TYPE *"
198 368ee2f8SDoug Moore.Fn RB_INSERT_PREV NAME "RB_HEAD *head" "struct TYPE *elm" "struct TYPE *prev"
199 368ee2f8SDoug Moore.Ft "struct TYPE *"
200 480565d5SRuslan Ermilov.Fn RB_REMOVE NAME "RB_HEAD *head" "struct TYPE *elm"
201 22823764SEdward Tomasz Napierala.Ft "struct TYPE *"
202 22823764SEdward Tomasz Napierala.Fn RB_REINSERT NAME "RB_HEAD *head" "struct TYPE *elm"
203 a8380d27SDoug Moore.Ft "void"
204 a8380d27SDoug Moore.Fn RB_AUGMENT NAME "struct TYPE *elm"
205 b16f993eSDoug Moore.Ft "bool"
206 b16f993eSDoug Moore.Fn RB_AUGMENT_CHECK NAME "struct TYPE *elm"
207 35557a0dSDoug Moore.Ft "void"
208 35557a0dSDoug Moore.Fn RB_UPDATE_AUGMENT NAME "struct TYPE *elm"
209 87f5f0ecSDag-Erling Smørgrav.Sh DESCRIPTION
210 480565d5SRuslan ErmilovThese macros define data structures for different types of trees:
211 e605dcc9SDoug Mooresplay trees and rank-balanced (wavl) trees.
212 87f5f0ecSDag-Erling Smørgrav.Pp
213 87f5f0ecSDag-Erling SmørgravIn the macro definitions,
214 87f5f0ecSDag-Erling Smørgrav.Fa TYPE
215 87f5f0ecSDag-Erling Smørgravis the name tag of a user defined structure that must contain a field of type
216 480565d5SRuslan Ermilov.Vt SPLAY_ENTRY ,
217 87f5f0ecSDag-Erling Smørgravor
218 480565d5SRuslan Ermilov.Vt RB_ENTRY ,
219 87f5f0ecSDag-Erling Smørgravnamed
220 87f5f0ecSDag-Erling Smørgrav.Fa ENTRYNAME .
221 87f5f0ecSDag-Erling SmørgravThe argument
222 87f5f0ecSDag-Erling Smørgrav.Fa HEADNAME
223 87f5f0ecSDag-Erling Smørgravis the name tag of a user defined structure that must be declared
224 87f5f0ecSDag-Erling Smørgravusing the macros
225 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_HEAD ,
226 87f5f0ecSDag-Erling Smørgravor
227 87f5f0ecSDag-Erling Smørgrav.Fn RB_HEAD .
228 87f5f0ecSDag-Erling SmørgravThe argument
229 87f5f0ecSDag-Erling Smørgrav.Fa NAME
230 87f5f0ecSDag-Erling Smørgravhas to be a unique name prefix for every tree that is defined.
231 87f5f0ecSDag-Erling Smørgrav.Pp
232 d72cd779SJason EvansThe function prototypes are declared with
233 480565d5SRuslan Ermilov.Fn SPLAY_PROTOTYPE ,
234 d72cd779SJason Evans.Fn RB_PROTOTYPE ,
235 87f5f0ecSDag-Erling Smørgravor
236 d72cd779SJason Evans.Fn RB_PROTOTYPE_STATIC .
237 d72cd779SJason EvansThe function bodies are generated with
238 480565d5SRuslan Ermilov.Fn SPLAY_GENERATE ,
239 d72cd779SJason Evans.Fn RB_GENERATE ,
240 87f5f0ecSDag-Erling Smørgravor
241 d72cd779SJason Evans.Fn RB_GENERATE_STATIC .
242 87f5f0ecSDag-Erling SmørgravSee the examples below for further explanation of how these macros are used.
243 87f5f0ecSDag-Erling Smørgrav.Sh SPLAY TREES
244 480565d5SRuslan ErmilovA splay tree is a self-organizing data structure.
245 480565d5SRuslan ErmilovEvery operation on the tree causes a splay to happen.
246 480565d5SRuslan ErmilovThe splay moves the requested
247 87f5f0ecSDag-Erling Smørgravnode to the root of the tree and partly rebalances it.
248 87f5f0ecSDag-Erling Smørgrav.Pp
249 87f5f0ecSDag-Erling SmørgravThis has the benefit that request locality causes faster lookups as
250 480565d5SRuslan Ermilovthe requested nodes move to the top of the tree.
251 480565d5SRuslan ErmilovOn the other hand, every lookup causes memory writes.
252 87f5f0ecSDag-Erling Smørgrav.Pp
253 480565d5SRuslan ErmilovThe Balance Theorem bounds the total access time for
254 480565d5SRuslan Ermilov.Ar m
255 480565d5SRuslan Ermilovoperations and
256 480565d5SRuslan Ermilov.Ar n
257 480565d5SRuslan Ermilovinserts on an initially empty tree as
258 480565d5SRuslan Ermilov.Fn O "\*[lp]m + n\*[rp]lg n" .
259 480565d5SRuslan ErmilovThe
260 480565d5SRuslan Ermilovamortized cost for a sequence of
261 480565d5SRuslan Ermilov.Ar m
262 480565d5SRuslan Ermilovaccesses to a splay tree is
263 480565d5SRuslan Ermilov.Fn O "lg n" .
264 87f5f0ecSDag-Erling Smørgrav.Pp
265 87f5f0ecSDag-Erling SmørgravA splay tree is headed by a structure defined by the
266 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_HEAD
267 87f5f0ecSDag-Erling Smørgravmacro.
268 87f5f0ecSDag-Erling SmørgravA
269 87f5f0ecSDag-Erling Smørgravstructure is declared as follows:
270 480565d5SRuslan Ermilov.Bd -ragged -offset indent
271 480565d5SRuslan Ermilov.Fn SPLAY_HEAD HEADNAME TYPE
272 480565d5SRuslan Ermilov.Va head ;
273 87f5f0ecSDag-Erling Smørgrav.Ed
274 87f5f0ecSDag-Erling Smørgrav.Pp
275 87f5f0ecSDag-Erling Smørgravwhere
276 87f5f0ecSDag-Erling Smørgrav.Fa HEADNAME
277 87f5f0ecSDag-Erling Smørgravis the name of the structure to be defined, and struct
278 87f5f0ecSDag-Erling Smørgrav.Fa TYPE
279 87f5f0ecSDag-Erling Smørgravis the type of the elements to be inserted into the tree.
280 87f5f0ecSDag-Erling Smørgrav.Pp
281 87f5f0ecSDag-Erling SmørgravThe
282 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_ENTRY
283 87f5f0ecSDag-Erling Smørgravmacro declares a structure that allows elements to be connected in the tree.
284 87f5f0ecSDag-Erling Smørgrav.Pp
285 87f5f0ecSDag-Erling SmørgravIn order to use the functions that manipulate the tree structure,
286 87f5f0ecSDag-Erling Smørgravtheir prototypes need to be declared with the
287 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_PROTOTYPE
288 87f5f0ecSDag-Erling Smørgravmacro,
289 87f5f0ecSDag-Erling Smørgravwhere
290 87f5f0ecSDag-Erling Smørgrav.Fa NAME
291 87f5f0ecSDag-Erling Smørgravis a unique identifier for this particular tree.
292 87f5f0ecSDag-Erling SmørgravThe
293 87f5f0ecSDag-Erling Smørgrav.Fa TYPE
294 87f5f0ecSDag-Erling Smørgravargument is the type of the structure that is being managed
295 87f5f0ecSDag-Erling Smørgravby the tree.
296 87f5f0ecSDag-Erling SmørgravThe
297 87f5f0ecSDag-Erling Smørgrav.Fa FIELD
298 87f5f0ecSDag-Erling Smørgravargument is the name of the element defined by
299 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_ENTRY .
300 87f5f0ecSDag-Erling Smørgrav.Pp
301 87f5f0ecSDag-Erling SmørgravThe function bodies are generated with the
302 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_GENERATE
303 480565d5SRuslan Ermilovmacro.
304 480565d5SRuslan ErmilovIt takes the same arguments as the
305 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_PROTOTYPE
306 87f5f0ecSDag-Erling Smørgravmacro, but should be used only once.
307 87f5f0ecSDag-Erling Smørgrav.Pp
308 87f5f0ecSDag-Erling SmørgravFinally,
309 87f5f0ecSDag-Erling Smørgravthe
310 87f5f0ecSDag-Erling Smørgrav.Fa CMP
311 480565d5SRuslan Ermilovargument is the name of a function used to compare tree nodes
312 480565d5SRuslan Ermilovwith each other.
313 480565d5SRuslan ErmilovThe function takes two arguments of type
314 480565d5SRuslan Ermilov.Vt "struct TYPE *" .
315 87f5f0ecSDag-Erling SmørgravIf the first argument is smaller than the second, the function returns a
316 480565d5SRuslan Ermilovvalue smaller than zero.
317 480565d5SRuslan ErmilovIf they are equal, the function returns zero.
318 480565d5SRuslan ErmilovOtherwise, it should return a value greater than zero.
319 480565d5SRuslan ErmilovThe compare
320 87f5f0ecSDag-Erling Smørgravfunction defines the order of the tree elements.
321 87f5f0ecSDag-Erling Smørgrav.Pp
322 87f5f0ecSDag-Erling SmørgravThe
323 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_INIT
324 87f5f0ecSDag-Erling Smørgravmacro initializes the tree referenced by
325 87f5f0ecSDag-Erling Smørgrav.Fa head .
326 87f5f0ecSDag-Erling Smørgrav.Pp
327 87f5f0ecSDag-Erling SmørgravThe splay tree can also be initialized statically by using the
328 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_INITIALIZER
329 87f5f0ecSDag-Erling Smørgravmacro like this:
330 480565d5SRuslan Ermilov.Bd -ragged -offset indent
331 480565d5SRuslan Ermilov.Fn SPLAY_HEAD HEADNAME TYPE
332 480565d5SRuslan Ermilov.Va head
333 480565d5SRuslan Ermilov=
334 480565d5SRuslan Ermilov.Fn SPLAY_INITIALIZER &head ;
335 87f5f0ecSDag-Erling Smørgrav.Ed
336 87f5f0ecSDag-Erling Smørgrav.Pp
337 87f5f0ecSDag-Erling SmørgravThe
338 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_INSERT
339 87f5f0ecSDag-Erling Smørgravmacro inserts the new element
340 87f5f0ecSDag-Erling Smørgrav.Fa elm
341 87f5f0ecSDag-Erling Smørgravinto the tree.
342 87f5f0ecSDag-Erling Smørgrav.Pp
343 87f5f0ecSDag-Erling SmørgravThe
344 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_REMOVE
345 87f5f0ecSDag-Erling Smørgravmacro removes the element
346 87f5f0ecSDag-Erling Smørgrav.Fa elm
347 87f5f0ecSDag-Erling Smørgravfrom the tree pointed by
348 87f5f0ecSDag-Erling Smørgrav.Fa head .
349 87f5f0ecSDag-Erling Smørgrav.Pp
350 87f5f0ecSDag-Erling SmørgravThe
351 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_FIND
352 87f5f0ecSDag-Erling Smørgravmacro can be used to find a particular element in the tree.
353 87f5f0ecSDag-Erling Smørgrav.Bd -literal -offset indent
354 87f5f0ecSDag-Erling Smørgravstruct TYPE find, *res;
355 87f5f0ecSDag-Erling Smørgravfind.key = 30;
356 87f5f0ecSDag-Erling Smørgravres = SPLAY_FIND(NAME, head, &find);
357 87f5f0ecSDag-Erling Smørgrav.Ed
358 87f5f0ecSDag-Erling Smørgrav.Pp
359 87f5f0ecSDag-Erling SmørgravThe
360 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_ROOT ,
361 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_MIN ,
362 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_MAX ,
363 87f5f0ecSDag-Erling Smørgravand
364 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_NEXT
365 87f5f0ecSDag-Erling Smørgravmacros can be used to traverse the tree:
366 87f5f0ecSDag-Erling Smørgrav.Bd -literal -offset indent
367 87f5f0ecSDag-Erling Smørgravfor (np = SPLAY_MIN(NAME, &head); np != NULL; np = SPLAY_NEXT(NAME, &head, np))
368 87f5f0ecSDag-Erling Smørgrav.Ed
369 87f5f0ecSDag-Erling Smørgrav.Pp
370 87f5f0ecSDag-Erling SmørgravOr, for simplicity, one can use the
371 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_FOREACH
372 87f5f0ecSDag-Erling Smørgravmacro:
373 480565d5SRuslan Ermilov.Bd -ragged -offset indent
374 480565d5SRuslan Ermilov.Fn SPLAY_FOREACH np NAME head
375 87f5f0ecSDag-Erling Smørgrav.Ed
376 87f5f0ecSDag-Erling Smørgrav.Pp
377 87f5f0ecSDag-Erling SmørgravThe
378 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_EMPTY
379 87f5f0ecSDag-Erling Smørgravmacro should be used to check whether a splay tree is empty.
380 e605dcc9SDoug Moore.Sh RANK-BALANCED TREES
381 e605dcc9SDoug MooreRank-balanced (RB) trees are a framework for defining height-balanced
382 e605dcc9SDoug Moorebinary search trees, including AVL and red-black trees.
383 e605dcc9SDoug MooreEach tree node has an associated rank.
384 e605dcc9SDoug MooreBalance conditions are expressed by conditions on the differences in
385 e605dcc9SDoug Moorerank between any node and its children.
386 e605dcc9SDoug MooreRank differences are stored in each tree node.
387 87f5f0ecSDag-Erling Smørgrav.Pp
388 e605dcc9SDoug MooreThe balance conditions implemented by the RB macros lead to weak AVL
389 e605dcc9SDoug Moore(wavl) trees, which combine the best aspects of AVL and red-black
390 e605dcc9SDoug Mooretrees.
391 e605dcc9SDoug MooreWavl trees rebalance after an insertion in the same way AVL trees do,
392 e605dcc9SDoug Moorewith the same worst-case time as red-black trees offer, and with
393 e605dcc9SDoug Moorebetter balance in the resulting tree.
394 e605dcc9SDoug MooreWavl trees rebalance after a removal in a way that requires less
395 e605dcc9SDoug Moorerestructuring, in the worst case, than either AVL or red-black trees
396 58f5de0dSMateusz Piotrowskido.
397 58f5de0dSMateusz PiotrowskiRemovals can lead to a tree almost as unbalanced as a red-black
398 e605dcc9SDoug Mooretree; insertions lead to a tree becoming as balanced as an AVL tree.
399 87f5f0ecSDag-Erling Smørgrav.Pp
400 e605dcc9SDoug MooreA rank-balanced tree is headed by a structure defined by the
401 87f5f0ecSDag-Erling Smørgrav.Fn RB_HEAD
402 87f5f0ecSDag-Erling Smørgravmacro.
403 87f5f0ecSDag-Erling SmørgravA
404 87f5f0ecSDag-Erling Smørgravstructure is declared as follows:
405 480565d5SRuslan Ermilov.Bd -ragged -offset indent
406 480565d5SRuslan Ermilov.Fn RB_HEAD HEADNAME TYPE
407 480565d5SRuslan Ermilov.Va head ;
408 87f5f0ecSDag-Erling Smørgrav.Ed
409 87f5f0ecSDag-Erling Smørgrav.Pp
410 87f5f0ecSDag-Erling Smørgravwhere
411 87f5f0ecSDag-Erling Smørgrav.Fa HEADNAME
412 87f5f0ecSDag-Erling Smørgravis the name of the structure to be defined, and struct
413 87f5f0ecSDag-Erling Smørgrav.Fa TYPE
414 87f5f0ecSDag-Erling Smørgravis the type of the elements to be inserted into the tree.
415 87f5f0ecSDag-Erling Smørgrav.Pp
416 87f5f0ecSDag-Erling SmørgravThe
417 87f5f0ecSDag-Erling Smørgrav.Fn RB_ENTRY
418 87f5f0ecSDag-Erling Smørgravmacro declares a structure that allows elements to be connected in the tree.
419 87f5f0ecSDag-Erling Smørgrav.Pp
420 87f5f0ecSDag-Erling SmørgravIn order to use the functions that manipulate the tree structure,
421 87f5f0ecSDag-Erling Smørgravtheir prototypes need to be declared with the
422 87f5f0ecSDag-Erling Smørgrav.Fn RB_PROTOTYPE
423 d72cd779SJason Evansor
424 d72cd779SJason Evans.Fn RB_PROTOTYPE_STATIC
425 87f5f0ecSDag-Erling Smørgravmacro,
426 87f5f0ecSDag-Erling Smørgravwhere
427 87f5f0ecSDag-Erling Smørgrav.Fa NAME
428 87f5f0ecSDag-Erling Smørgravis a unique identifier for this particular tree.
429 87f5f0ecSDag-Erling SmørgravThe
430 87f5f0ecSDag-Erling Smørgrav.Fa TYPE
431 87f5f0ecSDag-Erling Smørgravargument is the type of the structure that is being managed
432 87f5f0ecSDag-Erling Smørgravby the tree.
433 87f5f0ecSDag-Erling SmørgravThe
434 87f5f0ecSDag-Erling Smørgrav.Fa FIELD
435 87f5f0ecSDag-Erling Smørgravargument is the name of the element defined by
436 87f5f0ecSDag-Erling Smørgrav.Fn RB_ENTRY .
437 51782e3aSKonstantin BelousovIndividual prototypes can be declared with
438 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_INSERT ,
439 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_INSERT_COLOR ,
440 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_REMOVE ,
441 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_REMOVE_COLOR ,
442 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_FIND ,
443 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_NFIND ,
444 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_NEXT ,
445 51782e3aSKonstantin Belousov.Fn RB_PROTOTYPE_PREV ,
446 22823764SEdward Tomasz Napierala.Fn RB_PROTOTYPE_MINMAX ,
447 51782e3aSKonstantin Belousovand
448 22823764SEdward Tomasz Napierala.Fn RB_PROTOTYPE_REINSERT
449 eb49a6d3SEdward Tomasz Napieralain case not all functions are required.
450 eb49a6d3SEdward Tomasz NapieralaThe individual prototype macros expect
451 51782e3aSKonstantin Belousov.Fa NAME ,
452 51782e3aSKonstantin Belousov.Fa TYPE ,
453 51782e3aSKonstantin Belousovand
454 51782e3aSKonstantin Belousov.Fa ATTR
455 eb49a6d3SEdward Tomasz Napieralaarguments.
456 eb49a6d3SEdward Tomasz NapieralaThe
457 51782e3aSKonstantin Belousov.Fa ATTR
458 51782e3aSKonstantin Belousovargument must be empty for global functions or
459 51782e3aSKonstantin Belousov.Fa static
460 51782e3aSKonstantin Belousovfor static functions.
461 87f5f0ecSDag-Erling Smørgrav.Pp
462 87f5f0ecSDag-Erling SmørgravThe function bodies are generated with the
463 87f5f0ecSDag-Erling Smørgrav.Fn RB_GENERATE
464 d72cd779SJason Evansor
465 d72cd779SJason Evans.Fn RB_GENERATE_STATIC
466 480565d5SRuslan Ermilovmacro.
467 d72cd779SJason EvansThese macros take the same arguments as the
468 87f5f0ecSDag-Erling Smørgrav.Fn RB_PROTOTYPE
469 d72cd779SJason Evansand
470 d72cd779SJason Evans.Fn RB_PROTOTYPE_STATIC
471 d72cd779SJason Evansmacros, but should be used only once.
472 51782e3aSKonstantin BelousovAs an alternative individual function bodies are generated with the
473 51782e3aSKonstantin Belousov.Fn RB_GENERATE_INSERT ,
474 51782e3aSKonstantin Belousov.Fn RB_GENERATE_INSERT_COLOR ,
475 51782e3aSKonstantin Belousov.Fn RB_GENERATE_REMOVE ,
476 51782e3aSKonstantin Belousov.Fn RB_GENERATE_REMOVE_COLOR ,
477 51782e3aSKonstantin Belousov.Fn RB_GENERATE_FIND ,
478 51782e3aSKonstantin Belousov.Fn RB_GENERATE_NFIND ,
479 51782e3aSKonstantin Belousov.Fn RB_GENERATE_NEXT ,
480 51782e3aSKonstantin Belousov.Fn RB_GENERATE_PREV ,
481 22823764SEdward Tomasz Napierala.Fn RB_GENERATE_MINMAX ,
482 51782e3aSKonstantin Belousovand
483 22823764SEdward Tomasz Napierala.Fn RB_GENERATE_REINSERT
484 51782e3aSKonstantin Belousovmacros.
485 87f5f0ecSDag-Erling Smørgrav.Pp
486 87f5f0ecSDag-Erling SmørgravFinally,
487 87f5f0ecSDag-Erling Smørgravthe
488 87f5f0ecSDag-Erling Smørgrav.Fa CMP
489 9d44cd42SBenno Riceargument is the name of a function used to compare tree nodes
490 480565d5SRuslan Ermilovwith each other.
491 480565d5SRuslan ErmilovThe function takes two arguments of type
492 480565d5SRuslan Ermilov.Vt "struct TYPE *" .
493 87f5f0ecSDag-Erling SmørgravIf the first argument is smaller than the second, the function returns a
494 480565d5SRuslan Ermilovvalue smaller than zero.
495 480565d5SRuslan ErmilovIf they are equal, the function returns zero.
496 480565d5SRuslan ErmilovOtherwise, it should return a value greater than zero.
497 480565d5SRuslan ErmilovThe compare
498 87f5f0ecSDag-Erling Smørgravfunction defines the order of the tree elements.
499 87f5f0ecSDag-Erling Smørgrav.Pp
500 87f5f0ecSDag-Erling SmørgravThe
501 87f5f0ecSDag-Erling Smørgrav.Fn RB_INIT
502 87f5f0ecSDag-Erling Smørgravmacro initializes the tree referenced by
503 87f5f0ecSDag-Erling Smørgrav.Fa head .
504 87f5f0ecSDag-Erling Smørgrav.Pp
505 e605dcc9SDoug MooreThe rank-balanced tree can also be initialized statically by using the
506 87f5f0ecSDag-Erling Smørgrav.Fn RB_INITIALIZER
507 87f5f0ecSDag-Erling Smørgravmacro like this:
508 480565d5SRuslan Ermilov.Bd -ragged -offset indent
509 480565d5SRuslan Ermilov.Fn RB_HEAD HEADNAME TYPE
510 480565d5SRuslan Ermilov.Va head
511 480565d5SRuslan Ermilov=
512 480565d5SRuslan Ermilov.Fn RB_INITIALIZER &head ;
513 87f5f0ecSDag-Erling Smørgrav.Ed
514 87f5f0ecSDag-Erling Smørgrav.Pp
515 87f5f0ecSDag-Erling SmørgravThe
516 87f5f0ecSDag-Erling Smørgrav.Fn RB_INSERT
517 87f5f0ecSDag-Erling Smørgravmacro inserts the new element
518 87f5f0ecSDag-Erling Smørgrav.Fa elm
519 87f5f0ecSDag-Erling Smørgravinto the tree.
520 87f5f0ecSDag-Erling Smørgrav.Pp
521 87f5f0ecSDag-Erling SmørgravThe
522 368ee2f8SDoug Moore.Fn RB_INSERT_NEXT
523 368ee2f8SDoug Mooremacro inserts the new element
524 368ee2f8SDoug Moore.Fa elm
525 368ee2f8SDoug Mooreinto the tree immediately after a given element.
526 368ee2f8SDoug Moore.Pp
527 368ee2f8SDoug MooreThe
528 368ee2f8SDoug Moore.Fn RB_INSERT_PREV
529 368ee2f8SDoug Mooremacro inserts the new element
530 368ee2f8SDoug Moore.Fa elm
531 368ee2f8SDoug Mooreinto the tree immediately before a given element.
532 368ee2f8SDoug Moore.Pp
533 368ee2f8SDoug MooreThe
534 87f5f0ecSDag-Erling Smørgrav.Fn RB_REMOVE
535 87f5f0ecSDag-Erling Smørgravmacro removes the element
536 87f5f0ecSDag-Erling Smørgrav.Fa elm
537 87f5f0ecSDag-Erling Smørgravfrom the tree pointed by
538 87f5f0ecSDag-Erling Smørgrav.Fa head .
539 87f5f0ecSDag-Erling Smørgrav.Pp
540 87f5f0ecSDag-Erling SmørgravThe
541 87f5f0ecSDag-Erling Smørgrav.Fn RB_FIND
542 06115e08SJason Evansand
543 06115e08SJason Evans.Fn RB_NFIND
544 06115e08SJason Evansmacros can be used to find a particular element in the tree.
545 08349b18SKonstantin Belousov.Pp
546 08349b18SKonstantin BelousovThe
547 08349b18SKonstantin Belousov.Fn RB_FIND
548 08349b18SKonstantin Belousovmacro returns the element in the tree equal to the provided
549 08349b18SKonstantin Belousovkey, or
550 08349b18SKonstantin Belousov.Dv NULL
551 08349b18SKonstantin Belousovif there is no such element.
552 08349b18SKonstantin Belousov.Pp
553 08349b18SKonstantin BelousovThe
554 08349b18SKonstantin Belousov.Fn RB_NFIND
555 08349b18SKonstantin Belousovmacro returns the least element greater than or equal to the provided
556 08349b18SKonstantin Belousovkey, or
557 08349b18SKonstantin Belousov.Dv NULL
558 08349b18SKonstantin Belousovif there is no such element.
559 87f5f0ecSDag-Erling Smørgrav.Bd -literal -offset indent
560 08349b18SKonstantin Belousovstruct TYPE find, *res, *resn;
561 87f5f0ecSDag-Erling Smørgravfind.key = 30;
562 87f5f0ecSDag-Erling Smørgravres = RB_FIND(NAME, head, &find);
563 08349b18SKonstantin Belousovresn = RB_NFIND(NAME, head, &find);
564 87f5f0ecSDag-Erling Smørgrav.Ed
565 87f5f0ecSDag-Erling Smørgrav.Pp
566 87f5f0ecSDag-Erling SmørgravThe
567 87f5f0ecSDag-Erling Smørgrav.Fn RB_ROOT ,
568 87f5f0ecSDag-Erling Smørgrav.Fn RB_MIN ,
569 87f5f0ecSDag-Erling Smørgrav.Fn RB_MAX ,
570 8e4fd0a1SJason Evans.Fn RB_NEXT ,
571 87f5f0ecSDag-Erling Smørgravand
572 8e4fd0a1SJason Evans.Fn RB_PREV
573 87f5f0ecSDag-Erling Smørgravmacros can be used to traverse the tree:
574 480565d5SRuslan Ermilov.Pp
575 480565d5SRuslan Ermilov.Dl "for (np = RB_MIN(NAME, &head); np != NULL; np = RB_NEXT(NAME, &head, np))"
576 87f5f0ecSDag-Erling Smørgrav.Pp
577 87f5f0ecSDag-Erling SmørgravOr, for simplicity, one can use the
578 87f5f0ecSDag-Erling Smørgrav.Fn RB_FOREACH
579 8e4fd0a1SJason Evansor
580 8e4fd0a1SJason Evans.Fn RB_FOREACH_REVERSE
581 87f5f0ecSDag-Erling Smørgravmacro:
582 480565d5SRuslan Ermilov.Bd -ragged -offset indent
583 480565d5SRuslan Ermilov.Fn RB_FOREACH np NAME head
584 87f5f0ecSDag-Erling Smørgrav.Ed
585 87f5f0ecSDag-Erling Smørgrav.Pp
586 9dbae282SGleb SmirnoffThe macros
587 9dbae282SGleb Smirnoff.Fn RB_FOREACH_SAFE
588 9dbae282SGleb Smirnoffand
589 9dbae282SGleb Smirnoff.Fn RB_FOREACH_REVERSE_SAFE
590 9dbae282SGleb Smirnofftraverse the tree referenced by head
591 9dbae282SGleb Smirnoffin a forward or reverse direction respectively,
592 9dbae282SGleb Smirnoffassigning each element in turn to np.
593 9dbae282SGleb SmirnoffHowever, unlike their unsafe counterparts,
594 9dbae282SGleb Smirnoffthey permit both the removal of np
595 9dbae282SGleb Smirnoffas well as freeing it from within the loop safely
596 9dbae282SGleb Smirnoffwithout interfering with the traversal.
597 9dbae282SGleb Smirnoff.Pp
598 bff27689SBruce M SimpsonBoth
599 bff27689SBruce M Simpson.Fn RB_FOREACH_FROM
600 bff27689SBruce M Simpsonand
601 bff27689SBruce M Simpson.Fn RB_FOREACH_REVERSE_FROM
602 bff27689SBruce M Simpsonmay be used to continue an interrupted traversal
603 bff27689SBruce M Simpsonin a forward or reverse direction respectively.
604 639bf7bdSBruce M SimpsonThe head pointer is not required.
605 639bf7bdSBruce M SimpsonThe pointer to the node from where to resume the traversal
606 639bf7bdSBruce M Simpsonshould be passed as their last argument,
607 bff27689SBruce M Simpsonand will be overwritten to provide safe traversal.
608 bff27689SBruce M Simpson.Pp
609 87f5f0ecSDag-Erling SmørgravThe
610 87f5f0ecSDag-Erling Smørgrav.Fn RB_EMPTY
611 e605dcc9SDoug Mooremacro should be used to check whether a rank-balanced tree is empty.
612 22823764SEdward Tomasz Napierala.Pp
613 22823764SEdward Tomasz NapieralaThe
614 22823764SEdward Tomasz Napierala.Fn RB_REINSERT
615 22823764SEdward Tomasz Napieralamacro updates the position of the element
616 22823764SEdward Tomasz Napierala.Fa elm
617 22823764SEdward Tomasz Napieralain the tree.
618 22823764SEdward Tomasz NapieralaThis must be called if a member of a
619 22823764SEdward Tomasz Napierala.Nm tree
620 22823764SEdward Tomasz Napieralais modified in a way that affects comparison, such as by modifying
621 22823764SEdward Tomasz Napieralaa node's key.
622 22823764SEdward Tomasz NapieralaThis is a lower overhead alternative to removing the element
623 22823764SEdward Tomasz Napieralaand reinserting it again.
624 a8380d27SDoug Moore.Pp
625 a8380d27SDoug MooreThe
626 a8380d27SDoug Moore.Fn RB_AUGMENT
627 a8380d27SDoug Mooremacro updates augmentation data of the element
628 a8380d27SDoug Moore.Fa elm
629 a8380d27SDoug Moorein the tree.
630 a8380d27SDoug MooreBy default, it has no effect.
631 a8380d27SDoug MooreIt is not meant to be invoked by the RB user.
632 35557a0dSDoug MooreIf
633 35557a0dSDoug Moore.Fn RB_AUGMENT
634 35557a0dSDoug Mooreis defined by the RB user, then when an element is inserted or removed
635 35557a0dSDoug Moorefrom the tree, it is invoked for every element in the tree that is the
636 35557a0dSDoug Mooreroot of an altered subtree, working from the bottom of the tree up to
637 35557a0dSDoug Moorethe top.
638 a8380d27SDoug MooreIt is typically used to maintain some associative accumulation of tree
639 a8380d27SDoug Mooreelements, such as sums, minima, maxima, and the like.
640 35557a0dSDoug Moore.Pp
641 35557a0dSDoug MooreThe
642 b16f993eSDoug Moore.Fn RB_AUGMENT_CHECK
643 b16f993eSDoug Mooremacro updates augmentation data of the element
644 b16f993eSDoug Moore.Fa elm
645 b16f993eSDoug Moorein the tree.
646 b16f993eSDoug MooreBy default, it does nothing and returns false.
647 b16f993eSDoug MooreIf
648 b16f993eSDoug Moore.Fn RB_AUGMENT_CHECK
649 b16f993eSDoug Mooreis defined, then when an element is inserted or removed from the tree,
650 b16f993eSDoug Mooreit is invoked for every element in the tree that is the root of an
651 b16f993eSDoug Moorealtered subtree, working from the bottom of the tree up toward the
652 b16f993eSDoug Mooretop, until it returns false to indicate that it did not change the
653 b16f993eSDoug Mooreelement and so working further up the tree would change nothing.
654 b16f993eSDoug MooreIt is typically used to maintain some associative accumulation of tree
655 b16f993eSDoug Mooreelements, such as sums, minima, maxima, and the like.
656 b16f993eSDoug Moore.Pp
657 b16f993eSDoug MooreThe
658 35557a0dSDoug Moore.Fn RB_UPDATE_AUGMENT
659 35557a0dSDoug Mooremacro updates augmentation data of the element
660 35557a0dSDoug Moore.Fa elm
661 35557a0dSDoug Mooreand its ancestors in the tree.
662 624e5dc0SKonstantin BelousovIf
663 624e5dc0SKonstantin Belousov.Fn RB_AUGMENT
664 624e5dc0SKonstantin Belousovis defined by the RB user, then when an element in the
665 35557a0dSDoug Mooretree is changed, without changing the order of items in the tree,
666 35557a0dSDoug Mooreinvoking this function on that element restores consistency of the
667 35557a0dSDoug Mooreaugmentation state of the tree as if the element had been removed and
668 35557a0dSDoug Mooreinserted again.
669 ac0879c3SEdward Tomasz Napierala.Sh EXAMPLES
670 e605dcc9SDoug MooreThe following example demonstrates how to declare a rank-balanced tree
671 ac0879c3SEdward Tomasz Napieralaholding integers.
672 ac0879c3SEdward Tomasz NapieralaValues are inserted into it and the contents of the tree are printed
673 ac0879c3SEdward Tomasz Napieralain order.
674 a8380d27SDoug MooreTo maintain the sum of the values in the tree, each element maintains
675 a8380d27SDoug Moorethe sum of its value and the sums from its left and right subtrees.
676 ac0879c3SEdward Tomasz NapieralaLastly, the internal structure of the tree is printed.
677 ac0879c3SEdward Tomasz Napierala.Bd -literal -offset 3n
678 *503b7f94SMaxim Konovalov#define RB_AUGMENT(entry) sumaug(entry)
679 *503b7f94SMaxim Konovalov
680 ac0879c3SEdward Tomasz Napierala#include <sys/tree.h>
681 ac0879c3SEdward Tomasz Napierala#include <err.h>
682 ac0879c3SEdward Tomasz Napierala#include <stdio.h>
683 ac0879c3SEdward Tomasz Napierala#include <stdlib.h>
684 ac0879c3SEdward Tomasz Napierala
685 ac0879c3SEdward Tomasz Napieralastruct node {
686 ac0879c3SEdward Tomasz Napierala	RB_ENTRY(node) entry;
687 a8380d27SDoug Moore	int i, sum;
688 ac0879c3SEdward Tomasz Napierala};
689 ac0879c3SEdward Tomasz Napierala
690 ac0879c3SEdward Tomasz Napieralaint
691 ac0879c3SEdward Tomasz Napieralaintcmp(struct node *e1, struct node *e2)
692 ac0879c3SEdward Tomasz Napierala{
693 ac0879c3SEdward Tomasz Napierala	return (e1->i < e2->i ? -1 : e1->i > e2->i);
694 ac0879c3SEdward Tomasz Napierala}
695 ac0879c3SEdward Tomasz Napierala
696 *503b7f94SMaxim Konovalovvoid
697 a8380d27SDoug Mooresumaug(struct node *e)
698 a8380d27SDoug Moore{
699 a8380d27SDoug Moore	e->sum = e->i;
700 a8380d27SDoug Moore	if (RB_LEFT(e, entry) != NULL)
701 a8380d27SDoug Moore		e->sum += RB_LEFT(e, entry)->sum;
702 a8380d27SDoug Moore	if (RB_RIGHT(e, entry) != NULL)
703 a8380d27SDoug Moore		e->sum += RB_RIGHT(e, entry)->sum;
704 a8380d27SDoug Moore}
705 a8380d27SDoug Moore
706 ac0879c3SEdward Tomasz NapieralaRB_HEAD(inttree, node) head = RB_INITIALIZER(&head);
707 ac0879c3SEdward Tomasz NapieralaRB_GENERATE(inttree, node, entry, intcmp)
708 ac0879c3SEdward Tomasz Napierala
709 ac0879c3SEdward Tomasz Napieralaint testdata[] = {
710 ac0879c3SEdward Tomasz Napierala	20, 16, 17, 13, 3, 6, 1, 8, 2, 4, 10, 19, 5, 9, 12, 15, 18,
711 ac0879c3SEdward Tomasz Napierala	7, 11, 14
712 ac0879c3SEdward Tomasz Napierala};
713 ac0879c3SEdward Tomasz Napierala
714 ac0879c3SEdward Tomasz Napieralavoid
715 ac0879c3SEdward Tomasz Napieralaprint_tree(struct node *n)
716 ac0879c3SEdward Tomasz Napierala{
717 ac0879c3SEdward Tomasz Napierala	struct node *left, *right;
718 ac0879c3SEdward Tomasz Napierala
719 ac0879c3SEdward Tomasz Napierala	if (n == NULL) {
720 ac0879c3SEdward Tomasz Napierala		printf("nil");
721 ac0879c3SEdward Tomasz Napierala		return;
722 ac0879c3SEdward Tomasz Napierala	}
723 ac0879c3SEdward Tomasz Napierala	left = RB_LEFT(n, entry);
724 ac0879c3SEdward Tomasz Napierala	right = RB_RIGHT(n, entry);
725 ac0879c3SEdward Tomasz Napierala	if (left == NULL && right == NULL)
726 ac0879c3SEdward Tomasz Napierala		printf("%d", n->i);
727 ac0879c3SEdward Tomasz Napierala	else {
728 ac0879c3SEdward Tomasz Napierala		printf("%d(", n->i);
729 ac0879c3SEdward Tomasz Napierala		print_tree(left);
730 ac0879c3SEdward Tomasz Napierala		printf(",");
731 ac0879c3SEdward Tomasz Napierala		print_tree(right);
732 ac0879c3SEdward Tomasz Napierala		printf(")");
733 ac0879c3SEdward Tomasz Napierala	}
734 ac0879c3SEdward Tomasz Napierala}
735 ac0879c3SEdward Tomasz Napierala
736 ac0879c3SEdward Tomasz Napieralaint
737 ac0879c3SEdward Tomasz Napieralamain(void)
738 ac0879c3SEdward Tomasz Napierala{
739 ac0879c3SEdward Tomasz Napierala	int i;
740 ac0879c3SEdward Tomasz Napierala	struct node *n;
741 ac0879c3SEdward Tomasz Napierala
742 ac0879c3SEdward Tomasz Napierala	for (i = 0; i < sizeof(testdata) / sizeof(testdata[0]); i++) {
743 ac0879c3SEdward Tomasz Napierala		if ((n = malloc(sizeof(struct node))) == NULL)
744 ac0879c3SEdward Tomasz Napierala			err(1, NULL);
745 ac0879c3SEdward Tomasz Napierala		n->i = testdata[i];
746 ac0879c3SEdward Tomasz Napierala		RB_INSERT(inttree, &head, n);
747 ac0879c3SEdward Tomasz Napierala	}
748 ac0879c3SEdward Tomasz Napierala
749 ac0879c3SEdward Tomasz Napierala	RB_FOREACH(n, inttree, &head) {
750 ac0879c3SEdward Tomasz Napierala		printf("%d\en", n->i);
751 ac0879c3SEdward Tomasz Napierala	}
752 ac0879c3SEdward Tomasz Napierala	print_tree(RB_ROOT(&head));
753 *503b7f94SMaxim Konovalov	printf("\enSum of values = %d\en", RB_ROOT(&head)->sum);
754 ac0879c3SEdward Tomasz Napierala	return (0);
755 ac0879c3SEdward Tomasz Napierala}
756 ac0879c3SEdward Tomasz Napierala.Ed
757 87f5f0ecSDag-Erling Smørgrav.Sh NOTES
758 87f5f0ecSDag-Erling SmørgravTrying to free a tree in the following way is a common error:
759 87f5f0ecSDag-Erling Smørgrav.Bd -literal -offset indent
760 87f5f0ecSDag-Erling SmørgravSPLAY_FOREACH(var, NAME, head) {
761 87f5f0ecSDag-Erling Smørgrav	SPLAY_REMOVE(NAME, head, var);
762 87f5f0ecSDag-Erling Smørgrav	free(var);
763 87f5f0ecSDag-Erling Smørgrav}
764 87f5f0ecSDag-Erling Smørgravfree(head);
765 87f5f0ecSDag-Erling Smørgrav.Ed
766 87f5f0ecSDag-Erling Smørgrav.Pp
767 87f5f0ecSDag-Erling SmørgravSince
768 87f5f0ecSDag-Erling Smørgrav.Va var
769 480565d5SRuslan Ermilovis freed, the
770 87f5f0ecSDag-Erling Smørgrav.Fn FOREACH
771 87f5f0ecSDag-Erling Smørgravmacro refers to a pointer that may have been reallocated already.
772 87f5f0ecSDag-Erling SmørgravProper code needs a second variable.
773 87f5f0ecSDag-Erling Smørgrav.Bd -literal -offset indent
774 87f5f0ecSDag-Erling Smørgravfor (var = SPLAY_MIN(NAME, head); var != NULL; var = nxt) {
775 87f5f0ecSDag-Erling Smørgrav	nxt = SPLAY_NEXT(NAME, head, var);
776 87f5f0ecSDag-Erling Smørgrav	SPLAY_REMOVE(NAME, head, var);
777 87f5f0ecSDag-Erling Smørgrav	free(var);
778 87f5f0ecSDag-Erling Smørgrav}
779 87f5f0ecSDag-Erling Smørgrav.Ed
780 87f5f0ecSDag-Erling Smørgrav.Pp
781 87f5f0ecSDag-Erling SmørgravBoth
782 87f5f0ecSDag-Erling Smørgrav.Fn RB_INSERT
783 87f5f0ecSDag-Erling Smørgravand
784 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_INSERT
785 87f5f0ecSDag-Erling Smørgravreturn
786 480565d5SRuslan Ermilov.Dv NULL
787 87f5f0ecSDag-Erling Smørgravif the element was inserted in the tree successfully, otherwise they
788 87f5f0ecSDag-Erling Smørgravreturn a pointer to the element with the colliding key.
789 87f5f0ecSDag-Erling Smørgrav.Pp
790 87f5f0ecSDag-Erling SmørgravAccordingly,
791 87f5f0ecSDag-Erling Smørgrav.Fn RB_REMOVE
792 87f5f0ecSDag-Erling Smørgravand
793 87f5f0ecSDag-Erling Smørgrav.Fn SPLAY_REMOVE
794 87f5f0ecSDag-Erling Smørgravreturn the pointer to the removed element otherwise they return
795 480565d5SRuslan Ermilov.Dv NULL
796 87f5f0ecSDag-Erling Smørgravto indicate an error.
797 b9ec8f3bSSimon L. B. Nielsen.Sh SEE ALSO
798 fad4b12bSEdward Tomasz Napierala.Xr arb 3 ,
799 b9ec8f3bSSimon L. B. Nielsen.Xr queue 3
800 e605dcc9SDoug Moore.Rs
801 e605dcc9SDoug Moore.%A "Bernhard Haeupler"
802 e605dcc9SDoug Moore.%A "Siddhartha Sen"
803 e605dcc9SDoug Moore.%A "Robert E. Tarjan"
804 e605dcc9SDoug Moore.%T "Rank-Balanced Trees"
805 e605dcc9SDoug Moore.%U "http://sidsen.azurewebsites.net/papers/rb-trees-talg.pdf"
806 e605dcc9SDoug Moore.%J "ACM Transactions on Algorithms"
807 e605dcc9SDoug Moore.%V "11"
808 e605dcc9SDoug Moore.%N "4"
809 e605dcc9SDoug Moore.%D "June 2015"
810 e605dcc9SDoug Moore.Re
811 757a04bfSSergio Carlavilla Delgado.Sh HISTORY
812 757a04bfSSergio Carlavilla DelgadoThe tree macros first appeared in
813 757a04bfSSergio Carlavilla Delgado.Fx 4.6 .
814 87f5f0ecSDag-Erling Smørgrav.Sh AUTHORS
815 480565d5SRuslan ErmilovThe author of the tree macros is
816 480565d5SRuslan Ermilov.An Niels Provos .
817