You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
110 lines
2.5 KiB
110 lines
2.5 KiB
/* |
|
* 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 "alloc.h" |
|
#include "hash.h" |
|
|
|
struct oidtree_iter_data { |
|
oidtree_iter fn; |
|
void *arg; |
|
size_t *last_nibble_at; |
|
int algo; |
|
uint8_t last_byte; |
|
}; |
|
|
|
void oidtree_init(struct oidtree *ot) |
|
{ |
|
cb_init(&ot->tree); |
|
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); |
|
} |
|
} |
|
|
|
void oidtree_insert(struct oidtree *ot, const struct object_id *oid) |
|
{ |
|
struct cb_node *on; |
|
struct object_id k; |
|
|
|
if (!oid->algo) |
|
BUG("oidtree_insert requires oid->algo"); |
|
|
|
on = mem_pool_alloc(&ot->mem_pool, sizeof(*on) + sizeof(*oid)); |
|
|
|
/* |
|
* Clear the padding and copy the result in separate steps to |
|
* respect the 4-byte alignment needed by struct object_id. |
|
*/ |
|
oidcpy_with_padding(&k, oid); |
|
memcpy(on->k, &k, sizeof(k)); |
|
|
|
/* |
|
* n.b. Current callers won't get us duplicates, here. If a |
|
* future caller causes duplicates, there'll be a a small leak |
|
* that won't be freed until oidtree_clear. Currently it's not |
|
* worth maintaining a free list |
|
*/ |
|
cb_insert(&ot->tree, on, sizeof(*oid)); |
|
} |
|
|
|
|
|
int oidtree_contains(struct oidtree *ot, const struct object_id *oid) |
|
{ |
|
struct object_id k; |
|
size_t klen = sizeof(k); |
|
|
|
oidcpy_with_padding(&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)); |
|
|
|
return cb_lookup(&ot->tree, (const uint8_t *)&k, klen) ? 1 : 0; |
|
} |
|
|
|
static enum cb_next iter(struct cb_node *n, void *arg) |
|
{ |
|
struct oidtree_iter_data *x = arg; |
|
struct object_id k; |
|
|
|
/* Copy to provide 4-byte alignment needed by struct object_id. */ |
|
memcpy(&k, n->k, sizeof(k)); |
|
|
|
if (x->algo != GIT_HASH_UNKNOWN && x->algo != k.algo) |
|
return CB_CONTINUE; |
|
|
|
if (x->last_nibble_at) { |
|
if ((k.hash[*x->last_nibble_at] ^ x->last_byte) & 0xf0) |
|
return CB_CONTINUE; |
|
} |
|
|
|
return x->fn(&k, x->arg); |
|
} |
|
|
|
void oidtree_each(struct oidtree *ot, const struct object_id *oid, |
|
size_t oidhexsz, oidtree_iter fn, void *arg) |
|
{ |
|
size_t klen = oidhexsz / 2; |
|
struct oidtree_iter_data x = { 0 }; |
|
assert(oidhexsz <= GIT_MAX_HEXSZ); |
|
|
|
x.fn = fn; |
|
x.arg = arg; |
|
x.algo = oid->algo; |
|
if (oidhexsz & 1) { |
|
x.last_byte = oid->hash[klen]; |
|
x.last_nibble_at = &klen; |
|
} |
|
cb_each(&ot->tree, (const uint8_t *)oid, klen, iter, &x); |
|
}
|
|
|