void *data;
unsigned data_len;
unsigned keylen;
+ unsigned hash;
char key[];
};
struct dm_hash_table {
unsigned num_nodes;
- unsigned num_slots;
+ unsigned num_hint;
+ unsigned mask_slots; /* (slots - 1) -> used as hash mask */
+ unsigned collisions; /* Collissions of hash keys */
+ unsigned search; /* How many keys were searched */
+ unsigned found; /* How many nodes were found */
+ unsigned same_hash; /* Was there a colision with same masked hash and len ? */
struct dm_hash_node **slots;
};
unsigned new_size = 16u;
struct dm_hash_table *hc = zalloc(sizeof(*hc));
- if (!hc)
- return_0;
+ if (!hc) {
+ log_error("Failed to allocate memory for hash.");
+ return 0;
+ }
+
+ hc->num_hint = size_hint;
/* round size hint up to a power of two */
while (new_size < size_hint)
new_size = new_size << 1;
- hc->num_slots = new_size;
+ hc->mask_slots = new_size - 1;
len = sizeof(*(hc->slots)) * new_size;
- if (!(hc->slots = zalloc(len)))
- goto_bad;
+ if (!(hc->slots = zalloc(len))) {
+ free(hc);
+ log_error("Failed to allocate slots for hash.");
+ return 0;
+ }
return hc;
-
- bad:
- free(hc->slots);
- free(hc);
- return 0;
}
static void _free_nodes(struct dm_hash_table *t)
struct dm_hash_node *c, *n;
unsigned i;
- for (i = 0; i < t->num_slots; i++)
+#ifdef DEBUG
+ log_debug("Free hash hint:%d slots:%d nodes:%d (s:%d f:%d c:%d h:%d)",
+ t->num_hint, t->mask_slots + 1, t->num_nodes,
+ t->search, t->found, t->collisions, t->same_hash);
+#endif
+
+ if (!t->num_nodes)
+ return;
+
+ for (i = 0; i <= t->mask_slots; i++)
for (c = t->slots[i]; c; c = n) {
n = c->next;
free(c);
free(t);
}
-static struct dm_hash_node **_find(struct dm_hash_table *t, const void *key,
- uint32_t len)
+static struct dm_hash_node **_findh(struct dm_hash_table *t, const void *key,
+ uint32_t len, unsigned hash)
{
- unsigned h = _hash(key, len) & (t->num_slots - 1);
struct dm_hash_node **c;
- for (c = &t->slots[h]; *c; c = &((*c)->next)) {
- if ((*c)->keylen != len)
- continue;
-
- if (!memcmp(key, (*c)->key, len))
- break;
+ ++t->search;
+ for (c = &t->slots[hash & t->mask_slots]; *c; c = &((*c)->next)) {
+ if ((*c)->keylen == len && (*c)->hash == hash) {
+ if (!memcmp(key, (*c)->key, len)) {
+ ++t->found;
+ break;
+ }
+ ++t->same_hash;
+ }
+ ++t->collisions;
}
return c;
}
+static struct dm_hash_node **_find(struct dm_hash_table *t, const void *key,
+ uint32_t len)
+{
+ return _findh(t, key, len, _hash(key, len));
+}
+
void *dm_hash_lookup_binary(struct dm_hash_table *t, const void *key,
uint32_t len)
{
int dm_hash_insert_binary(struct dm_hash_table *t, const void *key,
uint32_t len, void *data)
{
- struct dm_hash_node **c = _find(t, key, len);
+ unsigned hash = _hash(key, len);
+ struct dm_hash_node **c = _findh(t, key, len, hash);
if (*c)
(*c)->data = data;
return 0;
n->data = data;
+ n->hash = hash;
n->next = 0;
*c = n;
t->num_nodes++;
struct dm_hash_node **c;
unsigned h;
- h = _hash(key, len) & (t->num_slots - 1);
+ h = _hash(key, len) & t->mask_slots;
for (c = &t->slots[h]; *c; c = &((*c)->next)) {
if ((*c)->keylen != len)
n->data = (void *)val;
n->data_len = val_len;
- h = _hash(key, len) & (t->num_slots - 1);
+ h = _hash(key, len) & t->mask_slots;
first = t->slots[h];
*count = 0;
- h = _hash(key, len) & (t->num_slots - 1);
+ h = _hash(key, len) & t->mask_slots;
for (c = &t->slots[h]; *c; c = &((*c)->next)) {
if ((*c)->keylen != len)
struct dm_hash_node *c, *n;
unsigned i;
- for (i = 0; i < t->num_slots; i++)
+ for (i = 0; i <= t->mask_slots; i++)
for (c = t->slots[i]; c; c = n) {
n = c->next;
f(c->data);
void dm_hash_wipe(struct dm_hash_table *t)
{
_free_nodes(t);
- memset(t->slots, 0, sizeof(struct dm_hash_node *) * t->num_slots);
- t->num_nodes = 0u;
+ memset(t->slots, 0, sizeof(struct dm_hash_node *) * (t->mask_slots + 1));
+ t->num_nodes = t->collisions = t->search = t->same_hash = 0u;
}
char *dm_hash_get_key(struct dm_hash_table *t __attribute__((unused)),
struct dm_hash_node *c = NULL;
unsigned i;
- for (i = s; i < t->num_slots && !c; i++)
+ for (i = s; i <= t->mask_slots && !c; i++)
c = t->slots[i];
return c;
struct dm_hash_node *dm_hash_get_next(struct dm_hash_table *t, struct dm_hash_node *n)
{
- unsigned h = _hash(n->key, n->keylen) & (t->num_slots - 1);
-
- return n->next ? n->next : _next_slot(t, h + 1);
+ return n->next ? n->next : _next_slot(t, (n->hash & t->mask_slots) + 1);
}