Skip to content

Commit 5c9f9bf

Browse files
peffgitster
authored andcommitted
rev-list: "adjust" results of "--count --use-bitmap-index -n"
If you ask rev-list for: git rev-list --count --use-bitmap-index HEAD we optimize out the actual traversal and just give you the number of bits set in the commit bitmap. This is faster, which is good. But if you ask to limit the size of the traversal, like: git rev-list --count --use-bitmap-index -n 100 HEAD we'll still output the full bitmapped number we found. On the surface, that might even seem OK. You explicitly asked to use the bitmap index, and it was cheap to compute the real answer, so we gave it to you. But there's something much more complicated going on under the hood. If we don't have a bitmap directly for HEAD, then we have to actually traverse backwards, looking for a bitmapped commit. And _that_ traversal is bounded by our `-n` count. This is a good thing, because it bounds the work we have to do, which is probably what the user wanted by asking for `-n`. But now it makes the output quite confusing. You might get many values: - your `-n` value, if we walked back and never found a bitmap (or fewer if there weren't that many commits) - the actual full count, if we found a bitmap root for every path of our traversal with in the `-n` limit - any number in between! We might have walked back and found _some_ bitmaps, but then cut off the traversal early with some commits not accounted for in the result. So you cannot even see a value higher than your `-n` and say "OK, bitmaps kicked in, this must be the real full count". The only sane thing is for git to just clamp the value to a maximum of the `-n` value, which means we should output the exact same results whether bitmaps are in use or not. The test in t5310 demonstrates this by using `-n 1`. Without this patch we fail in the full-bitmap case (where we do not have to traverse at all) but _not_ in the partial-bitmap case (where we have to walk down to find an actual bitmap). With this patch, both cases just work. I didn't implement the crazy in-between case, just because it's complicated to set up, and is really a subset of the full-count case, which we do cover. Signed-off-by: Jeff King <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent 7654286 commit 5c9f9bf

File tree

2 files changed

+9
-0
lines changed

2 files changed

+9
-0
lines changed

builtin/rev-list.c

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -355,8 +355,11 @@ int cmd_rev_list(int argc, const char **argv, const char *prefix)
355355
if (use_bitmap_index && !revs.prune) {
356356
if (revs.count && !revs.left_right && !revs.cherry_mark) {
357357
uint32_t commit_count;
358+
int max_count = revs.max_count;
358359
if (!prepare_bitmap_walk(&revs)) {
359360
count_bitmap_commit_list(&commit_count, NULL, NULL, NULL);
361+
if (max_count >= 0 && max_count < commit_count)
362+
commit_count = max_count;
360363
printf("%d\n", commit_count);
361364
return 0;
362365
}

t/t5310-pack-bitmaps.sh

Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -47,6 +47,12 @@ rev_list_tests() {
4747
test_cmp expect actual
4848
'
4949

50+
test_expect_success "counting commits with limit ($state)" '
51+
git rev-list --count -n 1 HEAD >expect &&
52+
git rev-list --use-bitmap-index --count -n 1 HEAD >actual &&
53+
test_cmp expect actual
54+
'
55+
5056
test_expect_success "counting non-linear history ($state)" '
5157
git rev-list --count other...master >expect &&
5258
git rev-list --use-bitmap-index --count other...master >actual &&

0 commit comments

Comments
 (0)