commit-graph: create write_commit_graph_context
The write_commit_graph() method is too large and complex. To simplify it, we should extract several helper functions. However, we will risk repeating a lot of declarations related to progress incidators and object id or commit lists. Create a new write_commit_graph_context struct that contains the core data structures used in this process. Replace the other local variables with the values inside the context object. Following this change, we will start to lift code segments wholesale out of the write_commit_graph() method and into helper functions. Signed-off-by: Derrick Stolee <dstolee@microsoft.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
parent
10bd0be173
commit
c9905beade
386
commit-graph.c
386
commit-graph.c
@ -518,14 +518,38 @@ struct tree *get_commit_tree_in_graph(struct repository *r, const struct commit
|
|||||||
return get_commit_tree_in_graph_one(r, r->objects->commit_graph, c);
|
return get_commit_tree_in_graph_one(r, r->objects->commit_graph, c);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct packed_commit_list {
|
||||||
|
struct commit **list;
|
||||||
|
int nr;
|
||||||
|
int alloc;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct packed_oid_list {
|
||||||
|
struct object_id *list;
|
||||||
|
int nr;
|
||||||
|
int alloc;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct write_commit_graph_context {
|
||||||
|
struct repository *r;
|
||||||
|
const char *obj_dir;
|
||||||
|
char *graph_name;
|
||||||
|
struct packed_oid_list oids;
|
||||||
|
struct packed_commit_list commits;
|
||||||
|
int num_extra_edges;
|
||||||
|
unsigned long approx_nr_objects;
|
||||||
|
struct progress *progress;
|
||||||
|
int progress_done;
|
||||||
|
uint64_t progress_cnt;
|
||||||
|
unsigned append:1,
|
||||||
|
report_progress:1;
|
||||||
|
};
|
||||||
|
|
||||||
static void write_graph_chunk_fanout(struct hashfile *f,
|
static void write_graph_chunk_fanout(struct hashfile *f,
|
||||||
struct commit **commits,
|
struct write_commit_graph_context *ctx)
|
||||||
int nr_commits,
|
|
||||||
struct progress *progress,
|
|
||||||
uint64_t *progress_cnt)
|
|
||||||
{
|
{
|
||||||
int i, count = 0;
|
int i, count = 0;
|
||||||
struct commit **list = commits;
|
struct commit **list = ctx->commits.list;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Write the first-level table (the list is sorted,
|
* Write the first-level table (the list is sorted,
|
||||||
@ -533,10 +557,10 @@ static void write_graph_chunk_fanout(struct hashfile *f,
|
|||||||
* having to do eight extra binary search iterations).
|
* having to do eight extra binary search iterations).
|
||||||
*/
|
*/
|
||||||
for (i = 0; i < 256; i++) {
|
for (i = 0; i < 256; i++) {
|
||||||
while (count < nr_commits) {
|
while (count < ctx->commits.nr) {
|
||||||
if ((*list)->object.oid.hash[0] != i)
|
if ((*list)->object.oid.hash[0] != i)
|
||||||
break;
|
break;
|
||||||
display_progress(progress, ++*progress_cnt);
|
display_progress(ctx->progress, ++ctx->progress_cnt);
|
||||||
count++;
|
count++;
|
||||||
list++;
|
list++;
|
||||||
}
|
}
|
||||||
@ -546,14 +570,12 @@ static void write_graph_chunk_fanout(struct hashfile *f,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void write_graph_chunk_oids(struct hashfile *f, int hash_len,
|
static void write_graph_chunk_oids(struct hashfile *f, int hash_len,
|
||||||
struct commit **commits, int nr_commits,
|
struct write_commit_graph_context *ctx)
|
||||||
struct progress *progress,
|
|
||||||
uint64_t *progress_cnt)
|
|
||||||
{
|
{
|
||||||
struct commit **list = commits;
|
struct commit **list = ctx->commits.list;
|
||||||
int count;
|
int count;
|
||||||
for (count = 0; count < nr_commits; count++, list++) {
|
for (count = 0; count < ctx->commits.nr; count++, list++) {
|
||||||
display_progress(progress, ++*progress_cnt);
|
display_progress(ctx->progress, ++ctx->progress_cnt);
|
||||||
hashwrite(f, (*list)->object.oid.hash, (int)hash_len);
|
hashwrite(f, (*list)->object.oid.hash, (int)hash_len);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -565,19 +587,17 @@ static const unsigned char *commit_to_sha1(size_t index, void *table)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void write_graph_chunk_data(struct hashfile *f, int hash_len,
|
static void write_graph_chunk_data(struct hashfile *f, int hash_len,
|
||||||
struct commit **commits, int nr_commits,
|
struct write_commit_graph_context *ctx)
|
||||||
struct progress *progress,
|
|
||||||
uint64_t *progress_cnt)
|
|
||||||
{
|
{
|
||||||
struct commit **list = commits;
|
struct commit **list = ctx->commits.list;
|
||||||
struct commit **last = commits + nr_commits;
|
struct commit **last = ctx->commits.list + ctx->commits.nr;
|
||||||
uint32_t num_extra_edges = 0;
|
uint32_t num_extra_edges = 0;
|
||||||
|
|
||||||
while (list < last) {
|
while (list < last) {
|
||||||
struct commit_list *parent;
|
struct commit_list *parent;
|
||||||
int edge_value;
|
int edge_value;
|
||||||
uint32_t packedDate[2];
|
uint32_t packedDate[2];
|
||||||
display_progress(progress, ++*progress_cnt);
|
display_progress(ctx->progress, ++ctx->progress_cnt);
|
||||||
|
|
||||||
parse_commit_no_graph(*list);
|
parse_commit_no_graph(*list);
|
||||||
hashwrite(f, get_commit_tree_oid(*list)->hash, hash_len);
|
hashwrite(f, get_commit_tree_oid(*list)->hash, hash_len);
|
||||||
@ -588,8 +608,8 @@ static void write_graph_chunk_data(struct hashfile *f, int hash_len,
|
|||||||
edge_value = GRAPH_PARENT_NONE;
|
edge_value = GRAPH_PARENT_NONE;
|
||||||
else {
|
else {
|
||||||
edge_value = sha1_pos(parent->item->object.oid.hash,
|
edge_value = sha1_pos(parent->item->object.oid.hash,
|
||||||
commits,
|
ctx->commits.list,
|
||||||
nr_commits,
|
ctx->commits.nr,
|
||||||
commit_to_sha1);
|
commit_to_sha1);
|
||||||
|
|
||||||
if (edge_value < 0)
|
if (edge_value < 0)
|
||||||
@ -609,8 +629,8 @@ static void write_graph_chunk_data(struct hashfile *f, int hash_len,
|
|||||||
edge_value = GRAPH_EXTRA_EDGES_NEEDED | num_extra_edges;
|
edge_value = GRAPH_EXTRA_EDGES_NEEDED | num_extra_edges;
|
||||||
else {
|
else {
|
||||||
edge_value = sha1_pos(parent->item->object.oid.hash,
|
edge_value = sha1_pos(parent->item->object.oid.hash,
|
||||||
commits,
|
ctx->commits.list,
|
||||||
nr_commits,
|
ctx->commits.nr,
|
||||||
commit_to_sha1);
|
commit_to_sha1);
|
||||||
if (edge_value < 0)
|
if (edge_value < 0)
|
||||||
BUG("missing parent %s for commit %s",
|
BUG("missing parent %s for commit %s",
|
||||||
@ -642,19 +662,16 @@ static void write_graph_chunk_data(struct hashfile *f, int hash_len,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void write_graph_chunk_extra_edges(struct hashfile *f,
|
static void write_graph_chunk_extra_edges(struct hashfile *f,
|
||||||
struct commit **commits,
|
struct write_commit_graph_context *ctx)
|
||||||
int nr_commits,
|
|
||||||
struct progress *progress,
|
|
||||||
uint64_t *progress_cnt)
|
|
||||||
{
|
{
|
||||||
struct commit **list = commits;
|
struct commit **list = ctx->commits.list;
|
||||||
struct commit **last = commits + nr_commits;
|
struct commit **last = ctx->commits.list + ctx->commits.nr;
|
||||||
struct commit_list *parent;
|
struct commit_list *parent;
|
||||||
|
|
||||||
while (list < last) {
|
while (list < last) {
|
||||||
int num_parents = 0;
|
int num_parents = 0;
|
||||||
|
|
||||||
display_progress(progress, ++*progress_cnt);
|
display_progress(ctx->progress, ++ctx->progress_cnt);
|
||||||
|
|
||||||
for (parent = (*list)->parents; num_parents < 3 && parent;
|
for (parent = (*list)->parents; num_parents < 3 && parent;
|
||||||
parent = parent->next)
|
parent = parent->next)
|
||||||
@ -668,8 +685,8 @@ static void write_graph_chunk_extra_edges(struct hashfile *f,
|
|||||||
/* Since num_parents > 2, this initializer is safe. */
|
/* Since num_parents > 2, this initializer is safe. */
|
||||||
for (parent = (*list)->parents->next; parent; parent = parent->next) {
|
for (parent = (*list)->parents->next; parent; parent = parent->next) {
|
||||||
int edge_value = sha1_pos(parent->item->object.oid.hash,
|
int edge_value = sha1_pos(parent->item->object.oid.hash,
|
||||||
commits,
|
ctx->commits.list,
|
||||||
nr_commits,
|
ctx->commits.nr,
|
||||||
commit_to_sha1);
|
commit_to_sha1);
|
||||||
|
|
||||||
if (edge_value < 0)
|
if (edge_value < 0)
|
||||||
@ -693,125 +710,111 @@ static int commit_compare(const void *_a, const void *_b)
|
|||||||
return oidcmp(a, b);
|
return oidcmp(a, b);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct packed_commit_list {
|
|
||||||
struct commit **list;
|
|
||||||
int nr;
|
|
||||||
int alloc;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct packed_oid_list {
|
|
||||||
struct object_id *list;
|
|
||||||
int nr;
|
|
||||||
int alloc;
|
|
||||||
struct progress *progress;
|
|
||||||
int progress_done;
|
|
||||||
};
|
|
||||||
|
|
||||||
static int add_packed_commits(const struct object_id *oid,
|
static int add_packed_commits(const struct object_id *oid,
|
||||||
struct packed_git *pack,
|
struct packed_git *pack,
|
||||||
uint32_t pos,
|
uint32_t pos,
|
||||||
void *data)
|
void *data)
|
||||||
{
|
{
|
||||||
struct packed_oid_list *list = (struct packed_oid_list*)data;
|
struct write_commit_graph_context *ctx = (struct write_commit_graph_context*)data;
|
||||||
enum object_type type;
|
enum object_type type;
|
||||||
off_t offset = nth_packed_object_offset(pack, pos);
|
off_t offset = nth_packed_object_offset(pack, pos);
|
||||||
struct object_info oi = OBJECT_INFO_INIT;
|
struct object_info oi = OBJECT_INFO_INIT;
|
||||||
|
|
||||||
if (list->progress)
|
if (ctx->progress)
|
||||||
display_progress(list->progress, ++list->progress_done);
|
display_progress(ctx->progress, ++ctx->progress_done);
|
||||||
|
|
||||||
oi.typep = &type;
|
oi.typep = &type;
|
||||||
if (packed_object_info(the_repository, pack, offset, &oi) < 0)
|
if (packed_object_info(ctx->r, pack, offset, &oi) < 0)
|
||||||
die(_("unable to get type of object %s"), oid_to_hex(oid));
|
die(_("unable to get type of object %s"), oid_to_hex(oid));
|
||||||
|
|
||||||
if (type != OBJ_COMMIT)
|
if (type != OBJ_COMMIT)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
ALLOC_GROW(list->list, list->nr + 1, list->alloc);
|
ALLOC_GROW(ctx->oids.list, ctx->oids.nr + 1, ctx->oids.alloc);
|
||||||
oidcpy(&(list->list[list->nr]), oid);
|
oidcpy(&(ctx->oids.list[ctx->oids.nr]), oid);
|
||||||
list->nr++;
|
ctx->oids.nr++;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void add_missing_parents(struct packed_oid_list *oids, struct commit *commit)
|
static void add_missing_parents(struct write_commit_graph_context *ctx, struct commit *commit)
|
||||||
{
|
{
|
||||||
struct commit_list *parent;
|
struct commit_list *parent;
|
||||||
for (parent = commit->parents; parent; parent = parent->next) {
|
for (parent = commit->parents; parent; parent = parent->next) {
|
||||||
if (!(parent->item->object.flags & UNINTERESTING)) {
|
if (!(parent->item->object.flags & UNINTERESTING)) {
|
||||||
ALLOC_GROW(oids->list, oids->nr + 1, oids->alloc);
|
ALLOC_GROW(ctx->oids.list, ctx->oids.nr + 1, ctx->oids.alloc);
|
||||||
oidcpy(&oids->list[oids->nr], &(parent->item->object.oid));
|
oidcpy(&ctx->oids.list[ctx->oids.nr], &(parent->item->object.oid));
|
||||||
oids->nr++;
|
ctx->oids.nr++;
|
||||||
parent->item->object.flags |= UNINTERESTING;
|
parent->item->object.flags |= UNINTERESTING;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void close_reachable(struct packed_oid_list *oids, int report_progress)
|
static void close_reachable(struct write_commit_graph_context *ctx)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
struct commit *commit;
|
struct commit *commit;
|
||||||
struct progress *progress = NULL;
|
|
||||||
|
|
||||||
if (report_progress)
|
if (ctx->report_progress)
|
||||||
progress = start_delayed_progress(
|
ctx->progress = start_delayed_progress(
|
||||||
_("Loading known commits in commit graph"), oids->nr);
|
_("Loading known commits in commit graph"),
|
||||||
for (i = 0; i < oids->nr; i++) {
|
ctx->oids.nr);
|
||||||
display_progress(progress, i + 1);
|
for (i = 0; i < ctx->oids.nr; i++) {
|
||||||
commit = lookup_commit(the_repository, &oids->list[i]);
|
display_progress(ctx->progress, i + 1);
|
||||||
|
commit = lookup_commit(ctx->r, &ctx->oids.list[i]);
|
||||||
if (commit)
|
if (commit)
|
||||||
commit->object.flags |= UNINTERESTING;
|
commit->object.flags |= UNINTERESTING;
|
||||||
}
|
}
|
||||||
stop_progress(&progress);
|
stop_progress(&ctx->progress);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* As this loop runs, oids->nr may grow, but not more
|
* As this loop runs, ctx->oids.nr may grow, but not more
|
||||||
* than the number of missing commits in the reachable
|
* than the number of missing commits in the reachable
|
||||||
* closure.
|
* closure.
|
||||||
*/
|
*/
|
||||||
if (report_progress)
|
if (ctx->report_progress)
|
||||||
progress = start_delayed_progress(
|
ctx->progress = start_delayed_progress(
|
||||||
_("Expanding reachable commits in commit graph"), oids->nr);
|
_("Expanding reachable commits in commit graph"),
|
||||||
for (i = 0; i < oids->nr; i++) {
|
ctx->oids.nr);
|
||||||
display_progress(progress, i + 1);
|
for (i = 0; i < ctx->oids.nr; i++) {
|
||||||
commit = lookup_commit(the_repository, &oids->list[i]);
|
display_progress(ctx->progress, i + 1);
|
||||||
|
commit = lookup_commit(ctx->r, &ctx->oids.list[i]);
|
||||||
|
|
||||||
if (commit && !parse_commit_no_graph(commit))
|
if (commit && !parse_commit_no_graph(commit))
|
||||||
add_missing_parents(oids, commit);
|
add_missing_parents(ctx, commit);
|
||||||
}
|
}
|
||||||
stop_progress(&progress);
|
stop_progress(&ctx->progress);
|
||||||
|
|
||||||
if (report_progress)
|
if (ctx->report_progress)
|
||||||
progress = start_delayed_progress(
|
ctx->progress = start_delayed_progress(
|
||||||
_("Clearing commit marks in commit graph"), oids->nr);
|
_("Clearing commit marks in commit graph"),
|
||||||
for (i = 0; i < oids->nr; i++) {
|
ctx->oids.nr);
|
||||||
display_progress(progress, i + 1);
|
for (i = 0; i < ctx->oids.nr; i++) {
|
||||||
commit = lookup_commit(the_repository, &oids->list[i]);
|
display_progress(ctx->progress, i + 1);
|
||||||
|
commit = lookup_commit(ctx->r, &ctx->oids.list[i]);
|
||||||
|
|
||||||
if (commit)
|
if (commit)
|
||||||
commit->object.flags &= ~UNINTERESTING;
|
commit->object.flags &= ~UNINTERESTING;
|
||||||
}
|
}
|
||||||
stop_progress(&progress);
|
stop_progress(&ctx->progress);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void compute_generation_numbers(struct packed_commit_list* commits,
|
static void compute_generation_numbers(struct write_commit_graph_context *ctx)
|
||||||
int report_progress)
|
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
struct commit_list *list = NULL;
|
struct commit_list *list = NULL;
|
||||||
struct progress *progress = NULL;
|
|
||||||
|
|
||||||
if (report_progress)
|
if (ctx->report_progress)
|
||||||
progress = start_progress(
|
ctx->progress = start_progress(
|
||||||
_("Computing commit graph generation numbers"),
|
_("Computing commit graph generation numbers"),
|
||||||
commits->nr);
|
ctx->commits.nr);
|
||||||
for (i = 0; i < commits->nr; i++) {
|
for (i = 0; i < ctx->commits.nr; i++) {
|
||||||
display_progress(progress, i + 1);
|
display_progress(ctx->progress, i + 1);
|
||||||
if (commits->list[i]->generation != GENERATION_NUMBER_INFINITY &&
|
if (ctx->commits.list[i]->generation != GENERATION_NUMBER_INFINITY &&
|
||||||
commits->list[i]->generation != GENERATION_NUMBER_ZERO)
|
ctx->commits.list[i]->generation != GENERATION_NUMBER_ZERO)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
commit_list_insert(commits->list[i], &list);
|
commit_list_insert(ctx->commits.list[i], &list);
|
||||||
while (list) {
|
while (list) {
|
||||||
struct commit *current = list->item;
|
struct commit *current = list->item;
|
||||||
struct commit_list *parent;
|
struct commit_list *parent;
|
||||||
@ -838,7 +841,7 @@ static void compute_generation_numbers(struct packed_commit_list* commits,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
stop_progress(&progress);
|
stop_progress(&ctx->progress);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int add_ref_to_list(const char *refname,
|
static int add_ref_to_list(const char *refname,
|
||||||
@ -869,8 +872,7 @@ int write_commit_graph(const char *obj_dir,
|
|||||||
struct string_list *commit_hex,
|
struct string_list *commit_hex,
|
||||||
unsigned int flags)
|
unsigned int flags)
|
||||||
{
|
{
|
||||||
struct packed_oid_list oids;
|
struct write_commit_graph_context *ctx;
|
||||||
struct packed_commit_list commits;
|
|
||||||
struct hashfile *f;
|
struct hashfile *f;
|
||||||
uint32_t i, count_distinct = 0;
|
uint32_t i, count_distinct = 0;
|
||||||
char *graph_name = NULL;
|
char *graph_name = NULL;
|
||||||
@ -878,44 +880,38 @@ int write_commit_graph(const char *obj_dir,
|
|||||||
uint32_t chunk_ids[5];
|
uint32_t chunk_ids[5];
|
||||||
uint64_t chunk_offsets[5];
|
uint64_t chunk_offsets[5];
|
||||||
int num_chunks;
|
int num_chunks;
|
||||||
int num_extra_edges;
|
|
||||||
struct commit_list *parent;
|
struct commit_list *parent;
|
||||||
struct progress *progress = NULL;
|
|
||||||
const unsigned hashsz = the_hash_algo->rawsz;
|
const unsigned hashsz = the_hash_algo->rawsz;
|
||||||
uint64_t progress_cnt = 0;
|
|
||||||
struct strbuf progress_title = STRBUF_INIT;
|
struct strbuf progress_title = STRBUF_INIT;
|
||||||
unsigned long approx_nr_objects;
|
|
||||||
int res = 0;
|
int res = 0;
|
||||||
int append = flags & COMMIT_GRAPH_APPEND;
|
|
||||||
int report_progress = flags & COMMIT_GRAPH_PROGRESS;
|
|
||||||
|
|
||||||
if (!commit_graph_compatible(the_repository))
|
if (!commit_graph_compatible(the_repository))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
oids.nr = 0;
|
ctx = xcalloc(1, sizeof(struct write_commit_graph_context));
|
||||||
approx_nr_objects = approximate_object_count();
|
ctx->r = the_repository;
|
||||||
oids.alloc = approx_nr_objects / 32;
|
ctx->obj_dir = obj_dir;
|
||||||
oids.progress = NULL;
|
ctx->append = flags & COMMIT_GRAPH_APPEND ? 1 : 0;
|
||||||
oids.progress_done = 0;
|
ctx->report_progress = flags & COMMIT_GRAPH_PROGRESS ? 1 : 0;
|
||||||
commits.list = NULL;
|
|
||||||
|
|
||||||
if (append) {
|
ctx->approx_nr_objects = approximate_object_count();
|
||||||
prepare_commit_graph_one(the_repository, obj_dir);
|
ctx->oids.alloc = ctx->approx_nr_objects / 32;
|
||||||
if (the_repository->objects->commit_graph)
|
|
||||||
oids.alloc += the_repository->objects->commit_graph->num_commits;
|
if (ctx->append) {
|
||||||
|
prepare_commit_graph_one(ctx->r, ctx->obj_dir);
|
||||||
|
if (ctx->r->objects->commit_graph)
|
||||||
|
ctx->oids.alloc += ctx->r->objects->commit_graph->num_commits;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (oids.alloc < 1024)
|
if (ctx->oids.alloc < 1024)
|
||||||
oids.alloc = 1024;
|
ctx->oids.alloc = 1024;
|
||||||
ALLOC_ARRAY(oids.list, oids.alloc);
|
ALLOC_ARRAY(ctx->oids.list, ctx->oids.alloc);
|
||||||
|
|
||||||
if (append && the_repository->objects->commit_graph) {
|
if (ctx->append && ctx->r->objects->commit_graph) {
|
||||||
struct commit_graph *commit_graph =
|
struct commit_graph *g = ctx->r->objects->commit_graph;
|
||||||
the_repository->objects->commit_graph;
|
for (i = 0; i < g->num_commits; i++) {
|
||||||
for (i = 0; i < commit_graph->num_commits; i++) {
|
const unsigned char *hash = g->chunk_oid_lookup + g->hash_len * i;
|
||||||
const unsigned char *hash = commit_graph->chunk_oid_lookup +
|
hashcpy(ctx->oids.list[ctx->oids.nr++].hash, hash);
|
||||||
commit_graph->hash_len * i;
|
|
||||||
hashcpy(oids.list[oids.nr++].hash, hash);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -924,14 +920,14 @@ int write_commit_graph(const char *obj_dir,
|
|||||||
int dirlen;
|
int dirlen;
|
||||||
strbuf_addf(&packname, "%s/pack/", obj_dir);
|
strbuf_addf(&packname, "%s/pack/", obj_dir);
|
||||||
dirlen = packname.len;
|
dirlen = packname.len;
|
||||||
if (report_progress) {
|
if (ctx->report_progress) {
|
||||||
strbuf_addf(&progress_title,
|
strbuf_addf(&progress_title,
|
||||||
Q_("Finding commits for commit graph in %d pack",
|
Q_("Finding commits for commit graph in %d pack",
|
||||||
"Finding commits for commit graph in %d packs",
|
"Finding commits for commit graph in %d packs",
|
||||||
pack_indexes->nr),
|
pack_indexes->nr),
|
||||||
pack_indexes->nr);
|
pack_indexes->nr);
|
||||||
oids.progress = start_delayed_progress(progress_title.buf, 0);
|
ctx->progress = start_delayed_progress(progress_title.buf, 0);
|
||||||
oids.progress_done = 0;
|
ctx->progress_done = 0;
|
||||||
}
|
}
|
||||||
for (i = 0; i < pack_indexes->nr; i++) {
|
for (i = 0; i < pack_indexes->nr; i++) {
|
||||||
struct packed_git *p;
|
struct packed_git *p;
|
||||||
@ -948,75 +944,76 @@ int write_commit_graph(const char *obj_dir,
|
|||||||
res = -1;
|
res = -1;
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
for_each_object_in_pack(p, add_packed_commits, &oids,
|
for_each_object_in_pack(p, add_packed_commits, ctx,
|
||||||
FOR_EACH_OBJECT_PACK_ORDER);
|
FOR_EACH_OBJECT_PACK_ORDER);
|
||||||
close_pack(p);
|
close_pack(p);
|
||||||
free(p);
|
free(p);
|
||||||
}
|
}
|
||||||
stop_progress(&oids.progress);
|
stop_progress(&ctx->progress);
|
||||||
strbuf_reset(&progress_title);
|
strbuf_reset(&progress_title);
|
||||||
strbuf_release(&packname);
|
strbuf_release(&packname);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (commit_hex) {
|
if (commit_hex) {
|
||||||
if (report_progress) {
|
if (ctx->report_progress) {
|
||||||
strbuf_addf(&progress_title,
|
strbuf_addf(&progress_title,
|
||||||
Q_("Finding commits for commit graph from %d ref",
|
Q_("Finding commits for commit graph from %d ref",
|
||||||
"Finding commits for commit graph from %d refs",
|
"Finding commits for commit graph from %d refs",
|
||||||
commit_hex->nr),
|
commit_hex->nr),
|
||||||
commit_hex->nr);
|
commit_hex->nr);
|
||||||
progress = start_delayed_progress(progress_title.buf,
|
ctx->progress = start_delayed_progress(
|
||||||
commit_hex->nr);
|
progress_title.buf,
|
||||||
|
commit_hex->nr);
|
||||||
}
|
}
|
||||||
for (i = 0; i < commit_hex->nr; i++) {
|
for (i = 0; i < commit_hex->nr; i++) {
|
||||||
const char *end;
|
const char *end;
|
||||||
struct object_id oid;
|
struct object_id oid;
|
||||||
struct commit *result;
|
struct commit *result;
|
||||||
|
|
||||||
display_progress(progress, i + 1);
|
display_progress(ctx->progress, i + 1);
|
||||||
if (commit_hex->items[i].string &&
|
if (commit_hex->items[i].string &&
|
||||||
parse_oid_hex(commit_hex->items[i].string, &oid, &end))
|
parse_oid_hex(commit_hex->items[i].string, &oid, &end))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
result = lookup_commit_reference_gently(the_repository, &oid, 1);
|
result = lookup_commit_reference_gently(ctx->r, &oid, 1);
|
||||||
|
|
||||||
if (result) {
|
if (result) {
|
||||||
ALLOC_GROW(oids.list, oids.nr + 1, oids.alloc);
|
ALLOC_GROW(ctx->oids.list, ctx->oids.nr + 1, ctx->oids.alloc);
|
||||||
oidcpy(&oids.list[oids.nr], &(result->object.oid));
|
oidcpy(&ctx->oids.list[ctx->oids.nr], &(result->object.oid));
|
||||||
oids.nr++;
|
ctx->oids.nr++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
stop_progress(&progress);
|
stop_progress(&ctx->progress);
|
||||||
strbuf_reset(&progress_title);
|
strbuf_reset(&progress_title);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!pack_indexes && !commit_hex) {
|
if (!pack_indexes && !commit_hex) {
|
||||||
if (report_progress)
|
if (ctx->report_progress)
|
||||||
oids.progress = start_delayed_progress(
|
ctx->progress = start_delayed_progress(
|
||||||
_("Finding commits for commit graph among packed objects"),
|
_("Finding commits for commit graph among packed objects"),
|
||||||
approx_nr_objects);
|
ctx->approx_nr_objects);
|
||||||
for_each_packed_object(add_packed_commits, &oids,
|
for_each_packed_object(add_packed_commits, ctx,
|
||||||
FOR_EACH_OBJECT_PACK_ORDER);
|
FOR_EACH_OBJECT_PACK_ORDER);
|
||||||
if (oids.progress_done < approx_nr_objects)
|
if (ctx->progress_done < ctx->approx_nr_objects)
|
||||||
display_progress(oids.progress, approx_nr_objects);
|
display_progress(ctx->progress, ctx->approx_nr_objects);
|
||||||
stop_progress(&oids.progress);
|
stop_progress(&ctx->progress);
|
||||||
}
|
}
|
||||||
|
|
||||||
close_reachable(&oids, report_progress);
|
close_reachable(ctx);
|
||||||
|
|
||||||
if (report_progress)
|
if (ctx->report_progress)
|
||||||
progress = start_delayed_progress(
|
ctx->progress = start_delayed_progress(
|
||||||
_("Counting distinct commits in commit graph"),
|
_("Counting distinct commits in commit graph"),
|
||||||
oids.nr);
|
ctx->oids.nr);
|
||||||
display_progress(progress, 0); /* TODO: Measure QSORT() progress */
|
display_progress(ctx->progress, 0); /* TODO: Measure QSORT() progress */
|
||||||
QSORT(oids.list, oids.nr, commit_compare);
|
QSORT(ctx->oids.list, ctx->oids.nr, commit_compare);
|
||||||
count_distinct = 1;
|
count_distinct = 1;
|
||||||
for (i = 1; i < oids.nr; i++) {
|
for (i = 1; i < ctx->oids.nr; i++) {
|
||||||
display_progress(progress, i + 1);
|
display_progress(ctx->progress, i + 1);
|
||||||
if (!oideq(&oids.list[i - 1], &oids.list[i]))
|
if (!oideq(&ctx->oids.list[i - 1], &ctx->oids.list[i]))
|
||||||
count_distinct++;
|
count_distinct++;
|
||||||
}
|
}
|
||||||
stop_progress(&progress);
|
stop_progress(&ctx->progress);
|
||||||
|
|
||||||
if (count_distinct >= GRAPH_EDGE_LAST_MASK) {
|
if (count_distinct >= GRAPH_EDGE_LAST_MASK) {
|
||||||
error(_("the commit graph format cannot write %d commits"), count_distinct);
|
error(_("the commit graph format cannot write %d commits"), count_distinct);
|
||||||
@ -1024,54 +1021,54 @@ int write_commit_graph(const char *obj_dir,
|
|||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
commits.nr = 0;
|
ctx->commits.alloc = count_distinct;
|
||||||
commits.alloc = count_distinct;
|
ALLOC_ARRAY(ctx->commits.list, ctx->commits.alloc);
|
||||||
ALLOC_ARRAY(commits.list, commits.alloc);
|
|
||||||
|
|
||||||
num_extra_edges = 0;
|
ctx->num_extra_edges = 0;
|
||||||
if (report_progress)
|
if (ctx->report_progress)
|
||||||
progress = start_delayed_progress(
|
ctx->progress = start_delayed_progress(
|
||||||
_("Finding extra edges in commit graph"),
|
_("Finding extra edges in commit graph"),
|
||||||
oids.nr);
|
ctx->oids.nr);
|
||||||
for (i = 0; i < oids.nr; i++) {
|
for (i = 0; i < ctx->oids.nr; i++) {
|
||||||
int num_parents = 0;
|
int num_parents = 0;
|
||||||
display_progress(progress, i + 1);
|
display_progress(ctx->progress, i + 1);
|
||||||
if (i > 0 && oideq(&oids.list[i - 1], &oids.list[i]))
|
if (i > 0 && oideq(&ctx->oids.list[i - 1], &ctx->oids.list[i]))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
commits.list[commits.nr] = lookup_commit(the_repository, &oids.list[i]);
|
ctx->commits.list[ctx->commits.nr] = lookup_commit(ctx->r, &ctx->oids.list[i]);
|
||||||
parse_commit_no_graph(commits.list[commits.nr]);
|
parse_commit_no_graph(ctx->commits.list[ctx->commits.nr]);
|
||||||
|
|
||||||
for (parent = commits.list[commits.nr]->parents;
|
for (parent = ctx->commits.list[ctx->commits.nr]->parents;
|
||||||
parent; parent = parent->next)
|
parent; parent = parent->next)
|
||||||
num_parents++;
|
num_parents++;
|
||||||
|
|
||||||
if (num_parents > 2)
|
if (num_parents > 2)
|
||||||
num_extra_edges += num_parents - 1;
|
ctx->num_extra_edges += num_parents - 1;
|
||||||
|
|
||||||
commits.nr++;
|
ctx->commits.nr++;
|
||||||
}
|
}
|
||||||
num_chunks = num_extra_edges ? 4 : 3;
|
stop_progress(&ctx->progress);
|
||||||
stop_progress(&progress);
|
|
||||||
|
|
||||||
if (commits.nr >= GRAPH_EDGE_LAST_MASK) {
|
if (ctx->commits.nr >= GRAPH_EDGE_LAST_MASK) {
|
||||||
error(_("too many commits to write graph"));
|
error(_("too many commits to write graph"));
|
||||||
res = -1;
|
res = -1;
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
compute_generation_numbers(&commits, report_progress);
|
compute_generation_numbers(ctx);
|
||||||
|
|
||||||
graph_name = get_commit_graph_filename(obj_dir);
|
num_chunks = ctx->num_extra_edges ? 4 : 3;
|
||||||
if (safe_create_leading_directories(graph_name)) {
|
|
||||||
UNLEAK(graph_name);
|
ctx->graph_name = get_commit_graph_filename(ctx->obj_dir);
|
||||||
|
if (safe_create_leading_directories(ctx->graph_name)) {
|
||||||
|
UNLEAK(ctx->graph_name);
|
||||||
error(_("unable to create leading directories of %s"),
|
error(_("unable to create leading directories of %s"),
|
||||||
graph_name);
|
ctx->graph_name);
|
||||||
res = -1;
|
res = -1;
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
hold_lock_file_for_update(&lk, graph_name, LOCK_DIE_ON_ERROR);
|
hold_lock_file_for_update(&lk, ctx->graph_name, LOCK_DIE_ON_ERROR);
|
||||||
f = hashfd(lk.tempfile->fd, lk.tempfile->filename.buf);
|
f = hashfd(lk.tempfile->fd, lk.tempfile->filename.buf);
|
||||||
|
|
||||||
hashwrite_be32(f, GRAPH_SIGNATURE);
|
hashwrite_be32(f, GRAPH_SIGNATURE);
|
||||||
@ -1084,7 +1081,7 @@ int write_commit_graph(const char *obj_dir,
|
|||||||
chunk_ids[0] = GRAPH_CHUNKID_OIDFANOUT;
|
chunk_ids[0] = GRAPH_CHUNKID_OIDFANOUT;
|
||||||
chunk_ids[1] = GRAPH_CHUNKID_OIDLOOKUP;
|
chunk_ids[1] = GRAPH_CHUNKID_OIDLOOKUP;
|
||||||
chunk_ids[2] = GRAPH_CHUNKID_DATA;
|
chunk_ids[2] = GRAPH_CHUNKID_DATA;
|
||||||
if (num_extra_edges)
|
if (ctx->num_extra_edges)
|
||||||
chunk_ids[3] = GRAPH_CHUNKID_EXTRAEDGES;
|
chunk_ids[3] = GRAPH_CHUNKID_EXTRAEDGES;
|
||||||
else
|
else
|
||||||
chunk_ids[3] = 0;
|
chunk_ids[3] = 0;
|
||||||
@ -1092,9 +1089,9 @@ int write_commit_graph(const char *obj_dir,
|
|||||||
|
|
||||||
chunk_offsets[0] = 8 + (num_chunks + 1) * GRAPH_CHUNKLOOKUP_WIDTH;
|
chunk_offsets[0] = 8 + (num_chunks + 1) * GRAPH_CHUNKLOOKUP_WIDTH;
|
||||||
chunk_offsets[1] = chunk_offsets[0] + GRAPH_FANOUT_SIZE;
|
chunk_offsets[1] = chunk_offsets[0] + GRAPH_FANOUT_SIZE;
|
||||||
chunk_offsets[2] = chunk_offsets[1] + hashsz * commits.nr;
|
chunk_offsets[2] = chunk_offsets[1] + hashsz * ctx->commits.nr;
|
||||||
chunk_offsets[3] = chunk_offsets[2] + (hashsz + 16) * commits.nr;
|
chunk_offsets[3] = chunk_offsets[2] + (hashsz + 16) * ctx->commits.nr;
|
||||||
chunk_offsets[4] = chunk_offsets[3] + 4 * num_extra_edges;
|
chunk_offsets[4] = chunk_offsets[3] + 4 * ctx->num_extra_edges;
|
||||||
|
|
||||||
for (i = 0; i <= num_chunks; i++) {
|
for (i = 0; i <= num_chunks; i++) {
|
||||||
uint32_t chunk_write[3];
|
uint32_t chunk_write[3];
|
||||||
@ -1105,32 +1102,33 @@ int write_commit_graph(const char *obj_dir,
|
|||||||
hashwrite(f, chunk_write, 12);
|
hashwrite(f, chunk_write, 12);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (report_progress) {
|
if (ctx->report_progress) {
|
||||||
strbuf_addf(&progress_title,
|
strbuf_addf(&progress_title,
|
||||||
Q_("Writing out commit graph in %d pass",
|
Q_("Writing out commit graph in %d pass",
|
||||||
"Writing out commit graph in %d passes",
|
"Writing out commit graph in %d passes",
|
||||||
num_chunks),
|
num_chunks),
|
||||||
num_chunks);
|
num_chunks);
|
||||||
progress = start_delayed_progress(
|
ctx->progress = start_delayed_progress(
|
||||||
progress_title.buf,
|
progress_title.buf,
|
||||||
num_chunks * commits.nr);
|
num_chunks * ctx->commits.nr);
|
||||||
}
|
}
|
||||||
write_graph_chunk_fanout(f, commits.list, commits.nr, progress, &progress_cnt);
|
write_graph_chunk_fanout(f, ctx);
|
||||||
write_graph_chunk_oids(f, hashsz, commits.list, commits.nr, progress, &progress_cnt);
|
write_graph_chunk_oids(f, hashsz, ctx);
|
||||||
write_graph_chunk_data(f, hashsz, commits.list, commits.nr, progress, &progress_cnt);
|
write_graph_chunk_data(f, hashsz, ctx);
|
||||||
if (num_extra_edges)
|
if (ctx->num_extra_edges)
|
||||||
write_graph_chunk_extra_edges(f, commits.list, commits.nr, progress, &progress_cnt);
|
write_graph_chunk_extra_edges(f, ctx);
|
||||||
stop_progress(&progress);
|
stop_progress(&ctx->progress);
|
||||||
strbuf_release(&progress_title);
|
strbuf_release(&progress_title);
|
||||||
|
|
||||||
close_commit_graph(the_repository);
|
close_commit_graph(ctx->r);
|
||||||
finalize_hashfile(f, NULL, CSUM_HASH_IN_STREAM | CSUM_FSYNC);
|
finalize_hashfile(f, NULL, CSUM_HASH_IN_STREAM | CSUM_FSYNC);
|
||||||
commit_lock_file(&lk);
|
commit_lock_file(&lk);
|
||||||
|
|
||||||
cleanup:
|
cleanup:
|
||||||
free(graph_name);
|
free(graph_name);
|
||||||
free(commits.list);
|
free(ctx->commits.list);
|
||||||
free(oids.list);
|
free(ctx->oids.list);
|
||||||
|
free(ctx);
|
||||||
|
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user