rebase: avoid computing unnecessary patch IDs
The `rebase` family of Git commands avoid applying patches that were
already integrated upstream. They do that by using the revision walking
option that computes the patch IDs of the two sides of the rebase
(local-only patches vs upstream-only ones) and skipping those local
patches whose patch ID matches one of the upstream ones.
In many cases, this causes unnecessary churn, as already the set of
paths touched by a given commit would suffice to determine that an
upstream patch has no local equivalent.
This hurts performance in particular when there are a lot of upstream
patches, and/or large ones.
Therefore, let's introduce the concept of a "diff-header-only" patch ID,
compare those first, and only evaluate the "full" patch ID lazily.
Please note that in contrast to the "full" patch IDs, those
"diff-header-only" patch IDs are prone to collide with one another, as
adjacent commits frequently touch the very same files. Hence we now
have to be careful to allow multiple hash entries with the same hash.
We accomplish that by using the hashmap_add() function that does not even
test for hash collisions. This also allows us to evaluate the full patch ID
lazily, i.e. only when we found commits with matching diff-header-only
patch IDs.
We add a performance test that demonstrates ~1-6% improvement. In
practice this will depend on various factors such as how many upstream
changes and how big those changes are along with whether file system
caches are cold or warm. As Git's test suite has no way of catching
performance regressions, we also add a regression test that verifies
that the full patch ID computation is skipped when the diff-header-only
computation suffices.
Signed-off-by: Kevin Willford <kcwillford@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2016-07-29 18:19:20 +02:00
|
|
|
#!/bin/sh
|
|
|
|
|
|
|
|
test_description='Tests rebase performance'
|
|
|
|
. ./perf-lib.sh
|
|
|
|
|
|
|
|
test_perf_default_repo
|
|
|
|
|
2017-05-05 16:57:13 +02:00
|
|
|
test_expect_success 'setup rebasing on top of a lot of changes' '
|
2018-11-09 22:19:23 +01:00
|
|
|
git checkout -f -B base &&
|
|
|
|
git checkout -B to-rebase &&
|
|
|
|
git checkout -B upstream &&
|
rebase: avoid computing unnecessary patch IDs
The `rebase` family of Git commands avoid applying patches that were
already integrated upstream. They do that by using the revision walking
option that computes the patch IDs of the two sides of the rebase
(local-only patches vs upstream-only ones) and skipping those local
patches whose patch ID matches one of the upstream ones.
In many cases, this causes unnecessary churn, as already the set of
paths touched by a given commit would suffice to determine that an
upstream patch has no local equivalent.
This hurts performance in particular when there are a lot of upstream
patches, and/or large ones.
Therefore, let's introduce the concept of a "diff-header-only" patch ID,
compare those first, and only evaluate the "full" patch ID lazily.
Please note that in contrast to the "full" patch IDs, those
"diff-header-only" patch IDs are prone to collide with one another, as
adjacent commits frequently touch the very same files. Hence we now
have to be careful to allow multiple hash entries with the same hash.
We accomplish that by using the hashmap_add() function that does not even
test for hash collisions. This also allows us to evaluate the full patch ID
lazily, i.e. only when we found commits with matching diff-header-only
patch IDs.
We add a performance test that demonstrates ~1-6% improvement. In
practice this will depend on various factors such as how many upstream
changes and how big those changes are along with whether file system
caches are cold or warm. As Git's test suite has no way of catching
performance regressions, we also add a regression test that verifies
that the full patch ID computation is skipped when the diff-header-only
computation suffices.
Signed-off-by: Kevin Willford <kcwillford@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2016-07-29 18:19:20 +02:00
|
|
|
for i in $(seq 100)
|
|
|
|
do
|
|
|
|
# simulate huge diffs
|
|
|
|
echo change$i >unrelated-file$i &&
|
|
|
|
seq 1000 >>unrelated-file$i &&
|
|
|
|
git add unrelated-file$i &&
|
|
|
|
test_tick &&
|
|
|
|
git commit -m commit$i unrelated-file$i &&
|
|
|
|
echo change$i >unrelated-file$i &&
|
|
|
|
seq 1000 | tac >>unrelated-file$i &&
|
|
|
|
git add unrelated-file$i &&
|
|
|
|
test_tick &&
|
|
|
|
git commit -m commit$i-reverse unrelated-file$i ||
|
|
|
|
break
|
|
|
|
done &&
|
|
|
|
git checkout to-rebase &&
|
|
|
|
test_commit our-patch interesting-file
|
|
|
|
'
|
|
|
|
|
|
|
|
test_perf 'rebase on top of a lot of unrelated changes' '
|
|
|
|
git rebase --onto upstream HEAD^ &&
|
|
|
|
git rebase --onto base HEAD^
|
|
|
|
'
|
|
|
|
|
2017-05-05 16:57:13 +02:00
|
|
|
test_expect_success 'setup rebasing many changes without split-index' '
|
|
|
|
git config core.splitIndex false &&
|
2018-11-09 22:19:23 +01:00
|
|
|
git checkout -B upstream2 to-rebase &&
|
|
|
|
git checkout -B to-rebase2 upstream
|
2017-05-05 16:57:13 +02:00
|
|
|
'
|
|
|
|
|
|
|
|
test_perf 'rebase a lot of unrelated changes without split-index' '
|
|
|
|
git rebase --onto upstream2 base &&
|
|
|
|
git rebase --onto base upstream2
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'setup rebasing many changes with split-index' '
|
|
|
|
git config core.splitIndex true
|
|
|
|
'
|
|
|
|
|
|
|
|
test_perf 'rebase a lot of unrelated changes with split-index' '
|
|
|
|
git rebase --onto upstream2 base &&
|
|
|
|
git rebase --onto base upstream2
|
|
|
|
'
|
|
|
|
|
rebase: avoid computing unnecessary patch IDs
The `rebase` family of Git commands avoid applying patches that were
already integrated upstream. They do that by using the revision walking
option that computes the patch IDs of the two sides of the rebase
(local-only patches vs upstream-only ones) and skipping those local
patches whose patch ID matches one of the upstream ones.
In many cases, this causes unnecessary churn, as already the set of
paths touched by a given commit would suffice to determine that an
upstream patch has no local equivalent.
This hurts performance in particular when there are a lot of upstream
patches, and/or large ones.
Therefore, let's introduce the concept of a "diff-header-only" patch ID,
compare those first, and only evaluate the "full" patch ID lazily.
Please note that in contrast to the "full" patch IDs, those
"diff-header-only" patch IDs are prone to collide with one another, as
adjacent commits frequently touch the very same files. Hence we now
have to be careful to allow multiple hash entries with the same hash.
We accomplish that by using the hashmap_add() function that does not even
test for hash collisions. This also allows us to evaluate the full patch ID
lazily, i.e. only when we found commits with matching diff-header-only
patch IDs.
We add a performance test that demonstrates ~1-6% improvement. In
practice this will depend on various factors such as how many upstream
changes and how big those changes are along with whether file system
caches are cold or warm. As Git's test suite has no way of catching
performance regressions, we also add a regression test that verifies
that the full patch ID computation is skipped when the diff-header-only
computation suffices.
Signed-off-by: Kevin Willford <kcwillford@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2016-07-29 18:19:20 +02:00
|
|
|
test_done
|