libcdsb/src/dict/extra.c

170 lines
4.2 KiB
C
Raw Normal View History

2022-08-14 21:06:38 +03:00
/* This software is licensed by the MIT License, see LICENSE file */
/* Copyright © 2022 Gregory Lirent */
#include "include.h"
/*#####################################################################################################################*/
static void dict_rehash(dict_t* s, size_t capacity) {
2022-08-18 02:27:26 +03:00
dnode_t **nodes, *c, *n, **p;
size_t i;
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
i = s->capacity;
nodes = calloc(sizeof(*nodes), capacity);
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
while (i--) {
c = s->nodes[i];
2022-08-15 17:31:01 +03:00
2022-08-18 02:27:26 +03:00
while (!is_null(c)) {
p = nodes + (vnode_hash(&c->key, c->key_type) % capacity);
n = c->prev;
c->prev = *p;
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
*p = c;
c = n;
2022-08-14 21:06:38 +03:00
}
}
free(s->nodes);
s->nodes = nodes;
2022-08-18 02:27:26 +03:00
s->capacity = capacity;
2022-08-14 21:06:38 +03:00
}
/*#####################################################################################################################*/
bool libcdsb_dict_shrink_to_fit(dict_t* s) {
size_t capacity;
capacity = (s->size / CAPACITY_BLOCK) + 1;
capacity *= CAPACITY_BLOCK;
while (((double)s->size / capacity) > 0.65)
capacity += CAPACITY_BLOCK;
if (capacity >= s->capacity)
return false;
dict_rehash(s, capacity);
return true;
}
bool libcdsb_dict_update(dict_t* x, const void* k, vtype kt, const void* v, vtype vt) {
2022-08-18 02:27:26 +03:00
dnode_t *c, **p;
2022-08-14 21:06:38 +03:00
2022-08-15 17:31:01 +03:00
if (!x->capacity || (double)x->size / x->capacity > REBUILD_POINT_MAX)
2022-08-14 21:06:38 +03:00
dict_rehash(x, x->capacity + CAPACITY_BLOCK);
2022-08-18 02:27:26 +03:00
c = *(p = x->nodes + (vtype_hash(k, kt) % x->capacity));
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
while (!is_null(c)) {
if (vtype_compare(k, kt, vnode_peek(&c->key, c->key_type), c->key_type) == 0) {
vnode_free(&c->value, c->value_type);
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
c->value = vnode_create(v, vt);
c->value_type = vt;
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
return true;
} else c = c->prev;
}
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
c = malloc(sizeof(*c));
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
c->prev = *p;
c->key = vnode_create(k, kt);
c->value = vnode_create(v, vt);
c->key_type = kt;
c->value_type = vt;
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
*p = c;
2022-08-15 17:31:01 +03:00
++x->size;
2022-08-14 21:06:38 +03:00
return false;
}
2022-08-18 02:27:26 +03:00
int libcdsb_dict_find(dict_t* x, const void* k, vtype t, void* dt, dict_access_callback callback, bool cut) {
dnode_t *c, **p;
int r;
void* key;
2022-08-14 21:06:38 +03:00
if (x->capacity) {
2022-08-18 02:27:26 +03:00
c = *(p = x->nodes + (vtype_hash(k, t) % x->capacity));
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
while (!is_null(c)) {
2022-08-14 21:06:38 +03:00
key = vnode_peek(&c->key, c->key_type);
2022-08-18 02:27:26 +03:00
if (vtype_compare(k, t, key, c->key_type) == 0) {
r = (callback) ? callback(key, c->key_type, vnode_peek(&c->value, c->value_type), c->value_type, dt) : 0;
2022-08-14 21:06:38 +03:00
if (cut) {
2022-08-18 02:27:26 +03:00
*p = c->prev;
vnode_free(&c->key, c->key_type);
vnode_free(&c->value, c->value_type);
2022-08-14 21:06:38 +03:00
free(c);
2022-08-15 17:31:01 +03:00
--x->size;
2022-08-14 21:06:38 +03:00
}
2022-08-18 02:27:26 +03:00
return r;
} else c = *(p = &c->prev);
2022-08-14 21:06:38 +03:00
}
}
return -1;
}
int libcdsb_dict_foreach(dict_t* x, void* dt, dict_access_callback callback, bool flush) {
2022-08-18 02:27:26 +03:00
dnode_t *c;
ssize_t i;
int r;
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
r = 0;
i = x->capacity;
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
while (i) {
c = x->nodes[--i];
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
while (!is_null(c)) {
r = callback(vnode_peek(&c->key, c->key_type), c->key_type,
vnode_peek(&c->value, c->value_type), c->value_type, dt);
2022-08-15 17:31:01 +03:00
2022-08-18 02:27:26 +03:00
c = c->prev;
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
if (r) {
if (!flush) goto end_;
else goto flush_loop_;
} else if (flush) {
vnode_free(&x->nodes[i]->key, x->nodes[i]->key_type);
vnode_free(&x->nodes[i]->value, x->nodes[i]->value_type);
free(x->nodes[i]);
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
x->nodes[i] = c;
}
2022-08-14 21:06:38 +03:00
}
}
if (flush) {
2022-08-18 02:27:26 +03:00
while (i) {
--i;
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
while (!is_null(x->nodes[i])) { flush_loop_:
vnode_free(&x->nodes[i]->key, x->nodes[i]->key_type);
vnode_free(&x->nodes[i]->value, x->nodes[i]->value_type);
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
c = x->nodes[i]->prev;
free(x->nodes[i]);
x->nodes[i] = c;
}
2022-08-14 21:06:38 +03:00
}
free(x->nodes);
memset(x, 0, sizeof(*x));
2022-08-18 02:27:26 +03:00
}
2022-08-14 21:06:38 +03:00
2022-08-18 02:27:26 +03:00
end_:
2022-08-14 21:06:38 +03:00
return r;
}