diff options
Diffstat (limited to 'libiberty/hashtab.c')
-rw-r--r-- | libiberty/hashtab.c | 68 |
1 files changed, 60 insertions, 8 deletions
diff --git a/libiberty/hashtab.c b/libiberty/hashtab.c index 0bc9e485ef0..16c5d3e4b12 100644 --- a/libiberty/hashtab.c +++ b/libiberty/hashtab.c @@ -144,6 +144,36 @@ htab_empty (htab) memset (htab->entries, 0, htab->size * sizeof (void *)); } +/* Similar to htab_find_slot, but without several unwanted side effects: + - Does not call htab->eq_f when it finds an existing entry. + - Does not change the count of elements/searches/collisions in the + hash table. + This function also assumes there are no deleted entries in the table. + HASH is the hash value for the element to be inserted. */ +static void ** +find_empty_slot_for_expand (htab, hash) + htab_t htab; + unsigned int hash; +{ + size_t size = htab->size; + unsigned int hash2 = 1 + hash % (size - 2); + unsigned int index = hash % size; + + for (;;) + { + void **slot = htab->entries + index; + if (*slot == EMPTY_ENTRY) + return slot; + + if (*slot == DELETED_ENTRY) + abort (); + + index += hash2; + if (index >= size) + index -= size; + } +} + /* The following function changes size of memory allocated for the entries and repeatedly inserts the table elements. The occupancy of the table after the call will be about 50%. Naturally the hash @@ -173,7 +203,7 @@ htab_expand (htab) void *x = *p; if (x != EMPTY_ENTRY && x != DELETED_ENTRY) { - void **q = htab_find_slot (htab, x, 1); + void **q = find_empty_slot_for_expand (htab, (*htab->hash_f) (x)); *q = x; } p++; @@ -186,16 +216,16 @@ htab_expand (htab) element. It cannot be used to insert or delete an element. */ void * -htab_find (htab, element) +htab_find_with_hash (htab, element, hash) htab_t htab; const void *element; + unsigned int hash; { - unsigned int index, hash, hash2; + unsigned int index, hash2; size_t size; htab->searches++; size = htab->size; - hash = (*htab->hash_f) (element); hash2 = 1 + hash % (size - 2); index = hash % size; @@ -214,6 +244,16 @@ htab_find (htab, element) } } +/* Like htab_find_slot_with_hash, but compute the hash value from the + element. */ +void * +htab_find (htab, element) + htab_t htab; + const void *element; +{ + return htab_find_with_hash (htab, element, (*htab->hash_f) (element)); +} + /* This function searches for a hash table slot containing an entry equal to the given element. To delete an entry, call this with INSERT = 0, then call htab_clear_slot on the slot returned (possibly @@ -221,20 +261,20 @@ htab_find (htab, element) INSERT = 1, then write the value you want into the returned slot. */ void ** -htab_find_slot (htab, element, insert) +htab_find_slot_with_hash (htab, element, hash, insert) htab_t htab; const void *element; + unsigned int hash; int insert; { void **first_deleted_slot; - unsigned int index, hash, hash2; + unsigned int index, hash2; size_t size; if (insert && htab->size * 3 <= htab->n_elements * 4) htab_expand (htab); size = htab->size; - hash = (*htab->hash_f) (element); hash2 = 1 + hash % (size - 2); index = hash % size; @@ -278,6 +318,18 @@ htab_find_slot (htab, element, insert) } } +/* Like htab_find_slot_with_hash, but compute the hash value from the + element. */ +void ** +htab_find_slot (htab, element, insert) + htab_t htab; + const void *element; + int insert; +{ + return htab_find_slot_with_hash (htab, element, (*htab->hash_f) (element), + insert); +} + /* This function deletes an element with the given value from hash table. If there is no matching element in the hash table, this function does nothing. */ @@ -336,7 +388,7 @@ htab_traverse (htab, callback, info) { void *x = *slot; if (x != EMPTY_ENTRY && x != DELETED_ENTRY) - if (!(*callback) (x, info)) + if (!(*callback) (slot, info)) break; } while (++slot < limit); |