Skip to content

Commit 4549162

Browse files
kjbracey2gitster
authored andcommitted
mergetools/p4merge: create a base if none available
Originally, with no base, Git gave P4Merge $LOCAL as a dummy base: p4merge "$LOCAL" "$LOCAL" "$REMOTE" "$MERGED" Commit 0a0ec7b changed this to: p4merge "empty file" "$LOCAL" "$REMOTE" "$MERGED" to avoid the problem of being unable to save in some circumstances with similar inputs. Unfortunately this approach produces much worse results on differing inputs. P4Merge really regards the blank file as the base, and once you have just a couple of differences between the two branches you end up with one a massive full-file conflict. The 3-way diff is not readable, and you have to invoke "difftool MERGE_HEAD HEAD" manually to get a useful view. The original approach appears to have invoked special 2-way merge behaviour in P4Merge that occurs only if the base filename is "" or equal to the left input. You get a good visual comparison, and it does not auto-resolve differences. (Normally if one branch matched the base, it would autoresolve to the other branch). But there appears to be no way of getting this 2-way behaviour and being able to reliably save. Having base==left appears to be triggering other assumptions. There are tricks the user can use to force the save icon on, but it's not intuitive. So we now follow a suggestion given in the original patch's discussion: generate a virtual base, consisting of the lines common to the two branches. This is the same as the technique used in resolve and octopus merges, so we relocate that code to a shared function. Note that if there are no differences at the same location, this technique can lead to automatic resolution without conflict, combining everything from the 2 files. As with the other merges using this technique, we assume the user will inspect the result before saving. Signed-off-by: Kevin Bracey <kevin@bracey.fi> Reviewed-by: David Aguilar <davvid@gmail.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
1 parent c699a7c commit 4549162

File tree

4 files changed

+28
-14
lines changed

4 files changed

+28
-14
lines changed

Documentation/git-sh-setup.txt

Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -82,6 +82,12 @@ get_author_ident_from_commit::
8282
outputs code for use with eval to set the GIT_AUTHOR_NAME,
8383
GIT_AUTHOR_EMAIL and GIT_AUTHOR_DATE variables for a given commit.
8484

85+
create_virtual_base::
86+
modifies the first file so only lines in common with the
87+
second file remain. If there is insufficient common material,
88+
then the first file is left empty. The result is suitable
89+
as a virtual base input for a 3-way merge.
90+
8591
GIT
8692
---
8793
Part of the linkgit:git[1] suite

git-merge-one-file.sh

Lines changed: 5 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -104,30 +104,22 @@ case "${1:-.}${2:-.}${3:-.}" in
104104
;;
105105
esac
106106

107-
src2=`git-unpack-file $3`
107+
src1=$(git-unpack-file $2)
108+
src2=$(git-unpack-file $3)
108109
case "$1" in
109110
'')
110111
echo "Added $4 in both, but differently."
111-
# This extracts OUR file in $orig, and uses git apply to
112-
# remove lines that are unique to ours.
113-
orig=`git-unpack-file $2`
114-
sz0=`wc -c <"$orig"`
115-
@@DIFF@@ -u -La/$orig -Lb/$orig $orig $src2 | git apply --no-add
116-
sz1=`wc -c <"$orig"`
117-
118-
# If we do not have enough common material, it is not
119-
# worth trying two-file merge using common subsections.
120-
expr $sz0 \< $sz1 \* 2 >/dev/null || : >$orig
112+
orig=$(git-unpack-file $2)
113+
create_virtual_base "$orig" "$src2"
121114
;;
122115
*)
123116
echo "Auto-merging $4"
124-
orig=`git-unpack-file $1`
117+
orig=$(git-unpack-file $1)
125118
;;
126119
esac
127120

128121
# Be careful for funny filename such as "-L" in "$4", which
129122
# would confuse "merge" greatly.
130-
src1=`git-unpack-file $2`
131123
git merge-file "$src1" "$orig" "$src2"
132124
ret=$?
133125
msg=

git-sh-setup.sh

Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -249,6 +249,18 @@ clear_local_git_env() {
249249
unset $(git rev-parse --local-env-vars)
250250
}
251251
252+
# Generate a virtual base file for a two-file merge. Uses git apply to
253+
# remove lines from $1 that are not in $2, leaving only common lines.
254+
create_virtual_base() {
255+
sz0=$(wc -c <"$1")
256+
@@DIFF@@ -u -La/"$1" -Lb/"$1" "$1" "$2" | git apply --no-add
257+
sz1=$(wc -c <"$1")
258+
259+
# If we do not have enough common material, it is not
260+
# worth trying two-file merge using common subsections.
261+
expr $sz0 \< $sz1 \* 2 >/dev/null || : >"$1"
262+
}
263+
252264
253265
# Platform specific tweaks to work around some commands
254266
case $(uname -s) in

mergetools/p4merge

Lines changed: 5 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,11 @@ diff_cmd () {
2121

2222
merge_cmd () {
2323
touch "$BACKUP"
24-
$base_present || >"$BASE"
24+
if ! $base_present
25+
then
26+
cp -- "$LOCAL" "$BASE"
27+
create_virtual_base "$BASE" "$REMOTE"
28+
fi
2529
"$merge_tool_path" "$BASE" "$REMOTE" "$LOCAL" "$MERGED"
2630
check_unchanged
2731
}

0 commit comments

Comments
 (0)