1.. SPDX-License-Identifier: GPL-2.0
2
3===================================
4Backporting and conflict resolution
5===================================
6
7:Author: Vegard Nossum <vegard.nossum@oracle.com>
8
9.. contents::
10    :local:
11    :depth: 3
12    :backlinks: none
13
14Introduction
15============
16
17Some developers may never really have to deal with backporting patches,
18merging branches, or resolving conflicts in their day-to-day work, so
19when a merge conflict does pop up, it can be daunting. Luckily,
20resolving conflicts is a skill like any other, and there are many useful
21techniques you can use to make the process smoother and increase your
22confidence in the result.
23
24This document aims to be a comprehensive, step-by-step guide to
25backporting and conflict resolution.
26
27Applying the patch to a tree
28============================
29
30Sometimes the patch you are backporting already exists as a git commit,
31in which case you just cherry-pick it directly using
32``git cherry-pick``. However, if the patch comes from an email, as it
33often does for the Linux kernel, you will need to apply it to a tree
34using ``git am``.
35
36If you've ever used ``git am``, you probably already know that it is
37quite picky about the patch applying perfectly to your source tree. In
38fact, you've probably had nightmares about ``.rej`` files and trying to
39edit the patch to make it apply.
40
41It is strongly recommended to instead find an appropriate base version
42where the patch applies cleanly and *then* cherry-pick it over to your
43destination tree, as this will make git output conflict markers and let
44you resolve conflicts with the help of git and any other conflict
45resolution tools you might prefer to use. For example, if you want to
46apply a patch that just arrived on LKML to an older stable kernel, you
47can apply it to the most recent mainline kernel and then cherry-pick it
48to your older stable branch.
49
50It's generally better to use the exact same base as the one the patch
51was generated from, but it doesn't really matter that much as long as it
52applies cleanly and isn't too far from the original base. The only
53problem with applying the patch to the "wrong" base is that it may pull
54in more unrelated changes in the context of the diff when cherry-picking
55it to the older branch.
56
57A good reason to prefer ``git cherry-pick`` over ``git am`` is that git
58knows the precise history of an existing commit, so it will know when
59code has moved around and changed the line numbers; this in turn makes
60it less likely to apply the patch to the wrong place (which can result
61in silent mistakes or messy conflicts).
62
63If you are using `b4`_. and you are applying the patch directly from an
64email, you can use ``b4 am`` with the options ``-g``/``--guess-base``
65and ``-3``/``--prep-3way`` to do some of this automatically (see the
66`b4 presentation`_ for more information). However, the rest of this
67article will assume that you are doing a plain ``git cherry-pick``.
68
69.. _b4: https://people.kernel.org/monsieuricon/introducing-b4-and-patch-attestation
70.. _b4 presentation: https://youtu.be/mF10hgVIx9o?t=2996
71
72Once you have the patch in git, you can go ahead and cherry-pick it into
73your source tree. Don't forget to cherry-pick with ``-x`` if you want a
74written record of where the patch came from!
75
76Note that if you are submiting a patch for stable, the format is
77slightly different; the first line after the subject line needs tobe
78either::
79
80    commit <upstream commit> upstream
81
82or::
83
84    [ Upstream commit <upstream commit> ]
85
86Resolving conflicts
87===================
88
89Uh-oh; the cherry-pick failed with a vaguely threatening message::
90
91    CONFLICT (content): Merge conflict
92
93What to do now?
94
95In general, conflicts appear when the context of the patch (i.e., the
96lines being changed and/or the lines surrounding the changes) doesn't
97match what's in the tree you are trying to apply the patch *to*.
98
99For backports, what likely happened was that the branch you are
100backporting from contains patches not in the branch you are backporting
101to. However, the reverse is also possible. In any case, the result is a
102conflict that needs to be resolved.
103
104If your attempted cherry-pick fails with a conflict, git automatically
105edits the files to include so-called conflict markers showing you where
106the conflict is and how the two branches have diverged. Resolving the
107conflict typically means editing the end result in such a way that it
108takes into account these other commits.
109
110Resolving the conflict can be done either by hand in a regular text
111editor or using a dedicated conflict resolution tool.
112
113Many people prefer to use their regular text editor and edit the
114conflict directly, as it may be easier to understand what you're doing
115and to control the final result. There are definitely pros and cons to
116each method, and sometimes there's value in using both.
117
118We will not cover using dedicated merge tools here beyond providing some
119pointers to various tools that you could use:
120
121-  `Emacs Ediff mode <https://www.emacswiki.org/emacs/EdiffMode>`__
122-  `vimdiff/gvimdiff <https://linux.die.net/man/1/vimdiff>`__
123-  `KDiff3 <http://kdiff3.sourceforge.net/>`__
124-  `TortoiseMerge <https://tortoisesvn.net/TortoiseMerge.html>`__
125-  `Meld <https://meldmerge.org/help/>`__
126-  `P4Merge <https://www.perforce.com/products/helix-core-apps/merge-diff-tool-p4merge>`__
127-  `Beyond Compare <https://www.scootersoftware.com/>`__
128-  `IntelliJ <https://www.jetbrains.com/help/idea/resolve-conflicts.html>`__
129-  `VSCode <https://code.visualstudio.com/docs/editor/versioncontrol>`__
130
131To configure git to work with these, see ``git mergetool --help`` or
132the official `git-mergetool documentation`_.
133
134.. _git-mergetool documentation: https://git-scm.com/docs/git-mergetool
135
136Prerequisite patches
137--------------------
138
139Most conflicts happen because the branch you are backporting to is
140missing some patches compared to the branch you are backporting *from*.
141In the more general case (such as merging two independent branches),
142development could have happened on either branch, or the branches have
143simply diverged -- perhaps your older branch had some other backports
144applied to it that themselves needed conflict resolutions, causing a
145divergence.
146
147It's important to always identify the commit or commits that caused the
148conflict, as otherwise you cannot be confident in the correctness of
149your resolution. As an added bonus, especially if the patch is in an
150area you're not that famliar with, the changelogs of these commits will
151often give you the context to understand the code and potential problems
152or pitfalls with your conflict resolution.
153
154git log
155~~~~~~~
156
157A good first step is to look at ``git log`` for the file that has the
158conflict -- this is usually sufficient when there aren't a lot of
159patches to the file, but may get confusing if the file is big and
160frequently patched. You should run ``git log`` on the range of commits
161between your currently checked-out branch (``HEAD``) and the parent of
162the patch you are picking (``<commit>``), i.e.::
163
164    git log HEAD..<commit>^ -- <path>
165
166Even better, if you want to restrict this output to a single function
167(because that's where the conflict appears), you can use the following
168syntax::
169
170    git log -L:'\<function\>':<path> HEAD..<commit>^
171
172.. note::
173     The ``\<`` and ``\>`` around the function name ensure that the
174     matches are anchored on a word boundary. This is important, as this
175     part is actually a regex and git only follows the first match, so
176     if you use ``-L:thread_stack:kernel/fork.c`` it may only give you
177     results for the function ``try_release_thread_stack_to_cache`` even
178     though there are many other functions in that file containing the
179     string ``thread_stack`` in their names.
180
181Another useful option for ``git log`` is ``-G``, which allows you to
182filter on certain strings appearing in the diffs of the commits you are
183listing::
184
185    git log -G'regex' HEAD..<commit>^ -- <path>
186
187This can also be a handy way to quickly find when something (e.g. a
188function call or a variable) was changed, added, or removed. The search
189string is a regular expression, which means you can potentially search
190for more specific things like assignments to a specific struct member::
191
192    git log -G'\->index\>.*='
193
194git blame
195~~~~~~~~~
196
197Another way to find prerequisite commits (albeit only the most recent
198one for a given conflict) is to run ``git blame``. In this case, you
199need to run it against the parent commit of the patch you are
200cherry-picking and the file where the conflict appared, i.e.::
201
202    git blame <commit>^ -- <path>
203
204This command also accepts the ``-L`` argument (for restricting the
205output to a single function), but in this case you specify the filename
206at the end of the command as usual::
207
208    git blame -L:'\<function\>' <commit>^ -- <path>
209
210Navigate to the place where the conflict occurred. The first column of
211the blame output is the commit ID of the patch that added a given line
212of code.
213
214It might be a good idea to ``git show`` these commits and see if they
215look like they might be the source of the conflict. Sometimes there will
216be more than one of these commits, either because multiple commits
217changed different lines of the same conflict area *or* because multiple
218subsequent patches changed the same line (or lines) multiple times. In
219the latter case, you may have to run ``git blame`` again and specify the
220older version of the file to look at in order to dig further back in
221the history of the file.
222
223Prerequisite vs. incidental patches
224~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
225
226Having found the patch that caused the conflict, you need to determine
227whether it is a prerequisite for the patch you are backporting or
228whether it is just incidental and can be skipped. An incidental patch
229would be one that touches the same code as the patch you are
230backporting, but does not change the semantics of the code in any
231material way. For example, a whitespace cleanup patch is completely
232incidental -- likewise, a patch that simply renames a function or a
233variable would be incidental as well. On the other hand, if the function
234being changed does not even exist in your current branch then this would
235not be incidental at all and you need to carefully consider whether the
236patch adding the function should be cherry-picked first.
237
238If you find that there is a necessary prerequisite patch, then you need
239to stop and cherry-pick that instead. If you've already resolved some
240conflicts in a different file and don't want to do it again, you can
241create a temporary copy of that file.
242
243To abort the current cherry-pick, go ahead and run
244``git cherry-pick --abort``, then restart the cherry-picking process
245with the commit ID of the prerequisite patch instead.
246
247Understanding conflict markers
248------------------------------
249
250Combined diffs
251~~~~~~~~~~~~~~
252
253Let's say you've decided against picking (or reverting) additional
254patches and you just want to resolve the conflict. Git will have
255inserted conflict markers into your file. Out of the box, this will look
256something like::
257
258    <<<<<<< HEAD
259    this is what's in your current tree before cherry-picking
260    =======
261    this is what the patch wants it to be after cherry-picking
262    >>>>>>> <commit>... title
263
264This is what you would see if you opened the file in your editor.
265However, if you were to run ``git diff`` without any arguments, the
266output would look something like this::
267
268    $ git diff
269    [...]
270    ++<<<<<<<< HEAD
271     +this is what's in your current tree before cherry-picking
272    ++========
273    + this is what the patch wants it to be after cherry-picking
274    ++>>>>>>>> <commit>... title
275
276When you are resolving a conflict, the behavior of ``git diff`` differs
277from its normal behavior. Notice the two columns of diff markers
278instead of the usual one; this is a so-called "`combined diff`_", here
279showing the 3-way diff (or diff-of-diffs) between
280
281#. the current branch (before cherry-picking) and the current working
282   directory, and
283#. the current branch (before cherry-picking) and the file as it looks
284   after the original patch has been applied.
285
286.. _combined diff: https://git-scm.com/docs/diff-format#_combined_diff_format
287
288
289Better diffs
290~~~~~~~~~~~~
291
2923-way combined diffs include all the other changes that happened to the
293file between your current branch and the branch you are cherry-picking
294from. While this is useful for spotting other changes that you need to
295take into account, this also makes the output of ``git diff`` somewhat
296intimidating and difficult to read. You may instead prefer to run
297``git diff HEAD`` (or ``git diff --ours``) which shows only the diff
298between the current branch before cherry-picking and the current working
299directory. It looks like this::
300
301    $ git diff HEAD
302    [...]
303    +<<<<<<<< HEAD
304     this is what's in your current tree before cherry-picking
305    +========
306    +this is what the patch wants it to be after cherry-picking
307    +>>>>>>>> <commit>... title
308
309As you can see, this reads just like any other diff and makes it clear
310which lines are in the current branch and which lines are being added
311because they are part of the merge conflict or the patch being
312cherry-picked.
313
314Merge styles and diff3
315~~~~~~~~~~~~~~~~~~~~~~
316
317The default conflict marker style shown above is known as the ``merge``
318style. There is also another style available, known as the ``diff3``
319style, which looks like this::
320
321    <<<<<<< HEAD
322    this is what is in your current tree before cherry-picking
323    ||||||| parent of <commit> (title)
324    this is what the patch expected to find there
325    =======
326    this is what the patch wants it to be after being applied
327    >>>>>>> <commit> (title)
328
329As you can see, this has 3 parts instead of 2, and includes what git
330expected to find there but didn't. It is *highly recommended* to use
331this conflict style as it makes it much clearer what the patch actually
332changed; i.e., it allows you to compare the before-and-after versions
333of the file for the commit you are cherry-picking. This allows you to
334make better decisions about how to resolve the conflict.
335
336To change conflict marker styles, you can use the following command::
337
338    git config merge.conflictStyle diff3
339
340There is a third option, ``zdiff3``, introduced in `Git 2.35`_,
341which has the same 3 sections as ``diff3``, but where common lines have
342been trimmed off, making the conflict area smaller in some cases.
343
344.. _Git 2.35: https://github.blog/2022-01-24-highlights-from-git-2-35/
345
346Iterating on conflict resolutions
347---------------------------------
348
349The first step in any conflict resolution process is to understand the
350patch you are backporting. For the Linux kernel this is especially
351important, since an incorrect change can lead to the whole system
352crashing -- or worse, an undetected security vulnerability.
353
354Understanding the patch can be easy or difficult depending on the patch
355itself, the changelog, and your familiarity with the code being changed.
356However, a good question for every change (or every hunk of the patch)
357might be: "Why is this hunk in the patch?" The answers to these
358questions will inform your conflict resolution.
359
360Resolution process
361~~~~~~~~~~~~~~~~~~
362
363Sometimes the easiest thing to do is to just remove all but the first
364part of the conflict, leaving the file essentially unchanged, and apply
365the changes by hand. Perhaps the patch is changing a function call
366argument from ``0`` to ``1`` while a conflicting change added an
367entirely new (and insignificant) parameter to the end of the parameter
368list; in that case, it's easy enough to change the argument from ``0``
369to ``1`` by hand and leave the rest of the arguments alone. This
370technique of manually applying changes is mostly useful if the conflict
371pulled in a lot of unrelated context that you don't really need to care
372about.
373
374For particularly nasty conflicts with many conflict markers, you can use
375``git add`` or ``git add -i`` to selectively stage your resolutions to
376get them out of the way; this also lets you use ``git diff HEAD`` to
377always see what remains to be resolved or ``git diff --cached`` to see
378what your patch looks like so far.
379
380Dealing with file renames
381~~~~~~~~~~~~~~~~~~~~~~~~~
382
383One of the most annoying things that can happen while backporting a
384patch is discovering that one of the files being patched has been
385renamed, as that typically means git won't even put in conflict markers,
386but will just throw up its hands and say (paraphrased): "Unmerged path!
387You do the work..."
388
389There are generally a few ways to deal with this. If the patch to the
390renamed file is small, like a one-line change, the easiest thing is to
391just go ahead and apply the change by hand and be done with it. On the
392other hand, if the change is big or complicated, you definitely don't
393want to do it by hand.
394
395As a first pass, you can try something like this, which will lower the
396rename detection threshold to 30% (by default, git uses 50%, meaning
397that two files need to have at least 50% in common for it to consider
398an add-delete pair to be a potential rename)::
399
400  git cherry-pick -strategy=recursive -Xrename-threshold=30
401
402Sometimes the right thing to do will be to also backport the patch that
403did the rename, but that's definitely not the most common case. Instead,
404what you can do is to temporarily rename the file in the branch you're
405backporting to (using ``git mv`` and committing the result), restart the
406attempt to cherry-pick the patch, rename the file back (``git mv`` and
407committing again), and finally squash the result using ``git rebase -i``
408(see the `rebase tutorial`_) so it appears as a single commit when you
409are done.
410
411.. _rebase tutorial: https://medium.com/@slamflipstrom/a-beginners-guide-to-squashing-commits-with-git-rebase-8185cf6e62ec
412
413Gotchas
414-------
415
416Function arguments
417~~~~~~~~~~~~~~~~~~
418
419Pay attention to changing function arguments! It's easy to gloss over
420details and think that two lines are the same but actually they differ
421in some small detail like which variable was passed as an argument
422(especially if the two variables are both a single character that look
423the same, like i and j).
424
425Error handling
426~~~~~~~~~~~~~~
427
428If you cherry-pick a patch that includes a ``goto`` statement (typically
429for error handling), it is absolutely imperative to double check that
430the target label is still correct in the branch you are backporting to.
431The same goes for added ``return``, ``break``, and ``continue``
432statements.
433
434Error handling is typically located at the bottom of the function, so it
435may not be part of the conflict even though could have been changed by
436other patches.
437
438A good way to ensure that you review the error paths is to always use
439``git diff -W`` and ``git show -W`` (AKA ``--function-context``) when
440inspecting your changes.  For C code, this will show you the whole
441function that's being changed in a patch. One of the things that often
442go wrong during backports is that something else in the function changed
443on either of the branches that you're backporting from or to. By
444including the whole function in the diff you get more context and can
445more easily spot problems that might otherwise go unnoticed.
446
447Refactored code
448~~~~~~~~~~~~~~~
449
450Something that happens quite often is that code gets refactored by
451"factoring out" a common code sequence or pattern into a helper
452function. When backporting patches to an area where such a refactoring
453has taken place, you effectively need to do the reverse when
454backporting: a patch to a single location may need to be applied to
455multiple locations in the backported version. (One giveaway for this
456scenario is that a function was renamed -- but that's not always the
457case.)
458
459To avoid incomplete backports, it's worth trying to figure out if the
460patch fixes a bug that appears in more than one place. One way to do
461this would be to use ``git grep``. (This is actually a good idea to do
462in general, not just for backports.) If you do find that the same kind
463of fix would apply to other places, it's also worth seeing if those
464places exist upstream -- if they don't, it's likely the patch may need
465to be adjusted. ``git log`` is your friend to figure out what happened
466to these areas as ``git blame`` won't show you code that has been
467removed.
468
469If you do find other instances of the same pattern in the upstream tree
470and you're not sure whether it's also a bug, it may be worth asking the
471patch author. It's not uncommon to find new bugs during backporting!
472
473Verifying the result
474====================
475
476colordiff
477---------
478
479Having committed a conflict-free new patch, you can now compare your
480patch to the original patch. It is highly recommended that you use a
481tool such as `colordiff`_ that can show two files side by side and color
482them according to the changes between them::
483
484    colordiff -yw -W 200 <(git diff -W <upstream commit>^-) <(git diff -W HEAD^-) | less -SR
485
486.. _colordiff: https://www.colordiff.org/
487
488Here, ``-y`` means to do a side-by-side comparison; ``-w`` ignores
489whitespace, and ``-W 200`` sets the width of the output (as otherwise it
490will use 130 by default, which is often a bit too little).
491
492The ``rev^-`` syntax is a handy shorthand for ``rev^..rev``, essentially
493giving you just the diff for that single commit; also see
494the official `git rev-parse documentation`_.
495
496.. _git rev-parse documentation: https://git-scm.com/docs/git-rev-parse#_other_rev_parent_shorthand_notations
497
498Again, note the inclusion of ``-W`` for ``git diff``; this ensures that
499you will see the full function for any function that has changed.
500
501One incredibly important thing that colordiff does is to highlight lines
502that are different. For example, if an error-handling ``goto`` has
503changed labels between the original and backported patch, colordiff will
504show these side-by-side but highlighted in a different color.  Thus, it
505is easy to see that the two ``goto`` statements are jumping to different
506labels. Likewise, lines that were not modified by either patch but
507differ in the context will also be highlighted and thus stand out during
508a manual inspection.
509
510Of course, this is just a visual inspection; the real test is building
511and running the patched kernel (or program).
512
513Build testing
514-------------
515
516We won't cover runtime testing here, but it can be a good idea to build
517just the files touched by the patch as a quick sanity check. For the
518Linux kernel you can build single files like this, assuming you have the
519``.config`` and build environment set up correctly::
520
521    make path/to/file.o
522
523Note that this won't discover linker errors, so you should still do a
524full build after verifying that the single file compiles. By compiling
525the single file first you can avoid having to wait for a full build *in
526case* there are compiler errors in any of the files you've changed.
527
528Runtime testing
529---------------
530
531Even a successful build or boot test is not necessarily enough to rule
532out a missing dependency somewhere. Even though the chances are small,
533there could be code changes where two independent changes to the same
534file result in no conflicts, no compile-time errors, and runtime errors
535only in exceptional cases.
536
537One concrete example of this was a pair of patches to the system call
538entry code where the first patch saved/restored a register and a later
539patch made use of the same register somewhere in the middle of this
540sequence. Since there was no overlap between the changes, one could
541cherry-pick the second patch, have no conflicts, and believe that
542everything was fine, when in fact the code was now scribbling over an
543unsaved register.
544
545Although the vast majority of errors will be caught during compilation
546or by superficially exercising the code, the only way to *really* verify
547a backport is to review the final patch with the same level of scrutiny
548as you would (or should) give to any other patch. Having unit tests and
549regression tests or other types of automatic testing can help increase
550the confidence in the correctness of a backport.
551
552Submitting backports to stable
553==============================
554
555As the stable maintainers try to cherry-pick mainline fixes onto their
556stable kernels, they may send out emails asking for backports when when
557encountering conflicts, see e.g.
558<https://lore.kernel.org/stable/2023101528-jawed-shelving-071a@gregkh/>.
559These emails typically include the exact steps you need to cherry-pick
560the patch to the correct tree and submit the patch.
561
562One thing to make sure is that your changelog conforms to the expected
563format::
564
565  <original patch title>
566  
567  [ Upstream commit <mainline rev> ]
568  
569  <rest of the original changelog>
570  [ <summary of the conflicts and their resolutions> ]
571  Signed-off-by: <your name and email>
572
573The "Upstream commit" line is sometimes slightly different depending on
574the stable version. Older version used this format::
575
576  commit <mainline rev> upstream.
577
578It is most common to indicate the kernel version the patch applies to
579in the email subject line (using e.g.
580``git send-email --subject-prefix='PATCH 6.1.y'``), but you can also put
581it in the Signed-off-by:-area or below the ``---`` line.
582
583The stable maintainers expect separate submissions for each active
584stable version, and each submission should also be tested separately.
585
586A few final words of advice
587===========================
588
5891) Approach the backporting process with humility.
5902) Understand the patch you are backporting; this means reading both
591   the changelog and the code.
5923) Be honest about your confidence in the result when submitting the
593   patch.
5944) Ask relevant maintainers for explicit acks.
595
596Examples
597========
598
599The above shows roughly the idealized process of backporting a patch.
600For a more concrete example, see this video tutorial where two patches
601are backported from mainline to stable:
602`Backporting Linux Kernel Patches`_.
603
604.. _Backporting Linux Kernel Patches: https://youtu.be/sBR7R1V2FeA
605