Skip to content

Commit 6936b58

Browse files
committed
diff -B -M: fix output for "copy and then rewrite" case
Starting from a single file, A, if you create B as a copy of A (and possibly make some edit) and then make extensive change to A, you will see: $ git diff -C --name-status C89 A B M A which is expected. However, if you ask the same question in a different way, you see this: $ git diff -B -M --name-status R89 A B M100 A telling us that A was rename-edited into B (as if "A will no longer exist as the result") and at the same time A itself was extensively edited. In this case, because the resulting tree still does have file A (even if it has contents vastly different from the original), we should use "C"opy, not "R"ename, to avoid hinting that A somehow goes away. Two existing tests were depending on the wrong behaviour, and fixed. Signed-off-by: Junio C Hamano <[email protected]>
1 parent eeff891 commit 6936b58

3 files changed

+11
-3
lines changed

diffcore-break.c

+7
Original file line numberDiff line numberDiff line change
@@ -246,6 +246,13 @@ static void merge_broken(struct diff_filepair *p,
246246

247247
dp = diff_queue(outq, d->one, c->two);
248248
dp->score = p->score;
249+
/*
250+
* We will be one extra user of the same src side of the
251+
* broken pair, if it was used as the rename source for other
252+
* paths elsewhere. Increment to mark that the path stays
253+
* in the resulting tree.
254+
*/
255+
d->one->rename_used++;
249256
diff_free_filespec_data(d->two);
250257
diff_free_filespec_data(c->one);
251258
free(d);

t/t4008-diff-break-rewrite.sh

+2-2
Original file line numberDiff line numberDiff line change
@@ -123,10 +123,10 @@ test_expect_success \
123123
'git diff-index -B -M "$tree" >current'
124124

125125
# file0 changed from regular to symlink. file1 is very close to the preimage of file0.
126-
# because we break file0, file1 can become a rename of it.
126+
# the change does not make file0 disappear, so file1 is denoted as a copy of file0
127127
cat >expected <<\EOF
128128
:100644 120000 f5deac7be59e7eeab8657fd9ae706fd6a57daed2 67be421f88824578857624f7b3dc75e99a8a1481 T file0
129-
:100644 100644 6ff87c4664981e4397625791c8ea3bbb5f2279a3 f5deac7be59e7eeab8657fd9ae706fd6a57daed2 R file0 file1
129+
:100644 100644 6ff87c4664981e4397625791c8ea3bbb5f2279a3 f5deac7be59e7eeab8657fd9ae706fd6a57daed2 C file0 file1
130130
EOF
131131

132132
test_expect_success \

t/t4023-diff-rename-typechange.sh

+2-1
Original file line numberDiff line numberDiff line change
@@ -76,7 +76,8 @@ test_expect_success 'moves and renames' '
7676
7777
git diff-tree three four -r --name-status -B -M | sort >actual &&
7878
{
79-
echo "R100 foo bar"
79+
# see -B -M (#6) in t4008
80+
echo "C100 foo bar"
8081
echo "T100 foo"
8182
} | sort >expect &&
8283
test_cmp expect actual

0 commit comments

Comments
 (0)