]>
git.cameronkatri.com Git - mandoc.git/blob - compat_ohash.c
33c57afcb06be90c9d5ecda02bbe8b20936e2481
11 /* $OpenBSD: ohash.c,v 1.1 2014/06/02 18:52:03 deraadt Exp $ */
13 /* Copyright (c) 1999, 2004 Marc Espie <espie@openbsd.org>
15 * Permission to use, copy, modify, and distribute this software for any
16 * purpose with or without fee is hereby granted, provided that the above
17 * copyright notice and this permission notice appear in all copies.
19 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
20 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
21 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
22 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
23 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
24 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
25 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
32 #include "compat_ohash.h"
34 struct _ohash_record
{
39 #define DELETED ((const char *)h)
40 #define NONE (h->size)
42 /* Don't bother changing the hash table if the change is small enough. */
43 #define MINSIZE (1UL << 4)
46 static void ohash_resize(struct ohash
*);
49 /* This handles the common case of variable length keys, where the
50 * key is stored at the end of the record.
53 ohash_create_entry(struct ohash_info
*i
, const char *start
, const char **end
)
58 *end
= start
+ strlen(start
);
59 p
= (i
->alloc
)(i
->key_offset
+ (*end
- start
) + 1, i
->data
);
61 memcpy(p
+i
->key_offset
, start
, *end
-start
);
62 p
[i
->key_offset
+ (*end
- start
)] = '\0';
67 /* hash_delete only frees the hash structure. Use hash_first/hash_next
68 * to free entries as well. */
70 ohash_delete(struct ohash
*h
)
72 (h
->info
.hfree
)(h
->t
, sizeof(struct _ohash_record
) * h
->size
,
80 ohash_resize(struct ohash
*h
)
82 struct _ohash_record
*n
;
86 if (4 * h
->deleted
< h
->total
)
88 else if (3 * h
->deleted
> 2 * h
->total
)
96 STAT_HASH_SIZE
+= ns
- h
->size
;
98 n
= (h
->info
.halloc
)(sizeof(struct _ohash_record
) * ns
, h
->info
.data
);
102 for (j
= 0; j
< h
->size
; j
++) {
103 if (h
->t
[j
].p
!= NULL
&& h
->t
[j
].p
!= DELETED
) {
105 incr
= ((h
->t
[j
].hv
% (ns
- 2)) & ~1) + 1;
106 while (n
[i
].p
!= NULL
) {
111 n
[i
].hv
= h
->t
[j
].hv
;
115 (h
->info
.hfree
)(h
->t
, sizeof(struct _ohash_record
) * h
->size
,
119 h
->total
-= h
->deleted
;
124 ohash_remove(struct ohash
*h
, unsigned int i
)
126 void *result
= (void *)h
->t
[i
].p
;
128 if (result
== NULL
|| result
== DELETED
)
136 if (h
->deleted
>= MINDELETED
&& 4 * h
->deleted
> h
->total
)
142 ohash_find(struct ohash
*h
, unsigned int i
)
144 if (h
->t
[i
].p
== DELETED
)
147 return (void *)h
->t
[i
].p
;
151 ohash_insert(struct ohash
*h
, unsigned int i
, void *p
)
156 if (h
->t
[i
].p
== DELETED
) {
161 /* Arbitrary resize boundary. Tweak if not efficient enough. */
162 if (++h
->total
* 4 > h
->size
* 3)
169 ohash_entries(struct ohash
*h
)
171 return h
->total
- h
->deleted
;
175 ohash_first(struct ohash
*h
, unsigned int *pos
)
178 return ohash_next(h
, pos
);
182 ohash_next(struct ohash
*h
, unsigned int *pos
)
184 for (; *pos
< h
->size
; (*pos
)++)
185 if (h
->t
[*pos
].p
!= DELETED
&& h
->t
[*pos
].p
!= NULL
)
186 return (void *)h
->t
[(*pos
)++].p
;
191 ohash_init(struct ohash
*h
, unsigned int size
, struct ohash_info
*info
)
193 h
->size
= 1UL << size
;
194 if (h
->size
< MINSIZE
)
197 STAT_HASH_CREATION
++;
198 STAT_HASH_SIZE
+= h
->size
;
200 /* Copy info so that caller may free it. */
201 h
->info
.key_offset
= info
->key_offset
;
202 h
->info
.halloc
= info
->halloc
;
203 h
->info
.hfree
= info
->hfree
;
204 h
->info
.alloc
= info
->alloc
;
205 h
->info
.data
= info
->data
;
206 h
->t
= (h
->info
.halloc
)(sizeof(struct _ohash_record
) * h
->size
,
208 h
->total
= h
->deleted
= 0;
212 ohash_interval(const char *s
, const char **e
)
223 k
= ((k
<< 2) | (k
>> 30)) ^ *s
++;
228 ohash_lookup_interval(struct ohash
*h
, const char *start
, const char *end
,
231 unsigned int i
, incr
;
239 incr
= ((hv
% (h
->size
-2)) & ~1) + 1;
240 while (h
->t
[i
].p
!= NULL
) {
244 if (h
->t
[i
].p
== DELETED
) {
247 } else if (h
->t
[i
].hv
== hv
&&
248 strncmp(h
->t
[i
].p
+h
->info
.key_offset
, start
,
250 (h
->t
[i
].p
+h
->info
.key_offset
)[end
-start
] == '\0') {
253 h
->t
[empty
].p
= h
->t
[i
].p
;
258 STAT_HASH_POSITIVE
++;
268 /* Found an empty position. */
276 ohash_lookup_memory(struct ohash
*h
, const char *k
, size_t size
, uint32_t hv
)
278 unsigned int i
, incr
;
286 incr
= ((hv
% (h
->size
-2)) & ~1) + 1;
287 while (h
->t
[i
].p
!= NULL
) {
291 if (h
->t
[i
].p
== DELETED
) {
294 } else if (h
->t
[i
].hv
== hv
&&
295 memcmp(h
->t
[i
].p
+h
->info
.key_offset
, k
, size
) == 0) {
298 h
->t
[empty
].p
= h
->t
[i
].p
;
303 STAT_HASH_POSITIVE
++;
312 /* Found an empty position. */
320 ohash_qlookup(struct ohash
*h
, const char *s
)
322 const char *e
= NULL
;
323 return ohash_qlookupi(h
, s
, &e
);
327 ohash_qlookupi(struct ohash
*h
, const char *s
, const char **e
)
331 hv
= ohash_interval(s
, e
);
332 return ohash_lookup_interval(h
, s
, *e
, hv
);
335 #endif /*!HAVE_OHASH*/