128#define AC_VERSION_KHASH_H "0.2.8"
139#define kh_inline __inline
141#define kh_inline inline
146#if (defined __clang__ && __clang_major__ >= 3) || (defined __GNUC__ && __GNUC__ >= 3)
147#define klib_unused __attribute__ ((__unused__))
156#define __ac_isempty(flag, i) ((flag[i>>4]>>((i&0xfU)<<1))&2)
157#define __ac_isdel(flag, i) ((flag[i>>4]>>((i&0xfU)<<1))&1)
158#define __ac_iseither(flag, i) ((flag[i>>4]>>((i&0xfU)<<1))&3)
159#define __ac_set_isdel_false(flag, i) (flag[i>>4]&=~(1ul<<((i&0xfU)<<1)))
160#define __ac_set_isempty_false(flag, i) (flag[i>>4]&=~(2ul<<((i&0xfU)<<1)))
161#define __ac_set_isboth_false(flag, i) (flag[i>>4]&=~(3ul<<((i&0xfU)<<1)))
162#define __ac_set_isdel_true(flag, i) (flag[i>>4]|=1ul<<((i&0xfU)<<1))
164#define __ac_fsize(m) ((m) < 16? 1 : (m)>>4)
167#define kroundup32(x) (--(x), (x)|=(x)>>1, (x)|=(x)>>2, (x)|=(x)>>4, (x)|=(x)>>8, (x)|=(x)>>16, ++(x))
171#define kcalloc(N,Z) calloc(N,Z)
174#define kmalloc(Z) malloc(Z)
177#define krealloc(P,Z) realloc(P,Z)
180#define kfree(P) free(P)
185#define __KHASH_TYPE(name, khkey_t, khval_t) \
186 typedef struct kh_##name##_s { \
187 khint_t n_buckets, size, n_occupied, upper_bound; \
193#define __KHASH_PROTOTYPES(name, khkey_t, khval_t) \
194 extern kh_##name##_t *kh_init_##name(void); \
195 extern void kh_destroy_##name(kh_##name##_t *h); \
196 extern void kh_clear_##name(kh_##name##_t *h); \
197 extern khint_t kh_get_##name(const kh_##name##_t *h, khkey_t key); \
198 extern int kh_resize_##name(kh_##name##_t *h, khint_t new_n_buckets); \
199 extern khint_t kh_put_##name(kh_##name##_t *h, khkey_t key, int *ret); \
200 extern void kh_del_##name(kh_##name##_t *h, khint_t x);
202#define __KHASH_IMPL(name, SCOPE, khkey_t, khval_t, kh_is_map, __hash_func, __hash_equal) \
203 SCOPE kh_##name##_t *kh_init_##name(void) { \
204 return (kh_##name##_t*)kcalloc(1, sizeof(kh_##name##_t)); \
206 SCOPE void kh_destroy_##name(kh_##name##_t *h) \
209 kfree((void *)h->keys); kfree(h->flags); \
210 kfree((void *)h->vals); \
214 SCOPE void kh_clear_##name(kh_##name##_t *h) \
216 if (h && h->flags) { \
217 memset(h->flags, 0xaa, __ac_fsize(h->n_buckets) * sizeof(khint32_t)); \
218 h->size = h->n_occupied = 0; \
221 SCOPE khint_t kh_get_##name(const kh_##name##_t *h, khkey_t key) \
223 if (h->n_buckets) { \
224 khint_t k, i, last, mask, step = 0; \
225 mask = h->n_buckets - 1; \
226 k = __hash_func(key); i = k & mask; \
228 while (!__ac_isempty(h->flags, i) && (__ac_isdel(h->flags, i) || !__hash_equal(h->keys[i], key))) { \
229 i = (i + (++step)) & mask; \
230 if (i == last) return h->n_buckets; \
232 return __ac_iseither(h->flags, i)? h->n_buckets : i; \
235 SCOPE int kh_resize_##name(kh_##name##_t *h, khint_t new_n_buckets) \
237 khint32_t *new_flags = 0; \
240 kroundup32(new_n_buckets); \
241 if (new_n_buckets < 4) new_n_buckets = 4; \
242 if (h->size >= (khint_t)(new_n_buckets * __ac_HASH_UPPER + 0.5)) j = 0; \
244 new_flags = (khint32_t*)kmalloc(__ac_fsize(new_n_buckets) * sizeof(khint32_t)); \
245 if (!new_flags) return -1; \
246 memset(new_flags, 0xaa, __ac_fsize(new_n_buckets) * sizeof(khint32_t)); \
247 if (h->n_buckets < new_n_buckets) { \
248 khkey_t *new_keys = (khkey_t*)krealloc((void *)h->keys, new_n_buckets * sizeof(khkey_t)); \
249 if (!new_keys) { kfree(new_flags); return -1; } \
250 h->keys = new_keys; \
252 khval_t *new_vals = (khval_t*)krealloc((void *)h->vals, new_n_buckets * sizeof(khval_t)); \
253 if (!new_vals) { kfree(new_flags); return -1; } \
254 h->vals = new_vals; \
260 for (j = 0; j != h->n_buckets; ++j) { \
261 if (__ac_iseither(h->flags, j) == 0) { \
262 khkey_t key = h->keys[j]; \
265 new_mask = new_n_buckets - 1; \
266 if (kh_is_map) val = h->vals[j]; \
267 __ac_set_isdel_true(h->flags, j); \
269 khint_t k, i, step = 0; \
270 k = __hash_func(key); \
272 while (!__ac_isempty(new_flags, i)) i = (i + (++step)) & new_mask; \
273 __ac_set_isempty_false(new_flags, i); \
274 if (i < h->n_buckets && __ac_iseither(h->flags, i) == 0) { \
275 { khkey_t tmp = h->keys[i]; h->keys[i] = key; key = tmp; } \
276 if (kh_is_map) { khval_t tmp = h->vals[i]; h->vals[i] = val; val = tmp; } \
277 __ac_set_isdel_true(h->flags, i); \
280 if (kh_is_map) h->vals[i] = val; \
286 if (h->n_buckets > new_n_buckets) { \
287 h->keys = (khkey_t*)krealloc((void *)h->keys, new_n_buckets * sizeof(khkey_t)); \
288 if (kh_is_map) h->vals = (khval_t*)krealloc((void *)h->vals, new_n_buckets * sizeof(khval_t)); \
291 h->flags = new_flags; \
292 h->n_buckets = new_n_buckets; \
293 h->n_occupied = h->size; \
294 h->upper_bound = (khint_t)(h->n_buckets * __ac_HASH_UPPER + 0.5); \
298 SCOPE khint_t kh_put_##name(kh_##name##_t *h, khkey_t key, int *ret) \
301 if (h->n_occupied >= h->upper_bound) { \
302 if (h->n_buckets > (h->size<<1)) { \
303 if (kh_resize_##name(h, h->n_buckets - 1) < 0) { \
304 *ret = -1; return h->n_buckets; \
306 } else if (kh_resize_##name(h, h->n_buckets + 1) < 0) { \
307 *ret = -1; return h->n_buckets; \
311 khint_t k, i, site, last, mask = h->n_buckets - 1, step = 0; \
312 x = site = h->n_buckets; k = __hash_func(key); i = k & mask; \
313 if (__ac_isempty(h->flags, i)) x = i; \
316 while (!__ac_isempty(h->flags, i) && (__ac_isdel(h->flags, i) || !__hash_equal(h->keys[i], key))) { \
317 if (__ac_isdel(h->flags, i)) site = i; \
318 i = (i + (++step)) & mask; \
319 if (i == last) { x = site; break; } \
321 if (x == h->n_buckets) { \
322 if (__ac_isempty(h->flags, i) && site != h->n_buckets) x = site; \
327 if (__ac_isempty(h->flags, x)) { \
329 __ac_set_isboth_false(h->flags, x); \
330 ++h->size; ++h->n_occupied; \
332 } else if (__ac_isdel(h->flags, x)) { \
334 __ac_set_isboth_false(h->flags, x); \
340 SCOPE void kh_del_##name(kh_##name##_t *h, khint_t x) \
342 if (x != h->n_buckets && !__ac_iseither(h->flags, x)) { \
343 __ac_set_isdel_true(h->flags, x); \
348#define KHASH_DECLARE(name, khkey_t, khval_t) \
349 __KHASH_TYPE(name, khkey_t, khval_t) \
350 __KHASH_PROTOTYPES(name, khkey_t, khval_t)
352#define KHASH_INIT2(name, SCOPE, khkey_t, khval_t, kh_is_map, __hash_func, __hash_equal) \
353 __KHASH_TYPE(name, khkey_t, khval_t) \
354 __KHASH_IMPL(name, SCOPE, khkey_t, khval_t, kh_is_map, __hash_func, __hash_equal)
356#define KHASH_INIT(name, khkey_t, khval_t, kh_is_map, __hash_func, __hash_equal) \
357 KHASH_INIT2(name, static kh_inline klib_unused, khkey_t, khval_t, kh_is_map, __hash_func, __hash_equal)
366#define kh_int_hash_func(key) (khint32_t)(key)
370#define kh_int_hash_equal(a, b) ((a) == (b))
376#define kh_int64_hash_func(key) (khint32_t)((key)>>33^(key)^(key)<<11)
380#define kh_int64_hash_equal(a, b) ((a) == (b))
397#define kh_str_hash_func(key) __ac_X31_hash_string(key)
401#define kh_str_hash_equal(a, b) (strcmp(a, b) == 0)
413#define kh_int_hash_func2(key) __ac_Wang_hash((khint_t)key)
423#define khash_t(name) kh_##name##_t
430#define kh_init(name) kh_init_##name()
437#define kh_destroy(name, h) kh_destroy_##name(h)
444#define kh_clear(name, h) kh_clear_##name(h)
452#define kh_resize(name, h, s) kh_resize_##name(h, s)
465#define kh_put(name, h, k, r) kh_put_##name(h, k, r)
474#define kh_get(name, h, k) kh_get_##name(h, k)
482#define kh_del(name, h, k) kh_del_##name(h, k)
490#define kh_exist(h, x) (!__ac_iseither((h)->flags, (x)))
498#define kh_key(h, x) ((h)->keys[x])
507#define kh_val(h, x) ((h)->vals[x])
512#define kh_value(h, x) ((h)->vals[x])
519#define kh_begin(h) (khint_t)(0)
526#define kh_end(h) ((h)->n_buckets)
533#define kh_size(h) ((h)->size)
540#define kh_n_buckets(h) ((h)->n_buckets)
549#define kh_foreach(h, kvar, vvar, code) { khint_t __i; \
550 for (__i = kh_begin(h); __i != kh_end(h); ++__i) { \
551 if (!kh_exist(h,__i)) continue; \
552 (kvar) = kh_key(h,__i); \
553 (vvar) = kh_val(h,__i); \
563#define kh_foreach_value(h, vvar, code) { khint_t __i; \
564 for (__i = kh_begin(h); __i != kh_end(h); ++__i) { \
565 if (!kh_exist(h,__i)) continue; \
566 (vvar) = kh_val(h,__i); \
576#define KHASH_SET_INIT_INT(name) \
577 KHASH_INIT(name, khint32_t, char, 0, kh_int_hash_func, kh_int_hash_equal)
584#define KHASH_MAP_INIT_INT(name, khval_t) \
585 KHASH_INIT(name, khint32_t, khval_t, 1, kh_int_hash_func, kh_int_hash_equal)
591#define KHASH_SET_INIT_INT64(name) \
592 KHASH_INIT(name, khint64_t, char, 0, kh_int64_hash_func, kh_int64_hash_equal)
599#define KHASH_MAP_INIT_INT64(name, khval_t) \
600 KHASH_INIT(name, khint64_t, khval_t, 1, kh_int64_hash_func, kh_int64_hash_equal)
607#define KHASH_SET_INIT_STR(name) \
608 KHASH_INIT(name, kh_cstr_t, char, 0, kh_str_hash_func, kh_str_hash_equal)
615#define KHASH_MAP_INIT_STR(name, khval_t) \
616 KHASH_INIT(name, kh_cstr_t, khval_t, 1, kh_str_hash_func, kh_str_hash_equal)
static const double __ac_HASH_UPPER
static khint_t __ac_X31_hash_string(const char *s)
static khint_t __ac_Wang_hash(khint_t key)