| /* |
| * 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" |
| |
| 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(&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 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)); |
| } |
| |
| bool oidtree_contains(struct oidtree *ot, const struct object_id *oid) |
| { |
| struct object_id k; |
| size_t klen = sizeof(k); |
| |
| 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)); |
| |
| return !!cb_lookup(&ot->tree, (const uint8_t *)&k, klen); |
| } |
| |
| 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_each_data *data = cb_data; |
| struct object_id k; |
| |
| /* Copy to provide 4-byte alignment needed by struct object_id. */ |
| memcpy(&k, n->k, sizeof(k)); |
| |
| if (data->algo != GIT_HASH_UNKNOWN && data->algo != k.algo) |
| return 0; |
| |
| if (data->last_nibble_at) { |
| if ((k.hash[*data->last_nibble_at] ^ data->last_byte) & 0xf0) |
| return 0; |
| } |
| |
| return data->cb(&k, 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); |
| } |