mirror of
https://github.com/git-for-windows/git.git
synced 2026-04-10 08:22:54 -05:00
The oidtree data structure is currently only used to store object IDs, without any associated data. So consequently, it can only really be used to track which object IDs exist, and we can use the tree structure to efficiently operate on OID prefixes. But there are valid use cases where we want to both: - Store object IDs in a sorted order. - Associated arbitrary data with them. Refactor the oidtree interface so that it allows us to store arbitrary payloads within the respective nodes. This will be used in the next commit. Signed-off-by: Patrick Steinhardt <ps@pks.im> Signed-off-by: Junio C Hamano <gitster@pobox.com>
133 lines
3.1 KiB
C
133 lines
3.1 KiB
C
/*
|
|
* 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);
|
|
}
|