Browse Source

Merge branch 'jc/maint-reflog-expire-unreachable'

* jc/maint-reflog-expire-unreachable:
  reflog --expire-unreachable: avoid merge-base computation
maint
Junio C Hamano 15 years ago
parent
commit
a9eb304129
  1. 94
      builtin/reflog.c

94
builtin/reflog.c

@ -36,6 +36,8 @@ struct expire_reflog_cb {
FILE *newlog; FILE *newlog;
const char *ref; const char *ref;
struct commit *ref_commit; struct commit *ref_commit;
struct commit_list *mark_list;
unsigned long mark_limit;
struct cmd_reflog_expire_cb *cmd; struct cmd_reflog_expire_cb *cmd;
unsigned char last_kept_sha1[20]; unsigned char last_kept_sha1[20];
}; };
@ -210,46 +212,23 @@ static int keep_entry(struct commit **it, unsigned char *sha1)
return 1; return 1;
} }


static int unreachable(struct expire_reflog_cb *cb, struct commit *commit, unsigned char *sha1)
{
/* /*
* We may or may not have the commit yet - if not, look it * Starting from commits in the cb->mark_list, mark commits that are
* up using the supplied sha1. * reachable from them. Stop the traversal at commits older than
* the expire_limit and queue them back, so that the caller can call
* us again to restart the traversal with longer expire_limit.
*/ */
if (!commit) { static void mark_reachable(struct expire_reflog_cb *cb)
if (is_null_sha1(sha1))
return 0;

commit = lookup_commit_reference_gently(sha1, 1);

/* Not a commit -- keep it */
if (!commit)
return 0;
}

/* Reachable from the current ref? Don't prune. */
if (commit->object.flags & REACHABLE)
return 0;
if (in_merge_bases(commit, &cb->ref_commit, 1))
return 0;

/* We can't reach it - prune it. */
return 1;
}

static void mark_reachable(struct commit *commit, unsigned long expire_limit)
{ {
/* struct commit *commit;
* We need to compute whether the commit on either side of a reflog struct commit_list *pending;
* entry is reachable from the tip of the ref for all entries. unsigned long expire_limit = cb->mark_limit;
* Mark commits that are reachable from the tip down to the struct commit_list *leftover = NULL;
* time threshold first; we know a commit marked thusly is
* reachable from the tip without running in_merge_bases()
* at all.
*/
struct commit_list *pending = NULL;


commit_list_insert(commit, &pending); for (pending = cb->mark_list; pending; pending = pending->next)
pending->item->object.flags &= ~REACHABLE;

pending = cb->mark_list;
while (pending) { while (pending) {
struct commit_list *entry = pending; struct commit_list *entry = pending;
struct commit_list *parent; struct commit_list *parent;
@ -261,8 +240,11 @@ static void mark_reachable(struct commit *commit, unsigned long expire_limit)
if (parse_commit(commit)) if (parse_commit(commit))
continue; continue;
commit->object.flags |= REACHABLE; commit->object.flags |= REACHABLE;
if (commit->date < expire_limit) if (commit->date < expire_limit) {
commit_list_insert(commit, &leftover);
continue; continue;
}
commit->object.flags |= REACHABLE;
parent = commit->parents; parent = commit->parents;
while (parent) { while (parent) {
commit = parent->item; commit = parent->item;
@ -272,6 +254,36 @@ static void mark_reachable(struct commit *commit, unsigned long expire_limit)
commit_list_insert(commit, &pending); commit_list_insert(commit, &pending);
} }
} }
cb->mark_list = leftover;
}

static int unreachable(struct expire_reflog_cb *cb, struct commit *commit, unsigned char *sha1)
{
/*
* We may or may not have the commit yet - if not, look it
* up using the supplied sha1.
*/
if (!commit) {
if (is_null_sha1(sha1))
return 0;

commit = lookup_commit_reference_gently(sha1, 1);

/* Not a commit -- keep it */
if (!commit)
return 0;
}

/* Reachable from the current ref? Don't prune. */
if (commit->object.flags & REACHABLE)
return 0;

if (cb->mark_list && cb->mark_limit) {
cb->mark_limit = 0; /* dig down to the root */
mark_reachable(cb);
}

return !(commit->object.flags & REACHABLE);
} }


static int expire_reflog_ent(unsigned char *osha1, unsigned char *nsha1, static int expire_reflog_ent(unsigned char *osha1, unsigned char *nsha1,
@ -348,8 +360,12 @@ static int expire_reflog(const char *ref, const unsigned char *sha1, int unused,
cb.ref_commit = lookup_commit_reference_gently(sha1, 1); cb.ref_commit = lookup_commit_reference_gently(sha1, 1);
cb.ref = ref; cb.ref = ref;
cb.cmd = cmd; cb.cmd = cmd;
if (cb.ref_commit) if (cb.ref_commit) {
mark_reachable(cb.ref_commit, cmd->expire_total); cb.mark_list = NULL;
commit_list_insert(cb.ref_commit, &cb.mark_list);
cb.mark_limit = cmd->expire_total;
mark_reachable(&cb);
}
for_each_reflog_ent(ref, expire_reflog_ent, &cb); for_each_reflog_ent(ref, expire_reflog_ent, &cb);
if (cb.ref_commit) if (cb.ref_commit)
clear_commit_marks(cb.ref_commit, REACHABLE); clear_commit_marks(cb.ref_commit, REACHABLE);

Loading…
Cancel
Save