1
2 /* Author : Stephen Smalley, <sds@epoch.ncsc.mil> */
3
4 /* FLASK */
5
6 /*
7 * Implementation of the hash table type.
8 */
9
10 #include <stdlib.h>
11 #include <string.h>
12 #include "hashtab.h"
13
hashtab_create(unsigned int (* hash_value)(hashtab_t h,const_hashtab_key_t key),int (* keycmp)(hashtab_t h,const_hashtab_key_t key1,const_hashtab_key_t key2),unsigned int size)14 hashtab_t hashtab_create(unsigned int (*hash_value) (hashtab_t h,
15 const_hashtab_key_t key),
16 int (*keycmp) (hashtab_t h,
17 const_hashtab_key_t key1,
18 const_hashtab_key_t key2),
19 unsigned int size)
20 {
21
22 hashtab_t p;
23 unsigned int i;
24
25 p = (hashtab_t) malloc(sizeof(hashtab_val_t));
26 if (p == NULL)
27 return p;
28
29 memset(p, 0, sizeof(hashtab_val_t));
30 p->size = size;
31 p->nel = 0;
32 p->hash_value = hash_value;
33 p->keycmp = keycmp;
34 p->htable = (hashtab_ptr_t *) malloc(sizeof(hashtab_ptr_t) * size);
35 if (p->htable == NULL) {
36 free(p);
37 return NULL;
38 }
39 for (i = 0; i < size; i++)
40 p->htable[i] = (hashtab_ptr_t) NULL;
41
42 return p;
43 }
44
hashtab_insert(hashtab_t h,hashtab_key_t key,hashtab_datum_t datum)45 int hashtab_insert(hashtab_t h, hashtab_key_t key, hashtab_datum_t datum)
46 {
47 int hvalue;
48 hashtab_ptr_t prev, cur, newnode;
49
50 if (!h)
51 return HASHTAB_OVERFLOW;
52
53 hvalue = h->hash_value(h, key);
54 prev = NULL;
55 cur = h->htable[hvalue];
56 while (cur && h->keycmp(h, key, cur->key) > 0) {
57 prev = cur;
58 cur = cur->next;
59 }
60
61 if (cur && (h->keycmp(h, key, cur->key) == 0))
62 return HASHTAB_PRESENT;
63
64 newnode = (hashtab_ptr_t) malloc(sizeof(hashtab_node_t));
65 if (newnode == NULL)
66 return HASHTAB_OVERFLOW;
67 memset(newnode, 0, sizeof(struct hashtab_node));
68 newnode->key = key;
69 newnode->datum = datum;
70 if (prev) {
71 newnode->next = prev->next;
72 prev->next = newnode;
73 } else {
74 newnode->next = h->htable[hvalue];
75 h->htable[hvalue] = newnode;
76 }
77
78 h->nel++;
79 return HASHTAB_SUCCESS;
80 }
81
hashtab_remove(hashtab_t h,hashtab_key_t key,void (* destroy)(hashtab_key_t k,hashtab_datum_t d,void * args),void * args)82 int hashtab_remove(hashtab_t h, hashtab_key_t key,
83 void (*destroy) (hashtab_key_t k,
84 hashtab_datum_t d, void *args), void *args)
85 {
86 int hvalue;
87 hashtab_ptr_t cur, last;
88
89 if (!h)
90 return HASHTAB_MISSING;
91
92 hvalue = h->hash_value(h, key);
93 last = NULL;
94 cur = h->htable[hvalue];
95 while (cur != NULL && h->keycmp(h, key, cur->key) > 0) {
96 last = cur;
97 cur = cur->next;
98 }
99
100 if (cur == NULL || (h->keycmp(h, key, cur->key) != 0))
101 return HASHTAB_MISSING;
102
103 if (last == NULL)
104 h->htable[hvalue] = cur->next;
105 else
106 last->next = cur->next;
107
108 if (destroy)
109 destroy(cur->key, cur->datum, args);
110 free(cur);
111 h->nel--;
112 return HASHTAB_SUCCESS;
113 }
114
hashtab_replace(hashtab_t h,hashtab_key_t key,hashtab_datum_t datum,void (* destroy)(hashtab_key_t k,hashtab_datum_t d,void * args),void * args)115 int hashtab_replace(hashtab_t h, hashtab_key_t key, hashtab_datum_t datum,
116 void (*destroy) (hashtab_key_t k,
117 hashtab_datum_t d, void *args), void *args)
118 {
119 int hvalue;
120 hashtab_ptr_t prev, cur, newnode;
121
122 if (!h)
123 return HASHTAB_OVERFLOW;
124
125 hvalue = h->hash_value(h, key);
126 prev = NULL;
127 cur = h->htable[hvalue];
128 while (cur != NULL && h->keycmp(h, key, cur->key) > 0) {
129 prev = cur;
130 cur = cur->next;
131 }
132
133 if (cur && (h->keycmp(h, key, cur->key) == 0)) {
134 if (destroy)
135 destroy(cur->key, cur->datum, args);
136 cur->key = key;
137 cur->datum = datum;
138 } else {
139 newnode = (hashtab_ptr_t) malloc(sizeof(hashtab_node_t));
140 if (newnode == NULL)
141 return HASHTAB_OVERFLOW;
142 memset(newnode, 0, sizeof(struct hashtab_node));
143 newnode->key = key;
144 newnode->datum = datum;
145 if (prev) {
146 newnode->next = prev->next;
147 prev->next = newnode;
148 } else {
149 newnode->next = h->htable[hvalue];
150 h->htable[hvalue] = newnode;
151 }
152 }
153
154 return HASHTAB_SUCCESS;
155 }
156
hashtab_search(hashtab_t h,const_hashtab_key_t key)157 hashtab_datum_t hashtab_search(hashtab_t h, const_hashtab_key_t key)
158 {
159
160 int hvalue;
161 hashtab_ptr_t cur;
162
163 if (!h)
164 return NULL;
165
166 hvalue = h->hash_value(h, key);
167 cur = h->htable[hvalue];
168 while (cur != NULL && h->keycmp(h, key, cur->key) > 0)
169 cur = cur->next;
170
171 if (cur == NULL || (h->keycmp(h, key, cur->key) != 0))
172 return NULL;
173
174 return cur->datum;
175 }
176
hashtab_destroy(hashtab_t h)177 void hashtab_destroy(hashtab_t h)
178 {
179 unsigned int i;
180 hashtab_ptr_t cur, temp;
181
182 if (!h)
183 return;
184
185 for (i = 0; i < h->size; i++) {
186 cur = h->htable[i];
187 while (cur != NULL) {
188 temp = cur;
189 cur = cur->next;
190 free(temp);
191 }
192 h->htable[i] = NULL;
193 }
194
195 free(h->htable);
196 h->htable = NULL;
197
198 free(h);
199 }
200
hashtab_map(hashtab_t h,int (* apply)(hashtab_key_t k,hashtab_datum_t d,void * args),void * args)201 int hashtab_map(hashtab_t h,
202 int (*apply) (hashtab_key_t k,
203 hashtab_datum_t d, void *args), void *args)
204 {
205 unsigned int i, ret;
206 hashtab_ptr_t cur;
207
208 if (!h)
209 return HASHTAB_SUCCESS;
210
211 for (i = 0; i < h->size; i++) {
212 cur = h->htable[i];
213 while (cur != NULL) {
214 ret = apply(cur->key, cur->datum, args);
215 if (ret)
216 return ret;
217 cur = cur->next;
218 }
219 }
220 return HASHTAB_SUCCESS;
221 }
222
hashtab_map_remove_on_error(hashtab_t h,int (* apply)(hashtab_key_t k,hashtab_datum_t d,void * args),void (* destroy)(hashtab_key_t k,hashtab_datum_t d,void * args),void * args)223 void hashtab_map_remove_on_error(hashtab_t h,
224 int (*apply) (hashtab_key_t k,
225 hashtab_datum_t d,
226 void *args),
227 void (*destroy) (hashtab_key_t k,
228 hashtab_datum_t d,
229 void *args), void *args)
230 {
231 unsigned int i;
232 int ret;
233 hashtab_ptr_t last, cur, temp;
234
235 if (!h)
236 return;
237
238 for (i = 0; i < h->size; i++) {
239 last = NULL;
240 cur = h->htable[i];
241 while (cur != NULL) {
242 ret = apply(cur->key, cur->datum, args);
243 if (ret) {
244 if (last) {
245 last->next = cur->next;
246 } else {
247 h->htable[i] = cur->next;
248 }
249
250 temp = cur;
251 cur = cur->next;
252 if (destroy)
253 destroy(temp->key, temp->datum, args);
254 free(temp);
255 h->nel--;
256 } else {
257 last = cur;
258 cur = cur->next;
259 }
260 }
261 }
262
263 return;
264 }
265
hashtab_hash_eval(hashtab_t h,char * tag)266 void hashtab_hash_eval(hashtab_t h, char *tag)
267 {
268 unsigned int i;
269 int chain_len, slots_used, max_chain_len;
270 hashtab_ptr_t cur;
271
272 slots_used = 0;
273 max_chain_len = 0;
274 for (i = 0; i < h->size; i++) {
275 cur = h->htable[i];
276 if (cur) {
277 slots_used++;
278 chain_len = 0;
279 while (cur) {
280 chain_len++;
281 cur = cur->next;
282 }
283
284 if (chain_len > max_chain_len)
285 max_chain_len = chain_len;
286 }
287 }
288
289 printf
290 ("%s: %d entries and %d/%d buckets used, longest chain length %d\n",
291 tag, h->nel, slots_used, h->size, max_chain_len);
292 }
293