diff options
author | Marvin Borner | 2023-05-06 20:19:23 +0200 |
---|---|---|
committer | Marvin Borner | 2023-05-06 20:19:23 +0200 |
commit | b31220aadc24ff137a4fe4bc39780ae63c58e11b (patch) | |
tree | 89e431ad99e85f581dbea56d61abac6794386f2a /src/store.c | |
parent | d19f7f5bb99ca1073edb3ce1b13e782a3f598b4a (diff) |
Start fresh
Diffstat (limited to 'src/store.c')
-rw-r--r-- | src/store.c | 857 |
1 files changed, 0 insertions, 857 deletions
diff --git a/src/store.c b/src/store.c deleted file mode 100644 index db3ccef..0000000 --- a/src/store.c +++ /dev/null @@ -1,857 +0,0 @@ -/* - * MIT License - * - * Copyright (c) 2020 Samuel Vogelsanger <vogelsangersamuel@gmail.com> - * Copyright (c) 2023 Marvin Borner <dev@marvinborner.de> - * - * Permission is hereby granted, free of charge, to any person obtaining a copy - * of this software and associated documentation files (the "Software"), to deal - * in the Software without restriction, including without limitation the rights - * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell - * copies of the Software, and to permit persons to whom the Software is - * furnished to do so, subject to the following conditions: - * - * The above copyright notice and this permission notice shall be included in all - * copies or substantial portions of the Software. - * - * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR - * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, - * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE - * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER - * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, - * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE - * SOFTWARE. - */ - -/* - * All the ref-counting specific code was marked with a "//reference counting" comment. If you need to modify this to - * work with your own memory policy, it is recommended to start looking at those places to understand when and where - * memory is allocated and freed. - */ - -#include <malloc.h> -#include <stdint.h> -#include <stdio.h> -#include <string.h> - -#include <store.h> -#include <gc.h> - -#define store_node_debug_fmt \ - "node{element_arity=%u, element_map=%08x, branch_arity=%u, branch_map=%08x, ref_count=%u}" -#define store_node_debug_args(node) \ - node->element_arity, node->element_map, node->branch_arity, \ - node->branch_map, node->ref_count - -#define HASH_PARTITION_WIDTH 5u -#define HASH_TOTAL_WIDTH (8 * sizeof(uint32_t)) - -/* - * Helper functions - */ - -static unsigned bitcount(uint32_t value) -{ - // taken from http://graphics.stanford.edu/~seander/bithacks.html#CountBitsSetParallel - value = value - - ((value >> 1u) & 0x55555555u); // reuse input as temporary - value = (value & 0x33333333u) + ((value >> 2u) & 0x33333333u); // temp - return (((value + (value >> 4u)) & 0xF0F0F0Fu) * 0x1010101u) >> - 24u; // count -} - -static uint32_t store_mask(uint32_t hash, unsigned shift) -{ - return (hash >> shift) & ((1u << HASH_PARTITION_WIDTH) - 1); -} - -static unsigned store_index(uint32_t bitmap, uint32_t bitpos) -{ - return bitcount(bitmap & (bitpos - 1)); -} - -/* - * Data structure definitions - */ - -struct kv { - STORE_KEY_T key; - STORE_VALUE_T val; -}; - -#define STORE_NODE_ELEMENT_T struct kv -#define STORE_NODE_BRANCH_T struct node * - -struct node { - uint8_t element_arity; - uint8_t branch_arity; - uint16_t ref_count; // reference counting - uint32_t element_map; - uint32_t branch_map; - STORE_NODE_ELEMENT_T content[]; -}; - -struct collision_node { - uint8_t element_arity; // MUST SHARE LAYOUT WITH struct node - uint8_t branch_arity; // MUST SHARE LAYOUT WITH struct node - uint16_t ref_count; // MUST SHARE LAYOUT WITH struct node // reference counting - STORE_NODE_ELEMENT_T content[]; -}; - -static struct node empty_node = { - .branch_arity = 0, - .element_arity = 0, - .ref_count = 1, - .branch_map = 0, - .element_map = 0, -}; - -#define STORE_NODE_ELEMENTS(node) (node)->content -#define STORE_NODE_BRANCHES(node) \ - ((STORE_NODE_BRANCH_T *)&(node)->content[(node)->element_arity]) - -#define STORE_NODE_ELEMENTS_SIZE(length) \ - (sizeof(STORE_NODE_ELEMENT_T) * (length)) -#define STORE_NODE_BRANCHES_SIZE(length) \ - (sizeof(STORE_NODE_BRANCH_T) * (length)) - -#define STORE_NODE_ELEMENT_AT(node, bitpos) \ - STORE_NODE_ELEMENTS(node)[store_index(node->element_map, bitpos)] -#define STORE_NODE_BRANCH_AT(node, bitpos) \ - STORE_NODE_BRANCHES(node)[store_index(node->branch_map, bitpos)] - -/* - * static function declarations - */ - -// node constructor -static struct node *node_new(uint32_t element_map, uint32_t branch_map, - STORE_NODE_ELEMENT_T const *elements, - uint8_t element_arity, - STORE_NODE_BRANCH_T const *branches, - uint8_t branch_arity); - -// collision node variant -static struct collision_node * -collision_node_new(const STORE_NODE_ELEMENT_T *values, uint8_t element_arity); - -// destructor -static void node_destroy(struct node *node); - -// reference counting -static struct node *store_node_acquire(struct node *node); - -// reference counting -static void store_node_release(struct node *node); - -// top-level functions -static STORE_VALUE_T node_get(struct node *node, STORE_EQUALSFN_T(equals), - STORE_KEY_T key, uint32_t hash, unsigned shift, - int *found); - -static struct node *node_update(struct node *node, STORE_HASHFN_T(hashfn), - STORE_EQUALSFN_T(equals), STORE_KEY_T key, - STORE_VALUE_T value, uint32_t hash, - unsigned shift, int *found); - -static struct node *node_assoc(struct node *node, STORE_HASHFN_T(hashfn), - STORE_EQUALSFN_T(equals), STORE_KEY_T key, - STORE_ASSOCFN_T(fn), void *user_data, - uint32_t hash, unsigned shift, int *found); - -// collision node variants -static STORE_VALUE_T collision_node_get(struct collision_node *node, - STORE_EQUALSFN_T(equals), - STORE_KEY_T key, int *found); - -static struct collision_node * -collision_node_update(struct collision_node *node, STORE_EQUALSFN_T(equals), - STORE_KEY_T key, STORE_VALUE_T value, int *found); - -static struct collision_node *collision_node_assoc(struct collision_node *node, - STORE_EQUALSFN_T(equals), - STORE_KEY_T key, - STORE_ASSOCFN_T(fn), - void *user_data, int *found); - -// helper functions for creation of modified nodes -static struct node *node_merge(uint32_t hash_l, STORE_KEY_T key_l, - STORE_VALUE_T value_l, uint32_t hash_r, - STORE_KEY_T key_r, STORE_VALUE_T value_r, - unsigned shift); - -static struct node *node_clone_update_branch(struct node *node, uint32_t bitpos, - struct node *branch); - -static struct node *node_clone_pushdown(struct node *node, uint32_t bitpos, - struct node *branch); - -static struct node *node_clone_insert_element(struct node *node, - uint32_t bitpos, STORE_KEY_T key, - STORE_VALUE_T value); - -static struct node *node_clone_update_element(struct node *node, - uint32_t bitpos, - STORE_VALUE_T value); - -// collision node variants -static struct collision_node * -collision_node_clone_insert_element(struct collision_node *node, - STORE_KEY_T key, STORE_VALUE_T value); - -static struct collision_node * -collision_node_clone_update_element(struct collision_node *node, unsigned index, - STORE_VALUE_T value); - -// equality -static int node_equals(struct node *left, struct node *right, - STORE_EQUALSFN_T(key_equals), - STORE_VALUE_EQUALSFN_T(value_equals), unsigned shift); - -static int collision_node_equals(struct collision_node *left, - struct collision_node *right, - STORE_EQUALSFN_T(key_equals), - STORE_VALUE_EQUALSFN_T(value_equals)); - -// store private constructor -static struct store *store_from(struct node *root, unsigned length, - STORE_HASHFN_T(hash), STORE_EQUALSFN_T(equals)); - -// iterator helper functions -static void iter_push(struct store_iter *iterator, struct node *node); - -static void iter_pop(struct store_iter *iterator); - -/* - * definitions - */ - -static void node_destroy(struct node *node) -{ - DEBUG_NOTICE(" destroying " store_node_debug_fmt "@%p\n", - store_node_debug_args(node), (void *)node); - - GC_free(node); -} - -// reference counting -static struct node *store_node_acquire(struct node *node) -{ - if (node == &empty_node) - return node; - node->ref_count++; - return node; -} - -// reference counting -static void store_node_release(struct node *node) -{ - if (node == &empty_node) - return; - - if (node->ref_count-- == 1) - node_destroy(node); -} - -/** - * WARNING: all branches in <code>branches</code> are "acquired", i.e. their reference count is incremented. - * Do not pass an "almost correct" list of branches. - */ -static struct node *node_new(uint32_t element_map, uint32_t branch_map, - STORE_NODE_ELEMENT_T const *elements, - uint8_t element_arity, - STORE_NODE_BRANCH_T const *branches, - uint8_t branch_arity) -{ - const size_t content_size = STORE_NODE_ELEMENTS_SIZE(element_arity) + - STORE_NODE_BRANCHES_SIZE(branch_arity); - struct node *result = GC_malloc(sizeof(*result) + content_size); - - result->element_arity = element_arity; - result->branch_arity = branch_arity; - result->ref_count = 0; - result->element_map = element_map; - result->branch_map = branch_map; - - if (elements) { - memcpy(STORE_NODE_ELEMENTS(result), elements, - STORE_NODE_ELEMENTS_SIZE(element_arity)); - } - - STORE_NODE_BRANCH_T *branches_dest = - (STORE_NODE_BRANCH_T *)STORE_NODE_BRANCHES(result); - // reference counting - for (int i = 0; i < branch_arity; ++i) { - branches_dest[i] = store_node_acquire(branches[i]); - } - - return result; -} - -static STORE_VALUE_T collision_node_get(struct collision_node *node, - STORE_EQUALSFN_T(equals), - STORE_KEY_T key, int *found) -{ - for (unsigned i = 0; i < node->element_arity; ++i) { - struct kv kv = node->content[i]; - if (equals(kv.key, key)) { - *found = 1; - return kv.val; - } - } - - *found = 0; - return (STORE_VALUE_T)0; -} - -static STORE_VALUE_T node_get(struct node *node, STORE_EQUALSFN_T(equals), - STORE_KEY_T key, uint32_t hash, unsigned shift, - int *found) -{ - if (shift >= HASH_TOTAL_WIDTH) - return collision_node_get((struct collision_node *)node, equals, - key, found); - - const uint32_t bitpos = 1u << store_mask(hash, shift); - - if (node->branch_map & bitpos) { - return node_get(STORE_NODE_BRANCH_AT(node, bitpos), equals, key, - hash, shift + HASH_PARTITION_WIDTH, found); - - } else if (node->element_map & bitpos) { - STORE_NODE_ELEMENT_T kv = STORE_NODE_ELEMENT_AT(node, bitpos); - if (equals(kv.key, key)) { - *found = 1; - return kv.val; - } - } - - *found = 0; - return (STORE_VALUE_T)0; -} - -static struct node *node_clone_insert_element(struct node *node, - uint32_t bitpos, STORE_KEY_T key, - STORE_VALUE_T value) -{ - STORE_NODE_ELEMENT_T elements[1u << HASH_PARTITION_WIDTH]; - const unsigned index = store_index(node->element_map, bitpos); - - // copy <element_arity> chunks in total - memcpy(elements, STORE_NODE_ELEMENTS(node), - STORE_NODE_ELEMENTS_SIZE(index)); // copy first <index> chunks - elements[index].key = (STORE_KEY_T)key; - elements[index].val = (STORE_VALUE_T)value; - memcpy(&elements[index + 1], // start copying into one-past-<index> - &STORE_NODE_ELEMENTS(node)[index], // start copying from <index> - STORE_NODE_ELEMENTS_SIZE( - node->element_arity - - index) // <index> chunks already copied, <element_arity> - <index> remaining - ); - - return node_new(node->element_map | bitpos, node->branch_map, elements, - node->element_arity + 1, STORE_NODE_BRANCHES(node), - node->branch_arity); -} - -static struct node *node_clone_update_element(struct node *node, - uint32_t bitpos, - STORE_VALUE_T value) -{ - STORE_NODE_ELEMENT_T elements[1u << HASH_PARTITION_WIDTH]; - const unsigned index = store_index(node->element_map, bitpos); - - memcpy(elements, STORE_NODE_ELEMENTS(node), - STORE_NODE_ELEMENTS_SIZE(node->element_arity)); - elements[index].val = (STORE_VALUE_T)value; - return node_new(node->element_map, node->branch_map, elements, - node->element_arity, STORE_NODE_BRANCHES(node), - node->branch_arity); -} - -static struct node *node_clone_update_branch(struct node *node, uint32_t bitpos, - struct node *branch) -{ - STORE_NODE_BRANCH_T branches[1u << HASH_PARTITION_WIDTH]; - const unsigned index = store_index(node->branch_map, bitpos); - - memcpy(branches, STORE_NODE_BRANCHES(node), - STORE_NODE_BRANCHES_SIZE(node->branch_arity)); - branches[index] = branch; - return node_new(node->element_map, node->branch_map, - STORE_NODE_ELEMENTS(node), node->element_arity, - branches, node->branch_arity); -} - -static struct node *node_clone_pushdown(struct node *node, uint32_t bitpos, - struct node *branch) -{ - STORE_NODE_ELEMENT_T elements[1u << HASH_PARTITION_WIDTH]; - STORE_NODE_BRANCH_T branches[1u << HASH_PARTITION_WIDTH]; - const unsigned element_index = store_index(node->element_map, bitpos); - const unsigned branch_index = store_index(node->branch_map, bitpos); - - memcpy(elements, STORE_NODE_ELEMENTS(node), - STORE_NODE_ELEMENTS_SIZE(element_index)); - memcpy(&elements[element_index], - &STORE_NODE_ELEMENTS(node)[element_index + 1], - STORE_NODE_ELEMENTS_SIZE(node->element_arity - - (element_index + 1))); - - memcpy(branches, STORE_NODE_BRANCHES(node), - STORE_NODE_BRANCHES_SIZE(branch_index)); - memcpy(&branches[branch_index + 1], - &STORE_NODE_BRANCHES(node)[branch_index], - STORE_NODE_BRANCHES_SIZE(node->branch_arity - branch_index)); - branches[branch_index] = branch; - - return node_new(node->element_map & ~bitpos, node->branch_map | bitpos, - elements, node->element_arity - 1, branches, - node->branch_arity + 1); -} - -static struct collision_node * -collision_node_new(const STORE_NODE_ELEMENT_T *values, uint8_t element_arity) -{ - size_t content_size = sizeof(STORE_NODE_ELEMENT_T) * element_arity; - struct collision_node *result = - GC_malloc(sizeof(*result) + content_size); - - result->element_arity = element_arity; - result->branch_arity = 0; - result->ref_count = 0; - - memcpy(result->content, values, - STORE_NODE_ELEMENTS_SIZE(element_arity)); - - return result; -} - -static struct node *node_merge(uint32_t hash_l, STORE_KEY_T key_l, - STORE_VALUE_T value_l, uint32_t hash_r, - STORE_KEY_T key_r, STORE_VALUE_T value_r, - unsigned shift) -{ - uint32_t bitpos_l = 1u << store_mask(hash_l, shift); - uint32_t bitpos_r = 1u << store_mask(hash_r, shift); - - if (shift >= HASH_TOTAL_WIDTH) { - STORE_NODE_ELEMENT_T elements[2]; - elements[0].key = (STORE_KEY_T)key_l; - elements[0].val = (STORE_VALUE_T)value_l; - elements[1].key = (STORE_KEY_T)key_r; - elements[1].val = (STORE_VALUE_T)value_r; - - return (struct node *)collision_node_new(elements, 2); - - } else if (bitpos_l != bitpos_r) { - STORE_NODE_ELEMENT_T elements[2]; - - if (bitpos_l <= bitpos_r) { - elements[0].key = (STORE_KEY_T)key_l; - elements[0].val = (STORE_VALUE_T)value_l; - elements[1].key = (STORE_KEY_T)key_r; - elements[1].val = (STORE_VALUE_T)value_r; - } else { - elements[0].key = (STORE_KEY_T)key_r; - elements[0].val = (STORE_VALUE_T)value_r; - elements[1].key = (STORE_KEY_T)key_l; - elements[1].val = (STORE_VALUE_T)value_l; - } - - return node_new(bitpos_l | bitpos_r, 0u, elements, 2, NULL, 0); - - } else { - struct node *sub_node = - node_merge(hash_l, key_l, value_l, hash_r, key_r, - value_r, shift + HASH_PARTITION_WIDTH); - - return node_new(0, bitpos_l, NULL, 0, &sub_node, 1); - } -} - -static struct collision_node * -collision_node_clone_update_element(struct collision_node *node, unsigned index, - STORE_VALUE_T value) -{ - STORE_NODE_ELEMENT_T elements[node->element_arity]; - - memcpy(elements, node->content, - STORE_NODE_ELEMENTS_SIZE(node->element_arity)); - elements[index].val = (STORE_VALUE_T)value; - - return collision_node_new(elements, node->element_arity); -} - -static struct collision_node * -collision_node_clone_insert_element(struct collision_node *node, - STORE_KEY_T key, STORE_VALUE_T value) -{ - STORE_NODE_ELEMENT_T elements[node->element_arity + 1]; - - memcpy(elements, node->content, - STORE_NODE_ELEMENTS_SIZE(node->element_arity)); - elements[node->element_arity].key = (STORE_KEY_T)key; - elements[node->element_arity].val = (STORE_VALUE_T)value; - - return collision_node_new(elements, node->element_arity + 1); -} - -static struct collision_node * -collision_node_update(struct collision_node *node, STORE_EQUALSFN_T(equals), - STORE_KEY_T key, STORE_VALUE_T value, int *found) -{ - for (unsigned i = 0; i < node->element_arity; ++i) { - struct kv kv = node->content[i]; - if (equals(kv.key, key)) { - *found = 1; - - return collision_node_clone_update_element(node, i, - value); - } - } - - return collision_node_clone_insert_element(node, key, value); -} - -static struct node *node_update(struct node *node, STORE_HASHFN_T(hashfn), - STORE_EQUALSFN_T(equals), STORE_KEY_T key, - STORE_VALUE_T value, uint32_t hash, - unsigned shift, int *found) -{ - if (shift >= HASH_TOTAL_WIDTH) - return (struct node *)collision_node_update( - (struct collision_node *)node, equals, key, value, - found); - - const uint32_t bitpos = 1u << store_mask(hash, shift); - - if (node->branch_map & bitpos) { - struct node *sub_node = STORE_NODE_BRANCH_AT(node, bitpos); - struct node *new_sub_node = - node_update(sub_node, hashfn, equals, key, value, hash, - shift + HASH_PARTITION_WIDTH, found); - return node_clone_update_branch(node, bitpos, new_sub_node); - - } else if (node->element_map & bitpos) { - STORE_KEY_T current_key = - STORE_NODE_ELEMENT_AT(node, bitpos).key; - - if (equals(current_key, key)) { - *found = 1; - return node_clone_update_element(node, bitpos, value); - - } else { - STORE_VALUE_T current_value = - STORE_NODE_ELEMENT_AT(node, bitpos).val; - struct node *sub_node = - node_merge(hashfn(current_key), current_key, - current_value, hashfn(key), key, - value, shift + HASH_PARTITION_WIDTH); - return node_clone_pushdown(node, bitpos, sub_node); - } - - } else { - return node_clone_insert_element(node, bitpos, key, value); - } -} - -static struct collision_node *collision_node_assoc(struct collision_node *node, - STORE_EQUALSFN_T(equals), - STORE_KEY_T key, - STORE_ASSOCFN_T(fn), - void *user_data, int *found) -{ - STORE_VALUE_T new_value; - for (unsigned i = 0; i < node->element_arity; ++i) { - struct kv kv = node->content[i]; - if (equals(kv.key, key)) { - *found = 1; - STORE_VALUE_T old_value = kv.val; - new_value = fn(key, old_value, user_data); - return collision_node_clone_update_element(node, i, - new_value); - } - } - - new_value = fn((STORE_KEY_T)0, (STORE_VALUE_T)0, user_data); - return collision_node_clone_insert_element(node, key, new_value); -} - -static struct node *node_assoc(struct node *node, STORE_HASHFN_T(hashfn), - STORE_EQUALSFN_T(equals), STORE_KEY_T key, - STORE_ASSOCFN_T(fn), void *user_data, - uint32_t hash, unsigned shift, int *found) -{ - if (shift >= HASH_TOTAL_WIDTH) - return (struct node *)collision_node_assoc( - (struct collision_node *)node, equals, key, fn, - user_data, found); - - const uint32_t bitpos = 1u << store_mask(hash, shift); - - if (node->branch_map & bitpos) { - struct node *sub_node = STORE_NODE_BRANCH_AT(node, bitpos); - struct node *new_sub_node = - node_assoc(sub_node, hashfn, equals, key, fn, user_data, - hash, shift + HASH_PARTITION_WIDTH, found); - return node_clone_update_branch(node, bitpos, new_sub_node); - - } else if (node->element_map & bitpos) { - STORE_KEY_T current_key = - STORE_NODE_ELEMENT_AT(node, bitpos).key; - - if (equals(current_key, key)) { - *found = 1; - STORE_VALUE_T old_value = - STORE_NODE_ELEMENT_AT(node, bitpos).val; - STORE_VALUE_T new_value = fn(key, old_value, user_data); - return node_clone_update_element(node, bitpos, - new_value); - - } else { - STORE_VALUE_T current_value = - STORE_NODE_ELEMENT_AT(node, bitpos).val; - STORE_VALUE_T new_value = - fn((STORE_KEY_T)0, (STORE_VALUE_T)0, user_data); - struct node *sub_node = - node_merge(hashfn(current_key), current_key, - current_value, hash, key, new_value, - shift + HASH_PARTITION_WIDTH); - return node_clone_pushdown(node, bitpos, sub_node); - } - - } else { - STORE_VALUE_T value = - fn((STORE_KEY_T)0, (STORE_VALUE_T)0, user_data); - return node_clone_insert_element(node, bitpos, key, value); - } -} - -static int collision_node_equals(struct collision_node *left, - struct collision_node *right, - STORE_EQUALSFN_T(key_equals), - STORE_VALUE_EQUALSFN_T(value_equals)) -{ - if (left == right) - return 1; - if (left->element_arity != right->element_arity) - return 0; - - for (unsigned left_i = 0; left_i < left->element_arity; ++left_i) { - struct kv left_element = STORE_NODE_ELEMENTS(left)[left_i]; - - for (unsigned right_i = 0; right_i < right->element_arity; - ++right_i) { - struct kv right_element = - STORE_NODE_ELEMENTS(right)[right_i]; - - if (key_equals(left_element.key, right_element.key) && - value_equals(left_element.val, right_element.val)) - goto found_matching_element; - } - return 0; // compared left_element to all elements in right node, no match. - - found_matching_element: - continue; - } - return 1; // compared all elements in left node, never had an element without match. -} - -static int node_equals(struct node *left, struct node *right, - STORE_EQUALSFN_T(key_equals), - STORE_VALUE_EQUALSFN_T(value_equals), unsigned shift) -{ - if (shift >= HASH_TOTAL_WIDTH) - return collision_node_equals((struct collision_node *)left, - (struct collision_node *)right, - key_equals, value_equals); - if (left == right) - return 1; - if (left->element_map != right->element_map) - return 0; - if (left->branch_map != right->branch_map) - return 0; - for (unsigned i = 0; i < left->element_arity; ++i) { - struct kv left_element = STORE_NODE_ELEMENTS(left)[i]; - struct kv right_element = STORE_NODE_ELEMENTS(right)[i]; - if (!key_equals(left_element.key, right_element.key) || - !value_equals(left_element.val, right_element.val)) - return 0; - } - for (unsigned i = 0; i < left->branch_arity; ++i) { - struct node *left_branch = STORE_NODE_BRANCHES(left)[i]; - struct node *right_branch = STORE_NODE_BRANCHES(right)[i]; - if (!node_equals(left_branch, right_branch, key_equals, - value_equals, shift + HASH_PARTITION_WIDTH)) - return 0; - } - return 1; -} - -static struct store *store_from(struct node *root, unsigned length, - STORE_HASHFN_T(hash), STORE_EQUALSFN_T(equals)) -{ - struct store *result = GC_malloc(sizeof(*result)); - result->ref_count = 0; - result->root = root; - result->length = length; - result->hash = hash; - result->equals = equals; - return result; -} - -void store_destroy(struct store **store) -{ - DEBUG_NOTICE("destroying store@%p\n", (void *)*store); - - store_node_release((*store)->root); - GC_free(*store); - *store = NULL; -} - -struct store *store_new(STORE_HASHFN_T(hash), STORE_EQUALSFN_T(equals)) -{ - return store_from((struct node *)&empty_node, 0, hash, equals); -} - -struct store *store_acquire(struct store *store) -{ - store->ref_count++; - DEBUG_NOTICE("ACQ %p: %d\n", (void *)store, store->ref_count); - return store; -} - -void store_release(struct store **store) -{ - DEBUG_NOTICE("REL %p: %d\n", (void *)*store, (*store)->ref_count - 1); - if ((*store)->ref_count-- == 1) - store_destroy(store); -} - -struct store *store_of(STORE_HASHFN_T(hash), STORE_EQUALSFN_T(equals), - STORE_KEY_T *keys, STORE_VALUE_T *values, size_t length) -{ - struct store *result = store_new(hash, equals); - while (length--) { - struct store *tmp = - store_set(result, keys[length], values[length], NULL); - store_destroy(&result); - result = tmp; - } - return result; -} - -unsigned store_length(const struct store *store) -{ - return store->length; -} - -struct store *store_set(const struct store *store, STORE_KEY_T key, - STORE_VALUE_T value, int *replaced) -{ - const uint32_t hash = store->hash(key); - int found = 0; - int *found_p = replaced ? replaced : &found; - *found_p = 0; - struct node *new_root = store_node_acquire( - node_update(store->root, store->hash, store->equals, key, value, - hash, 0, found_p)); - return store_from(new_root, store->length + (*found_p ? 0 : 1), - store->hash, store->equals); -} - -STORE_VALUE_T store_get(const struct store *store, STORE_KEY_T key, int *found) -{ - uint32_t hash = store->hash(key); - int tmp = 0; - return node_get(store->root, store->equals, key, hash, 0, - found ? found : &tmp); -} - -struct store *store_assoc(const struct store *store, STORE_KEY_T key, - STORE_ASSOCFN_T(fn), void *user_data) -{ - const uint32_t hash = store->hash(key); - int found = 0; - struct node *new_root = store_node_acquire( - node_assoc(store->root, store->hash, store->equals, key, fn, - user_data, hash, 0, &found)); - return store_from(new_root, store->length + (found ? 0 : 1), - store->hash, store->equals); -} - -int store_equals(const struct store *left, const struct store *right, - STORE_VALUE_EQUALSFN_T(value_equals)) -{ - if (left == right) - return 1; - else if (store_length(left) != store_length(right)) - return 0; - else - return node_equals(left->root, right->root, left->equals, - value_equals, 0); -} - -void store_iter_init(struct store_iter *iterator, const struct store *store) -{ - iterator->stack_level = 0; - iterator->element_cursor = 0; - iterator->element_arity = store->root->element_arity; - iterator->branch_cursor_stack[0] = 0; - iterator->branch_arity_stack[0] = store->root->branch_arity; - iterator->node_stack[0] = store->root; -} - -static void iter_push(struct store_iter *iterator, struct node *node) -{ - iterator->stack_level += 1; - iterator->element_cursor = 0; - iterator->element_arity = node->element_arity; - iterator->branch_cursor_stack[iterator->stack_level] = 0; - iterator->branch_arity_stack[iterator->stack_level] = - node->branch_arity; - iterator->node_stack[iterator->stack_level] = node; -} - -static void iter_pop(struct store_iter *iterator) -{ - iterator->stack_level -= 1; -} - -int store_iter_next(struct store_iter *iterator, STORE_KEY_T *key, - STORE_VALUE_T *value) -{ - if (iterator->stack_level == -1) - return 0; - - struct node *current_node = iterator->node_stack[iterator->stack_level]; - unsigned *branch_cursor = - iterator->branch_cursor_stack + iterator->stack_level; - if (*branch_cursor == 0 && - iterator->element_cursor < - current_node->element_arity) { // todo: write test for this - *key = STORE_NODE_ELEMENTS( - current_node)[iterator->element_cursor] - .key; - *value = STORE_NODE_ELEMENTS( - current_node)[iterator->element_cursor] - .val; - ++iterator->element_cursor; - return 1; - - } else { - if (*branch_cursor < - iterator->branch_arity_stack[iterator->stack_level]) { - iter_push(iterator, - STORE_NODE_BRANCHES( - current_node)[*branch_cursor]); - ++*branch_cursor; - return store_iter_next(iterator, key, value); - - } else { - iter_pop(iterator); - return store_iter_next(iterator, key, value); - } - } -} |