MOXA linux-2.6.x / linux-2.6.9-uc0 from sdlinux-moxaart.tgz
[linux-2.6.9-moxart.git] / security / selinux / ss / hashtab.c
blob2a6752a8d016b9347c8326699c1bdf94a8f310d0
1 /*
2 * Implementation of the hash table type.
4 * Author : Stephen Smalley, <sds@epoch.ncsc.mil>
5 */
6 #include <linux/kernel.h>
7 #include <linux/slab.h>
8 #include <linux/errno.h>
9 #include "hashtab.h"
11 struct hashtab *hashtab_create(u32 (*hash_value)(struct hashtab *h, void *key),
12 int (*keycmp)(struct hashtab *h, void *key1, void *key2),
13 u32 size)
15 struct hashtab *p;
16 u32 i;
18 p = kmalloc(sizeof(*p), GFP_KERNEL);
19 if (p == NULL)
20 return p;
22 memset(p, 0, sizeof(*p));
23 p->size = size;
24 p->nel = 0;
25 p->hash_value = hash_value;
26 p->keycmp = keycmp;
27 p->htable = kmalloc(sizeof(*(p->htable)) * size, GFP_KERNEL);
28 if (p->htable == NULL) {
29 kfree(p);
30 return NULL;
33 for (i = 0; i < size; i++)
34 p->htable[i] = NULL;
36 return p;
39 int hashtab_insert(struct hashtab *h, void *key, void *datum)
41 u32 hvalue;
42 struct hashtab_node *prev, *cur, *newnode;
44 if (!h || h->nel == HASHTAB_MAX_NODES)
45 return -EINVAL;
47 hvalue = h->hash_value(h, key);
48 prev = NULL;
49 cur = h->htable[hvalue];
50 while (cur && h->keycmp(h, key, cur->key) > 0) {
51 prev = cur;
52 cur = cur->next;
55 if (cur && (h->keycmp(h, key, cur->key) == 0))
56 return -EEXIST;
58 newnode = kmalloc(sizeof(*newnode), GFP_KERNEL);
59 if (newnode == NULL)
60 return -ENOMEM;
61 memset(newnode, 0, sizeof(*newnode));
62 newnode->key = key;
63 newnode->datum = datum;
64 if (prev) {
65 newnode->next = prev->next;
66 prev->next = newnode;
67 } else {
68 newnode->next = h->htable[hvalue];
69 h->htable[hvalue] = newnode;
72 h->nel++;
73 return 0;
76 int hashtab_remove(struct hashtab *h, void *key,
77 void (*destroy)(void *k, void *d, void *args),
78 void *args)
80 u32 hvalue;
81 struct hashtab_node *cur, *last;
83 if (!h)
84 return -EINVAL;
86 hvalue = h->hash_value(h, key);
87 last = NULL;
88 cur = h->htable[hvalue];
89 while (cur != NULL && h->keycmp(h, key, cur->key) > 0) {
90 last = cur;
91 cur = cur->next;
94 if (cur == NULL || (h->keycmp(h, key, cur->key) != 0))
95 return -ENOENT;
97 if (last == NULL)
98 h->htable[hvalue] = cur->next;
99 else
100 last->next = cur->next;
102 if (destroy)
103 destroy(cur->key, cur->datum, args);
104 kfree(cur);
105 h->nel--;
106 return 0;
109 int hashtab_replace(struct hashtab *h, void *key, void *datum,
110 void (*destroy)(void *k, void *d, void *args),
111 void *args)
113 u32 hvalue;
114 struct hashtab_node *prev, *cur, *newnode;
116 if (!h)
117 return -EINVAL;
119 hvalue = h->hash_value(h, key);
120 prev = NULL;
121 cur = h->htable[hvalue];
122 while (cur != NULL && h->keycmp(h, key, cur->key) > 0) {
123 prev = cur;
124 cur = cur->next;
127 if (cur && (h->keycmp(h, key, cur->key) == 0)) {
128 if (destroy)
129 destroy(cur->key, cur->datum, args);
130 cur->key = key;
131 cur->datum = datum;
132 } else {
133 newnode = kmalloc(sizeof(*newnode), GFP_KERNEL);
134 if (newnode == NULL)
135 return -ENOMEM;
136 memset(newnode, 0, sizeof(*newnode));
137 newnode->key = key;
138 newnode->datum = datum;
139 if (prev) {
140 newnode->next = prev->next;
141 prev->next = newnode;
142 } else {
143 newnode->next = h->htable[hvalue];
144 h->htable[hvalue] = newnode;
148 return 0;
151 void *hashtab_search(struct hashtab *h, void *key)
153 u32 hvalue;
154 struct hashtab_node *cur;
156 if (!h)
157 return NULL;
159 hvalue = h->hash_value(h, key);
160 cur = h->htable[hvalue];
161 while (cur != NULL && h->keycmp(h, key, cur->key) > 0)
162 cur = cur->next;
164 if (cur == NULL || (h->keycmp(h, key, cur->key) != 0))
165 return NULL;
167 return cur->datum;
170 void hashtab_destroy(struct hashtab *h)
172 u32 i;
173 struct hashtab_node *cur, *temp;
175 if (!h)
176 return;
178 for (i = 0; i < h->size; i++) {
179 cur = h->htable[i];
180 while (cur != NULL) {
181 temp = cur;
182 cur = cur->next;
183 kfree(temp);
185 h->htable[i] = NULL;
188 kfree(h->htable);
189 h->htable = NULL;
191 kfree(h);
194 int hashtab_map(struct hashtab *h,
195 int (*apply)(void *k, void *d, void *args),
196 void *args)
198 u32 i;
199 int ret;
200 struct hashtab_node *cur;
202 if (!h)
203 return 0;
205 for (i = 0; i < h->size; i++) {
206 cur = h->htable[i];
207 while (cur != NULL) {
208 ret = apply(cur->key, cur->datum, args);
209 if (ret)
210 return ret;
211 cur = cur->next;
214 return 0;
218 void hashtab_map_remove_on_error(struct hashtab *h,
219 int (*apply)(void *k, void *d, void *args),
220 void (*destroy)(void *k, void *d, void *args),
221 void *args)
223 u32 i;
224 int ret;
225 struct hashtab_node *last, *cur, *temp;
227 if (!h)
228 return;
230 for (i = 0; i < h->size; i++) {
231 last = NULL;
232 cur = h->htable[i];
233 while (cur != NULL) {
234 ret = apply(cur->key, cur->datum, args);
235 if (ret) {
236 if (last)
237 last->next = cur->next;
238 else
239 h->htable[i] = cur->next;
241 temp = cur;
242 cur = cur->next;
243 if (destroy)
244 destroy(temp->key, temp->datum, args);
245 kfree(temp);
246 h->nel--;
247 } else {
248 last = cur;
249 cur = cur->next;
253 return;
256 void hashtab_stat(struct hashtab *h, struct hashtab_info *info)
258 u32 i, chain_len, slots_used, max_chain_len;
259 struct hashtab_node *cur;
261 slots_used = 0;
262 max_chain_len = 0;
263 for (slots_used = max_chain_len = i = 0; i < h->size; i++) {
264 cur = h->htable[i];
265 if (cur) {
266 slots_used++;
267 chain_len = 0;
268 while (cur) {
269 chain_len++;
270 cur = cur->next;
273 if (chain_len > max_chain_len)
274 max_chain_len = chain_len;
278 info->slots_used = slots_used;
279 info->max_chain_len = max_chain_len;