On Thu, May 19, 2022 at 6:15 AM ZheNing Hu <adlternative@xxxxxxxxx> wrote: > > Elijah Newren <newren@xxxxxxxxx> 于2022年5月13日周五 13:16写道: > > > > On Thu, May 12, 2022 at 8:39 AM Junio C Hamano <gitster@xxxxxxxxx> wrote: > > > > > > Elijah Newren <newren@xxxxxxxxx> writes: > > > > > > > Merge with strategy octopus failed. > > > > > > > > Also, if we check `git status`: > > > > > > > > $ git status > > > > On branch main > > > > Unmerged paths: > > > > (use "git restore --staged <file>..." to unstage) > > > > (use "git add <file>..." to mark resolution) > > > > both modified: base > > > > > > > > no changes added to commit (use "git add" and/or "git commit -a") > > > > > > > > And in git-merge-octopus.sh we see: > > > > > > > > case "$OCTOPUS_FAILURE" in > > > > 1) > > > > # We allow only last one to have a hand-resolvable > > > > # conflicts. Last round failed and we still had > > > > # a head to merge. > > > > gettextln "Automated merge did not work." > > > > gettextln "Should not be doing an octopus." > > > > exit 2 > > > > esac > > > > > > > > and in builtin/merge.c, we see: > > > > > > > > /* > > > > * The backend exits with 1 when conflicts are > > > > * left to be resolved, with 2 when it does not > > > > * handle the given merge at all. > > > > */ > > > > > > > > Which means git-merge-octopus.sh is claiming it can't handle this type > > > > of merge, and some other merge strategy should be tried, and > > > > implicitly that it didn't leave any conflicts to be resolved because > > > > it can't handle this merge. > > > > > > Correct. Near the beginning of the loop you found the above > > > comment, there is this code: > > > > > > if (use_strategies_nr == 1 || > > > /* > > > * Stash away the local changes so that we can try more than one. > > > */ > > > save_state(&stash)) > > > oidclr(&stash); > > > > > > for (i = 0; !merge_was_ok && i < use_strategies_nr; i++) { > > > int ret, cnt; > > > if (i) { > > > printf(_("Rewinding the tree to pristine...\n")); > > > restore_state(&head_commit->object.oid, &stash); > > > } > > > > Side-comment, which becomes important below: The save/restore code in > > builtin/merge.c appears to be broken to me. As noted in the code > > above, stash will be set to null_oid() if save_state() returns > > non-zero (which happens when "stash create" has no output, which > > happens if there is _initially_ no state to save, i.e. if there are no > > local changes before the merge started). restore_state() is a no-op > > whenever stash is the null_oid, meaning in that case it won't actually > > rewind the tree to a pristine state to undo the changes of the > > previous merge attempt. So, if: > > > > * The user had no local changes before starting the merge > > * Multiple merge strategies are applicable > > * The first merge strategy makes index/working-tree changes, but > > returns with exit status 2 > > > > Then the restore_state() called before the second merge strategy will > > do nothing, and the second merge strategy will be working on an index > > and working tree with garbage leftover from the first merge strategy. > > While this may have never been triggered (in what case do we have > > multiple merge strategies that all return an exit status of 2?), I > > suspect we want to fix this problem with something like this: > > > > diff --git a/builtin/merge.c b/builtin/merge.c > > index f178f5a3ee..7f3650fb09 100644 > > --- a/builtin/merge.c > > +++ b/builtin/merge.c > > @@ -378,11 +378,11 @@ static void restore_state(const struct object_id *head, > > struct strbuf sb = STRBUF_INIT; > > const char *args[] = { "stash", "apply", NULL, NULL }; > > > > + reset_hard(head, 1); > > + > > if (is_null_oid(stash)) > > return; > > > > - reset_hard(head, 1); > > - > > args[2] = oid_to_hex(stash); > > > > /* > > > > It looks like this code can go back to the old state now? > > I think if we can handle it as special case when: > > 1. Use octopus merge without other strategies (no friend). > 2. Merge failed, so we don't have the best strategy. > > Then we force restore the original state? Yeah, I've got some patches ready; just didn't submit yet as I was hoping to get time to update the in-core merge series. https://github.com/gitgitgadget/git/pull/1231