2019-04-13 13:52:29 +00:00
|
|
|
#include "hashmap.h"
|
|
|
|
#include <string.h>
|
|
|
|
|
2021-01-09 16:14:14 +00:00
|
|
|
struct ext2fs_hashmap {
|
|
|
|
uint32_t size;
|
|
|
|
uint32_t(*hash)(const void *key, size_t len);
|
|
|
|
void(*free)(void*);
|
|
|
|
struct ext2fs_hashmap_entry *first;
|
|
|
|
struct ext2fs_hashmap_entry *last;
|
|
|
|
#if __GNUC_PREREQ (4, 8)
|
|
|
|
#pragma GCC diagnostic push
|
|
|
|
#pragma GCC diagnostic ignored "-Wpedantic"
|
|
|
|
#endif
|
|
|
|
struct ext2fs_hashmap_entry *entries[0];
|
|
|
|
#if __GNUC_PREREQ (4, 8)
|
|
|
|
#pragma GCC diagnostic pop
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2019-04-13 13:52:29 +00:00
|
|
|
uint32_t ext2fs_djb2_hash(const void *str, size_t size)
|
|
|
|
{
|
|
|
|
int c;
|
|
|
|
const char *s = str;
|
|
|
|
uint32_t hash = 5381;
|
|
|
|
|
|
|
|
while (size-- > 0) {
|
|
|
|
c = *s++;
|
|
|
|
hash = ((hash << 5) + hash) + c;
|
|
|
|
}
|
|
|
|
return hash;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ext2fs_hashmap *ext2fs_hashmap_create(
|
|
|
|
uint32_t(*hash_fct)(const void*, size_t),
|
|
|
|
void(*free_fct)(void*), size_t size)
|
|
|
|
{
|
|
|
|
struct ext2fs_hashmap *h = calloc(sizeof(struct ext2fs_hashmap) +
|
|
|
|
sizeof(struct ext2fs_hashmap_entry) * size, 1);
|
2019-04-16 19:44:13 +00:00
|
|
|
if (!h)
|
|
|
|
return h;
|
2019-04-13 13:52:29 +00:00
|
|
|
h->size = size;
|
|
|
|
h->free = free_fct;
|
|
|
|
h->hash = hash_fct;
|
|
|
|
h->first = h->last = NULL;
|
|
|
|
return h;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ext2fs_hashmap_add(struct ext2fs_hashmap *h, void *data, const void *key,
|
|
|
|
size_t key_len)
|
|
|
|
{
|
|
|
|
uint32_t hash = h->hash(key, key_len) % h->size;
|
|
|
|
struct ext2fs_hashmap_entry *e = malloc(sizeof(*e));
|
|
|
|
|
2019-04-16 19:44:13 +00:00
|
|
|
if (!e)
|
|
|
|
return;
|
2019-04-13 13:52:29 +00:00
|
|
|
e->data = data;
|
|
|
|
e->key = key;
|
|
|
|
e->key_len = key_len;
|
|
|
|
e->next = h->entries[hash];
|
|
|
|
h->entries[hash] = e;
|
|
|
|
|
|
|
|
e->list_prev = NULL;
|
|
|
|
e->list_next = h->first;
|
|
|
|
if (h->first)
|
|
|
|
h->first->list_prev = e;
|
|
|
|
h->first = e;
|
|
|
|
if (!h->last)
|
|
|
|
h->last = e;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *ext2fs_hashmap_lookup(struct ext2fs_hashmap *h, const void *key,
|
|
|
|
size_t key_len)
|
|
|
|
{
|
|
|
|
struct ext2fs_hashmap_entry *iter;
|
|
|
|
uint32_t hash = h->hash(key, key_len) % h->size;
|
|
|
|
|
|
|
|
for (iter = h->entries[hash]; iter; iter = iter->next)
|
|
|
|
if (iter->key_len == key_len && !memcmp(iter->key, key, key_len))
|
|
|
|
return iter->data;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *ext2fs_hashmap_iter_in_order(struct ext2fs_hashmap *h,
|
|
|
|
struct ext2fs_hashmap_entry **it)
|
|
|
|
{
|
|
|
|
*it = *it ? (*it)->list_next : h->first;
|
|
|
|
return *it ? (*it)->data : NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ext2fs_hashmap_free(struct ext2fs_hashmap *h)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < h->size; ++i) {
|
|
|
|
struct ext2fs_hashmap_entry *it = h->entries[i];
|
|
|
|
while (it) {
|
|
|
|
struct ext2fs_hashmap_entry *tmp = it->next;
|
|
|
|
if (h->free)
|
|
|
|
h->free(it->data);
|
|
|
|
free(it);
|
|
|
|
it = tmp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(h);
|
|
|
|
}
|