Skip to content

Commit 694b7a1

Browse files
mhaggergitster
authored andcommitted
repack_without_ref(): write peeled refs in the rewritten file
When a reference that existed in the packed-refs file is deleted, the packed-refs file must be rewritten. Previously, the file was rewritten without any peeled refs, even if the file contained peeled refs when it was read. This was not a bug, because the packed-refs file header didn't claim that the file contained peeled values. But it had a performance cost, because the repository would lose the benefit of having precomputed peeled references until pack-refs was run again. Teach repack_without_ref() to write peeled refs to the packed-refs file (regardless of whether they were present in the old version of the file). This means that if the old version of the packed-refs file was not fully peeled, then repack_without_ref() will have to peel references. To avoid the expense of reading lots of loose references, we take two shortcuts relative to pack-refs: * If the peeled value of a reference is already known (i.e., because it was read from the old version of the packed-refs file), then output that peeled value again without any checks. This is the usual code path and should avoid any noticeable overhead. (This is different than pack-refs, which always re-peels references.) * We don't verify that the packed ref is still current. It could be that a packed references is overridden by a loose reference, in which case the packed ref is no longer needed and might even refer to an object that has been garbage collected. But we don't check; instead, we just try to peel all references. If peeling is successful, the peeled value is written out (even though it might not be needed any more); if not, then the reference is silently omitted from the output. The extra overhead of peeling references in repack_without_ref() should only be incurred the first time the packed-refs file is written by a version of Git that knows about the "fully-peeled" attribute. Signed-off-by: Michael Haggerty <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent c995de6 commit 694b7a1

File tree

2 files changed

+24
-1
lines changed

2 files changed

+24
-1
lines changed

refs.c

Lines changed: 23 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -876,6 +876,13 @@ void invalidate_ref_cache(const char *submodule)
876876
/* The length of a peeled reference line in packed-refs, including EOL: */
877877
#define PEELED_LINE_LENGTH 42
878878

879+
/*
880+
* The packed-refs header line that we write out. Perhaps other
881+
* traits will be added later. The trailing space is required.
882+
*/
883+
static const char PACKED_REFS_HEADER[] =
884+
"# pack-refs with: peeled fully-peeled \n";
885+
879886
/*
880887
* Parse one line from a packed-refs file. Write the SHA1 to sha1.
881888
* Return a pointer to the refname within the line (null-terminated),
@@ -1391,6 +1398,12 @@ static enum peel_status peel_object(const unsigned char *name, unsigned char *sh
13911398

13921399
/*
13931400
* Peel the entry (if possible) and return its new peel_status.
1401+
*
1402+
* It is OK to call this function with a packed reference entry that
1403+
* might be stale and might even refer to an object that has since
1404+
* been garbage-collected. In such a case, if the entry has
1405+
* REF_KNOWS_PEELED then leave the status unchanged and return
1406+
* PEEL_PEELED or PEEL_NON_TAG; otherwise, return PEEL_INVALID.
13941407
*/
13951408
static enum peel_status peel_entry(struct ref_entry *entry)
13961409
{
@@ -1993,6 +2006,15 @@ static int repack_ref_fn(struct ref_entry *entry, void *cb_data)
19932006
if (len > sizeof(line))
19942007
die("too long a refname '%s'", entry->name);
19952008
write_or_die(*fd, line, len);
2009+
if (!peel_entry(entry)) {
2010+
/* This reference could be peeled; write the peeled value: */
2011+
if (snprintf(line, sizeof(line), "^%s\n",
2012+
sha1_to_hex(entry->u.value.peeled)) !=
2013+
PEELED_LINE_LENGTH)
2014+
die("internal error");
2015+
write_or_die(*fd, line, PEELED_LINE_LENGTH);
2016+
}
2017+
19962018
return 0;
19972019
}
19982020

@@ -2023,6 +2045,7 @@ static int repack_without_ref(const char *refname)
20232045
rollback_lock_file(&packlock);
20242046
return 0;
20252047
}
2048+
write_or_die(fd, PACKED_REFS_HEADER, strlen(PACKED_REFS_HEADER));
20262049
do_for_each_entry_in_dir(packed, 0, repack_ref_fn, &fd);
20272050
return commit_lock_file(&packlock);
20282051
}

t/t3211-peel-ref.sh

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -61,7 +61,7 @@ test_expect_success 'refs are peeled outside of refs/tags (old packed)' '
6161
test_cmp expect actual
6262
'
6363

64-
test_expect_failure 'peeled refs survive deletion of packed ref' '
64+
test_expect_success 'peeled refs survive deletion of packed ref' '
6565
git pack-refs --all &&
6666
cp .git/packed-refs fully-peeled &&
6767
git branch yadda &&

0 commit comments

Comments
 (0)