mirror of
https://github.com/git-for-windows/git.git
synced 2026-05-05 17:40:24 -05:00
Let's not walk off the end of the array. Also, avoid an early `malloc()` in `add_prefix_entry()` (avoiding reuse of the same data structure for lookup as for adding a new item), and strengthen the condition for the bug condition. Signed-off-by: Johannes Schindelin <johannes.schindelin@gmx.de>
110 lines
2.9 KiB
C
110 lines
2.9 KiB
C
#include "cache.h"
|
|
#include "prefix-map.h"
|
|
|
|
static int map_cmp(const void *unused_cmp_data,
|
|
const void *entry,
|
|
const void *entry_or_key,
|
|
const void *unused_keydata)
|
|
{
|
|
const struct prefix_map_entry *a = entry;
|
|
const struct prefix_map_entry *b = entry_or_key;
|
|
|
|
return a->prefix_length != b->prefix_length ||
|
|
strncmp(a->name, b->name, a->prefix_length);
|
|
}
|
|
|
|
static void add_prefix_entry(struct hashmap *map, const char *name,
|
|
size_t prefix_length, struct prefix_item *item)
|
|
{
|
|
struct prefix_map_entry *result = xmalloc(sizeof(*result));
|
|
result->name = name;
|
|
result->prefix_length = prefix_length;
|
|
result->item = item;
|
|
hashmap_entry_init(result, memhash(name, prefix_length));
|
|
hashmap_add(map, result);
|
|
}
|
|
|
|
static void init_prefix_map(struct prefix_map *prefix_map,
|
|
int min_prefix_length, int max_prefix_length)
|
|
{
|
|
hashmap_init(&prefix_map->map, map_cmp, NULL, 0);
|
|
prefix_map->min_length = min_prefix_length;
|
|
prefix_map->max_length = max_prefix_length;
|
|
}
|
|
|
|
static void add_prefix_item(struct prefix_map *prefix_map,
|
|
struct prefix_item *item)
|
|
{
|
|
struct prefix_map_entry e = { { NULL } }, *e2;
|
|
int j;
|
|
|
|
e.item = item;
|
|
e.name = item->name;
|
|
|
|
for (j = prefix_map->min_length;
|
|
j <= prefix_map->max_length && e.name[j]; j++) {
|
|
/* Avoid breaking UTF-8 multi-byte sequences */
|
|
if (!isascii(e.name[j]))
|
|
break;
|
|
|
|
e.prefix_length = j;
|
|
hashmap_entry_init(&e, memhash(e.name, j));
|
|
e2 = hashmap_get(&prefix_map->map, &e, NULL);
|
|
if (!e2) {
|
|
/* prefix is unique at this stage */
|
|
item->prefix_length = j;
|
|
add_prefix_entry(&prefix_map->map, e.name, j, item);
|
|
break;
|
|
}
|
|
|
|
if (!e2->item)
|
|
continue; /* non-unique prefix */
|
|
|
|
if (j != e2->item->prefix_length || memcmp(e.name, e2->name, j))
|
|
BUG("unexpected prefix length: %d != %d (%s != %s)",
|
|
j, (int)e2->item->prefix_length, e.name, e2->name);
|
|
|
|
/* skip common prefix */
|
|
for (; j < prefix_map->max_length && e.name[j]; j++) {
|
|
if (e.item->name[j] != e2->item->name[j])
|
|
break;
|
|
add_prefix_entry(&prefix_map->map, e.name, j + 1,
|
|
NULL);
|
|
}
|
|
|
|
/* e2 no longer refers to a unique prefix */
|
|
if (j < prefix_map->max_length && e2->name[j]) {
|
|
/* found a new unique prefix for e2's item */
|
|
e2->item->prefix_length = j + 1;
|
|
add_prefix_entry(&prefix_map->map, e2->name, j + 1,
|
|
e2->item);
|
|
}
|
|
else
|
|
e2->item->prefix_length = 0;
|
|
e2->item = NULL;
|
|
|
|
if (j < prefix_map->max_length && e.name[j]) {
|
|
/* found a unique prefix for the item */
|
|
e.item->prefix_length = j + 1;
|
|
add_prefix_entry(&prefix_map->map, e.name, j + 1,
|
|
e.item);
|
|
} else
|
|
/* item has no (short enough) unique prefix */
|
|
e.item->prefix_length = 0;
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
void find_unique_prefixes(struct prefix_item **list, size_t nr,
|
|
int min_length, int max_length)
|
|
{
|
|
int i;
|
|
struct prefix_map prefix_map;
|
|
|
|
init_prefix_map(&prefix_map, min_length, max_length);
|
|
for (i = 0; i < nr; i++)
|
|
add_prefix_item(&prefix_map, list[i]);
|
|
hashmap_free(&prefix_map.map, 1);
|
|
}
|