dmake: do not set MAKEFLAGS=k
[unleashed/tickless.git] / usr / src / lib / libast / common / man / cdt.3
blobf353e3f0b13b5c7620f22018b48a035955cf291f
1 .fp 5 CW
2 .TH LIBCDT 3
3 .SH NAME
4 \fBCdt\fR \- container data types
5 .SH SYNOPSIS
6 .de Tp
7 .fl
8 .ne 2
9 .TP
11 .de Ss
12 .fl
13 .ne 2
14 .SS "\\$1"
16 .de Cs
17 .nf
18 .ft 5
20 .de Ce
21 .ft 1
22 .fi
24 .ta 1.0i 2.0i 3.0i 4.0i 5.0i
25 .Cs
26 #include <cdt.h>
27 .Ce
28 .Ss "DICTIONARY TYPES"
29 .Cs
30 Void_t*;
31 Dt_t;
32 Dtdisc_t;
33 Dtmethod_t;
34 Dtlink_t;
35 Dtstat_t;
36 .Ce
37 .Ss "DICTIONARY CONTROL"
38 .Cs
39 Dt_t*       dtopen(const Dtdisc_t* disc, const Dtmethod_t* meth);
40 int         dtclose(Dt_t* dt);
41 void        dtclear(dt);
42 Dtmethod_t* dtmethod(Dt_t* dt, const Dtmethod_t* meth);
43 Dtdisc_t*   dtdisc(Dt_t* dt, const Dtdisc_t* disc, int type);
44 Dt_t*       dtview(Dt_t* dt, Dt_t* view);
45 int         dttreeset(Dt_t* dt, int minp, int balance);
46 .Ce
47 .Ss "STORAGE METHODS"
48 .Cs
49 Dtmethod_t* Dtset;
50 Dtmethod_t* Dtbag;
51 Dtmethod_t* Dtoset;
52 Dtmethod_t* Dtobag;
53 Dtmethod_t* Dtlist;
54 Dtmethod_t* Dtstack;
55 Dtmethod_t* Dtqueue;
56 .Ce
57 .Ss "DISCIPLINE"
58 .Cs
59 #define DTOFFSET(struct_s,member)
60 #define DTDISC(disc,key,size,link,makef,freef,comparf,hashf,memoryf,eventf)
61 typedef Void_t*      (*Dtmake_f)(Dt_t*, Void_t*, Dtdisc_t*);
62 typedef void         (*Dtfree_f)(Dt_t*, Void_t*, Dtdisc_t*);
63 typedef int          (*Dtcompar_f)(Dt_t*, Void_t*, Void_t*, Dtdisc_t*);
64 typedef unsigned int (*Dthash_f)(Dt_t*, Void_t*, Dtdisc_t*);
65 typedef Void_t*      (*Dtmemory_f)(Dt_t*, Void_t*, size_t, Dtdisc_t*);
66 typedef int          (*Dtevent_f)(Dt_t*, int, Void_t*, Dtdisc_t*);
67 .Ce
68 .Ss "OBJECT OPERATIONS"
69 .Cs
70 Void_t*   dtinsert(Dt_t* dt, Void_t* obj);
71 Void_t*   dtdelete(Dt_t* dt, Void_t* obj);
72 Void_t*   dtattach(Dt_t* dt, Void_t* obj);
73 Void_t*   dtdetach(Dt_t* dt, Void_t* obj);
74 Void_t*   dtsearch(Dt_t* dt, Void_t* obj);
75 Void_t*   dtmatch(Dt_t* dt, Void_t* key);
76 Void_t*   dtfirst(Dt_t* dt);
77 Void_t*   dtnext(Dt_t* dt, Void_t* obj);
78 Void_t*   dtlast(Dt_t* dt);
79 Void_t*   dtprev(Dt_t* dt, Void_t* obj);
80 Void_t*   dtfinger(Dt_t* dt);
81 Void_t*   dtrenew(Dt_t* dt, Void_t* obj);
82 int       dtwalk(Dt_t* dt, int (*userf)(Dt_t*, Void_t*, Void_t*), Void_t*);
83 Dtlink_t* dtflatten(Dt_t* dt);
84 Dtlink_t* dtlink(Dt_t*, Dtlink_t* link);
85 Void_t*   dtobj(Dt_t* dt, Dtlink_t* link);
86 Dtlink_t* dtextract(Dt_t* dt);
87 int       dtrestore(Dt_t* dt, Dtlink_t* link);
89 #define   DTTREESEARCH(Dt_t* dt, Void_t* obj, action)
90 #define   DTTREEMATCH(Dt_t* dt, Void_t* key, action)
91 .Ce
92 .Ss "DICTIONARY STATUS"
93 .Cs
94 Dt_t*     dtvnext(Dt_t* dt);
95 int       dtvcount(Dt_t* dt);
96 Dt_t*     dtvhere(Dt_t* dt);
97 int       dtsize(Dt_t* dt);
98 int       dtstat(Dt_t* dt, Dtstat_t*, int all);
99 .Ce
100 .Ss "HASH FUNCTIONS"
102 unsigned int dtstrhash(unsigned int h, char* str, int n);
103 unsigned int dtcharhash(unsigned int h, unsigned char c);
105 .SH DESCRIPTION
107 \fICdt\fP manages run-time dictionaries using standard container data types:
108 unordered set/multiset, ordered set/multiset, list, stack, and queue.
110 .Ss "DICTIONARY TYPES"
112 .Ss "  Void_t*"
113 This type is used to pass objects between \fICdt\fP and application code.
114 \f5Void_t\fP is defined as \f5void\fP for ANSI-C and C++
115 and \f5char\fP for other compilation environments.
117 .Ss "  Dt_t"
118 This is the type of a dictionary handle.
120 .Ss "  Dtdisc_t"
121 This defines the type of a discipline structure which describes
122 object lay-out and manipulation functions.
124 .Ss "  Dtmethod_t"
125 This defines the type of a container method.
127 .Ss "  Dtlink_t"
128 This is the type of a dictionary object holder (see \f5dtdisc()\fP.)
130 .Ss "  Dtstat_t"
131 This is the type of a structure to return dictionary statistics (see \f5dtstat()\fP.)
133 .Ss "DICTIONARY CONTROL"
135 .Ss "  Dt_t* dtopen(const Dtdisc_t* disc, const Dtmethod_t* meth)"
136 This creates a new dictionary.
137 \f5disc\fP is a discipline structure to describe object format.
138 \f5meth\fP specifies a manipulation method.
139 \f5dtopen()\fP returns the new dictionary or \f5NULL\fP on error.
140 See also the events \f5DT_OPEN\fP and \f5DT_ENDOPEN\fP below.
142 .Ss "  int dtclose(Dt_t* dt)"
143 This deletes \f5dt\fP and its objects.
144 Note that \f5dtclose()\fP fails if \f5dt\fP is being viewed by
145 some other dictionaries (see \f5dtview()\fP).
146 \f5dtclose()\fP returns \f50\fP on success and \f5-1\fP on error.
147 See also the events \f5DT_CLOSE\fP and \f5DT_ENDCLOSE\fP below.
149 .Ss "  void dtclear(Dt_t* dt)"
150 This deletes all objects in \f5dt\fP without closing \f5dt\fP.
152 .Ss "  Dtmethod_t dtmethod(Dt_t* dt, const Dtmethod_t* meth)"
153 If \f5meth\fP is \f5NULL\fP, \f5dtmethod()\fP returns the current method.
154 Otherwise, it changes the storage method of \f5dt\fP to \f5meth\fP.
155 Object order remains the same during a
156 method switch among \f5Dtlist\fP, \f5Dtstack\fP and \f5Dtqueue\fP.
157 Switching to and from \f5Dtset/Dtbag\fP and \f5Dtoset/Dtobag\fP may cause
158 objects to be rehashed, reordered, or removed as the case requires.
159 \f5dtmethod()\fP returns the previous method or \f5NULL\fP on error.
161 .Ss "  Dtdisc_t* dtdisc(Dt_t* dt, const Dtdisc_t* disc, int type)"
162 If \f5disc\fP is \f5NULL\fP, \f5dtdisc()\fP returns the current discipline.
163 Otherwise, it changes the discipline of \f5dt\fP to \f5disc\fP.
164 Objects may be rehashed, reordered, or removed as appropriate.
165 \f5type\fP can be any bit combination of \f5DT_SAMECMP\fP and \f5DT_SAMEHASH\fP.
166 \f5DT_SAMECMP\fP means that objects will compare exactly the same as before
167 thus obviating the need for reordering or removing new duplicates.
168 \f5DT_SAMEHASH\fP means that hash values of objects remain the same
169 thus obviating the need to rehash.
170 \f5dtdisc()\fP returns the previous discipline on success
171 and \f5NULL\fP on error.
173 .Ss "  Dt_t* dtview(Dt_t* dt, Dt_t* view)"
174 A viewpath allows a search or walk starting from a dictionary to continue to another.
175 \f5dtview()\fP first terminates any current view from \f5dt\fP to another dictionary.
176 Then, if \f5view\fP is \f5NULL\fP, \f5dtview\fP returns the terminated view dictionary.
177 If \f5view\fP is not \f5NULL\fP, a viewpath from \f5dt\fP to \f5view\fP is established.
178 \f5dtview()\fP returns \f5dt\fP on success and \f5NULL\fP on error.
180 If two dictionaries on the same viewpath have the same values for the discipline fields
181 \f5Dtdisc_t.link\fP, \f5Dtdisc_t.key\fP, \f5Dtdisc_t.size\fP, and \f5Dtdisc_t.hashf\fP,
182 it is expected that key hashing will be the same.
183 If not, undefined behaviors may result during a search or a walk.
185 .Ss "  int dttreeset(Dt_t* dt, int minp, int balance)"
186 This function only applies to dictionaries operated under the method \f5Dtoset\fP
187 which uses top-down splay trees (see below). It returns 0 on success and -1 on error.
189 \f5minp\fP:
190 This parameter defines the minimum path length before a search path is adjusted.
191 For example, \f5minp\fP equal 0 would mean that search paths are always adjusted.
192 If \f5minp\fP is negative, the minimum search path is internally computed based
193 on a function of the current dictionary size. This computed value is such that
194 if the tree is balanced, it will never require adjusting.
196 \f5balance\fP:
197 If this is non-zero, the tree will be made balanced.
199 .Ss "STORAGE METHODS"
201 Storage methods are of type \f5Dtmethod_t*\fP.
202 \fICdt\fP supports the following methods:
204 .Ss "  Dtoset"
205 .Ss "  Dtobag"
206 Objects are ordered by comparisons.
207 \f5Dtoset\fP keeps unique objects.
208 \f5Dtobag\fP allows repeatable objects.
210 .Ss "  Dtset"
211 .Ss "  Dtbag"
212 Objects are unordered.
213 \f5Dtset\fP keeps unique objects.
214 \f5Dtbag\fP allows repeatable objects and always keeps them together
215 (note the effect on dictionary walking.)
216 These methods use a hash table with chaining to manage the objects.
217 See also the event \f5DT_HASHSIZE\fP below on how to manage hash table
218 resizing when objects are inserted.
220 .Ss "  Dtlist"
221 Objects are kept in a list.
222 New objects are inserted either
223 in front of \fIcurrent object\fP (see \f5dtfinger()\fP) if this is defined
224 or at list front if there is no current object.
226 .Ss "  Dtstack"
227 Objects are kept in a stack, i.e., in reverse order of insertion.
228 Thus, the last object inserted is at stack top
229 and will be the first to be deleted.
231 .Ss "  Dtqueue"
232 Objects are kept in a queue, i.e., in order of insertion.
233 Thus, the first object inserted is at queue head
234 and will be the first to be deleted.
236 .Ss "DISCIPLINE"
238 Object format and associated management functions are
239 defined in the type \f5Dtdisc_t\fP:
241     typedef struct
242     { int        key, size;
243       int        link;
244       Dtmake_f   makef;
245       Dtfree_f   freef;
246       Dtcompar_f comparf;
247       Dthash_f   hashf;
248       Dtmemory_f memoryf;
249       Dtevent_f  eventf;
250     } Dtdisc_t;
252 .Ss "  int key, size"
253 Each object \f5obj\fP is identified by a key used for object comparison or hashing.
254 \f5key\fP should be non-negative and defines an offset into \f5obj\fP.
255 If \f5size\fP is negative, the key is a null-terminated
256 string with starting address \f5*(Void_t**)((char*)obj+key)\fP.
257 If \f5size\fP is zero, the key is a null-terminated string with starting address
258 \f5(Void_t*)((char*)obj+key)\fP.
259 Finally, if \f5size\fP is positive, the key is a byte array of length \f5size\fP
260 starting at \f5(Void_t*)((char*)obj+key)\fP.
262 .Ss "  int link"
263 Let \f5obj\fP be an object to be inserted into \f5dt\fP as discussed below.
264 If \f5link\fP is negative, an internally allocated object holder is used
265 to hold \f5obj\fP. Otherwise, \f5obj\fP should have
266 a \f5Dtlink_t\fP structure embedded \f5link\fP bytes into it,
267 i.e., at address \f5(Dtlink_t*)((char*)obj+link)\fP.
269 .Ss "  Void_t* (*makef)(Dt_t* dt, Void_t* obj, Dtdisc_t* disc)"
270 If \f5makef\fP is not \f5NULL\fP,
271 \f5dtinsert(dt,obj)\fP will call it
272 to make a copy of \f5obj\fP suitable for insertion into \f5dt\fP.
273 If \f5makef\fP is \f5NULL\fP, \f5obj\fP itself will be inserted into \f5dt\fP.
275 .Ss "  void (*freef)(Dt_t* dt, Void_t* obj, Dtdisc_t* disc)"
276 If not \f5NULL\fP,
277 \f5freef\fP is used to destroy data associated with \f5obj\fP.
279 .Ss "int (*comparf)(Dt_t* dt, Void_t* key1, Void_t* key2, Dtdisc_t* disc)"
280 If not \f5NULL\fP, \f5comparf\fP is used to compare two keys.
281 Its return value should be \f5<0\fP, \f5=0\fP, or \f5>0\fP to indicate
282 whether \f5key1\fP is smaller, equal to, or larger than \f5key2\fP.
283 All three values are significant for method \f5Dtoset\fP and \f5Dtobag\fP.
284 For other methods, a zero value
285 indicates equality and a non-zero value indicates inequality.
286 If \f5(*comparf)()\fP is \f5NULL\fP, an internal function is used
287 to compare the keys as defined by the \f5Dtdisc_t.size\fP field.
289 .Ss "  unsigned int (*hashf)(Dt_t* dt, Void_t* key, Dtdisc_t* disc)"
290 If not \f5NULL\fP,
291 \f5hashf\fP is used to compute the hash value of \f5key\fP.
292 It is required that keys compared equal will also have same hash values.
293 If \f5hashf\fP is \f5NULL\fP, an internal function is used to hash
294 the key as defined by the \f5Dtdisc_t.size\fP field.
296 .Ss "  Void_t* (*memoryf)(Dt_t* dt, Void_t* addr, size_t size, Dtdisc_t* disc)"
297 If not \f5NULL\fP, \f5memoryf\fP is used to allocate and free memory.
298 When \f5addr\fP is \f5NULL\fP, a memory segment of size \f5size\fP is requested. 
299 If \f5addr\fP is not \f5NULL\fP and \f5size\fP is zero, \f5addr\fP is to be freed.
300 If \f5addr\fP is not \f5NULL\fP and \f5size\fP is positive,
301 \f5addr\fP is to be resized to the given size.
302 If \f5memoryf\fP is \f5NULL\fP, \fImalloc(3)\fP is used.
303 When dictionaries share memory,
304 a record of the first allocated memory segment should be kept
305 so that it can be used to initialize other dictionaries (see below.)
307 .Ss "  int (*eventf)(Dt_t* dt, int type, Void_t* data, Dtdisc_t* disc)"
308 If not \f5NULL\fP, \f5eventf\fP announces various events.
309 Each event may have particular handling of the return values from \f5eventf\fP.
310 But a negative return value typically means failure.
311 Following are the events:
313 \f5DT_OPEN\fP:
314 \f5dt\fP is being opened.
315 If \f5eventf\fP returns negative, the opening process terminates with failure.
316 If \f5eventf\fP returns zero, the opening process proceeds normally.
317 A positive return value indicates special treatment of memory as follows.
318 If \f5*(Void_t**)data\fP is set to point to some memory segment
319 as discussed in \f5memoryf\fP, that segment of memory is used to start
320 the dictionary. If \f5*(Void_t**)data\fP is not set, this indicates that
321 \f5dtopen()\fP should allocate the dictionary handle itself with
322 \f5memoryf\fP so that all memories pertained to the dictionary are allocated
323 via this function.
325 \f5DT_ENDOPEN\fP:
326 This event announces that \f5dtopen()\fP has successfully opened
327 a dictionary and is about to return. The \f5data\fP argument of
328 \f5eventf\fP should be the new dictionary handle itself.
330 \f5DT_CLOSE\fP:
331 \f5dt\fP is about to be closed. If \f5eventf\fP returns zero,
332 the closing process proceeds normally. This means that all objects
333 in the dictionary will be deleted and all associated memories freed.
334 If \f5eventf\fP returns a positive value, objects will not be deleted.
335 However, the dictionary handle itself will still be deallocated
336 unless it was allocated via \f5memoryf\fP during \f5dtopen()\fP.
337 This allows shared/persistent memory dictionaries to retain
338 the relevant memories across dictionary openings and closings.
340 \f5DT_ENDCLOSE\fP:
341 This event announces that \f5dtclose()\fP has successfully closed
342 a dictionary and is about to return.
344 \f5DT_DISC\fP:
345 The discipline of \f5dt\fP is being changed to a new one given in
346 \f5(Dtdisc_t*)data\fP.
348 \f5DT_METH\fP:
349 The method of \f5dt\fP is being changed to a new one given in
350 \f5(Dtmethod_t*)data\fP.
352 \f5DT_HASHSIZE\fP:
353 The hash table (for \f5Dtset\fP and \f5Dtbag\fP) is being resized.
354 In this case, \f5*(int*)data\fP has the current size of the table.
355 The application can set the new table size by first changing
356 \f5*(int*)data\fP to the desired size, then return a positive value.
357 The application can also fix the table size at the current value
358 forever by setting \f5*(int*)data\fP to a negative value, then
359 again return a positive value. A non-positive return value from
360 the event handling function means that Cdt will be responsible
361 for choosing the hash table size.
363 .Ss "#define DTOFFSET(struct_s,member)"
364 This macro function computes the offset of \f5member\fP from the start
365 of structure \f5struct_s\fP. It is useful for getting the offset of
366 a \f5Dtlink_t\fP embedded inside an object.
368 .Ss "#define DTDISC(disc,key,size,link,makef,freef,comparf,hashf,memoryf,eventf)"
369 This macro function initializes the discipline pointed to by \f5disc\fP
370 with the given values.
372 .Ss "OBJECT OPERATIONS"
374 .Ss "  Void_t* dtinsert(Dt_t* dt, Void_t* obj)"
375 This inserts an object prototyped by \f5obj\fP into \f5dt\fP.
376 If there is an existing object in \f5dt\fP matching \f5obj\fP
377 and the storage method is \f5Dtset\fP or \f5Dtoset\fP,
378 \f5dtinsert()\fP will simply return the matching object.
379 Otherwise, a new object is inserted according to the method in use.
380 See \f5Dtdisc_t.makef\fP for object construction.
381 \f5dtinsert()\fP returns the new object, a matching object as noted,
382 or \f5NULL\fP on error.
384 .Ss "  Void_t* dtdelete(Dt_t* dt, Void_t* obj)"
385 If \f5obj\fP is \f5NULL\fP, methods \f5Dtstack\fP and \f5Dtqueue\fP
386 delete respectively stack top or queue head while other methods do nothing.
387 If \f5obj\fP is not \f5NULL\fP, there are two cases.
388 If the method in use is not \f5Dtbag\fP or \f5Dtobag\fP,
389 the first object matching \f5obj\fP is deleted.
390 On the other hand, if the method in use is \f5Dtbag\fP or \f5Dtobag\fP,
391 the library check to see if \f5obj\fP is in the dictionary and delete it.
392 If \f5obj\fP is not in the dictionary, some object matching it will be deleted.
393 See \f5Dtdisc_t.freef\fP for object destruction.
394 \f5dtdelete()\fP returns the deleted object (even if it was deallocated)
395 or \f5NULL\fP on error.
397 .Ss "  Void_t* dtattach(Dt_t* dt, Void_t* obj)"
398 This function is similar to \f5dtinsert()\fP but \f5obj\fP itself
399 will be inserted into \f5dt\fP even if a discipline
400 function \f5makef\fP is defined.
402 .Ss "  Void_t* dtdetach(Dt_t* dt, Void_t* obj)"
403 This function is similar to \f5dtdelete()\fP but the object to be deleted
404 from \f5dt\fP will not be freed (via the discipline \f5freef\fP function).
406 .Ss "  Void_t* dtsearch(Dt_t* dt, Void_t* obj)"
407 .Ss "  Void_t* dtmatch(Dt_t* dt, Void_t* key)"
408 These functions find an object matching \f5obj\fP or \f5key\fP either from \f5dt\fP or
409 from some dictionary accessible from \f5dt\fP via a viewpath (see \f5dtview()\fP.)
410 \f5dtsearch()\fP and \f5dtmatch()\fP return the matching object or
411 \f5NULL\fP on failure.
413 .Ss "  Void_t* dtfirst(Dt_t* dt)"
414 .Ss "  Void_t* dtnext(Dt_t* dt, Void_t* obj)"
415 \f5dtfirst()\fP returns the first object in \f5dt\fP.
416 \f5dtnext()\fP returns the object following \f5obj\fP.
417 Objects are ordered based on the storage method in use.
418 For \f5Dtoset\fP and \f5Dtobag\fP, objects are ordered by object comparisons.
419 For \f5Dtstack\fP, objects are ordered in reverse order of insertion.
420 For \f5Dtqueue\fP, objects are ordered in order of insertion.
421 For \f5Dtlist\fP, objects are ordered by list position.
422 For \f5Dtset\fP and \f5Dtbag\fP,
423 objects are ordered by some internal order (more below).
424 Thus, objects in a dictionary or a viewpath can be walked using 
425 a \f5for(;;)\fP loop as below.
427     for(obj = dtfirst(dt); obj; obj = dtnext(dt,obj))
429 When a dictionary uses \f5Dtset\fP or \f5Dtbag\fP,
430 the object order is determined upon a call to \f5dtfirst()\fP/\f5dtlast()\fP.
431 This order is frozen until a call \f5dtnext()\fP/\f5dtprev()\fP returns \f5NULL\fP
432 or when these same functions are called with a \f5NULL\fP object argument.
433 It is important that a \f5dtfirst()/dtlast()\fP call be
434 balanced by a \f5dtnext()/dtprev()\fP call as described.
435 Nested loops will require multiple balancing, once per loop.
436 If loop balancing is not done carefully, either performance is degraded
437 or unexpected behaviors may result.
438 .Ss "  Void_t* dtlast(Dt_t* dt)"
439 .Ss "  Void_t* dtprev(Dt_t* dt, Void_t* obj)"
440 \f5dtlast()\fP and \f5dtprev()\fP are like \f5dtfirst()\fP and \f5dtnext()\fP
441 but work in reverse order.
442 Note that dictionaries on a viewpath are still walked in order
443 but objects in each dictionary are walked in reverse order.
445 .Ss "  Void_t* dtfinger(Dt_t* dt)"
446 This function returns the \fIcurrent object\fP of \f5dt\fP, if any.
447 The current object is defined after a successful call to one of
448 \f5dtsearch()\fP, \f5dtmatch()\fP, \f5dtinsert()\fP,
449 \f5dtfirst()\fP, \f5dtnext()\fP, \f5dtlast()\fP, or \f5dtprev()\fP.
450 As a side effect of this implementation of \fICdt\fP,
451 when a dictionary is based on \f5Dtoset\fP and \f5Dtobag\fP,
452 the current object is always defined and is the root of the tree.
454 .Ss "  Void_t* dtrenew(Dt_t* dt, Void_t* obj)"
455 This function repositions and perhaps rehashes
456 an object \f5obj\fP after its key has been changed.
457 \f5dtrenew()\fP only works if \f5obj\fP is the current object (see \f5dtfinger()\fP).
459 .Ss "  dtwalk(Dt_t* dt, int (*userf)(Dt_t*, Void_t*, Void_t*), Void_t* data)"
460 This function calls \f5(*userf)(walk,obj,data)\fP on each object in \f5dt\fP and
461 other dictionaries viewable from it.
462 \f5walk\fP is the dictionary containing \f5obj\fP.
463 If \f5userf()\fP returns a \f5<0\fP value,
464 \f5dtwalk()\fP terminates and returns the same value.
465 \f5dtwalk()\fP returns \f50\fP on completion.
467 .Ss "  Dtlink_t* dtflatten(Dt_t* dt)"
468 .Ss "  Dtlink_t* dtlink(Dt_t* dt, Dtlink_t* link)"
469 .Ss "  Void_t* dtobj(Dt_t* dt, Dtlink_t* link)"
470 Using \f5dtfirst()/dtnext()\fP or \f5dtlast()/dtprev()\fP
471 to walk a single dictionary can incur significant cost due to function calls.
472 For efficient walking of a single directory (i.e., no viewpathing),
473 \f5dtflatten()\fP and \f5dtlink()\fP can be used.
474 Objects in \f5dt\fP are made into a linked list and walked as follows:
476     for(link = dtflatten(dt); link; link = dtlink(dt,link) )
479 Note that \f5dtflatten()\fP returns a list of type \f5Dtlink_t*\fP,
480 not \f5Void_t*\fP. That is, it returns a dictionary holder pointer,
481 not a user object pointer
482 (although both are the same if the discipline field \f5link\fP is zero.)
483 The macro function \f5dtlink()\fP
484 returns the dictionary holder object following \f5link\fP.
485 The macro function \f5dtobj(dt,link)\fP
486 returns the user object associated with \f5link\fP,
487 Beware that the flattened object list is unflattened on any
488 dictionary operations other than \f5dtlink()\fP.
490 .Ss "  Dtlink_t* dtextract(Dt_t* dt)"
491 .Ss "  int dtrestore(Dt_t* dt, Dtlink_t* link)"
492 \f5dtextract()\fP extracts all objects from \f5dt\fP and makes it appear empty.
493 \f5dtrestore()\fP repopulates \f5dt\fP with
494 objects previously obtained via \f5dtextract()\fP.
495 \f5dtrestore()\fP will fail if \f5dt\fP is not empty.
496 These functions can be used
497 to share a same \f5dt\fP handle among many sets of objects.
498 They are useful to reduce dictionary overhead
499 in an application that creates many concurrent dictionaries.
500 It is important that the same discipline and method are in use at both
501 extraction and restoration. Otherwise, undefined behaviors may result.
503 .Ss "  #define   DTTREESEARCH(Dt_t* dt, Void_t* obj, action)"
504 .Ss "  #define   DTTREEMATCH(Dt_t* dt, Void_t* key, action)"
505 These macro functions are analogues of \f5dtsearch()\fP and \f5dtmatch()\fP
506 but they can only be used on a dictionary based on a binary
507 search tree, i.e., \f5Dtoset\fP or \f5Dtobag\fP.
509 \f5obj\fP or \f5key\fP:
510 These are used to find a matching object. If there is no match,
511 the result is \f5NULL\fP.
513 \f5action\fP:
514 The matching object \f5o\fP (which may be \f5NULL\fP) will be processed as follow:
517     action (o);
520 Since \f5action\fP is used verbatim, it can be any C code
521 fragment combinable with \f5(o)\fP to form a syntactically correct C statement.
522 For example, suppose that the matching object is an integer, the below code
523 accumulates the integer value in a variable \f5total\fP:
526     DTTREEMATCH(dt, key, total += (int));
530 .Ss "DICTIONARY INFORMATION"
532 .Ss "  Dt_t* dtvnext(Dt_t* dt)"
533 This returns the dictionary that \f5dt\fP is viewing, if any.
534 .Ss "  int dtvcount(Dt_t* dt)"
535 This returns the number of dictionaries that view \f5dt\fP.
536 .Ss "  Dt_t* dtvhere(Dt_t* dt)"
537 This returns the dictionary \f5v\fP viewable from \f5dt\fP
538 where an object was found from the most recent search or walk operation.
539 .Ss "  int dtsize(Dt_t* dt)"
540 This function returns the number of objects stored in \f5dt\fP.
542 .Ss "  int dtstat(Dt_t *dt, Dtstat_t* st, int all)"
543 This function reports dictionary statistics.
544 If \f5all\fP is non-zero, all fields of \f5st\fP are filled.
545 Otherwise, only the \f5dt_type\fP and \f5dt_size\fP fields are filled.
546 It returns \f50\fP on success and \f5-1\fP on error.
548 \f5Dtstat_t\fP contains the below fields:
550 \f5int dt_type\fP:
551 This is one of \f5DT_SET\fP, \f5DT_BAG\fP, \f5DT_OSET\fP, \f5DT_OBAG\fP,
552 \f5DT_LIST\fP, \f5DT_STACK\fP, and \f5DT_QUEUE\fP.
554 \f5int dt_size\fP:
555 This contains the number of objects in the dictionary.
557 \f5int dt_n\fP:
558 For \f5Dtset\fP and \f5Dtbag\fP,
559 this is the number of non-empty chains in the hash table.
560 For \f5Dtoset\fP and \f5Dtobag\fP,
561 this is the deepest level in the tree (counting from zero.)
562 Each level in the tree contains all nodes of equal distance from the root node.
563 \f5dt_n\fP and the below two fields are undefined for other methods.
565 \f5int dt_max\fP:
566 For \f5Dtbag\fP and \f5Dtset\fP, this is the size of a largest chain.
567 For \f5Dtoset\fP and \f5Dtobag\fP, this is the size of a largest level.
569 \f5int* dt_count\fP:
570 For \f5Dtset\fP and \f5Dtbag\fP,
571 this is the list of counts for chains of particular sizes.
572 For example, \f5dt_count[1]\fP is the number of chains of size \f51\fP.
573 For \f5Dtoset\fP and \f5Dtobag\fP, this is the list of sizes of the levels.
574 For example, \f5dt_count[1]\fP is the size of level \f51\fP.
576 .Ss "HASH FUNCTIONS"
578 .Ss "  unsigned int dtcharhash(unsigned int h, char c)"
579 .Ss "  unsigned int dtstrhash(unsigned int h, char* str, int n)"
580 These functions compute hash values from bytes or strings.
581 \f5dtcharhash()\fP computes a new hash value from byte \f5c\fP and seed value \f5h\fP.
582 \f5dtstrhash()\fP computes a new hash value from string \f5str\fP and seed value \f5h\fP.
583 If \f5n\fP is positive, \f5str\fP is a byte array of length \f5n\fP;
584 otherwise, \f5str\fP is a null-terminated string.
586 .SH IMPLEMENTATION NOTES
587 \f5Dtset\fP and \f5Dtbag\fP are based on hash tables with
588 move-to-front collision chains.
589 \f5Dtoset\fP and \f5Dtobag\fP are based on top-down splay trees.
590 \f5Dtlist\fP, \f5Dtstack\fP and \f5Dtqueue\fP are based on doubly linked list.
592 .SH AUTHOR
593 Kiem-Phong Vo, kpv@research.att.com