2002-10-29 07:03:13 +03:00
|
|
|
/* Generic hash table
|
|
|
|
**
|
2002-07-09 16:24:59 +04:00
|
|
|
** Copyright 2001, Travis Geiselbrecht. All rights reserved.
|
|
|
|
** Distributed under the terms of the NewOS License.
|
|
|
|
*/
|
|
|
|
|
2002-10-30 02:07:06 +03:00
|
|
|
#include <malloc.h>
|
2002-07-09 16:24:59 +04:00
|
|
|
#include <debug.h>
|
|
|
|
#include <Errors.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <khash.h>
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
// ToDo: this file apparently contains two different hash implementations
|
|
|
|
// get rid of one of them, and update the external code.
|
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
struct hash_table {
|
|
|
|
struct hash_elem **table;
|
2002-10-29 07:03:13 +03:00
|
|
|
int next_ptr_offset;
|
|
|
|
unsigned int table_size;
|
|
|
|
int num_elems;
|
|
|
|
int flags;
|
|
|
|
int (*compare_func)(void *e, const void *key);
|
|
|
|
unsigned int (*hash_func)(void *e, const void *key, unsigned int range);
|
2002-07-09 16:24:59 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
// XXX gross hack
|
|
|
|
#define NEXT_ADDR(t, e) ((void *)(((unsigned long)(e)) + (t)->next_ptr_offset))
|
|
|
|
#define NEXT(t, e) ((void *)(*(unsigned long *)NEXT_ADDR(t, e)))
|
|
|
|
#define PUT_IN_NEXT(t, e, val) (*(unsigned long *)NEXT_ADDR(t, e) = (long)(val))
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
void *
|
|
|
|
hash_init(unsigned int table_size, int next_ptr_offset,
|
2002-07-09 16:24:59 +04:00
|
|
|
int compare_func(void *e, const void *key),
|
|
|
|
unsigned int hash_func(void *e, const void *key, unsigned int range))
|
|
|
|
{
|
|
|
|
struct hash_table *t;
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
t = (struct hash_table *)malloc(sizeof(struct hash_table));
|
2002-10-29 07:03:13 +03:00
|
|
|
if (t == NULL)
|
2002-07-09 16:24:59 +04:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
t->table = (struct hash_elem **)malloc(sizeof(void *) * table_size);
|
2002-11-28 05:25:04 +03:00
|
|
|
if (t->table == NULL) {
|
|
|
|
free(t);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
for (i = 0; i<table_size; i++)
|
2002-07-09 16:24:59 +04:00
|
|
|
t->table[i] = NULL;
|
|
|
|
t->table_size = table_size;
|
|
|
|
t->next_ptr_offset = next_ptr_offset;
|
|
|
|
t->flags = 0;
|
|
|
|
t->num_elems = 0;
|
|
|
|
t->compare_func = compare_func;
|
|
|
|
t->hash_func = hash_func;
|
|
|
|
|
|
|
|
// dprintf("hash_init: created table 0x%x, next_ptr_offset %d, compare_func 0x%x, hash_func 0x%x\n",
|
|
|
|
// t, next_ptr_offset, compare_func, hash_func);
|
|
|
|
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
int
|
|
|
|
hash_uninit(void *_hash_table)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
struct hash_table *t = (struct hash_table *)_hash_table;
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
if(t->num_elems > 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
free(t->table);
|
|
|
|
free(t);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
int
|
|
|
|
hash_insert(void *_hash_table, void *e)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
struct hash_table *t = (struct hash_table *)_hash_table;
|
|
|
|
unsigned int hash;
|
|
|
|
|
|
|
|
// dprintf("hash_insert: table 0x%x, element 0x%x\n", t, e);
|
|
|
|
|
|
|
|
hash = t->hash_func(e, NULL, t->table_size);
|
|
|
|
PUT_IN_NEXT(t, e, t->table[hash]);
|
|
|
|
t->table[hash] = (struct hash_elem *)e;
|
|
|
|
t->num_elems++;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
int hash_remove(void *_hash_table, void *e)
|
|
|
|
{
|
|
|
|
struct hash_table *t = (struct hash_table *)_hash_table;
|
|
|
|
void *i, *last_i;
|
|
|
|
unsigned int hash;
|
|
|
|
|
|
|
|
hash = t->hash_func(e, NULL, t->table_size);
|
|
|
|
last_i = NULL;
|
2002-10-29 07:03:13 +03:00
|
|
|
for (i = t->table[hash]; i != NULL; last_i = i, i = NEXT(t, i)) {
|
|
|
|
if (i == e) {
|
|
|
|
if (last_i != NULL)
|
2002-07-09 16:24:59 +04:00
|
|
|
PUT_IN_NEXT(t, last_i, NEXT(t, i));
|
|
|
|
else
|
|
|
|
t->table[hash] = (struct hash_elem *)NEXT(t, i);
|
|
|
|
t->num_elems--;
|
|
|
|
return B_NO_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-07-12 02:21:56 +04:00
|
|
|
return B_ERROR;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
void *
|
|
|
|
hash_find(void *_hash_table, void *e)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
struct hash_table *t = (struct hash_table *)_hash_table;
|
|
|
|
void *i;
|
|
|
|
unsigned int hash;
|
|
|
|
|
|
|
|
hash = t->hash_func(e, NULL, t->table_size);
|
2002-10-29 07:03:13 +03:00
|
|
|
for (i = t->table[hash]; i != NULL; i = NEXT(t, i)) {
|
|
|
|
if (i == e)
|
2002-07-09 16:24:59 +04:00
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
void *
|
|
|
|
hash_lookup(void *_hash_table, const void *key)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
struct hash_table *t = (struct hash_table *)_hash_table;
|
|
|
|
void *i;
|
|
|
|
unsigned int hash;
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
if (t->compare_func == NULL)
|
2002-07-09 16:24:59 +04:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
hash = t->hash_func(NULL, key, t->table_size);
|
2002-10-29 07:03:13 +03:00
|
|
|
for (i = t->table[hash]; i != NULL; i = NEXT(t, i)) {
|
|
|
|
if (t->compare_func(i, key) == 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
struct hash_iterator *
|
|
|
|
hash_open(void *_hash_table, struct hash_iterator *i)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
struct hash_table *t = (struct hash_table *)_hash_table;
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
if (i == NULL) {
|
2002-07-09 16:24:59 +04:00
|
|
|
i = (struct hash_iterator *)malloc(sizeof(struct hash_iterator));
|
2002-10-29 07:03:13 +03:00
|
|
|
if (i == NULL)
|
2002-07-09 16:24:59 +04:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
hash_rewind(t, i);
|
|
|
|
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
hash_close(void *_hash_table, struct hash_iterator *i, bool freeIterator)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2002-10-29 07:03:13 +03:00
|
|
|
if (freeIterator)
|
2002-07-09 16:24:59 +04:00
|
|
|
free(i);
|
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
hash_rewind(void *_hash_table, struct hash_iterator *i)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
i->ptr = NULL;
|
|
|
|
i->bucket = -1;
|
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
void *
|
|
|
|
hash_next(void *_hash_table, struct hash_iterator *i)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
struct hash_table *t = (struct hash_table *)_hash_table;
|
|
|
|
unsigned int index;
|
|
|
|
|
|
|
|
restart:
|
2002-10-29 07:03:13 +03:00
|
|
|
if (!i->ptr) {
|
|
|
|
for (index = (unsigned int)(i->bucket + 1); index < t->table_size; index++) {
|
|
|
|
if (t->table[index]) {
|
2002-07-09 16:24:59 +04:00
|
|
|
i->bucket = index;
|
|
|
|
i->ptr = t->table[index];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
i->ptr = NEXT(t, i->ptr);
|
2002-10-29 07:03:13 +03:00
|
|
|
if (!i->ptr)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto restart;
|
|
|
|
}
|
|
|
|
|
|
|
|
return i->ptr;
|
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
unsigned int
|
|
|
|
hash_hash_str( const char *str )
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
char ch;
|
|
|
|
unsigned int hash = 0;
|
2002-10-29 07:03:13 +03:00
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
// we assume hash to be at least 32 bits
|
2002-10-29 07:03:13 +03:00
|
|
|
while ((ch = *str++) != 0) {
|
2002-07-09 16:24:59 +04:00
|
|
|
hash ^= hash >> 28;
|
|
|
|
hash <<= 4;
|
|
|
|
hash ^= ch;
|
|
|
|
}
|
2002-10-29 07:03:13 +03:00
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
return hash;
|
|
|
|
}
|
2002-07-19 05:27:22 +04:00
|
|
|
|
2002-11-28 20:34:55 +03:00
|
|
|
|
|
|
|
|
|
|
|
|
2002-07-19 05:27:22 +04:00
|
|
|
#define MAX_INITIAL 15;
|
|
|
|
|
|
|
|
/*
|
|
|
|
static void nhash_this(hash_table_index *hi, const void **key, ssize_t *klen,
|
|
|
|
void **val)
|
|
|
|
{
|
|
|
|
if (key) *key = hi->this->key;
|
|
|
|
if (klen) *klen = hi->this->klen;
|
|
|
|
if (val) *val = (void*)hi->this->val;
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
new_hash_table *
|
|
|
|
hash_make(void)
|
2002-07-19 05:27:22 +04:00
|
|
|
{
|
2002-11-28 05:25:04 +03:00
|
|
|
status_t rv;
|
2002-07-19 05:27:22 +04:00
|
|
|
new_hash_table *nn;
|
|
|
|
|
2002-10-30 02:07:06 +03:00
|
|
|
nn = (new_hash_table *)malloc(sizeof(new_hash_table));
|
2002-11-28 05:25:04 +03:00
|
|
|
if (nn == NULL)
|
2002-07-19 05:27:22 +04:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
nn->count = 0;
|
|
|
|
nn->max = MAX_INITIAL;
|
|
|
|
|
2002-10-30 02:07:06 +03:00
|
|
|
nn->array = (hash_entry **)malloc(sizeof(hash_entry) * (nn->max + 1));
|
2002-11-28 05:25:04 +03:00
|
|
|
if (nn == NULL) {
|
|
|
|
free(nn);
|
|
|
|
return NULL;
|
|
|
|
}
|
2002-07-19 05:27:22 +04:00
|
|
|
memset(nn->array, 0, sizeof(hash_entry) * (nn->max +1));
|
2002-11-28 05:25:04 +03:00
|
|
|
rv = pool_init(&nn->pool, sizeof(hash_entry));
|
|
|
|
if (rv < B_OK || nn->pool == NULL) {
|
|
|
|
free(nn->array);
|
|
|
|
free(nn);
|
2002-07-19 05:27:22 +04:00
|
|
|
return NULL;
|
2002-11-28 05:25:04 +03:00
|
|
|
}
|
2002-07-19 05:27:22 +04:00
|
|
|
return nn;
|
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
static hash_index *
|
|
|
|
new_hash_next(hash_index *hi)
|
2002-07-19 05:27:22 +04:00
|
|
|
{
|
|
|
|
hi->this_idx = hi->next;
|
|
|
|
while (!hi->this_idx) {
|
|
|
|
if (hi->index > hi->nh->max)
|
|
|
|
return NULL;
|
|
|
|
hi->this_idx = hi->nh->array[hi->index++];
|
|
|
|
}
|
|
|
|
hi->next = hi->this_idx->next;
|
|
|
|
return hi;
|
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
static hash_index *
|
|
|
|
new_hash_first(new_hash_table *nh)
|
2002-07-19 05:27:22 +04:00
|
|
|
{
|
|
|
|
hash_index *hi = &nh->iterator;
|
|
|
|
hi->nh = nh;
|
|
|
|
hi->index = 0;
|
|
|
|
hi->this_idx = hi->next = NULL;
|
|
|
|
return new_hash_next(hi);
|
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
static void
|
|
|
|
expand_array(new_hash_table *nh)
|
2002-07-19 05:27:22 +04:00
|
|
|
{
|
|
|
|
hash_index *hi;
|
|
|
|
hash_entry **new_array;
|
2002-11-28 05:25:04 +03:00
|
|
|
int new_max = (nh->max + 1) * 2 - 1;
|
2002-07-19 05:27:22 +04:00
|
|
|
int i;
|
|
|
|
|
2002-11-28 05:25:04 +03:00
|
|
|
new_array = (hash_entry **)malloc(sizeof(hash_entry) * new_max + 1);
|
|
|
|
if (new_array == NULL)
|
|
|
|
panic("khash, expand_array failed\n"); // XXX stupid, this function should return an error if it failes
|
|
|
|
memset(new_array, 0, sizeof(hash_entry) * new_max + 1);
|
2002-07-19 05:27:22 +04:00
|
|
|
for (hi = new_hash_first(nh); hi; hi = new_hash_next(hi)) {
|
|
|
|
i = hi->this_idx->hash & new_max;
|
|
|
|
hi->this_idx->next = new_array[i];
|
|
|
|
new_array[i] = hi->this_idx;
|
|
|
|
}
|
2002-10-30 02:07:06 +03:00
|
|
|
free(nh->array);
|
2002-07-19 05:27:22 +04:00
|
|
|
nh->array = new_array;
|
|
|
|
nh->max = new_max;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
static hash_entry **
|
|
|
|
find_entry(new_hash_table *nh, const void *key, ssize_t klen, const void *val)
|
2002-07-19 05:27:22 +04:00
|
|
|
{
|
|
|
|
hash_entry **hep;
|
|
|
|
hash_entry *he;
|
|
|
|
const unsigned char *p;
|
|
|
|
int hash = 0;
|
|
|
|
ssize_t i;
|
|
|
|
|
2002-11-28 20:34:55 +03:00
|
|
|
ASSERT(nh != NULL);
|
2002-07-19 05:27:22 +04:00
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
for (p = key, i = klen; i; i--, p++)
|
2002-07-19 05:27:22 +04:00
|
|
|
hash = hash * 33 + *p;
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
for (hep = &nh->array[hash & nh->max], he = *hep; he; hep = &he->next, he = *hep) {
|
2002-11-28 18:05:58 +03:00
|
|
|
// dprintf("khash, find_entry looking at hep %p, he %p\n", hep, he);
|
2002-11-28 20:34:55 +03:00
|
|
|
if (he->hash == hash && he->klen == klen && memcmp(he->key, key, klen) == 0) {
|
|
|
|
break;
|
2002-07-19 05:27:22 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-11-28 20:34:55 +03:00
|
|
|
if (val == 0) /* val == 0 means we only lookup, and don't insert */
|
|
|
|
return he ? hep : NULL;
|
2002-07-19 05:27:22 +04:00
|
|
|
|
|
|
|
/* add a new linked-list entry */
|
|
|
|
he = (hash_entry *)pool_get(nh->pool);
|
2002-11-28 20:34:55 +03:00
|
|
|
if (he == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* copy the key */
|
|
|
|
he->key = malloc(klen);
|
|
|
|
if (he->key == NULL) {
|
|
|
|
pool_put(nh->pool, he);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
memcpy(he->key, key, klen);
|
|
|
|
|
2002-07-19 05:27:22 +04:00
|
|
|
he->next = NULL;
|
|
|
|
he->hash = hash;
|
|
|
|
he->klen = klen;
|
|
|
|
he->val = val;
|
|
|
|
*hep = he;
|
|
|
|
nh->count++;
|
|
|
|
return hep;
|
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
void *
|
|
|
|
hash_get(new_hash_table *nh, const void *key, ssize_t klen)
|
2002-07-19 05:27:22 +04:00
|
|
|
{
|
2002-11-28 05:25:04 +03:00
|
|
|
hash_entry **hepp;
|
|
|
|
hash_entry *hep;
|
2002-11-28 20:34:55 +03:00
|
|
|
|
|
|
|
ASSERT(nh != NULL);
|
2002-11-28 05:25:04 +03:00
|
|
|
|
|
|
|
hepp = find_entry(nh, key, klen, NULL);
|
2002-11-28 18:05:58 +03:00
|
|
|
// dprintf("khash, find_entry returned %p\n", hepp);
|
2002-11-28 05:25:04 +03:00
|
|
|
if (hepp == NULL)
|
|
|
|
return NULL;
|
2002-10-29 07:03:13 +03:00
|
|
|
|
2002-11-28 05:25:04 +03:00
|
|
|
hep = *hepp;
|
|
|
|
if (hep == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return (void *) hep->val; /* XXX casting away the const */
|
2002-07-19 05:27:22 +04:00
|
|
|
}
|
|
|
|
|
2002-10-29 07:03:13 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
hash_set(new_hash_table *nh, const void *key, ssize_t klen, const void *val)
|
2002-07-19 05:27:22 +04:00
|
|
|
{
|
|
|
|
hash_entry **hep;
|
|
|
|
hash_entry *old;
|
2002-11-28 20:34:55 +03:00
|
|
|
|
|
|
|
ASSERT(nh != NULL);
|
|
|
|
|
2002-07-19 05:27:22 +04:00
|
|
|
hep = find_entry(nh, key, klen, val);
|
2002-11-28 20:34:55 +03:00
|
|
|
ASSERT(hep != NULL);
|
2002-07-19 05:27:22 +04:00
|
|
|
|
|
|
|
if (*hep) {
|
|
|
|
if (!val) {
|
|
|
|
/* delete it */
|
|
|
|
old = *hep;
|
|
|
|
*hep = (*hep)->next;
|
|
|
|
--nh->count;
|
|
|
|
pool_put(nh->pool, old);
|
|
|
|
} else {
|
|
|
|
/* replace it */
|
|
|
|
(*hep)->val = val;
|
|
|
|
if (nh->count > nh->max)
|
|
|
|
expand_array(nh);
|
|
|
|
}
|
|
|
|
}
|
2002-11-28 20:34:55 +03:00
|
|
|
else ASSERT(false);
|
2002-07-19 05:27:22 +04:00
|
|
|
}
|