23 #ifndef KHASH_DEFAULT_SIZE 24 # define KHASH_DEFAULT_SIZE 32 26 #define KHASH_MIN_SIZE 8 28 #define UPPER_BOUND(x) ((x)>>2|(x)>>1) 33 static const uint8_t __m_empty[] = {0x02, 0x08, 0x20, 0x80};
34 static const uint8_t __m_del[] = {0x01, 0x04, 0x10, 0x40};
35 static const uint8_t __m_either[] = {0x03, 0x0c, 0x30, 0xc0};
38 #define __ac_isempty(ed_flag, i) (ed_flag[(i)/4]&__m_empty[(i)%4]) 39 #define __ac_isdel(ed_flag, i) (ed_flag[(i)/4]&__m_del[(i)%4]) 40 #define __ac_iseither(ed_flag, i) (ed_flag[(i)/4]&__m_either[(i)%4]) 41 #define khash_power2(v) do { \ 50 #define khash_mask(h) ((h)->n_buckets-1) 51 #define khash_upper_bound(h) (UPPER_BOUND((h)->n_buckets)) 60 #define KHASH_DECLARE(name, khkey_t, khval_t, kh_is_map) \ 61 typedef struct kh_##name { \ 69 void kh_alloc_##name(mrb_state *mrb, kh_##name##_t *h); \ 70 kh_##name##_t *kh_init_##name##_size(mrb_state *mrb, khint_t size); \ 71 kh_##name##_t *kh_init_##name(mrb_state *mrb); \ 72 void kh_destroy_##name(mrb_state *mrb, kh_##name##_t *h); \ 73 void kh_clear_##name(mrb_state *mrb, kh_##name##_t *h); \ 74 khint_t kh_get_##name(mrb_state *mrb, kh_##name##_t *h, khkey_t key); \ 75 khint_t kh_put_##name(mrb_state *mrb, kh_##name##_t *h, khkey_t key, int *ret); \ 76 void kh_resize_##name(mrb_state *mrb, kh_##name##_t *h, khint_t new_n_buckets); \ 77 void kh_del_##name(mrb_state *mrb, kh_##name##_t *h, khint_t x); \ 78 kh_##name##_t *kh_copy_##name(mrb_state *mrb, kh_##name##_t *h); 81 kh_fill_flags(uint8_t *p, uint8_t c,
size_t len)
97 #define KHASH_DEFINE(name, khkey_t, khval_t, kh_is_map, __hash_func, __hash_equal) \ 98 void kh_alloc_##name(mrb_state *mrb, kh_##name##_t *h) \ 100 khint_t sz = h->n_buckets; \ 101 size_t len = sizeof(khkey_t) + (kh_is_map ? sizeof(khval_t) : 0); \ 102 uint8_t *p = (uint8_t*)mrb_malloc(mrb, sizeof(uint8_t)*sz/4+len*sz); \ 103 h->size = h->n_occupied = 0; \ 104 h->keys = (khkey_t *)p; \ 105 h->vals = kh_is_map ? (khval_t *)(p+sizeof(khkey_t)*sz) : NULL; \ 106 h->ed_flags = p+len*sz; \ 107 kh_fill_flags(h->ed_flags, 0xaa, sz/4); \ 109 kh_##name##_t *kh_init_##name##_size(mrb_state *mrb, khint_t size) { \ 110 kh_##name##_t *h = (kh_##name##_t*)mrb_calloc(mrb, 1, sizeof(kh_##name##_t)); \ 111 if (size < KHASH_MIN_SIZE) \ 112 size = KHASH_MIN_SIZE; \ 113 khash_power2(size); \ 114 h->n_buckets = size; \ 115 kh_alloc_##name(mrb, h); \ 118 kh_##name##_t *kh_init_##name(mrb_state *mrb) { \ 119 return kh_init_##name##_size(mrb, KHASH_DEFAULT_SIZE); \ 121 void kh_destroy_##name(mrb_state *mrb, kh_##name##_t *h) \ 124 mrb_free(mrb, h->keys); \ 128 void kh_clear_##name(mrb_state *mrb, kh_##name##_t *h) \ 131 if (h && h->ed_flags) { \ 132 kh_fill_flags(h->ed_flags, 0xaa, h->n_buckets/4); \ 133 h->size = h->n_occupied = 0; \ 136 khint_t kh_get_##name(mrb_state *mrb, kh_##name##_t *h, khkey_t key) \ 138 khint_t k = __hash_func(mrb,key) & khash_mask(h), step = 0; \ 140 while (!__ac_isempty(h->ed_flags, k)) { \ 141 if (!__ac_isdel(h->ed_flags, k)) { \ 142 if (__hash_equal(mrb,h->keys[k], key)) return k; \ 144 k = (k+(++step)) & khash_mask(h); \ 148 void kh_resize_##name(mrb_state *mrb, kh_##name##_t *h, khint_t new_n_buckets) \ 150 if (new_n_buckets < KHASH_MIN_SIZE) \ 151 new_n_buckets = KHASH_MIN_SIZE; \ 152 khash_power2(new_n_buckets); \ 155 uint8_t *old_ed_flags = h->ed_flags; \ 156 khkey_t *old_keys = h->keys; \ 157 khval_t *old_vals = h->vals; \ 158 khint_t old_n_buckets = h->n_buckets; \ 160 hh.n_buckets = new_n_buckets; \ 161 kh_alloc_##name(mrb, &hh); \ 163 for (i=0 ; i<old_n_buckets ; i++) { \ 164 if (!__ac_iseither(old_ed_flags, i)) { \ 165 khint_t k = kh_put_##name(mrb, &hh, old_keys[i], NULL); \ 166 if (kh_is_map) kh_value(&hh,k) = old_vals[i]; \ 171 mrb_free(mrb, old_keys); \ 174 khint_t kh_put_##name(mrb_state *mrb, kh_##name##_t *h, khkey_t key, int *ret) \ 176 khint_t k, del_k, step = 0; \ 177 if (h->n_occupied >= khash_upper_bound(h)) { \ 178 kh_resize_##name(mrb, h, h->n_buckets*2); \ 180 k = __hash_func(mrb,key) & khash_mask(h); \ 182 while (!__ac_isempty(h->ed_flags, k)) { \ 183 if (!__ac_isdel(h->ed_flags, k)) { \ 184 if (__hash_equal(mrb,h->keys[k], key)) { \ 189 else if (del_k == kh_end(h)) { \ 192 k = (k+(++step)) & khash_mask(h); \ 194 if (del_k != kh_end(h)) { \ 196 h->keys[del_k] = key; \ 197 h->ed_flags[del_k/4] &= ~__m_del[del_k%4]; \ 205 h->ed_flags[k/4] &= ~__m_empty[k%4]; \ 212 void kh_del_##name(mrb_state *mrb, kh_##name##_t *h, khint_t x) \ 215 mrb_assert(x != h->n_buckets && !__ac_iseither(h->ed_flags, x)); \ 216 h->ed_flags[x/4] |= __m_del[x%4]; \ 219 kh_##name##_t *kh_copy_##name(mrb_state *mrb, kh_##name##_t *h) \ 224 h2 = kh_init_##name(mrb); \ 225 for (k = kh_begin(h); k != kh_end(h); k++) { \ 226 if (kh_exist(h, k)) { \ 227 k2 = kh_put_##name(mrb, h2, kh_key(h, k), NULL); \ 228 if (kh_is_map) kh_value(h2, k2) = kh_value(h, k); \ 235 #define khash_t(name) kh_##name##_t 237 #define kh_init_size(name,mrb,size) kh_init_##name##_size(mrb,size) 238 #define kh_init(name,mrb) kh_init_##name(mrb) 239 #define kh_destroy(name, mrb, h) kh_destroy_##name(mrb, h) 240 #define kh_clear(name, mrb, h) kh_clear_##name(mrb, h) 241 #define kh_resize(name, mrb, h, s) kh_resize_##name(mrb, h, s) 242 #define kh_put(name, mrb, h, k) kh_put_##name(mrb, h, k, NULL) 243 #define kh_put2(name, mrb, h, k, r) kh_put_##name(mrb, h, k, r) 244 #define kh_get(name, mrb, h, k) kh_get_##name(mrb, h, k) 245 #define kh_del(name, mrb, h, k) kh_del_##name(mrb, h, k) 246 #define kh_copy(name, mrb, h) kh_copy_##name(mrb, h) 248 #define kh_exist(h, x) (!__ac_iseither((h)->ed_flags, (x))) 249 #define kh_key(h, x) ((h)->keys[x]) 250 #define kh_val(h, x) ((h)->vals[x]) 251 #define kh_value(h, x) ((h)->vals[x]) 252 #define kh_begin(h) (khint_t)(0) 253 #define kh_end(h) ((h)->n_buckets) 254 #define kh_size(h) ((h)->size) 255 #define kh_n_buckets(h) ((h)->n_buckets) 257 #define kh_int_hash_func(mrb,key) (khint_t)((key)^((key)<<2)^((key)>>2)) 258 #define kh_int_hash_equal(mrb,a, b) (a == b) 259 #define kh_int64_hash_func(mrb,key) (khint_t)((key)>>33^(key)^(key)<<11) 260 #define kh_int64_hash_equal(mrb,a, b) (a == b) 261 static inline khint_t __ac_X31_hash_string(
const char *s)
264 if (h)
for (++s ; *s; ++s) h = (h << 5) - h + *s;
267 #define kh_str_hash_func(mrb,key) __ac_X31_hash_string(key) 268 #define kh_str_hash_equal(mrb,a, b) (strcmp(a, b) == 0) 270 typedef const char *kh_cstr_t;
uint32_t khint_t
khash definitions used in mruby's hash table.
Definition: khash.h:20
#define MRB_BEGIN_DECL
Start declarations in C mode.
Definition: common.h:26
#define MRB_END_DECL
End declarations in C mode.
Definition: common.h:28
mruby common platform definition"