Blame src/hash_table.c

Packit 8fb591
/**
Packit 8fb591
 * @file hash_table.c
Packit 8fb591
 * @author Radek Krejci <rkrejci@cesnet.cz>
Packit 8fb591
 * @brief libyang dictionary for storing strings and generic hash table
Packit 8fb591
 *
Packit 8fb591
 * Copyright (c) 2015 - 2018 CESNET, z.s.p.o.
Packit 8fb591
 *
Packit 8fb591
 * This source code is licensed under BSD 3-Clause License (the "License").
Packit 8fb591
 * You may not use this file except in compliance with the License.
Packit 8fb591
 * You may obtain a copy of the License at
Packit 8fb591
 *
Packit 8fb591
 *     https://opensource.org/licenses/BSD-3-Clause
Packit 8fb591
 */
Packit 8fb591
Packit 8fb591
#include <string.h>
Packit 8fb591
#include <stdint.h>
Packit 8fb591
#include <stdlib.h>
Packit 8fb591
#include <pthread.h>
Packit 8fb591
#include <assert.h>
Packit 8fb591
Packit 8fb591
#include "common.h"
Packit 8fb591
#include "context.h"
Packit 8fb591
#include "hash_table.h"
Packit 8fb591
Packit 8fb591
static int
Packit 8fb591
lydict_val_eq(void *val1_p, void *val2_p, int UNUSED(mod), void *cb_data)
Packit 8fb591
{
Packit 8fb591
    if (!val1_p || !val2_p) {
Packit 8fb591
        LOGARG;
Packit 8fb591
        return 0;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    const char *str1 = ((struct dict_rec *)val1_p)->value;
Packit 8fb591
    const char *str2 = ((struct dict_rec *)val2_p)->value;
Packit 8fb591
Packit 8fb591
    if (!str1 || !str2 || !cb_data) {
Packit 8fb591
        LOGARG;
Packit 8fb591
        return 0;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    if (strncmp(str1, str2, *(size_t *)cb_data) == 0) {
Packit 8fb591
        return 1;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    return 0;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
void
Packit 8fb591
lydict_init(struct dict_table *dict)
Packit 8fb591
{
Packit 8fb591
    if (!dict) {
Packit 8fb591
        LOGARG;
Packit 8fb591
        return;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    dict->hash_tab = lyht_new(1024, sizeof(struct dict_rec), lydict_val_eq, NULL, 1);
Packit 8fb591
    LY_CHECK_ERR_RETURN(!dict->hash_tab, LOGINT(NULL), );
Packit 8fb591
    pthread_mutex_init(&dict->lock, NULL);
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
void
Packit 8fb591
lydict_clean(struct dict_table *dict)
Packit 8fb591
{
Packit 8fb591
    unsigned int i;
Packit 8fb591
    struct dict_rec *dict_rec  = NULL;
Packit 8fb591
    struct ht_rec *rec = NULL;
Packit 8fb591
Packit 8fb591
    if (!dict) {
Packit 8fb591
        LOGARG;
Packit 8fb591
        return;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    for (i = 0; i < dict->hash_tab->size; i++) {
Packit 8fb591
        /* get ith record */
Packit 8fb591
        rec = (struct ht_rec *)&dict->hash_tab->recs[i * dict->hash_tab->rec_size];
Packit 8fb591
        if (rec->hits == 1) {
Packit 8fb591
            /*
Packit 8fb591
             * this should not happen, all records inserted into
Packit 8fb591
             * dictionary are supposed to be removed using lydict_remove()
Packit 8fb591
             * before calling lydict_clean()
Packit 8fb591
             */
Packit 8fb591
            dict_rec  = (struct dict_rec *)rec->val;
Packit 8fb591
            LOGWRN(NULL, "String \"%s\" not freed from the dictionary, refcount %d", dict_rec->value, dict_rec->refcount);
Packit 8fb591
            /* if record wasn't removed before free string allocated for that record */
Packit 8fb591
#ifdef NDEBUG
Packit 8fb591
            free(dict_rec->value);
Packit 8fb591
#endif
Packit 8fb591
        }
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    /* free table and destroy mutex */
Packit 8fb591
    lyht_free(dict->hash_tab);
Packit 8fb591
    pthread_mutex_destroy(&dict->lock);
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
/*
Packit 8fb591
 * Bob Jenkin's one-at-a-time hash
Packit 8fb591
 * http://www.burtleburtle.net/bob/hash/doobs.html
Packit 8fb591
 *
Packit 8fb591
 * Spooky hash is faster, but it works only for little endian architectures.
Packit 8fb591
 */
Packit 8fb591
static uint32_t
Packit 8fb591
dict_hash(const char *key, size_t len)
Packit 8fb591
{
Packit 8fb591
    uint32_t hash, i;
Packit 8fb591
Packit 8fb591
    for (hash = i = 0; i < len; ++i) {
Packit 8fb591
        hash += key[i];
Packit 8fb591
        hash += (hash << 10);
Packit 8fb591
        hash ^= (hash >> 6);
Packit 8fb591
    }
Packit 8fb591
    hash += (hash << 3);
Packit 8fb591
    hash ^= (hash >> 11);
Packit 8fb591
    hash += (hash << 15);
Packit 8fb591
    return hash;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
/*
Packit 8fb591
 * Usage:
Packit 8fb591
 * - init hash to 0
Packit 8fb591
 * - repeatedly call dict_hash_multi(), provide hash from the last call
Packit 8fb591
 * - call dict_hash_multi() with key_part = NULL to finish the hash
Packit 8fb591
 */
Packit 8fb591
uint32_t
Packit 8fb591
dict_hash_multi(uint32_t hash, const char *key_part, size_t len)
Packit 8fb591
{
Packit 8fb591
    uint32_t i;
Packit 8fb591
Packit 8fb591
    if (key_part) {
Packit 8fb591
        for (i = 0; i < len; ++i) {
Packit 8fb591
            hash += key_part[i];
Packit 8fb591
            hash += (hash << 10);
Packit 8fb591
            hash ^= (hash >> 6);
Packit 8fb591
        }
Packit 8fb591
    } else {
Packit 8fb591
        hash += (hash << 3);
Packit 8fb591
        hash ^= (hash >> 11);
Packit 8fb591
        hash += (hash << 15);
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    return hash;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
API void
Packit 8fb591
lydict_remove(struct ly_ctx *ctx, const char *value)
Packit 8fb591
{
Packit 8fb591
    size_t len;
Packit 8fb591
    int ret;
Packit 8fb591
    uint32_t hash;
Packit 8fb591
    struct dict_rec rec, *match = NULL;
Packit 8fb591
    char *val_p;
Packit 8fb591
Packit 8fb591
    if (!value || !ctx) {
Packit 8fb591
        return;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    len = strlen(value);
Packit 8fb591
    hash = dict_hash(value, len);
Packit 8fb591
Packit 8fb591
    /* create record for lyht_find call */
Packit 8fb591
    rec.value = (char *)value;
Packit 8fb591
    rec.refcount = 0;
Packit 8fb591
Packit 8fb591
    pthread_mutex_lock(&ctx->dict.lock);
Packit 8fb591
    /* set len as data for compare callback */
Packit 8fb591
    lyht_set_cb_data(ctx->dict.hash_tab, (void *)&len;;
Packit 8fb591
    /* check if value is already inserted */
Packit 8fb591
    ret = lyht_find(ctx->dict.hash_tab, &rec, hash, (void **)&match);
Packit 8fb591
Packit 8fb591
    if (ret == 0) {
Packit 8fb591
        LY_CHECK_ERR_GOTO(!match, LOGINT(ctx), finish);
Packit 8fb591
Packit 8fb591
        /* if value is already in dictionary, decrement reference counter */
Packit 8fb591
        match->refcount--;
Packit 8fb591
        if (match->refcount == 0) {
Packit 8fb591
            /*
Packit 8fb591
             * remove record
Packit 8fb591
             * save pointer to stored string before lyht_remove to
Packit 8fb591
             * free it after it is removed from hash table
Packit 8fb591
             */
Packit 8fb591
            val_p = match->value;
Packit 8fb591
            ret = lyht_remove(ctx->dict.hash_tab, &rec, hash);
Packit 8fb591
            free(val_p);
Packit 8fb591
            LY_CHECK_ERR_GOTO(ret, LOGINT(ctx), finish);
Packit 8fb591
        }
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
finish:
Packit 8fb591
    pthread_mutex_unlock(&ctx->dict.lock);
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
static char *
Packit 8fb591
dict_insert(struct ly_ctx *ctx, char *value, size_t len, int zerocopy)
Packit 8fb591
{
Packit 8fb591
    struct dict_rec *match = NULL, rec;
Packit 8fb591
    int ret = 0;
Packit 8fb591
    uint32_t hash;
Packit 8fb591
Packit 8fb591
    hash = dict_hash(value, len);
Packit 8fb591
    /* set len as data for compare callback */
Packit 8fb591
    lyht_set_cb_data(ctx->dict.hash_tab, (void *)&len;;
Packit 8fb591
    /* create record for lyht_insert */
Packit 8fb591
    rec.value = value;
Packit 8fb591
    rec.refcount = 1;
Packit 8fb591
Packit 8fb591
    LOGDBG(LY_LDGDICT, "inserting \"%s\"", rec.value);
Packit 8fb591
    ret = lyht_insert(ctx->dict.hash_tab, (void *)&rec, hash, (void **)&match);
Packit 8fb591
    if (ret == 1) {
Packit 8fb591
        match->refcount++;
Packit 8fb591
        if (zerocopy) {
Packit 8fb591
            free(value);
Packit 8fb591
        }
Packit 8fb591
    } else if (ret == 0) {
Packit 8fb591
        if (!zerocopy) {
Packit 8fb591
            /*
Packit 8fb591
             * allocate string for new record
Packit 8fb591
             * record is already inserted in hash table
Packit 8fb591
             */
Packit 8fb591
            match->value = malloc(sizeof *match->value * (len + 1));
Packit 8fb591
            LY_CHECK_ERR_RETURN(!match->value, LOGMEM(ctx), NULL);
Packit 8fb591
            memcpy(match->value, value, len);
Packit 8fb591
            match->value[len] = '\0';
Packit 8fb591
        }
Packit 8fb591
    } else {
Packit 8fb591
        /* lyht_insert returned error */
Packit 8fb591
        LOGINT(ctx);
Packit 8fb591
        return NULL;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    return match->value;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
API const char *
Packit 8fb591
lydict_insert(struct ly_ctx *ctx, const char *value, size_t len)
Packit 8fb591
{
Packit 8fb591
    const char *result;
Packit 8fb591
Packit 8fb591
    if (!value) {
Packit 8fb591
        return NULL;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    if (!len) {
Packit 8fb591
        len = strlen(value);
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    pthread_mutex_lock(&ctx->dict.lock);
Packit 8fb591
    result = dict_insert(ctx, (char *)value, len, 0);
Packit 8fb591
    pthread_mutex_unlock(&ctx->dict.lock);
Packit 8fb591
Packit 8fb591
    return result;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
API const char *
Packit 8fb591
lydict_insert_zc(struct ly_ctx *ctx, char *value)
Packit 8fb591
{
Packit 8fb591
    const char *result;
Packit 8fb591
Packit 8fb591
    if (!value) {
Packit 8fb591
        return NULL;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    pthread_mutex_lock(&ctx->dict.lock);
Packit 8fb591
    result = dict_insert(ctx, value, strlen(value), 1);
Packit 8fb591
    pthread_mutex_unlock(&ctx->dict.lock);
Packit 8fb591
Packit 8fb591
    return result;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
struct ht_rec *
Packit 8fb591
lyht_get_rec(unsigned char *recs, uint16_t rec_size, uint32_t idx)
Packit 8fb591
{
Packit 8fb591
    return (struct ht_rec *)&recs[idx * rec_size];
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
struct hash_table *
Packit 8fb591
lyht_new(uint32_t size, uint16_t val_size, values_equal_cb val_equal, void *cb_data, int resize)
Packit 8fb591
{
Packit 8fb591
    struct hash_table *ht;
Packit 8fb591
Packit 8fb591
    /* check that 2^x == size (power of 2) */
Packit 8fb591
    assert(size && !(size & (size - 1)));
Packit 8fb591
    assert(val_equal && val_size);
Packit 8fb591
    assert(resize == 0 || resize == 1);
Packit 8fb591
Packit 8fb591
    if (size < LYHT_MIN_SIZE) {
Packit 8fb591
        size = LYHT_MIN_SIZE;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    ht = malloc(sizeof *ht);
Packit 8fb591
    LY_CHECK_ERR_RETURN(!ht, LOGMEM(NULL), NULL);
Packit 8fb591
Packit 8fb591
    ht->used = 0;
Packit 8fb591
    ht->size = size;
Packit 8fb591
    ht->val_equal = val_equal;
Packit 8fb591
    ht->cb_data = cb_data;
Packit 8fb591
    ht->resize = (uint16_t)resize;
Packit 8fb591
Packit 8fb591
    ht->rec_size = (sizeof(struct ht_rec) - 1) + val_size;
Packit 8fb591
    /* allocate the records correctly */
Packit 8fb591
    ht->recs = calloc(size, ht->rec_size);
Packit 8fb591
    LY_CHECK_ERR_RETURN(!ht->recs, free(ht); LOGMEM(NULL), NULL);
Packit 8fb591
Packit 8fb591
    return ht;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
values_equal_cb
Packit 8fb591
lyht_set_cb(struct hash_table *ht, values_equal_cb new_val_equal)
Packit 8fb591
{
Packit 8fb591
    values_equal_cb prev;
Packit 8fb591
Packit 8fb591
    prev = ht->val_equal;
Packit 8fb591
    ht->val_equal = new_val_equal;
Packit 8fb591
    return prev;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
void *
Packit 8fb591
lyht_set_cb_data(struct hash_table *ht, void *new_cb_data)
Packit 8fb591
{
Packit 8fb591
    void *prev;
Packit 8fb591
Packit 8fb591
    prev = ht->cb_data;
Packit 8fb591
    ht->cb_data = new_cb_data;
Packit 8fb591
    return prev;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
struct hash_table *
Packit 8fb591
lyht_dup(const struct hash_table *orig)
Packit 8fb591
{
Packit 8fb591
    struct hash_table *ht;
Packit 8fb591
Packit 8fb591
    if (!orig) {
Packit 8fb591
        return NULL;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    ht = lyht_new(orig->size, orig->rec_size - (sizeof(struct ht_rec) - 1), orig->val_equal, orig->cb_data, orig->resize ? 1 : 0);
Packit 8fb591
    if (!ht) {
Packit 8fb591
        return NULL;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    memcpy(ht->recs, orig->recs, orig->used * orig->rec_size);
Packit 8fb591
    ht->used = orig->used;
Packit 8fb591
    return ht;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
void
Packit 8fb591
lyht_free(struct hash_table *ht)
Packit 8fb591
{
Packit 8fb591
    if (ht) {
Packit 8fb591
        free(ht->recs);
Packit 8fb591
        free(ht);
Packit 8fb591
    }
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
static int
Packit 8fb591
lyht_resize(struct hash_table *ht, int enlarge)
Packit 8fb591
{
Packit 8fb591
    struct ht_rec *rec;
Packit 8fb591
    unsigned char *old_recs;
Packit 8fb591
    uint32_t i, old_size;
Packit 8fb591
    int ret;
Packit 8fb591
Packit 8fb591
    old_recs = ht->recs;
Packit 8fb591
    old_size = ht->size;
Packit 8fb591
Packit 8fb591
    if (enlarge) {
Packit 8fb591
        /* double the size */
Packit 8fb591
        ht->size <<= 1;
Packit 8fb591
    } else {
Packit 8fb591
        /* half the size */
Packit 8fb591
        ht->size >>= 1;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    ht->recs = calloc(ht->size, ht->rec_size);
Packit 8fb591
    LY_CHECK_ERR_RETURN(!ht->recs, LOGMEM(NULL); ht->recs = old_recs; ht->size = old_size, -1);
Packit 8fb591
Packit 8fb591
    /* reset used, it will increase again */
Packit 8fb591
    ht->used = 0;
Packit 8fb591
Packit 8fb591
    /* add all the old records into the new records array */
Packit 8fb591
    for (i = 0; i < old_size; ++i) {
Packit 8fb591
        rec = lyht_get_rec(old_recs, ht->rec_size, i);
Packit 8fb591
        if (rec->hits > 0) {
Packit 8fb591
            ret = lyht_insert(ht, rec->val, rec->hash, NULL);
Packit 8fb591
            assert(!ret);
Packit 8fb591
            (void)ret;
Packit 8fb591
        }
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    /* final touches */
Packit 8fb591
    free(old_recs);
Packit 8fb591
    return 0;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
/* return: 0 - hash found, returned its record,
Packit 8fb591
 *         1 - hash not found, returned the record where it would be inserted */
Packit 8fb591
static int
Packit 8fb591
lyht_find_first(struct hash_table *ht, uint32_t hash, struct ht_rec **rec_p)
Packit 8fb591
{
Packit 8fb591
    struct ht_rec *rec;
Packit 8fb591
    uint32_t i, idx;
Packit 8fb591
Packit 8fb591
    if (rec_p) {
Packit 8fb591
        *rec_p = NULL;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    idx = i = hash & (ht->size - 1);
Packit 8fb591
    rec = lyht_get_rec(ht->recs, ht->rec_size, idx);
Packit 8fb591
Packit 8fb591
    /* skip through overflow and deleted records */
Packit 8fb591
    while ((rec->hits != 0) && ((rec->hits == -1) || ((rec->hash & (ht->size - 1)) != idx))) {
Packit 8fb591
        if ((rec->hits == -1) && rec_p && !(*rec_p)) {
Packit 8fb591
            /* remember this record for return */
Packit 8fb591
            *rec_p = rec;
Packit 8fb591
        }
Packit 8fb591
        i = (i + 1) % ht->size;
Packit 8fb591
        if (i == idx) {
Packit 8fb591
            /* we went through all the records (very unlikely, but possible when many records are invalid),
Packit 8fb591
             * just return not found */
Packit 8fb591
            assert(!rec_p || *rec_p);
Packit 8fb591
            return 1;
Packit 8fb591
        }
Packit 8fb591
        rec = lyht_get_rec(ht->recs, ht->rec_size, i);
Packit 8fb591
    }
Packit 8fb591
    if (rec->hits == 0) {
Packit 8fb591
        /* we could not find the value */
Packit 8fb591
        if (rec_p && !*rec_p) {
Packit 8fb591
            *rec_p = rec;
Packit 8fb591
        }
Packit 8fb591
        return 1;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    /* we have found a record with equal (shortened) hash */
Packit 8fb591
    if (rec_p) {
Packit 8fb591
        *rec_p = rec;
Packit 8fb591
    }
Packit 8fb591
    return 0;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
/**
Packit 8fb591
 * @brief Search for the next collision.
Packit 8fb591
 *
Packit 8fb591
 * @param[in] ht Hash table to search in.
Packit 8fb591
 * @param[in,out] last Last returned collision record.
Packit 8fb591
 * @param[in] first First collision record (hits > 1).
Packit 8fb591
 * @return 0 when hash collision found, \p last points to this next collision,
Packit 8fb591
 *         1 when hash collision not found, \p last points to the record where it would be inserted.
Packit 8fb591
 */
Packit 8fb591
static int
Packit 8fb591
lyht_find_collision(struct hash_table *ht, struct ht_rec **last, struct ht_rec *first)
Packit 8fb591
{
Packit 8fb591
    struct ht_rec *empty = NULL;
Packit 8fb591
    uint32_t i, idx;
Packit 8fb591
Packit 8fb591
    assert(last && *last);
Packit 8fb591
Packit 8fb591
    idx = (*last)->hash & (ht->size - 1);
Packit 8fb591
    i = (((unsigned char *)*last) - ht->recs) / ht->rec_size;
Packit 8fb591
Packit 8fb591
    do {
Packit 8fb591
        i = (i + 1) % ht->size;
Packit 8fb591
        *last = lyht_get_rec(ht->recs, ht->rec_size, i);
Packit 8fb591
        if (*last == first) {
Packit 8fb591
            /* we went through all the records (very unlikely, but possible when many records are invalid),
Packit 8fb591
             * just return an invalid record */
Packit 8fb591
            assert(empty);
Packit 8fb591
            *last = empty;
Packit 8fb591
            return 1;
Packit 8fb591
        }
Packit 8fb591
Packit 8fb591
        if (((*last)->hits == -1) && !empty) {
Packit 8fb591
            empty = *last;
Packit 8fb591
        }
Packit 8fb591
    } while (((*last)->hits != 0) && (((*last)->hits == -1) || (((*last)->hash & (ht->size - 1)) != idx)));
Packit 8fb591
Packit 8fb591
    if ((*last)->hits > 0) {
Packit 8fb591
        /* we found a collision */
Packit 8fb591
        assert((*last)->hits == 1);
Packit 8fb591
        return 0;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    /* no next collision found, return the record where it would be inserted */
Packit 8fb591
    if (empty) {
Packit 8fb591
        *last = empty;
Packit 8fb591
    } /* else (*last)->hits == 0, it is already correct */
Packit 8fb591
    return 1;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
int
Packit 8fb591
lyht_find(struct hash_table *ht, void *val_p, uint32_t hash, void **match_p)
Packit 8fb591
{
Packit 8fb591
    struct ht_rec *rec, *crec;
Packit 8fb591
    uint32_t i, c;
Packit 8fb591
    int r;
Packit 8fb591
Packit 8fb591
    if (lyht_find_first(ht, hash, &rec)) {
Packit 8fb591
        /* not found */
Packit 8fb591
        return 1;
Packit 8fb591
    }
Packit 8fb591
    if ((rec->hash == hash) && ht->val_equal(val_p, &rec->val, 0, ht->cb_data)) {
Packit 8fb591
        /* even the value matches */
Packit 8fb591
        if (match_p) {
Packit 8fb591
            *match_p = rec->val;
Packit 8fb591
        }
Packit 8fb591
        return 0;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    /* some collisions, we need to go through them, too */
Packit 8fb591
    crec = rec;
Packit 8fb591
    c = rec->hits;
Packit 8fb591
    for (i = 1; i < c; ++i) {
Packit 8fb591
        r = lyht_find_collision(ht, &rec, crec);
Packit 8fb591
        assert(!r);
Packit 8fb591
        (void)r;
Packit 8fb591
Packit 8fb591
        /* compare values */
Packit 8fb591
        if ((rec->hash == hash) && ht->val_equal(val_p, &rec->val, 0, ht->cb_data)) {
Packit 8fb591
            if (match_p) {
Packit 8fb591
                *match_p = rec->val;
Packit 8fb591
            }
Packit 8fb591
            return 0;
Packit 8fb591
        }
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    /* not found even in collisions */
Packit 8fb591
    return 1;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
int
Packit 8fb591
lyht_find_next(struct hash_table *ht, void *val_p, uint32_t hash, void **match_p)
Packit 8fb591
{
Packit 8fb591
    struct ht_rec *rec, *crec;
Packit 8fb591
    uint32_t i, c;
Packit 8fb591
    int r, found = 0;
Packit 8fb591
Packit 8fb591
    if (lyht_find_first(ht, hash, &rec)) {
Packit 8fb591
        /* not found, cannot happen */
Packit 8fb591
        assert(0);
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    if ((rec->hash == hash) && ht->val_equal(val_p, &rec->val, 1, ht->cb_data)) {
Packit 8fb591
        /* previously returned value */
Packit 8fb591
        found = 1;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    if (rec->hits == 1) {
Packit 8fb591
        /* there are no more similar values */
Packit 8fb591
        assert(rec->hash == hash);
Packit 8fb591
        assert(found);
Packit 8fb591
        return 1;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    /* go through collisions and find next one after the previous one */
Packit 8fb591
    crec = rec;
Packit 8fb591
    c = rec->hits;
Packit 8fb591
    for (i = 1; i < c; ++i) {
Packit 8fb591
        r = lyht_find_collision(ht, &rec, crec);
Packit 8fb591
        assert(!r);
Packit 8fb591
        (void)r;
Packit 8fb591
Packit 8fb591
        if (rec->hash != hash) {
Packit 8fb591
            /* a normal collision, we are not interested in those */
Packit 8fb591
            continue;
Packit 8fb591
        }
Packit 8fb591
Packit 8fb591
        if (found) {
Packit 8fb591
            /* next value with equal hash, found our value */
Packit 8fb591
            if (match_p) {
Packit 8fb591
                *match_p = rec->val;
Packit 8fb591
            }
Packit 8fb591
            return 0;
Packit 8fb591
        }
Packit 8fb591
Packit 8fb591
        if (!ht->val_equal(val_p, &rec->val, 1, ht->cb_data)) {
Packit 8fb591
            /* already returned value, skip */
Packit 8fb591
            continue;
Packit 8fb591
        }
Packit 8fb591
Packit 8fb591
        /* this one was returned previously, continue looking */
Packit 8fb591
        found = 1;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    /* the last equal value was already returned */
Packit 8fb591
    assert(found);
Packit 8fb591
    return 1;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
int
Packit 8fb591
lyht_insert_with_resize_cb(struct hash_table *ht, void *val_p, uint32_t hash,
Packit 8fb591
                           values_equal_cb resize_val_equal, void **match_p)
Packit 8fb591
{
Packit 8fb591
    struct ht_rec *rec, *crec = NULL;
Packit 8fb591
    int32_t i;
Packit 8fb591
    int r, ret;
Packit 8fb591
    values_equal_cb old_val_equal;
Packit 8fb591
Packit 8fb591
    if (!lyht_find_first(ht, hash, &rec)) {
Packit 8fb591
        /* we found matching shortened hash */
Packit 8fb591
        if ((rec->hash == hash) && ht->val_equal(val_p, &rec->val, 1, ht->cb_data)) {
Packit 8fb591
            /* even the value matches */
Packit 8fb591
            if (match_p) {
Packit 8fb591
                *match_p = (void *)&rec->val;
Packit 8fb591
            }
Packit 8fb591
            return 1;
Packit 8fb591
        }
Packit 8fb591
Packit 8fb591
        /* some collisions, we need to go through them, too */
Packit 8fb591
        crec = rec;
Packit 8fb591
        for (i = 1; i < crec->hits; ++i) {
Packit 8fb591
            r = lyht_find_collision(ht, &rec, crec);
Packit 8fb591
            assert(!r);
Packit 8fb591
Packit 8fb591
            /* compare values */
Packit 8fb591
            if ((rec->hash == hash) && ht->val_equal(val_p, &rec->val, 1, ht->cb_data)) {
Packit 8fb591
                if (match_p) {
Packit 8fb591
                    *match_p = (void *)&rec->val;
Packit 8fb591
                }
Packit 8fb591
                return 1;
Packit 8fb591
            }
Packit 8fb591
        }
Packit 8fb591
Packit 8fb591
        /* value not found, get the record where it will be inserted */
Packit 8fb591
        r = lyht_find_collision(ht, &rec, crec);
Packit 8fb591
        assert(r);
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    /* insert it into the returned record */
Packit 8fb591
    assert(rec->hits < 1);
Packit 8fb591
    rec->hash = hash;
Packit 8fb591
    rec->hits = 1;
Packit 8fb591
    memcpy(&rec->val, val_p, ht->rec_size - (sizeof(struct ht_rec) - 1));
Packit 8fb591
    if (match_p) {
Packit 8fb591
        *match_p = (void *)&rec->val;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    if (crec) {
Packit 8fb591
        /* there was a collision, increase hits */
Packit 8fb591
        if (crec->hits == INT32_MAX) {
Packit 8fb591
            LOGINT(NULL);
Packit 8fb591
        }
Packit 8fb591
        ++crec->hits;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    /* check size & enlarge if needed */
Packit 8fb591
    ret = 0;
Packit 8fb591
    ++ht->used;
Packit 8fb591
    if (ht->resize) {
Packit 8fb591
        r = (ht->used * 100) / ht->size;
Packit 8fb591
        if ((ht->resize == 1) && (r >= LYHT_FIRST_SHRINK_PERCENTAGE)) {
Packit 8fb591
            /* enable shrinking */
Packit 8fb591
            ht->resize = 2;
Packit 8fb591
        }
Packit 8fb591
        if ((ht->resize == 2) && (r >= LYHT_ENLARGE_PERCENTAGE)) {
Packit 8fb591
            if (resize_val_equal) {
Packit 8fb591
                old_val_equal = lyht_set_cb(ht, resize_val_equal);
Packit 8fb591
            }
Packit 8fb591
Packit 8fb591
            /* enlarge */
Packit 8fb591
            ret = lyht_resize(ht, 1);
Packit 8fb591
            /* if hash_table was resized, we need to find new matching value */
Packit 8fb591
            if (ret == 0 && match_p) {
Packit 8fb591
                lyht_find(ht, val_p, hash, match_p);
Packit 8fb591
            }
Packit 8fb591
Packit 8fb591
            if (resize_val_equal) {
Packit 8fb591
                lyht_set_cb(ht, old_val_equal);
Packit 8fb591
            }
Packit 8fb591
        }
Packit 8fb591
    }
Packit 8fb591
    return ret;
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
int
Packit 8fb591
lyht_insert(struct hash_table *ht, void *val_p, uint32_t hash, void **match_p)
Packit 8fb591
{
Packit 8fb591
    return lyht_insert_with_resize_cb(ht, val_p, hash, NULL, match_p);
Packit 8fb591
}
Packit 8fb591
Packit 8fb591
int
Packit 8fb591
lyht_remove(struct hash_table *ht, void *val_p, uint32_t hash)
Packit 8fb591
{
Packit 8fb591
    struct ht_rec *rec, *crec;
Packit 8fb591
    int32_t i;
Packit 8fb591
    int first_matched = 0, r, ret;
Packit 8fb591
Packit 8fb591
    if (lyht_find_first(ht, hash, &rec)) {
Packit 8fb591
        /* hash not found */
Packit 8fb591
        return 1;
Packit 8fb591
    }
Packit 8fb591
    if ((rec->hash == hash) && ht->val_equal(val_p, &rec->val, 1, ht->cb_data)) {
Packit 8fb591
        /* even the value matches */
Packit 8fb591
        first_matched = 1;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    /* we always need to go through collisions */
Packit 8fb591
    crec = rec;
Packit 8fb591
    for (i = 1; i < crec->hits; ++i) {
Packit 8fb591
        r = lyht_find_collision(ht, &rec, crec);
Packit 8fb591
        assert(!r);
Packit 8fb591
Packit 8fb591
        /* compare values */
Packit 8fb591
        if (!first_matched && (rec->hash == hash) && ht->val_equal(val_p, &rec->val, 1, ht->cb_data)) {
Packit 8fb591
            break;
Packit 8fb591
        }
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    if (i < crec->hits) {
Packit 8fb591
        /* one of collisions matched, reduce collision count, remove the record */
Packit 8fb591
        assert(!first_matched);
Packit 8fb591
        --crec->hits;
Packit 8fb591
        rec->hits = -1;
Packit 8fb591
    } else if (first_matched) {
Packit 8fb591
        /* the first record matches */
Packit 8fb591
        if (crec != rec) {
Packit 8fb591
            /* ... so put the last collision in its place */
Packit 8fb591
            rec->hits = crec->hits - 1;
Packit 8fb591
            memcpy(crec, rec, ht->rec_size);
Packit 8fb591
        }
Packit 8fb591
        rec->hits = -1;
Packit 8fb591
    } else {
Packit 8fb591
        /* value not found even in collisions */
Packit 8fb591
        assert(!first_matched);
Packit 8fb591
        return 1;
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    /* check size & shrink if needed */
Packit 8fb591
    ret = 0;
Packit 8fb591
    --ht->used;
Packit 8fb591
    if (ht->resize == 2) {
Packit 8fb591
        r = (ht->used * 100) / ht->size;
Packit 8fb591
        if ((r < LYHT_SHRINK_PERCENTAGE) && (ht->size > LYHT_MIN_SIZE)) {
Packit 8fb591
            /* shrink */
Packit 8fb591
            ret = lyht_resize(ht, 0);
Packit 8fb591
        }
Packit 8fb591
    }
Packit 8fb591
Packit 8fb591
    return ret;
Packit 8fb591
}