/* * A wrapper around cbtree which stores oids * May be used to replace oid-array for prefix (abbreviation) matches */ #include "git-compat-util.h" #include "oidtree.h" #include "hash.h" struct oidtree_node { struct cb_node base; struct object_id key; void *data; }; void oidtree_init(struct oidtree *ot) { cb_init(&ot->tree, offsetof(struct oidtree_node, key)); mem_pool_init(&ot->mem_pool, 0); } void oidtree_clear(struct oidtree *ot) { if (ot) { mem_pool_discard(&ot->mem_pool, 0); oidtree_init(ot); } } struct oidtree_data { struct object_id oid; }; void oidtree_insert(struct oidtree *ot, const struct object_id *oid, void *data) { struct oidtree_node *on; struct cb_node *node; if (!oid->algo) BUG("oidtree_insert requires oid->algo"); on = mem_pool_alloc(&ot->mem_pool, sizeof(*on)); oidcpy(&on->key, oid); on->data = data; /* * n.b. Current callers won't get us duplicates, here. If a * future caller causes duplicates, there'll be a small leak * that won't be freed until oidtree_clear. Currently it's not * worth maintaining a free list */ node = cb_insert(&ot->tree, &on->base, sizeof(*oid)); if (node) { struct oidtree_node *preexisting = container_of(node, struct oidtree_node, base); preexisting->data = data; } } static struct oidtree_node *oidtree_lookup(struct oidtree *ot, const struct object_id *oid) { struct object_id k; size_t klen = sizeof(k); struct cb_node *node; oidcpy(&k, oid); if (oid->algo == GIT_HASH_UNKNOWN) klen -= sizeof(oid->algo); /* cb_lookup relies on memcmp on the struct, so order matters: */ klen += BUILD_ASSERT_OR_ZERO(offsetof(struct object_id, hash) < offsetof(struct object_id, algo)); node = cb_lookup(&ot->tree, (const uint8_t *)&k, klen); return node ? container_of(node, struct oidtree_node, base) : NULL; } bool oidtree_contains(struct oidtree *ot, const struct object_id *oid) { struct oidtree_node *node = oidtree_lookup(ot, oid); return node ? 1 : 0; } void *oidtree_get(struct oidtree *ot, const struct object_id *oid) { struct oidtree_node *node = oidtree_lookup(ot, oid); return node ? node->data : NULL; } struct oidtree_each_data { oidtree_each_cb cb; void *cb_data; size_t *last_nibble_at; uint32_t algo; uint8_t last_byte; }; static int iter(struct cb_node *n, void *cb_data) { struct oidtree_node *node = container_of(n, struct oidtree_node, base); struct oidtree_each_data *data = cb_data; if (data->algo != GIT_HASH_UNKNOWN && data->algo != node->key.algo) return 0; if (data->last_nibble_at) { if ((node->key.hash[*data->last_nibble_at] ^ data->last_byte) & 0xf0) return 0; } return data->cb(&node->key, node->data, data->cb_data); } int oidtree_each(struct oidtree *ot, const struct object_id *prefix, size_t prefix_hex_len, oidtree_each_cb cb, void *cb_data) { struct oidtree_each_data data = { .cb = cb, .cb_data = cb_data, .algo = prefix->algo, }; size_t klen = prefix_hex_len / 2; assert(prefix_hex_len <= GIT_MAX_HEXSZ); if (prefix_hex_len & 1) { data.last_byte = prefix->hash[klen]; data.last_nibble_at = &klen; } return cb_each(&ot->tree, prefix->hash, klen, iter, &data); }