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.
87 lines
2.0 KiB
87 lines
2.0 KiB
#include "cache.h" |
|
#include "object.h" |
|
#include "decorate.h" |
|
|
|
int track_object_refs = 0; |
|
|
|
static struct decoration ref_decorate; |
|
|
|
struct object_refs *lookup_object_refs(struct object *base) |
|
{ |
|
return lookup_decoration(&ref_decorate, base); |
|
} |
|
|
|
static void add_object_refs(struct object *obj, struct object_refs *refs) |
|
{ |
|
if (add_decoration(&ref_decorate, obj, refs)) |
|
die("object %s tried to add refs twice!", sha1_to_hex(obj->sha1)); |
|
} |
|
|
|
struct object_refs *alloc_object_refs(unsigned count) |
|
{ |
|
struct object_refs *refs; |
|
size_t size = sizeof(*refs) + count*sizeof(struct object *); |
|
|
|
refs = xcalloc(1, size); |
|
refs->count = count; |
|
return refs; |
|
} |
|
|
|
static int compare_object_pointers(const void *a, const void *b) |
|
{ |
|
const struct object * const *pa = a; |
|
const struct object * const *pb = b; |
|
if (*pa == *pb) |
|
return 0; |
|
else if (*pa < *pb) |
|
return -1; |
|
else |
|
return 1; |
|
} |
|
|
|
void set_object_refs(struct object *obj, struct object_refs *refs) |
|
{ |
|
unsigned int i, j; |
|
|
|
/* Do not install empty list of references */ |
|
if (refs->count < 1) { |
|
free(refs); |
|
return; |
|
} |
|
|
|
/* Sort the list and filter out duplicates */ |
|
qsort(refs->ref, refs->count, sizeof(refs->ref[0]), |
|
compare_object_pointers); |
|
for (i = j = 1; i < refs->count; i++) { |
|
if (refs->ref[i] != refs->ref[i - 1]) |
|
refs->ref[j++] = refs->ref[i]; |
|
} |
|
if (j < refs->count) { |
|
/* Duplicates were found - reallocate list */ |
|
size_t size = sizeof(*refs) + j*sizeof(struct object *); |
|
refs->count = j; |
|
refs = xrealloc(refs, size); |
|
} |
|
|
|
for (i = 0; i < refs->count; i++) |
|
refs->ref[i]->used = 1; |
|
add_object_refs(obj, refs); |
|
} |
|
|
|
void mark_reachable(struct object *obj, unsigned int mask) |
|
{ |
|
const struct object_refs *refs; |
|
|
|
if (!track_object_refs) |
|
die("cannot do reachability with object refs turned off"); |
|
/* If we've been here already, don't bother */ |
|
if (obj->flags & mask) |
|
return; |
|
obj->flags |= mask; |
|
refs = lookup_object_refs(obj); |
|
if (refs) { |
|
unsigned i; |
|
for (i = 0; i < refs->count; i++) |
|
mark_reachable(refs->ref[i], mask); |
|
} |
|
}
|
|
|