| 1 | #ifndef _CDT_H |
| 2 | #define _CDT_H 1 |
| 3 | |
| 4 | #ifdef __cplusplus |
| 5 | extern "C" { |
| 6 | #endif |
| 7 | |
| 8 | /* Public interface for the dictionary library |
| 9 | ** |
| 10 | ** Written by Kiem-Phong Vo |
| 11 | */ |
| 12 | |
| 13 | #define CDT_VERSION 20050420L |
| 14 | |
| 15 | #include <stddef.h> /* size_t */ |
| 16 | #include <string.h> |
| 17 | |
| 18 | #ifdef _WIN32 |
| 19 | # ifdef EXPORT_CDT |
| 20 | # define CDT_API __declspec(dllexport) |
| 21 | # else |
| 22 | # define CDT_API __declspec(dllimport) |
| 23 | # endif |
| 24 | #else |
| 25 | # define CDT_API extern |
| 26 | #endif |
| 27 | |
| 28 | typedef struct _dtlink_s Dtlink_t; |
| 29 | typedef struct _dthold_s Dthold_t; |
| 30 | typedef struct _dtdisc_s Dtdisc_t; |
| 31 | typedef struct _dtmethod_s Dtmethod_t; |
| 32 | typedef struct _dtdata_s Dtdata_t; |
| 33 | typedef struct _dt_s Dt_t; |
| 34 | typedef struct _dt_s Dict_t; /* for libdict compatibility */ |
| 35 | typedef struct _dtstat_s Dtstat_t; |
| 36 | typedef void* (*Dtmemory_f)(Dt_t*,void*,size_t,Dtdisc_t*); |
| 37 | typedef void* (*Dtsearch_f)(Dt_t*,void*,int); |
| 38 | typedef void* (*Dtmake_f)(Dt_t*,void*,Dtdisc_t*); |
| 39 | typedef void (*Dtfree_f)(Dt_t*,void*,Dtdisc_t*); |
| 40 | typedef int (*Dtcompar_f)(Dt_t*,void*,void*,Dtdisc_t*); |
| 41 | typedef unsigned int (*Dthash_f)(Dt_t*,void*,Dtdisc_t*); |
| 42 | typedef int (*Dtevent_f)(Dt_t*,int,void*,Dtdisc_t*); |
| 43 | |
| 44 | struct _dtlink_s |
| 45 | { Dtlink_t* right; /* right child */ |
| 46 | union |
| 47 | { unsigned int _hash; /* hash value */ |
| 48 | Dtlink_t* _left; /* left child */ |
| 49 | } hl; |
| 50 | }; |
| 51 | |
| 52 | /* private structure to hold an object */ |
| 53 | struct _dthold_s |
| 54 | { Dtlink_t hdr; /* header */ |
| 55 | void* obj; /* user object */ |
| 56 | }; |
| 57 | |
| 58 | /* method to manipulate dictionary structure */ |
| 59 | struct _dtmethod_s |
| 60 | { Dtsearch_f searchf; /* search function */ |
| 61 | int type; /* type of operation */ |
| 62 | }; |
| 63 | |
| 64 | /* stuff that may be in shared memory */ |
| 65 | struct _dtdata_s |
| 66 | { int type; /* type of dictionary */ |
| 67 | Dtlink_t* here; /* finger to last search element */ |
| 68 | union |
| 69 | { Dtlink_t** _htab; /* hash table */ |
| 70 | Dtlink_t* _head; /* linked list */ |
| 71 | } hh; |
| 72 | int ntab; /* number of hash slots */ |
| 73 | int size; /* number of objects */ |
| 74 | int loop; /* number of nested loops */ |
| 75 | int minp; /* min path before splay, always even */ |
| 76 | /* for hash dt, > 0: fixed table size */ |
| 77 | }; |
| 78 | |
| 79 | /* structure to hold methods that manipulate an object */ |
| 80 | struct _dtdisc_s |
| 81 | { int key; /* where the key begins in an object */ |
| 82 | int size; /* key size and type */ |
| 83 | int link; /* offset to Dtlink_t field */ |
| 84 | Dtmake_f makef; /* object constructor */ |
| 85 | Dtfree_f freef; /* object destructor */ |
| 86 | Dtcompar_f comparf;/* to compare two objects */ |
| 87 | Dthash_f hashf; /* to compute hash value of an object */ |
| 88 | Dtmemory_f memoryf;/* to allocate/free memory */ |
| 89 | Dtevent_f eventf; /* to process events */ |
| 90 | }; |
| 91 | |
| 92 | #define DTDISC(dc,ky,sz,lk,mkf,frf,cmpf,hshf,memf,evf) \ |
| 93 | ( (dc)->key = (ky), (dc)->size = (sz), (dc)->link = (lk), \ |
| 94 | (dc)->makef = (mkf), (dc)->freef = (frf), \ |
| 95 | (dc)->comparf = (cmpf), (dc)->hashf = (hshf), \ |
| 96 | (dc)->memoryf = (memf), (dc)->eventf = (evf) ) |
| 97 | |
| 98 | /* the dictionary structure itself */ |
| 99 | struct _dt_s |
| 100 | { Dtsearch_f searchf;/* search function */ |
| 101 | Dtdisc_t* disc; /* method to manipulate objs */ |
| 102 | Dtdata_t* data; /* sharable data */ |
| 103 | Dtmemory_f memoryf;/* function to alloc/free memory */ |
| 104 | Dtmethod_t* meth; /* dictionary method */ |
| 105 | int type; /* type information */ |
| 106 | int nview; /* number of parent view dictionaries */ |
| 107 | Dt_t* view; /* next on viewpath */ |
| 108 | Dt_t* walk; /* dictionary being walked */ |
| 109 | void* user; /* for user's usage */ |
| 110 | }; |
| 111 | |
| 112 | /* structure to get status of a dictionary */ |
| 113 | struct _dtstat_s |
| 114 | { int dt_meth; /* method type */ |
| 115 | int dt_size; /* number of elements */ |
| 116 | int dt_n; /* number of chains or levels */ |
| 117 | int dt_max; /* max size of a chain or a level */ |
| 118 | int* dt_count; /* counts of chains or levels by size */ |
| 119 | }; |
| 120 | |
| 121 | /* flag set if the last search operation actually found the object */ |
| 122 | #define DT_FOUND 0100000 |
| 123 | |
| 124 | /* supported storage methods */ |
| 125 | #define DT_SET 0000001 /* set with unique elements */ |
| 126 | #define DT_BAG 0000002 /* multiset */ |
| 127 | #define DT_OSET 0000004 /* ordered set (self-adjusting tree) */ |
| 128 | #define DT_OBAG 0000010 /* ordered multiset */ |
| 129 | #define DT_LIST 0000020 /* linked list */ |
| 130 | #define DT_STACK 0000040 /* stack: insert/delete at top */ |
| 131 | #define DT_QUEUE 0000100 /* queue: insert at top, delete at tail */ |
| 132 | #define DT_DEQUE 0000200 /* deque: insert at top, append at tail */ |
| 133 | #define DT_METHODS 0000377 /* all currently supported methods */ |
| 134 | |
| 135 | /* asserts to dtdisc() */ |
| 136 | #define DT_SAMECMP 0000001 /* compare methods equivalent */ |
| 137 | #define DT_SAMEHASH 0000002 /* hash methods equivalent */ |
| 138 | |
| 139 | /* types of search */ |
| 140 | #define DT_INSERT 0000001 /* insert object if not found */ |
| 141 | #define DT_DELETE 0000002 /* delete object if found */ |
| 142 | #define DT_SEARCH 0000004 /* look for an object */ |
| 143 | #define DT_NEXT 0000010 /* look for next element */ |
| 144 | #define DT_PREV 0000020 /* find previous element */ |
| 145 | #define DT_RENEW 0000040 /* renewing an object */ |
| 146 | #define DT_CLEAR 0000100 /* clearing all objects */ |
| 147 | #define DT_FIRST 0000200 /* get first object */ |
| 148 | #define DT_LAST 0000400 /* get last object */ |
| 149 | #define DT_MATCH 0001000 /* find object matching key */ |
| 150 | #define DT_VSEARCH 0002000 /* search using internal representation */ |
| 151 | #define DT_ATTACH 0004000 /* attach an object to the dictionary */ |
| 152 | #define DT_DETACH 0010000 /* detach an object from the dictionary */ |
| 153 | #define DT_APPEND 0020000 /* used on Dtlist to append an object */ |
| 154 | |
| 155 | /* events */ |
| 156 | #define DT_OPEN 1 /* a dictionary is being opened */ |
| 157 | #define DT_CLOSE 2 /* a dictionary is being closed */ |
| 158 | #define DT_DISC 3 /* discipline is about to be changed */ |
| 159 | #define DT_METH 4 /* method is about to be changed */ |
| 160 | #define DT_ENDOPEN 5 /* dtopen() is done */ |
| 161 | #define DT_ENDCLOSE 6 /* dtclose() is done */ |
| 162 | #define DT_HASHSIZE 7 /* setting hash table size */ |
| 163 | |
| 164 | CDT_API Dtmethod_t* Dtset; |
| 165 | CDT_API Dtmethod_t* Dtbag; |
| 166 | CDT_API Dtmethod_t* Dtoset; |
| 167 | CDT_API Dtmethod_t* Dtobag; |
| 168 | CDT_API Dtmethod_t* Dtlist; |
| 169 | CDT_API Dtmethod_t* Dtstack; |
| 170 | CDT_API Dtmethod_t* Dtqueue; |
| 171 | CDT_API Dtmethod_t* Dtdeque; |
| 172 | |
| 173 | /* compatibility stuff; will go away */ |
| 174 | #ifndef KPVDEL |
| 175 | CDT_API Dtmethod_t* Dtorder; |
| 176 | CDT_API Dtmethod_t* Dttree; |
| 177 | CDT_API Dtmethod_t* Dthash; |
| 178 | CDT_API Dtmethod_t _Dttree; |
| 179 | CDT_API Dtmethod_t _Dthash; |
| 180 | CDT_API Dtmethod_t _Dtlist; |
| 181 | CDT_API Dtmethod_t _Dtqueue; |
| 182 | CDT_API Dtmethod_t _Dtstack; |
| 183 | #endif |
| 184 | |
| 185 | CDT_API Dt_t* dtopen(Dtdisc_t*, Dtmethod_t*); |
| 186 | CDT_API int dtclose(Dt_t*); |
| 187 | CDT_API Dt_t* dtview(Dt_t*, Dt_t*); |
| 188 | CDT_API Dtdisc_t* dtdisc(Dt_t* dt, Dtdisc_t*, int); |
| 189 | CDT_API Dtmethod_t* dtmethod(Dt_t*, Dtmethod_t*); |
| 190 | |
| 191 | CDT_API Dtlink_t* dtflatten(Dt_t*); |
| 192 | CDT_API Dtlink_t* (Dt_t*); |
| 193 | CDT_API int dtrestore(Dt_t*, Dtlink_t*); |
| 194 | |
| 195 | CDT_API int dtwalk(Dt_t*, int(*)(Dt_t*,void*,void*), void*); |
| 196 | |
| 197 | CDT_API void* dtrenew(Dt_t*, void*); |
| 198 | |
| 199 | CDT_API int dtsize(Dt_t*); |
| 200 | CDT_API int dtstat(Dt_t*, Dtstat_t*, int); |
| 201 | CDT_API unsigned int dtstrhash(unsigned int, void*, int); |
| 202 | |
| 203 | /* internal functions for translating among holder, object and key */ |
| 204 | #define _DT(dt) ((Dt_t*)(dt)) |
| 205 | #define _DTDSC(dc,ky,sz,lk,cmpf) \ |
| 206 | (ky = dc->key, sz = dc->size, lk = dc->link, cmpf = dc->comparf) |
| 207 | #define _DTLNK(o,lk) ((Dtlink_t*)((char*)(o) + lk) ) |
| 208 | #define _DTOBJ(e,lk) (lk < 0 ? ((Dthold_t*)(e))->obj : (void*)((char*)(e) - lk) ) |
| 209 | #define _DTKEY(o,ky,sz) (void*)(sz < 0 ? *((char**)((char*)(o)+ky)) : ((char*)(o)+ky)) |
| 210 | |
| 211 | #define _DTCMP(dt,k1,k2,dc,cmpf,sz) \ |
| 212 | (cmpf ? (*cmpf)(dt,k1,k2,dc) : \ |
| 213 | (sz <= 0 ? strcmp(k1,k2) : memcmp(k1,k2,sz)) ) |
| 214 | #define _DTHSH(dt,ky,dc,sz) (dc->hashf ? (*dc->hashf)(dt,ky,dc) : dtstrhash(0,ky,sz) ) |
| 215 | |
| 216 | /* special search function for tree structure only */ |
| 217 | #define _DTMTCH(dt,key,action) \ |
| 218 | do { Dtlink_t* _e; void *_o, *_k, *_key; Dtdisc_t* _dc; \ |
| 219 | int _ky, _sz, _lk, _cmp; Dtcompar_f _cmpf; \ |
| 220 | _dc = (dt)->disc; _DTDSC(_dc, _ky, _sz, _lk, _cmpf); \ |
| 221 | _key = (key); \ |
| 222 | for(_e = (dt)->data->here; _e; _e = _cmp < 0 ? _e->hl._left : _e->right) \ |
| 223 | { _o = _DTOBJ(_e, _lk); _k = _DTKEY(_o, _ky, _sz); \ |
| 224 | if((_cmp = _DTCMP((dt), _key, _k, _dc, _cmpf, _sz)) == 0) \ |
| 225 | break; \ |
| 226 | } \ |
| 227 | action (_e ? _o : (void*)0); \ |
| 228 | } while(0) |
| 229 | |
| 230 | #define _DTSRCH(dt,obj,action) \ |
| 231 | do { Dtlink_t* _e; void *_o, *_k, *_key; Dtdisc_t* _dc; \ |
| 232 | int _ky, _sz, _lk, _cmp; Dtcompar_f _cmpf; \ |
| 233 | _dc = (dt)->disc; _DTDSC(_dc, _ky, _sz, _lk, _cmpf); \ |
| 234 | _key = _DTKEY(obj, _ky, _sz); \ |
| 235 | for(_e = (dt)->data->here; _e; _e = _cmp < 0 ? _e->hl._left : _e->right) \ |
| 236 | { _o = _DTOBJ(_e, _lk); _k = _DTKEY(_o, _ky, _sz); \ |
| 237 | if((_cmp = _DTCMP((dt), _key, _k, _dc, _cmpf, _sz)) == 0) \ |
| 238 | break; \ |
| 239 | } \ |
| 240 | action (_e ? _o : (void*)0); \ |
| 241 | } while(0) |
| 242 | |
| 243 | #define DTTREEMATCH(dt,key,action) _DTMTCH(_DT(dt),(void*)(key),action) |
| 244 | #define DTTREESEARCH(dt,obj,action) _DTSRCH(_DT(dt),(void*)(obj),action) |
| 245 | |
| 246 | #define dtvnext(d) (_DT(d)->view) |
| 247 | #define dtvcount(d) (_DT(d)->nview) |
| 248 | #define dtvhere(d) (_DT(d)->walk) |
| 249 | |
| 250 | #define dtlink(d,e) (((Dtlink_t*)(e))->right) |
| 251 | #define dtobj(d,e) _DTOBJ((e), _DT(d)->disc->link) |
| 252 | #define dtfinger(d) (_DT(d)->data->here ? dtobj((d),_DT(d)->data->here):(void*)(0)) |
| 253 | |
| 254 | #define dtfirst(d) (*(_DT(d)->searchf))((d),(void*)(0),DT_FIRST) |
| 255 | #define dtnext(d,o) (*(_DT(d)->searchf))((d),(void*)(o),DT_NEXT) |
| 256 | #define dtleast(d,o) (*(_DT(d)->searchf))((d),(void*)(o),DT_SEARCH|DT_NEXT) |
| 257 | #define dtlast(d) (*(_DT(d)->searchf))((d),(void*)(0),DT_LAST) |
| 258 | #define dtprev(d,o) (*(_DT(d)->searchf))((d),(void*)(o),DT_PREV) |
| 259 | #define dtmost(d,o) (*(_DT(d)->searchf))((d),(void*)(o),DT_SEARCH|DT_PREV) |
| 260 | #define dtsearch(d,o) (*(_DT(d)->searchf))((d),(void*)(o),DT_SEARCH) |
| 261 | #define dtmatch(d,o) (*(_DT(d)->searchf))((d),(void*)(o),DT_MATCH) |
| 262 | #define dtinsert(d,o) (*(_DT(d)->searchf))((d),(void*)(o),DT_INSERT) |
| 263 | #define dtappend(d,o) (*(_DT(d)->searchf))((d),(void*)(o),DT_INSERT|DT_APPEND) |
| 264 | #define dtdelete(d,o) (*(_DT(d)->searchf))((d),(void*)(o),DT_DELETE) |
| 265 | #define dtattach(d,o) (*(_DT(d)->searchf))((d),(void*)(o),DT_ATTACH) |
| 266 | #define dtdetach(d,o) (*(_DT(d)->searchf))((d),(void*)(o),DT_DETACH) |
| 267 | #define dtclear(d) (*(_DT(d)->searchf))((d),(void*)(0),DT_CLEAR) |
| 268 | #define dtfound(d) (_DT(d)->type & DT_FOUND) |
| 269 | |
| 270 | #define DT_PRIME 17109811 /* 2#00000001 00000101 00010011 00110011 */ |
| 271 | #define dtcharhash(h,c) (((unsigned int)(h) + (unsigned int)(c)) * DT_PRIME ) |
| 272 | |
| 273 | #ifdef __cplusplus |
| 274 | } |
| 275 | #endif |
| 276 | |
| 277 | #endif /* _CDT_H */ |
| 278 | |