aboutsummaryrefslogtreecommitdiff
path: root/src/store.c
diff options
context:
space:
mode:
authorMarvin Borner2023-05-06 20:19:23 +0200
committerMarvin Borner2023-05-06 20:19:23 +0200
commitb31220aadc24ff137a4fe4bc39780ae63c58e11b (patch)
tree89e431ad99e85f581dbea56d61abac6794386f2a /src/store.c
parentd19f7f5bb99ca1073edb3ce1b13e782a3f598b4a (diff)
Start fresh
Diffstat (limited to 'src/store.c')
-rw-r--r--src/store.c857
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);
- }
- }
-}