| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339 | /* Hash table implementation. * * This file implements in memory hash tables with insert/del/replace/find/ * get-random-element operations. Hash tables will auto resize if needed * tables of power of two in size are used, collisions are handled by * chaining. See the source code for more information... :) * * Copyright (c) 2006-2010, Salvatore Sanfilippo <antirez at gmail dot com> * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * *   * Redistributions of source code must retain the above copyright notice, *     this list of conditions and the following disclaimer. *   * Redistributions in binary form must reproduce the above copyright *     notice, this list of conditions and the following disclaimer in the *     documentation and/or other materials provided with the distribution. *   * Neither the name of Redis nor the names of its contributors may be used *     to endorse or promote products derived from this software without *     specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */#include "fmacros.h"#include "alloc.h"#include <stdlib.h>#include <assert.h>#include <limits.h>#include "dict.h"/* -------------------------- private prototypes ---------------------------- */static int _dictExpandIfNeeded(dict *ht);static unsigned long _dictNextPower(unsigned long size);static int _dictKeyIndex(dict *ht, const void *key);static int _dictInit(dict *ht, dictType *type, void *privDataPtr);/* -------------------------- hash functions -------------------------------- *//* Generic hash function (a popular one from Bernstein). * I tested a few and this was the best. */static unsigned int dictGenHashFunction(const unsigned char *buf, int len) {    unsigned int hash = 5381;    while (len--)        hash = ((hash << 5) + hash) + (*buf++); /* hash * 33 + c */    return hash;}/* ----------------------------- API implementation ------------------------- *//* Reset an hashtable already initialized with ht_init(). * NOTE: This function should only called by ht_destroy(). */static void _dictReset(dict *ht) {    ht->table = NULL;    ht->size = 0;    ht->sizemask = 0;    ht->used = 0;}/* Create a new hash table */static dict *dictCreate(dictType *type, void *privDataPtr) {    dict *ht = hi_malloc(sizeof(*ht));    _dictInit(ht,type,privDataPtr);    return ht;}/* Initialize the hash table */static int _dictInit(dict *ht, dictType *type, void *privDataPtr) {    _dictReset(ht);    ht->type = type;    ht->privdata = privDataPtr;    return DICT_OK;}/* Expand or create the hashtable */static int dictExpand(dict *ht, unsigned long size) {    dict n; /* the new hashtable */    unsigned long realsize = _dictNextPower(size), i;    /* the size is invalid if it is smaller than the number of     * elements already inside the hashtable */    if (ht->used > size)        return DICT_ERR;    _dictInit(&n, ht->type, ht->privdata);    n.size = realsize;    n.sizemask = realsize-1;    n.table = calloc(realsize,sizeof(dictEntry*));    /* Copy all the elements from the old to the new table:     * note that if the old hash table is empty ht->size is zero,     * so dictExpand just creates an hash table. */    n.used = ht->used;    for (i = 0; i < ht->size && ht->used > 0; i++) {        dictEntry *he, *nextHe;        if (ht->table[i] == NULL) continue;        /* For each hash entry on this slot... */        he = ht->table[i];        while(he) {            unsigned int h;            nextHe = he->next;            /* Get the new element index */            h = dictHashKey(ht, he->key) & n.sizemask;            he->next = n.table[h];            n.table[h] = he;            ht->used--;            /* Pass to the next element */            he = nextHe;        }    }    assert(ht->used == 0);    free(ht->table);    /* Remap the new hashtable in the old */    *ht = n;    return DICT_OK;}/* Add an element to the target hash table */static int dictAdd(dict *ht, void *key, void *val) {    int index;    dictEntry *entry;    /* Get the index of the new element, or -1 if     * the element already exists. */    if ((index = _dictKeyIndex(ht, key)) == -1)        return DICT_ERR;    /* Allocates the memory and stores key */    entry = hi_malloc(sizeof(*entry));    entry->next = ht->table[index];    ht->table[index] = entry;    /* Set the hash entry fields. */    dictSetHashKey(ht, entry, key);    dictSetHashVal(ht, entry, val);    ht->used++;    return DICT_OK;}/* Add an element, discarding the old if the key already exists. * Return 1 if the key was added from scratch, 0 if there was already an * element with such key and dictReplace() just performed a value update * operation. */static int dictReplace(dict *ht, void *key, void *val) {    dictEntry *entry, auxentry;    /* Try to add the element. If the key     * does not exists dictAdd will succeed. */    if (dictAdd(ht, key, val) == DICT_OK)        return 1;    /* It already exists, get the entry */    entry = dictFind(ht, key);    /* Free the old value and set the new one */    /* Set the new value and free the old one. Note that it is important     * to do that in this order, as the value may just be exactly the same     * as the previous one. In this context, think to reference counting,     * you want to increment (set), and then decrement (free), and not the     * reverse. */    auxentry = *entry;    dictSetHashVal(ht, entry, val);    dictFreeEntryVal(ht, &auxentry);    return 0;}/* Search and remove an element */static int dictDelete(dict *ht, const void *key) {    unsigned int h;    dictEntry *de, *prevde;    if (ht->size == 0)        return DICT_ERR;    h = dictHashKey(ht, key) & ht->sizemask;    de = ht->table[h];    prevde = NULL;    while(de) {        if (dictCompareHashKeys(ht,key,de->key)) {            /* Unlink the element from the list */            if (prevde)                prevde->next = de->next;            else                ht->table[h] = de->next;            dictFreeEntryKey(ht,de);            dictFreeEntryVal(ht,de);            free(de);            ht->used--;            return DICT_OK;        }        prevde = de;        de = de->next;    }    return DICT_ERR; /* not found */}/* Destroy an entire hash table */static int _dictClear(dict *ht) {    unsigned long i;    /* Free all the elements */    for (i = 0; i < ht->size && ht->used > 0; i++) {        dictEntry *he, *nextHe;        if ((he = ht->table[i]) == NULL) continue;        while(he) {            nextHe = he->next;            dictFreeEntryKey(ht, he);            dictFreeEntryVal(ht, he);            free(he);            ht->used--;            he = nextHe;        }    }    /* Free the table and the allocated cache structure */    free(ht->table);    /* Re-initialize the table */    _dictReset(ht);    return DICT_OK; /* never fails */}/* Clear & Release the hash table */static void dictRelease(dict *ht) {    _dictClear(ht);    free(ht);}static dictEntry *dictFind(dict *ht, const void *key) {    dictEntry *he;    unsigned int h;    if (ht->size == 0) return NULL;    h = dictHashKey(ht, key) & ht->sizemask;    he = ht->table[h];    while(he) {        if (dictCompareHashKeys(ht, key, he->key))            return he;        he = he->next;    }    return NULL;}static dictIterator *dictGetIterator(dict *ht) {    dictIterator *iter = hi_malloc(sizeof(*iter));    iter->ht = ht;    iter->index = -1;    iter->entry = NULL;    iter->nextEntry = NULL;    return iter;}static dictEntry *dictNext(dictIterator *iter) {    while (1) {        if (iter->entry == NULL) {            iter->index++;            if (iter->index >=                    (signed)iter->ht->size) break;            iter->entry = iter->ht->table[iter->index];        } else {            iter->entry = iter->nextEntry;        }        if (iter->entry) {            /* We need to save the 'next' here, the iterator user             * may delete the entry we are returning. */            iter->nextEntry = iter->entry->next;            return iter->entry;        }    }    return NULL;}static void dictReleaseIterator(dictIterator *iter) {    free(iter);}/* ------------------------- private functions ------------------------------ *//* Expand the hash table if needed */static int _dictExpandIfNeeded(dict *ht) {    /* If the hash table is empty expand it to the initial size,     * if the table is "full" dobule its size. */    if (ht->size == 0)        return dictExpand(ht, DICT_HT_INITIAL_SIZE);    if (ht->used == ht->size)        return dictExpand(ht, ht->size*2);    return DICT_OK;}/* Our hash table capability is a power of two */static unsigned long _dictNextPower(unsigned long size) {    unsigned long i = DICT_HT_INITIAL_SIZE;    if (size >= LONG_MAX) return LONG_MAX;    while(1) {        if (i >= size)            return i;        i *= 2;    }}/* Returns the index of a free slot that can be populated with * an hash entry for the given 'key'. * If the key already exists, -1 is returned. */static int _dictKeyIndex(dict *ht, const void *key) {    unsigned int h;    dictEntry *he;    /* Expand the hashtable if needed */    if (_dictExpandIfNeeded(ht) == DICT_ERR)        return -1;    /* Compute the key hash value */    h = dictHashKey(ht, key) & ht->sizemask;    /* Search if this slot does not already contain the given key */    he = ht->table[h];    while(he) {        if (dictCompareHashKeys(ht, key, he->key))            return -1;        he = he->next;    }    return h;}
 |