1*c9945492SAndroid Build Coastguard Worker #define _GNU_SOURCE
2*c9945492SAndroid Build Coastguard Worker #include <stdlib.h>
3*c9945492SAndroid Build Coastguard Worker #include <string.h>
4*c9945492SAndroid Build Coastguard Worker #include <search.h>
5*c9945492SAndroid Build Coastguard Worker
6*c9945492SAndroid Build Coastguard Worker /*
7*c9945492SAndroid Build Coastguard Worker open addressing hash table with 2^n table size
8*c9945492SAndroid Build Coastguard Worker quadratic probing is used in case of hash collision
9*c9945492SAndroid Build Coastguard Worker tab indices and hash are size_t
10*c9945492SAndroid Build Coastguard Worker after resize fails with ENOMEM the state of tab is still usable
11*c9945492SAndroid Build Coastguard Worker
12*c9945492SAndroid Build Coastguard Worker with the posix api items cannot be iterated and length cannot be queried
13*c9945492SAndroid Build Coastguard Worker */
14*c9945492SAndroid Build Coastguard Worker
15*c9945492SAndroid Build Coastguard Worker #define MINSIZE 8
16*c9945492SAndroid Build Coastguard Worker #define MAXSIZE ((size_t)-1/2 + 1)
17*c9945492SAndroid Build Coastguard Worker
18*c9945492SAndroid Build Coastguard Worker struct __tab {
19*c9945492SAndroid Build Coastguard Worker ENTRY *entries;
20*c9945492SAndroid Build Coastguard Worker size_t mask;
21*c9945492SAndroid Build Coastguard Worker size_t used;
22*c9945492SAndroid Build Coastguard Worker };
23*c9945492SAndroid Build Coastguard Worker
24*c9945492SAndroid Build Coastguard Worker static struct hsearch_data htab;
25*c9945492SAndroid Build Coastguard Worker
26*c9945492SAndroid Build Coastguard Worker static int __hcreate_r(size_t, struct hsearch_data *);
27*c9945492SAndroid Build Coastguard Worker static void __hdestroy_r(struct hsearch_data *);
28*c9945492SAndroid Build Coastguard Worker static int __hsearch_r(ENTRY, ACTION, ENTRY **, struct hsearch_data *);
29*c9945492SAndroid Build Coastguard Worker
keyhash(char * k)30*c9945492SAndroid Build Coastguard Worker static size_t keyhash(char *k)
31*c9945492SAndroid Build Coastguard Worker {
32*c9945492SAndroid Build Coastguard Worker unsigned char *p = (void *)k;
33*c9945492SAndroid Build Coastguard Worker size_t h = 0;
34*c9945492SAndroid Build Coastguard Worker
35*c9945492SAndroid Build Coastguard Worker while (*p)
36*c9945492SAndroid Build Coastguard Worker h = 31*h + *p++;
37*c9945492SAndroid Build Coastguard Worker return h;
38*c9945492SAndroid Build Coastguard Worker }
39*c9945492SAndroid Build Coastguard Worker
resize(size_t nel,struct hsearch_data * htab)40*c9945492SAndroid Build Coastguard Worker static int resize(size_t nel, struct hsearch_data *htab)
41*c9945492SAndroid Build Coastguard Worker {
42*c9945492SAndroid Build Coastguard Worker size_t newsize;
43*c9945492SAndroid Build Coastguard Worker size_t i, j;
44*c9945492SAndroid Build Coastguard Worker size_t oldsize = htab->__tab->mask + 1;
45*c9945492SAndroid Build Coastguard Worker ENTRY *e, *newe;
46*c9945492SAndroid Build Coastguard Worker ENTRY *oldtab = htab->__tab->entries;
47*c9945492SAndroid Build Coastguard Worker
48*c9945492SAndroid Build Coastguard Worker if (nel > MAXSIZE)
49*c9945492SAndroid Build Coastguard Worker nel = MAXSIZE;
50*c9945492SAndroid Build Coastguard Worker for (newsize = MINSIZE; newsize < nel; newsize *= 2);
51*c9945492SAndroid Build Coastguard Worker htab->__tab->entries = calloc(newsize, sizeof *htab->__tab->entries);
52*c9945492SAndroid Build Coastguard Worker if (!htab->__tab->entries) {
53*c9945492SAndroid Build Coastguard Worker htab->__tab->entries = oldtab;
54*c9945492SAndroid Build Coastguard Worker return 0;
55*c9945492SAndroid Build Coastguard Worker }
56*c9945492SAndroid Build Coastguard Worker htab->__tab->mask = newsize - 1;
57*c9945492SAndroid Build Coastguard Worker if (!oldtab)
58*c9945492SAndroid Build Coastguard Worker return 1;
59*c9945492SAndroid Build Coastguard Worker for (e = oldtab; e < oldtab + oldsize; e++)
60*c9945492SAndroid Build Coastguard Worker if (e->key) {
61*c9945492SAndroid Build Coastguard Worker for (i=keyhash(e->key),j=1; ; i+=j++) {
62*c9945492SAndroid Build Coastguard Worker newe = htab->__tab->entries + (i & htab->__tab->mask);
63*c9945492SAndroid Build Coastguard Worker if (!newe->key)
64*c9945492SAndroid Build Coastguard Worker break;
65*c9945492SAndroid Build Coastguard Worker }
66*c9945492SAndroid Build Coastguard Worker *newe = *e;
67*c9945492SAndroid Build Coastguard Worker }
68*c9945492SAndroid Build Coastguard Worker free(oldtab);
69*c9945492SAndroid Build Coastguard Worker return 1;
70*c9945492SAndroid Build Coastguard Worker }
71*c9945492SAndroid Build Coastguard Worker
hcreate(size_t nel)72*c9945492SAndroid Build Coastguard Worker int hcreate(size_t nel)
73*c9945492SAndroid Build Coastguard Worker {
74*c9945492SAndroid Build Coastguard Worker return __hcreate_r(nel, &htab);
75*c9945492SAndroid Build Coastguard Worker }
76*c9945492SAndroid Build Coastguard Worker
hdestroy(void)77*c9945492SAndroid Build Coastguard Worker void hdestroy(void)
78*c9945492SAndroid Build Coastguard Worker {
79*c9945492SAndroid Build Coastguard Worker __hdestroy_r(&htab);
80*c9945492SAndroid Build Coastguard Worker }
81*c9945492SAndroid Build Coastguard Worker
lookup(char * key,size_t hash,struct hsearch_data * htab)82*c9945492SAndroid Build Coastguard Worker static ENTRY *lookup(char *key, size_t hash, struct hsearch_data *htab)
83*c9945492SAndroid Build Coastguard Worker {
84*c9945492SAndroid Build Coastguard Worker size_t i, j;
85*c9945492SAndroid Build Coastguard Worker ENTRY *e;
86*c9945492SAndroid Build Coastguard Worker
87*c9945492SAndroid Build Coastguard Worker for (i=hash,j=1; ; i+=j++) {
88*c9945492SAndroid Build Coastguard Worker e = htab->__tab->entries + (i & htab->__tab->mask);
89*c9945492SAndroid Build Coastguard Worker if (!e->key || strcmp(e->key, key) == 0)
90*c9945492SAndroid Build Coastguard Worker break;
91*c9945492SAndroid Build Coastguard Worker }
92*c9945492SAndroid Build Coastguard Worker return e;
93*c9945492SAndroid Build Coastguard Worker }
94*c9945492SAndroid Build Coastguard Worker
hsearch(ENTRY item,ACTION action)95*c9945492SAndroid Build Coastguard Worker ENTRY *hsearch(ENTRY item, ACTION action)
96*c9945492SAndroid Build Coastguard Worker {
97*c9945492SAndroid Build Coastguard Worker ENTRY *e;
98*c9945492SAndroid Build Coastguard Worker
99*c9945492SAndroid Build Coastguard Worker __hsearch_r(item, action, &e, &htab);
100*c9945492SAndroid Build Coastguard Worker return e;
101*c9945492SAndroid Build Coastguard Worker }
102*c9945492SAndroid Build Coastguard Worker
__hcreate_r(size_t nel,struct hsearch_data * htab)103*c9945492SAndroid Build Coastguard Worker static int __hcreate_r(size_t nel, struct hsearch_data *htab)
104*c9945492SAndroid Build Coastguard Worker {
105*c9945492SAndroid Build Coastguard Worker int r;
106*c9945492SAndroid Build Coastguard Worker
107*c9945492SAndroid Build Coastguard Worker htab->__tab = calloc(1, sizeof *htab->__tab);
108*c9945492SAndroid Build Coastguard Worker if (!htab->__tab)
109*c9945492SAndroid Build Coastguard Worker return 0;
110*c9945492SAndroid Build Coastguard Worker r = resize(nel, htab);
111*c9945492SAndroid Build Coastguard Worker if (r == 0) {
112*c9945492SAndroid Build Coastguard Worker free(htab->__tab);
113*c9945492SAndroid Build Coastguard Worker htab->__tab = 0;
114*c9945492SAndroid Build Coastguard Worker }
115*c9945492SAndroid Build Coastguard Worker return r;
116*c9945492SAndroid Build Coastguard Worker }
117*c9945492SAndroid Build Coastguard Worker weak_alias(__hcreate_r, hcreate_r);
118*c9945492SAndroid Build Coastguard Worker
__hdestroy_r(struct hsearch_data * htab)119*c9945492SAndroid Build Coastguard Worker static void __hdestroy_r(struct hsearch_data *htab)
120*c9945492SAndroid Build Coastguard Worker {
121*c9945492SAndroid Build Coastguard Worker if (htab->__tab) free(htab->__tab->entries);
122*c9945492SAndroid Build Coastguard Worker free(htab->__tab);
123*c9945492SAndroid Build Coastguard Worker htab->__tab = 0;
124*c9945492SAndroid Build Coastguard Worker }
125*c9945492SAndroid Build Coastguard Worker weak_alias(__hdestroy_r, hdestroy_r);
126*c9945492SAndroid Build Coastguard Worker
__hsearch_r(ENTRY item,ACTION action,ENTRY ** retval,struct hsearch_data * htab)127*c9945492SAndroid Build Coastguard Worker static int __hsearch_r(ENTRY item, ACTION action, ENTRY **retval, struct hsearch_data *htab)
128*c9945492SAndroid Build Coastguard Worker {
129*c9945492SAndroid Build Coastguard Worker size_t hash = keyhash(item.key);
130*c9945492SAndroid Build Coastguard Worker ENTRY *e = lookup(item.key, hash, htab);
131*c9945492SAndroid Build Coastguard Worker
132*c9945492SAndroid Build Coastguard Worker if (e->key) {
133*c9945492SAndroid Build Coastguard Worker *retval = e;
134*c9945492SAndroid Build Coastguard Worker return 1;
135*c9945492SAndroid Build Coastguard Worker }
136*c9945492SAndroid Build Coastguard Worker if (action == FIND) {
137*c9945492SAndroid Build Coastguard Worker *retval = 0;
138*c9945492SAndroid Build Coastguard Worker return 0;
139*c9945492SAndroid Build Coastguard Worker }
140*c9945492SAndroid Build Coastguard Worker *e = item;
141*c9945492SAndroid Build Coastguard Worker if (++htab->__tab->used > htab->__tab->mask - htab->__tab->mask/4) {
142*c9945492SAndroid Build Coastguard Worker if (!resize(2*htab->__tab->used, htab)) {
143*c9945492SAndroid Build Coastguard Worker htab->__tab->used--;
144*c9945492SAndroid Build Coastguard Worker e->key = 0;
145*c9945492SAndroid Build Coastguard Worker *retval = 0;
146*c9945492SAndroid Build Coastguard Worker return 0;
147*c9945492SAndroid Build Coastguard Worker }
148*c9945492SAndroid Build Coastguard Worker e = lookup(item.key, hash, htab);
149*c9945492SAndroid Build Coastguard Worker }
150*c9945492SAndroid Build Coastguard Worker *retval = e;
151*c9945492SAndroid Build Coastguard Worker return 1;
152*c9945492SAndroid Build Coastguard Worker }
153*c9945492SAndroid Build Coastguard Worker weak_alias(__hsearch_r, hsearch_r);
154