Re: rebase - "fixup!" conflict applied at the wrong code location, why ?

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi Philippe,

On Mon, Jul 13, 2020 at 8:42 PM Philippe Blain
<levraiphilippeblain@xxxxxxxxx> wrote:
>
> Hi Elijah,
>
> > Le 9 juil. 2020 à 00:07, Elijah Newren <newren@xxxxxxxxx> a écrit :
> >
> > On Wed, Jul 8, 2020 at 5:07 PM Philippe Blain
> > <levraiphilippeblain@xxxxxxxxx> wrote:
> >>
> >> Hello,
> >>
> >> I've been working on a branch for a while. I've been using `git commit --fixup`  and `git commit --squash`
> >> when I noticed that I had forgotten to add something to a previous commit.
> >> Today I did `git rebase --autosquash` to clean up my history, and the rebase failed at the
> >> first 'fixup!' commit with a conflict. However, the conflict is not located at the right place
> >> in the code (it's not in the right subroutine!). This is very surprising to me, and I would
> >> like to understand why it happens.
> >>
> >> Steps to reproduce:
> >>
> >> git clone -b branch-to-be-rebased https://github.com/phil-blain/CICE.git cice
> >> cd cice
> >> git rebase -i --autosquash my-first-commit
> >> # save the todo list without modifications
> >>  Auto-merging <file>
> >>  CONFLICT (content): Merge conflict in <file>
> >>  error: could not apply e8bfa55... fixup! <commit message of f4e1ae6>
> >> # the rebase stops at f4e1ae6
> >> git diff
> >> # tangential question : for some reason the hunk header does not appear here, I don't know why...
> >> git diff -2  # but it appears here
> >> git grep -p -e '<<<<<<< HEAD' -e '>>>>>>> e8bfa55...'  # or here
> >> # ok, the conflict appears in subroutine 'picard_solver'
> >> git show REBASE_HEAD -U5
> >> # but the original "fixup!" commit only modifies the subroutine 'anderson_solver' !!
> >>
> >> I would have expected that the conflict be created around lines 1118-1132
> >> (line numbers in f4e1ae6), in the 'anderson_solver' subroutine.
> >>
> >> I don't know if this plays a part here, but commit f4e1ae6 (where the rebase stops)
> >> is the commit where the 'anderson_solver' subroutine is added to the code...
> >>
> >> Thanks,
> >>
> >> Philippe.
> >
> > If you take a look where the rebase stops, you see:
>
> First, thanks a lot for your answer. I have a few questions.
>
> >
> > $ git ls-files -u
> > 100644 ee4377f1ec6836fa05573976a473373906c37d9f 1
> > cicecore/cicedynB/dynamics/ice_dyn_vp.F90
> > 100644 30c699ac371c2a751052fa98d04317e84a96ec47 2
> > cicecore/cicedynB/dynamics/ice_dyn_vp.F90
> > 100644 276f224e9048fe0bbd7c25822695049547362c87 3
> > cicecore/cicedynB/dynamics/ice_dyn_vp.F90
> >
> > The difference from the merge base to "other" (index 3) is pretty
> > tiny, you just moved one line in the "anderson_solver" subroutine
> > about 10 lines down.
>
> Yes, the output from
> $ git diff :1:cicecore/cicedynB/dynamics/ice_dyn_vp.F90 :3:cicecore/cicedynB/dynamics/ice_dyn_vp.F90
>
> seems to be the same as the one from
> $ git show REBASE_HEAD
>
> This is a little confusing to me, in the sense that I don't understand why
> the merge-base is what it is. At this point, I do
>
> $ git merge-base HEAD REBASE_HEAD
> f4e1ae67b7d6ca36c6f3ea7c9da43d81caf24067
>
> Ok, so 'git merge-base' finds that the merge-base between HEAD and
> REBASE_HEAD is HEAD; this makes sense to me (no previous commits
> have been rewritten so far, so REBASE_HEAD is directly ahead of HEAD).
> But, if I try to find the commit that contains
> the blob ee4377f1ec6836fa05573976a473373906c37d9f (index 1), I find
> REBASE_HEAD's parent:
>
> $ git log --all --find-object=ee4377f1ec6836fa05573976a473373906c37d9f --format='commmit %H%ntree %T%nparent %P%n%n    %s%n'
> commmit e8bfa557d3c81b75116d6557784b0439b792a308
> tree f6fecb8193c3b877f22bcb8f4d8d2c203e17f06f
> parent 7a8d5a82984dfedd7fac1d7ed7c7fbd1781c1f61
>
>     fixup! Add Anderson acceleration for Picard iteration
>
> commmit 7a8d5a82984dfedd7fac1d7ed7c7fbd1781c1f61
> tree 11fd096851015c0c16b793d9bbb5db039776483b
> parent 63d4c73c1dd973f620307833bd363a1d5069d090
>
>     ice_dyn_vp: introduce 'CICE_USE_LAPACK' preprocessor macro
>
> $ blob=ee4377f1ec6836fa05573976a473373906c37d9f
> $ parent=7a8d5a82984dfedd7fac1d7ed7c7fbd1781c1f61
> $ git ls-tree -r $parent^{tree} | grep $blob
> 100644 blob ee4377f1ec6836fa05573976a473373906c37d9f    cicecore/cicedynB/dynamics/ice_dyn_vp.F90
>
> I don't understand why at this point of the rebase, Git
> determines that the merge-base between HEAD and
> REBASE_HEAD is REBASE_HEAD's parent... this commit
> is not even an ancestor of HEAD (or maybe I don't understand
> what the "merge-base" is in this context?)...

First, remember that this is a rebase or cherry-pick, and not a merge
(we are not creating a new merge commit, but just a commit that
"cherry-picks" the changes from one commit).  But since the wording
and the machinery comes from a merge, let's start by discussing how
merge works.

When you merge two branches (e.g. 'git merge other' will merge branch
'other' into HEAD), git determines the merge-base, i.e. the point of
common history between two commits.  For simplicity, let's assume
there is a common point of history and exactly one common point of
history for these two commits.  Given these assumptions, we have a
merge-base and two commits.  Let's label the merge-base as A, the
commit at HEAD 'B', and the commit at 'other', 'C'.  So we are doing a
three-way merge between A, B, and C.

The thing about a three-way merge is besides thinking of it as a
three-way merge of A, B, and C, there are two other equivalent ways to
look at it: applying the changes between A and B to C, or applying the
changes between A and C to B.  These alternate ways of looking at
things come in handy when looking at rebases and cherry-picks.

When cherry-picking or rebasing a commit, we do NOT want to use the
normal merge-base of HEAD and the commit being picked; if we did that,
it would merge ALL the changes in the history of the commit being
picked into the current branch instead of just the changes from the
one commit we want to cherry-pick/rebase.  Noting that we only want
the changes from that one commit, we can word it similarly to above by
stating that we want to apply the changes between REBASE_HEAD^1 and
REBASE_HEAD to HEAD.  Looking at the above wording for three-way
merges ("applying the changes between A and C to B"), that means we
want to do a three-way merge of REBASE_HEAD^1, HEAD, and REBASE_HEAD.
And when we're done and create a new commit, we do not make a merge
commit but instead record HEAD as the only parent.  Hopefully, this
makes it clear why we use the parent of REBASE_HEAD as the
"merge-base" in the merge machinery; it's not really a "merge-base"
because we're not doing a merge, but we're using the merge machinery
with a special commit as the merge-base because it gives the results
we want.

Hope that helps,
Elijah




[Index of Archives]     [Linux Kernel Development]     [Gcc Help]     [IETF Annouce]     [DCCP]     [Netdev]     [Networking]     [Security]     [V4L]     [Bugtraq]     [Yosemite]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Linux SCSI]     [Fedora Users]

  Powered by Linux