summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--libihash/ihash.c60
-rw-r--r--libihash/ihash.h36
2 files changed, 84 insertions, 12 deletions
diff --git a/libihash/ihash.c b/libihash/ihash.c
index 598d3412..451f8db6 100644
--- a/libihash/ihash.c
+++ b/libihash/ihash.c
@@ -1,5 +1,5 @@
/* ihash.c - Integer-keyed hash table functions.
- Copyright (C) 1993-1997, 2001, 2003, 2004, 2006
+ Copyright (C) 1993-1997, 2001, 2003, 2004, 2006, 2014, 2015
Free Software Foundation, Inc.
Written by Michael I. Bushnell.
Revised by Miles Bader <miles@gnu.org>.
@@ -32,6 +32,23 @@
#include "ihash.h"
+/* This function is used to hash the key. */
+static inline hurd_ihash_key_t
+hash (hurd_ihash_t ht, hurd_ihash_key_t k)
+{
+ return ht->fct_hash ? ht->fct_hash ((const void *) k) : k;
+}
+
+/* This function is used to compare the key. Returns true if A is
+ equal to B. */
+static inline int
+compare (hurd_ihash_t ht, hurd_ihash_key_t a, hurd_ihash_key_t b)
+{
+ return
+ ht->fct_cmp ? (a && ht->fct_cmp ((const void *) a, (const void *) b))
+ : a == b;
+}
+
/* Return 1 if the slot with the index IDX in the hash table HT is
empty, and 0 otherwise. */
static inline int
@@ -46,7 +63,7 @@ index_empty (hurd_ihash_t ht, unsigned int idx)
static inline int
index_valid (hurd_ihash_t ht, unsigned int idx, hurd_ihash_key_t key)
{
- return !index_empty (ht, idx) && ht->items[idx].key == key;
+ return !index_empty (ht, idx) && compare (ht, ht->items[idx].key, key);
}
@@ -60,9 +77,10 @@ find_index (hurd_ihash_t ht, hurd_ihash_key_t key)
unsigned int up_idx;
unsigned int mask = ht->size - 1;
- idx = key & mask;
+ idx = hash (ht, key) & mask;
- if (ht->items[idx].value == _HURD_IHASH_EMPTY || ht->items[idx].key == key)
+ if (ht->items[idx].value == _HURD_IHASH_EMPTY
+ || compare (ht, ht->items[idx].key, key))
return idx;
up_idx = idx;
@@ -71,7 +89,7 @@ find_index (hurd_ihash_t ht, hurd_ihash_key_t key)
{
up_idx = (up_idx + 1) & mask;
if (ht->items[up_idx].value == _HURD_IHASH_EMPTY
- || ht->items[up_idx].key == key)
+ || compare (ht, ht->items[up_idx].key, key))
return up_idx;
}
while (up_idx != idx);
@@ -88,9 +106,11 @@ find_index (hurd_ihash_t ht, hurd_ihash_key_t key)
static inline void
locp_remove (hurd_ihash_t ht, hurd_ihash_locp_t locp)
{
+ struct _hurd_ihash_item *item = locp;
if (ht->cleanup)
- (*ht->cleanup) (*locp, ht->cleanup_data);
- *locp = _HURD_IHASH_DELETED;
+ (*ht->cleanup) (item->value, ht->cleanup_data);
+ item->value = _HURD_IHASH_DELETED;
+ item->key = 0;
ht->nr_items--;
}
@@ -106,6 +126,8 @@ hurd_ihash_init (hurd_ihash_t ht, intptr_t locp_offs)
ht->locp_offset = locp_offs;
ht->max_load = HURD_IHASH_MAX_LOAD_DEFAULT;
ht->cleanup = 0;
+ ht->fct_hash = NULL;
+ ht->fct_cmp = NULL;
}
@@ -166,6 +188,21 @@ hurd_ihash_set_cleanup (hurd_ihash_t ht, hurd_ihash_cleanup_t cleanup,
}
+/* Use the generalized key interface. Must be called before any item
+ is inserted into the table. */
+void
+hurd_ihash_set_gki (hurd_ihash_t ht,
+ hurd_ihash_fct_hash_t fct_hash,
+ hurd_ihash_fct_cmp_t fct_cmp)
+{
+ assert (ht->size == 0 || !"called after insertion");
+ assert (fct_hash);
+ assert (fct_cmp);
+ ht->fct_hash = fct_hash;
+ ht->fct_cmp = fct_cmp;
+}
+
+
/* Set the maximum load factor in binary percent to MAX_LOAD, which
should be between 64 and 128. The default is
HURD_IHASH_MAX_LOAD_DEFAULT. New elements are only added to the
@@ -199,10 +236,11 @@ add_one (hurd_ihash_t ht, hurd_ihash_key_t key, hurd_ihash_value_t value)
unsigned int first_free;
unsigned int mask = ht->size - 1;
- idx = key & mask;
+ idx = hash (ht, key) & mask;
first_free = idx;
- if (ht->items[idx].value != _HURD_IHASH_EMPTY && ht->items[idx].key != key)
+ if (ht->items[idx].value != _HURD_IHASH_EMPTY
+ && ! compare (ht, ht->items[idx].key, key))
{
unsigned int up_idx = idx;
@@ -210,7 +248,7 @@ add_one (hurd_ihash_t ht, hurd_ihash_key_t key, hurd_ihash_value_t value)
{
up_idx = (up_idx + 1) & mask;
if (ht->items[up_idx].value == _HURD_IHASH_EMPTY
- || ht->items[up_idx].key == key)
+ || compare (ht, ht->items[up_idx].key, key))
{
idx = up_idx;
break;
@@ -278,7 +316,7 @@ hurd_ihash_locp_add (hurd_ihash_t ht, hurd_ihash_locp_t locp,
}
else
{
- assert (item->key == key);
+ assert (compare (ht, item->key, key));
if (ht->cleanup)
(*ht->cleanup) (locp, ht->cleanup_data);
}
diff --git a/libihash/ihash.h b/libihash/ihash.h
index fdfc3673..28fefe80 100644
--- a/libihash/ihash.h
+++ b/libihash/ihash.h
@@ -1,5 +1,5 @@
/* ihash.h - Integer keyed hash table interface.
- Copyright (C) 1995, 2003, 2004 Free Software Foundation, Inc.
+ Copyright (C) 1995, 2003, 2004, 2014, 2015 Free Software Foundation, Inc.
Written by Miles Bader <miles@gnu.org>.
Revised by Marcus Brinkmann <marcus@gnu.org>.
@@ -57,6 +57,20 @@ typedef uintptr_t hurd_ihash_key_t;
typedef hurd_ihash_value_t *hurd_ihash_locp_t;
+/* We support non-integer keys using the generalized key interface.
+
+ To use it, supply a pair of functions matching the following
+ specification, and use pointers to the key instead of the key
+ itself in all calls to libihash. */
+
+/* The type of a function computing a hash for the given key. */
+typedef hurd_ihash_key_t (*hurd_ihash_fct_hash_t) (const void *);
+
+/* The type of a function comparing two given keys. Return true if
+ both keys are equal. */
+typedef int (*hurd_ihash_fct_cmp_t) (const void *, const void *);
+
+
/* The type of the cleanup function, which is called for every value
removed from the hash table. */
typedef void (*hurd_ihash_cleanup_t) (hurd_ihash_value_t value, void *arg);
@@ -95,6 +109,10 @@ struct hurd_ihash
second argument. This does not happen if CLEANUP is NULL. */
hurd_ihash_cleanup_t cleanup;
void *cleanup_data;
+
+ /* User-supplied functions for the generalized key interface. */
+ hurd_ihash_fct_hash_t fct_hash;
+ hurd_ihash_fct_cmp_t fct_cmp;
};
typedef struct hurd_ihash *hurd_ihash_t;
@@ -118,6 +136,16 @@ typedef struct hurd_ihash *hurd_ihash_t;
.max_load = HURD_IHASH_MAX_LOAD_DEFAULT, \
.locp_offset = (locp_offs)}
+#define HURD_IHASH_INITIALIZER_GKI(locp_offs, f_clean, f_clean_data, \
+ f_hash, f_compare) \
+ { .nr_items = 0, .size = 0, \
+ .cleanup = (f_clean), \
+ .cleanup_data = (f_clean_data), \
+ .max_load = HURD_IHASH_MAX_LOAD_DEFAULT, \
+ .locp_offset = (locp_offs), \
+ .fct_hash = (f_hash), \
+ .fct_cmp = (f_compare)} \
+
/* Initialize the hash table at address HT. If LOCP_OFFSET is not
HURD_IHASH_NO_LOCP, then this is an offset (in bytes) from the
address of a hash value where a location pointer can be found. The
@@ -152,6 +180,12 @@ void hurd_ihash_free (hurd_ihash_t ht);
void hurd_ihash_set_cleanup (hurd_ihash_t ht, hurd_ihash_cleanup_t cleanup,
void *cleanup_data);
+/* Use the generalized key interface. Must be called before any item
+ is inserted into the table. */
+void hurd_ihash_set_gki (hurd_ihash_t ht,
+ hurd_ihash_fct_hash_t fct_hash,
+ hurd_ihash_fct_cmp_t fct_cmp);
+
/* Set the maximum load factor in binary percent to MAX_LOAD, which
should be between 64 and 128. The default is
HURD_IHASH_MAX_LOAD_DEFAULT. New elements are only added to the