Skip to content

Commit 02471e7

Browse files
dschogitster
authored andcommitted
rebase --autosquash: fix a potential segfault
When rearranging the todo list so that the fixups/squashes are reordered just after the commits they intend to fix up, we use two arrays to maintain that list: `next` and `tail`. The idea is that `next[i]`, if set to a non-negative value, contains the index of the item that should be rearranged just after the `i`th item. To avoid having to walk the entire `next` chain when appending another fixup/squash, we also store the end of the `next` chain in `tail[i]`. The logic we currently use to update these array items is based on the assumption that given a fixup/squash item at index `i`, we just found the index `i2` indicating the first item in that fixup chain. However, as reported by Paul Ganssle, that need not be true: the special form `fixup! <commit-hash>` is allowed to point to _another_ fixup commit in the middle of the fixup chain. Example: * 0192a To fixup * 02f12 fixup! To fixup * 03763 fixup! To fixup * 04ecb fixup! 02f12 Note how the fourth commit targets the second commit, which is already a fixup that targets the first commit. Previously, we would update `next` and `tail` under our assumption that every `fixup!` commit would find the start of the `fixup!`/`squash!` chain. This would lead to a segmentation fault because we would actually end up with a `next[i]` pointing to a `fixup!` but the corresponding `tail[i]` pointing nowhere, which would the lead to a segmentation fault. Let's fix this by _inserting_, rather than _appending_, the item. In other words, if we make a given line successor of another line, we do not simply forget any previously set successor of the latter, but make it a successor of the former. In the above example, at the point when we insert 04ecb just after 02f12, 03763 would already be recorded as a successor of 04ecb, and we now "squeeze in" 04ecb. To complete the idea, we now no longer assume that `next[i]` pointing to a line means that `last[i]` points to a line, too. Instead, we extend the concept of `last` to cover also partial `fixup!`/`squash!` chains, i.e. chains starting in the middle of a larger such chain. In the above example, after processing all lines, `last[0]` (corresponding to 0192a) would point to 03763, which indeed is the end of the overall `fixup!` chain, and `last[1]` (corresponding to 02f12) would point to 04ecb (which is the last `fixup!` targeting 02f12, but it has 03763 as successor, i.e. it is not the end of overall `fixup!` chain). Reported-by: Paul Ganssle <[email protected]> Helped-by: Jeff King <[email protected]> Signed-off-by: Johannes Schindelin <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent f2a0490 commit 02471e7

File tree

2 files changed

+21
-2
lines changed

2 files changed

+21
-2
lines changed

sequencer.c

Lines changed: 5 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -5034,10 +5034,13 @@ static int todo_list_rearrange_squash(struct todo_list *todo_list)
50345034
todo_list->items[i].command =
50355035
starts_with(subject, "fixup!") ?
50365036
TODO_FIXUP : TODO_SQUASH;
5037-
if (next[i2] < 0)
5037+
if (tail[i2] < 0) {
5038+
next[i] = next[i2];
50385039
next[i2] = i;
5039-
else
5040+
} else {
5041+
next[i] = next[tail[i2]];
50405042
next[tail[i2]] = i;
5043+
}
50415044
tail[i2] = i;
50425045
} else if (!hashmap_get_from_hash(&subject2item,
50435046
strhash(subject), subject)) {

t/t3415-rebase-autosquash.sh

Lines changed: 16 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -349,4 +349,20 @@ test_expect_success 'abort last squash' '
349349
! grep first actual
350350
'
351351

352+
test_expect_success 'fixup a fixup' '
353+
echo 0to-fixup >file0 &&
354+
test_tick &&
355+
git commit -m "to-fixup" file0 &&
356+
test_tick &&
357+
git commit --squash HEAD -m X --allow-empty &&
358+
test_tick &&
359+
git commit --squash HEAD^ -m Y --allow-empty &&
360+
test_tick &&
361+
git commit -m "squash! $(git rev-parse HEAD^)" -m Z --allow-empty &&
362+
test_tick &&
363+
git commit -m "squash! $(git rev-parse HEAD^^)" -m W --allow-empty &&
364+
git rebase -ki --autosquash HEAD~5 &&
365+
test XZWY = $(git show | tr -cd W-Z)
366+
'
367+
352368
test_done

0 commit comments

Comments
 (0)