[PATCH] The diff-raw format updates.

Update the diff-raw format as Linus and I discussed, except that
it does not use sequence of underscore '_' letters to express
nonexistence.  All '0' mode is used for that purpose instead.

The new diff-raw format can express rename/copy, and the earlier
restriction that -M and -C _must_ be used with the patch format
output is no longer necessary.  The patch makes -M and -C flags
independent of -p flag, so you need to say git-whatchanged -M -p
to get the diff/patch format.

Updated are both documentations and tests.

Signed-off-by: Junio C Hamano <junkio@cox.net>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
This commit is contained in:
Junio C Hamano 2005-05-21 19:42:18 -07:00 committed by Linus Torvalds
parent 38c6f78059
commit 81e50eabf0
16 changed files with 326 additions and 317 deletions

View File

@ -16,25 +16,30 @@ git-diff-tree [-r] <tree-ish-1> <tree-ish-2> [<pattern>...]::
git-diff-files [<pattern>...]:: git-diff-files [<pattern>...]::
compares the cache and the files on the filesystem. compares the cache and the files on the filesystem.
The following desription uses "old" and "new" to mean those
compared entities.
For files in old but not in new (i.e. removed): An output line is formatted this way:
-<mode> \t <type> \t <object> \t <path> ':' <mode> ' ' <mode> ' ' <sha1> ' ' <sha1> I <path> I <path> L
For files not in old but in new (i.e. added): By default, I and L are '\t' and '\n' respectively. When '-z'
flag is in effect, both I and L are '\0'.
+<mode> \t <type> \t <object> \t <path> In each <mode>, <sha1> and <path> pair, left hand side describes
the left hand side of what is being compared (<tree-ish> in
git-diff-cache, <tree-ish-1> in git-diff-tree, cache contents in
git-diff-files). Non-existence is shown by having 000000 in the
<mode> column. That is, 000000 appears as the first <mode> for
newly created files, and as the second <mode> for deleted files.
For files that differ: Usually two <path> are the same. When rename/copy detection is
used, however, an "create" and another "delete" records can be
merged into a single record that has two <path>, old name and
new name.
*<old-mode>-><new-mode> \t <type> \t <old-sha1>-><new-sha1> \t <path> <sha1> is shown as all 0's if new is a file on the filesystem
and it is out of sync with the cache. Example:
<new-sha1> is shown as all 0's if new is a file on the :100644 100644 5be4a4...... 000000...... file.c file.c
filesystem and it is out of sync with the cache. Example:
*100644->100644 blob 5be4a4.......->000000....... file.c
Generating patches with -p Generating patches with -p

View File

@ -34,10 +34,10 @@ OPTIONS
\0 line termination on output \0 line termination on output
-M:: -M::
Detect renames; implies -p. Detect renames.
-C:: -C::
Detect copies as well as renames; implies -p. Detect copies as well as renames.
-S<string>:: -S<string>::
Look for differences that contains the change in <string>. Look for differences that contains the change in <string>.

View File

@ -30,10 +30,10 @@ OPTIONS
Output diff in reverse. Output diff in reverse.
-M:: -M::
Detect renames; implies -p. Detect renames.
-C:: -C::
Detect copies as well as renames; implies -p. Detect copies as well as renames.
-S<string>:: -S<string>::
Look for differences that contains the change in <string>. Look for differences that contains the change in <string>.

View File

@ -34,11 +34,10 @@ OPTIONS
git-diff-tree, this flag implies '-r' as well. git-diff-tree, this flag implies '-r' as well.
-M:: -M::
Detect renames; implies -p, in turn implying also '-r'. Detect renames.
-C:: -C::
Detect copies as well as renames; implies -p, in turn Detect copies as well as renames.
implying also '-r'.
-R:: -R::
Output diff in reverse. Output diff in reverse.

View File

@ -2,9 +2,8 @@
#include "diff.h" #include "diff.h"
static int cached_only = 0; static int cached_only = 0;
static int generate_patch = 0; static int diff_output_format = DIFF_FORMAT_HUMAN;
static int match_nonexisting = 0; static int match_nonexisting = 0;
static int line_termination = '\n';
static int detect_rename = 0; static int detect_rename = 0;
static int reverse_diff = 0; static int reverse_diff = 0;
static int diff_score_opt = 0; static int diff_score_opt = 0;
@ -174,22 +173,21 @@ int main(int argc, char **argv)
continue; continue;
} }
if (!strcmp(arg, "-p")) { if (!strcmp(arg, "-p")) {
generate_patch = 1; diff_output_format = DIFF_FORMAT_PATCH;
continue; continue;
} }
if (!strncmp(arg, "-M", 2)) { if (!strncmp(arg, "-M", 2)) {
generate_patch = detect_rename = 1; detect_rename = 1;
diff_score_opt = diff_scoreopt_parse(arg); diff_score_opt = diff_scoreopt_parse(arg);
continue; continue;
} }
if (!strncmp(arg, "-C", 2)) { if (!strncmp(arg, "-C", 2)) {
generate_patch = 1;
detect_rename = 2; detect_rename = 2;
diff_score_opt = diff_scoreopt_parse(arg); diff_score_opt = diff_scoreopt_parse(arg);
continue; continue;
} }
if (!strcmp(arg, "-z")) { if (!strcmp(arg, "-z")) {
line_termination = '\0'; diff_output_format = DIFF_FORMAT_MACHINE;
continue; continue;
} }
if (!strcmp(arg, "-R")) { if (!strcmp(arg, "-R")) {
@ -214,7 +212,7 @@ int main(int argc, char **argv)
if (argc != 2 || get_sha1(argv[1], tree_sha1)) if (argc != 2 || get_sha1(argv[1], tree_sha1))
usage(diff_cache_usage); usage(diff_cache_usage);
diff_setup(reverse_diff, (generate_patch ? -1 : line_termination)); diff_setup(reverse_diff, diff_output_format);
mark_merge_entries(); mark_merge_entries();

View File

@ -9,8 +9,7 @@
static const char *diff_files_usage = static const char *diff_files_usage =
"git-diff-files [-p] [-q] [-r] [-z] [-M] [-C] [-R] [-S<string>] [paths...]"; "git-diff-files [-p] [-q] [-r] [-z] [-M] [-C] [-R] [-S<string>] [paths...]";
static int generate_patch = 0; static int diff_output_format = DIFF_FORMAT_HUMAN;
static int line_termination = '\n';
static int detect_rename = 0; static int detect_rename = 0;
static int reverse_diff = 0; static int reverse_diff = 0;
static int diff_score_opt = 0; static int diff_score_opt = 0;
@ -57,7 +56,7 @@ int main(int argc, char **argv)
while (1 < argc && argv[1][0] == '-') { while (1 < argc && argv[1][0] == '-') {
if (!strcmp(argv[1], "-p")) if (!strcmp(argv[1], "-p"))
generate_patch = 1; diff_output_format = DIFF_FORMAT_PATCH;
else if (!strcmp(argv[1], "-q")) else if (!strcmp(argv[1], "-q"))
silent = 1; silent = 1;
else if (!strcmp(argv[1], "-r")) else if (!strcmp(argv[1], "-r"))
@ -65,19 +64,18 @@ int main(int argc, char **argv)
else if (!strcmp(argv[1], "-s")) else if (!strcmp(argv[1], "-s"))
; /* no-op */ ; /* no-op */
else if (!strcmp(argv[1], "-z")) else if (!strcmp(argv[1], "-z"))
line_termination = 0; diff_output_format = DIFF_FORMAT_MACHINE;
else if (!strcmp(argv[1], "-R")) else if (!strcmp(argv[1], "-R"))
reverse_diff = 1; reverse_diff = 1;
else if (!strcmp(argv[1], "-S")) else if (!strcmp(argv[1], "-S"))
pickaxe = argv[1] + 2; pickaxe = argv[1] + 2;
else if (!strncmp(argv[1], "-M", 2)) { else if (!strncmp(argv[1], "-M", 2)) {
diff_score_opt = diff_scoreopt_parse(argv[1]); diff_score_opt = diff_scoreopt_parse(argv[1]);
detect_rename = generate_patch = 1; detect_rename = 1;
} }
else if (!strncmp(argv[1], "-C", 2)) { else if (!strncmp(argv[1], "-C", 2)) {
diff_score_opt = diff_scoreopt_parse(argv[1]); diff_score_opt = diff_scoreopt_parse(argv[1]);
detect_rename = 2; detect_rename = 2;
generate_patch = 1;
} }
else else
usage(diff_files_usage); usage(diff_files_usage);
@ -92,7 +90,7 @@ int main(int argc, char **argv)
exit(1); exit(1);
} }
diff_setup(reverse_diff, (generate_patch ? -1 : line_termination)); diff_setup(reverse_diff, diff_output_format);
for (i = 0; i < entries; i++) { for (i = 0; i < entries; i++) {
struct stat st; struct stat st;

View File

@ -8,88 +8,51 @@
static int detect_rename = 0; static int detect_rename = 0;
static int diff_score_opt = 0; static int diff_score_opt = 0;
static int generate_patch = 1;
static const char *pickaxe = NULL; static const char *pickaxe = NULL;
static int diff_output_style = DIFF_FORMAT_PATCH;
static int line_termination = '\n';
static int inter_name_termination = '\t';
static int parse_oneside_change(const char *cp, int *mode, static int parse_diff_raw(char *buf1, char *buf2, char *buf3)
unsigned char *sha1, char *path)
{ {
int ch, m; char old_path[PATH_MAX];
m = 0;
while ((ch = *cp) && '0' <= ch && ch <= '7') {
m = (m << 3) | (ch - '0');
cp++;
}
*mode = m;
if (strncmp(cp, "\tblob\t", 6) && strncmp(cp, " blob ", 6) &&
strncmp(cp, "\ttree\t", 6) && strncmp(cp, " tree ", 6))
return -1;
cp += 6;
if (get_sha1_hex(cp, sha1))
return -1;
cp += 40;
if ((*cp != '\t') && *cp != ' ')
return -1;
strcpy(path, ++cp);
return 0;
}
static int parse_diff_raw_output(const char *buf)
{
char path[PATH_MAX];
unsigned char old_sha1[20], new_sha1[20]; unsigned char old_sha1[20], new_sha1[20];
const char *cp = buf; char *ep;
char *cp = buf1;
int ch, old_mode, new_mode; int ch, old_mode, new_mode;
switch (*cp++) { old_mode = new_mode = 0;
case 'U': while ((ch = *cp) && ('0' <= ch && ch <= '7')) {
diff_unmerge(cp + 1); old_mode = (old_mode << 3) | (ch - '0');
break; cp++;
case '+':
if (parse_oneside_change(cp, &new_mode, new_sha1, path))
return -1;
diff_addremove('+', new_mode, new_sha1, path, NULL);
break;
case '-':
if (parse_oneside_change(cp, &old_mode, old_sha1, path))
return -1;
diff_addremove('-', old_mode, old_sha1, path, NULL);
break;
case '*':
old_mode = new_mode = 0;
while ((ch = *cp) && ('0' <= ch && ch <= '7')) {
old_mode = (old_mode << 3) | (ch - '0');
cp++;
}
if (strncmp(cp, "->", 2))
return -1;
cp += 2;
while ((ch = *cp) && ('0' <= ch && ch <= '7')) {
new_mode = (new_mode << 3) | (ch - '0');
cp++;
}
if (strncmp(cp, "\tblob\t", 6) && strncmp(cp, " blob ", 6) &&
strncmp(cp, "\ttree\t", 6) && strncmp(cp, " tree ", 6))
return -1;
cp += 6;
if (get_sha1_hex(cp, old_sha1))
return -1;
cp += 40;
if (strncmp(cp, "->", 2))
return -1;
cp += 2;
if (get_sha1_hex(cp, new_sha1))
return -1;
cp += 40;
if ((*cp != '\t') && *cp != ' ')
return -1;
strcpy(path, ++cp);
diff_change(old_mode, new_mode, old_sha1, new_sha1, path, NULL);
break;
default:
return -1;
} }
if (*cp++ != ' ')
return -1;
while ((ch = *cp) && ('0' <= ch && ch <= '7')) {
new_mode = (new_mode << 3) | (ch - '0');
cp++;
}
if (*cp++ != ' ')
return -1;
if (get_sha1_hex(cp, old_sha1))
return -1;
cp += 40;
if (*cp++ != ' ')
return -1;
if (get_sha1_hex(cp, new_sha1))
return -1;
cp += 40;
if (*cp++ != inter_name_termination)
return -1;
if (buf2)
cp = buf2;
ep = strchr(cp, inter_name_termination);
if (!ep)
return -1;
*ep++ = 0;
strcpy(old_path, cp);
diff_guif(old_mode, new_mode, old_sha1, new_sha1,
old_path, buf3 ? buf3 : ep);
return 0; return 0;
} }
@ -97,19 +60,22 @@ static const char *diff_helper_usage =
"git-diff-helper [-z] [-R] [-M] [-C] [-S<string>] paths..."; "git-diff-helper [-z] [-R] [-M] [-C] [-S<string>] paths...";
int main(int ac, const char **av) { int main(int ac, const char **av) {
struct strbuf sb; struct strbuf sb1, sb2, sb3;
int reverse = 0; int reverse_diff = 0;
int line_termination = '\n';
strbuf_init(&sb); strbuf_init(&sb1);
strbuf_init(&sb2);
strbuf_init(&sb3);
while (1 < ac && av[1][0] == '-') { while (1 < ac && av[1][0] == '-') {
if (av[1][1] == 'R') if (av[1][1] == 'R')
reverse = 1; reverse_diff = 1;
else if (av[1][1] == 'z') else if (av[1][1] == 'z')
line_termination = 0; line_termination = inter_name_termination = 0;
else if (av[1][1] == 'p') /* hidden from the help */ else if (av[1][1] == 'p') /* hidden from the help */
generate_patch = 0; diff_output_style = DIFF_FORMAT_HUMAN;
else if (av[1][1] == 'P') /* hidden from the help */
diff_output_style = DIFF_FORMAT_MACHINE;
else if (av[1][1] == 'M') { else if (av[1][1] == 'M') {
detect_rename = 1; detect_rename = 1;
diff_score_opt = diff_scoreopt_parse(av[1]); diff_score_opt = diff_scoreopt_parse(av[1]);
@ -127,19 +93,38 @@ int main(int ac, const char **av) {
} }
/* the remaining parameters are paths patterns */ /* the remaining parameters are paths patterns */
diff_setup(reverse, (generate_patch ? -1 : line_termination)); diff_setup(reverse_diff, diff_output_style);
while (1) { while (1) {
int status; int status;
read_line(&sb, stdin, line_termination); read_line(&sb1, stdin, line_termination);
if (sb.eof) if (sb1.eof)
break; break;
status = parse_diff_raw_output(sb.buf); switch (sb1.buf[0]) {
case 'U':
diff_unmerge(sb1.buf + 2);
continue;
case ':':
break;
default:
goto unrecognized;
}
if (!line_termination) {
read_line(&sb2, stdin, line_termination);
if (sb2.eof)
break;
read_line(&sb3, stdin, line_termination);
if (sb3.eof)
break;
status = parse_diff_raw(sb1.buf+1, sb2.buf, sb3.buf);
}
else
status = parse_diff_raw(sb1.buf+1, NULL, NULL);
if (status) { if (status) {
unrecognized:
diff_flush(av+1, ac-1); diff_flush(av+1, ac-1);
printf("%s%c", sb.buf, line_termination); printf("%s%c", sb1.buf, line_termination);
} }
} }
if (detect_rename) if (detect_rename)
diff_detect_rename(detect_rename, diff_score_opt); diff_detect_rename(detect_rename, diff_score_opt);
if (pickaxe) if (pickaxe)

View File

@ -8,8 +8,7 @@ static int verbose_header = 0;
static int ignore_merges = 1; static int ignore_merges = 1;
static int recursive = 0; static int recursive = 0;
static int read_stdin = 0; static int read_stdin = 0;
static int line_termination = '\n'; static int diff_output_format = DIFF_FORMAT_HUMAN;
static int generate_patch = 0;
static int detect_rename = 0; static int detect_rename = 0;
static int reverse_diff = 0; static int reverse_diff = 0;
static int diff_score_opt = 0; static int diff_score_opt = 0;
@ -269,7 +268,7 @@ static int diff_tree_sha1(const unsigned char *old, const unsigned char *new, co
static void call_diff_setup(void) static void call_diff_setup(void)
{ {
diff_setup(reverse_diff, (generate_patch ? -1 : line_termination)); diff_setup(reverse_diff, diff_output_format);
} }
static void call_diff_flush(void) static void call_diff_flush(void)
@ -501,7 +500,8 @@ int main(int argc, char **argv)
continue; continue;
} }
if (!strcmp(arg, "-p")) { if (!strcmp(arg, "-p")) {
recursive = generate_patch = 1; diff_output_format = DIFF_FORMAT_PATCH;
recursive = 1;
continue; continue;
} }
if (!strncmp(arg, "-S", 2)) { if (!strncmp(arg, "-S", 2)) {
@ -509,18 +509,17 @@ int main(int argc, char **argv)
continue; continue;
} }
if (!strncmp(arg, "-M", 2)) { if (!strncmp(arg, "-M", 2)) {
detect_rename = recursive = generate_patch = 1; detect_rename = 1;
diff_score_opt = diff_scoreopt_parse(arg); diff_score_opt = diff_scoreopt_parse(arg);
continue; continue;
} }
if (!strncmp(arg, "-C", 2)) { if (!strncmp(arg, "-C", 2)) {
detect_rename = 2; detect_rename = 2;
recursive = generate_patch = 1;
diff_score_opt = diff_scoreopt_parse(arg); diff_score_opt = diff_scoreopt_parse(arg);
continue; continue;
} }
if (!strcmp(arg, "-z")) { if (!strcmp(arg, "-z")) {
line_termination = '\0'; diff_output_format = DIFF_FORMAT_MACHINE;
continue; continue;
} }
if (!strcmp(arg, "-m")) { if (!strcmp(arg, "-m")) {

132
diff.c
View File

@ -13,7 +13,9 @@ static const char *diff_opts = "-pu";
static unsigned char null_sha1[20] = { 0, }; static unsigned char null_sha1[20] = { 0, };
static int reverse_diff; static int reverse_diff;
static int diff_raw_output = -1; static int generate_patch;
static int line_termination = '\n';
static int inter_name_termination = '\t';
static const char **pathspec; static const char **pathspec;
static int speccnt; static int speccnt;
@ -163,20 +165,23 @@ struct diff_filespec *alloc_filespec(const char *path)
struct diff_filespec *spec = xmalloc(sizeof(*spec) + namelen + 1); struct diff_filespec *spec = xmalloc(sizeof(*spec) + namelen + 1);
spec->path = (char *)(spec + 1); spec->path = (char *)(spec + 1);
strcpy(spec->path, path); strcpy(spec->path, path);
spec->should_free = spec->should_munmap = spec->file_valid = 0; spec->should_free = spec->should_munmap = 0;
spec->xfrm_flags = 0; spec->xfrm_flags = 0;
spec->size = 0; spec->size = 0;
spec->data = 0; spec->data = 0;
spec->mode = 0;
memset(spec->sha1, 0, 20);
return spec; return spec;
} }
void fill_filespec(struct diff_filespec *spec, const unsigned char *sha1, void fill_filespec(struct diff_filespec *spec, const unsigned char *sha1,
unsigned short mode) unsigned short mode)
{ {
spec->mode = mode; if (mode) { /* just playing defensive */
memcpy(spec->sha1, sha1, 20); spec->mode = mode;
spec->sha1_valid = !!memcmp(sha1, null_sha1, 20); memcpy(spec->sha1, sha1, 20);
spec->file_valid = 1; spec->sha1_valid = !!memcmp(sha1, null_sha1, 20);
}
} }
/* /*
@ -231,7 +236,7 @@ static int work_tree_matches(const char *name, const unsigned char *sha1)
int diff_populate_filespec(struct diff_filespec *s) int diff_populate_filespec(struct diff_filespec *s)
{ {
int err = 0; int err = 0;
if (!s->file_valid) if (!DIFF_FILE_VALID(s))
die("internal error: asking to populate invalid file."); die("internal error: asking to populate invalid file.");
if (S_ISDIR(s->mode)) if (S_ISDIR(s->mode))
return -1; return -1;
@ -316,7 +321,7 @@ static void prepare_temp_file(const char *name,
struct diff_tempfile *temp, struct diff_tempfile *temp,
struct diff_filespec *one) struct diff_filespec *one)
{ {
if (!one->file_valid) { if (!DIFF_FILE_VALID(one)) {
not_a_valid_file: not_a_valid_file:
/* A '-' entry produces this for file-2, and /* A '-' entry produces this for file-2, and
* a '+' entry produces this for file-1. * a '+' entry produces this for file-1.
@ -509,10 +514,22 @@ int diff_scoreopt_parse(const char *opt)
return MAX_SCORE * num / scale; return MAX_SCORE * num / scale;
} }
void diff_setup(int reverse_diff_, int diff_raw_output_) void diff_setup(int reverse_diff_, int diff_output_style)
{ {
reverse_diff = reverse_diff_; reverse_diff = reverse_diff_;
diff_raw_output = diff_raw_output_; generate_patch = 0;
switch (diff_output_style) {
case DIFF_FORMAT_HUMAN:
line_termination = '\n';
inter_name_termination = '\t';
break;
case DIFF_FORMAT_MACHINE:
line_termination = inter_name_termination = 0;
break;
case DIFF_FORMAT_PATCH:
generate_patch = 1;
break;
}
} }
struct diff_queue_struct diff_queued_diff; struct diff_queue_struct diff_queued_diff;
@ -536,43 +553,17 @@ struct diff_filepair *diff_queue(struct diff_queue_struct *queue,
return dp; return dp;
} }
static const char *git_object_type(unsigned mode)
{
return S_ISDIR(mode) ? "tree" : "blob";
}
static void diff_flush_raw(struct diff_filepair *p) static void diff_flush_raw(struct diff_filepair *p)
{ {
struct diff_filespec *it; /*
int addremove; * We used to reject rename/copy but new diff-raw can express them.
*/
/* raw output does not have a way to express rename nor copy */ printf(":%06o %06o %s ",
if (strcmp(p->one->path, p->two->path)) p->one->mode, p->two->mode, sha1_to_hex(p->one->sha1));
return; printf("%s%c%s%c%s%c",
sha1_to_hex(p->two->sha1), inter_name_termination,
if (p->one->file_valid && p->two->file_valid) { p->one->path, inter_name_termination,
char hex[41]; p->two->path, line_termination);
strcpy(hex, sha1_to_hex(p->one->sha1));
printf("*%06o->%06o %s %s->%s %s%c",
p->one->mode, p->two->mode,
git_object_type(p->one->mode),
hex, sha1_to_hex(p->two->sha1),
p->one->path, diff_raw_output);
return;
}
if (p->one->file_valid) {
it = p->one;
addremove = '-';
} else {
it = p->two;
addremove = '+';
}
printf("%c%06o %s %s %s%c",
addremove,
it->mode, git_object_type(it->mode),
sha1_to_hex(it->sha1), it->path, diff_raw_output);
} }
static void diff_flush_patch(struct diff_filepair *p) static void diff_flush_patch(struct diff_filepair *p)
@ -581,8 +572,8 @@ static void diff_flush_patch(struct diff_filepair *p)
name = p->one->path; name = p->one->path;
other = (strcmp(name, p->two->path) ? p->two->path : NULL); other = (strcmp(name, p->two->path) ? p->two->path : NULL);
if ((p->one->file_valid && S_ISDIR(p->one->mode)) || if ((DIFF_FILE_VALID(p->one) && S_ISDIR(p->one->mode)) ||
(p->two->file_valid && S_ISDIR(p->two->mode))) (DIFF_FILE_VALID(p->two) && S_ISDIR(p->two->mode)))
return; /* no tree diffs in patch format */ return; /* no tree diffs in patch format */
run_external_diff(name, other, p->one, p->two, p->xfrm_msg); run_external_diff(name, other, p->one, p->two, p->xfrm_msg);
@ -596,11 +587,12 @@ static int identical(struct diff_filespec *one, struct diff_filespec *two)
* and filter and clean them up here before producing the output. * and filter and clean them up here before producing the output.
*/ */
if (!one->file_valid && !two->file_valid) if (!DIFF_FILE_VALID(one) && !DIFF_FILE_VALID(two))
return 1; /* not interesting */ return 1; /* not interesting */
/* deletion, addition, mode change and renames are all interesting. */ /* deletion, addition, mode change and renames are all interesting. */
if ((one->file_valid != two->file_valid) || (one->mode != two->mode) || if (DIFF_FILE_VALID(one) != DIFF_FILE_VALID(two) ||
(one->mode != two->mode) ||
strcmp(one->path, two->path)) strcmp(one->path, two->path))
return 0; return 0;
@ -619,10 +611,10 @@ static void diff_flush_one(struct diff_filepair *p)
{ {
if (identical(p->one, p->two)) if (identical(p->one, p->two))
return; return;
if (0 <= diff_raw_output) if (generate_patch)
diff_flush_raw(p);
else
diff_flush_patch(p); diff_flush_patch(p);
else
diff_flush_raw(p);
} }
int diff_queue_is_empty(void) int diff_queue_is_empty(void)
@ -697,10 +689,35 @@ void diff_addremove(int addremove, unsigned mode,
diff_queue(&diff_queued_diff, one, two); diff_queue(&diff_queued_diff, one, two);
} }
void diff_guif(unsigned old_mode,
unsigned new_mode,
const unsigned char *old_sha1,
const unsigned char *new_sha1,
const char *old_path,
const char *new_path)
{
struct diff_filespec *one, *two;
if (reverse_diff) {
unsigned tmp;
const unsigned char *tmp_c;
tmp = old_mode; old_mode = new_mode; new_mode = tmp;
tmp_c = old_sha1; old_sha1 = new_sha1; new_sha1 = tmp_c;
}
one = alloc_filespec(old_path);
two = alloc_filespec(new_path);
if (old_mode)
fill_filespec(one, old_sha1, old_mode);
if (new_mode)
fill_filespec(two, new_sha1, new_mode);
diff_queue(&diff_queued_diff, one, two);
}
void diff_change(unsigned old_mode, unsigned new_mode, void diff_change(unsigned old_mode, unsigned new_mode,
const unsigned char *old_sha1, const unsigned char *old_sha1,
const unsigned char *new_sha1, const unsigned char *new_sha1,
const char *base, const char *path) { const char *base, const char *path)
{
char concatpath[PATH_MAX]; char concatpath[PATH_MAX];
struct diff_filespec *one, *two; struct diff_filespec *one, *two;
@ -722,9 +739,8 @@ void diff_change(unsigned old_mode, unsigned new_mode,
void diff_unmerge(const char *path) void diff_unmerge(const char *path)
{ {
if (0 <= diff_raw_output) { if (generate_patch)
printf("U %s%c", path, diff_raw_output); run_external_diff(path, NULL, NULL, NULL, NULL);
return; else
} printf("U %s%c", path, line_termination);
run_external_diff(path, NULL, NULL, NULL, NULL);
} }

12
diff.h
View File

@ -15,11 +15,21 @@ extern void diff_change(unsigned mode1, unsigned mode2,
const unsigned char *sha2, const unsigned char *sha2,
const char *base, const char *path); const char *base, const char *path);
extern void diff_guif(unsigned mode1,
unsigned mode2,
const unsigned char *sha1,
const unsigned char *sha2,
const char *path1,
const char *path2);
extern void diff_unmerge(const char *path); extern void diff_unmerge(const char *path);
extern int diff_scoreopt_parse(const char *opt); extern int diff_scoreopt_parse(const char *opt);
extern void diff_setup(int reverse, int diff_raw_output); #define DIFF_FORMAT_HUMAN 0
#define DIFF_FORMAT_MACHINE 1
#define DIFF_FORMAT_PATCH 2
extern void diff_setup(int reverse, int diff_output_style);
extern void diff_detect_rename(int, int); extern void diff_detect_rename(int, int);
extern void diff_pickaxe(const char *); extern void diff_pickaxe(const char *);

View File

@ -32,14 +32,14 @@ void diff_pickaxe(const char *needle)
for (i = 0; i < q->nr; i++) { for (i = 0; i < q->nr; i++) {
struct diff_filepair *p = q->queue[i]; struct diff_filepair *p = q->queue[i];
if (!p->one->file_valid) { if (!DIFF_FILE_VALID(p->one)) {
if (!p->two->file_valid) if (!DIFF_FILE_VALID(p->two))
continue; /* ignore nonsense */ continue; /* ignore nonsense */
/* created */ /* created */
if (contains(p->two, needle, len)) if (contains(p->two, needle, len))
diff_queue(&outq, p->one, p->two); diff_queue(&outq, p->one, p->two);
} }
else if (!p->two->file_valid) { else if (!DIFF_FILE_VALID(p->two)) {
if (contains(p->one, needle, len)) if (contains(p->one, needle, len))
diff_queue(&outq, p->one, p->two); diff_queue(&outq, p->one, p->two);
} }

View File

@ -142,7 +142,7 @@ static void debug_filespec(struct diff_filespec *s, int x, const char *one)
fprintf(stderr, "queue[%d] %s (%s) %s %06o %s\n", fprintf(stderr, "queue[%d] %s (%s) %s %06o %s\n",
x, one, x, one,
s->path, s->path,
s->file_valid ? "valid" : "invalid", DIFF_FILE_VALID(s) ? "valid" : "invalid",
s->mode, s->mode,
s->sha1_valid ? sha1_to_hex(s->sha1) : ""); s->sha1_valid ? sha1_to_hex(s->sha1) : "");
fprintf(stderr, "queue[%d] %s size %lu flags %d\n", fprintf(stderr, "queue[%d] %s size %lu flags %d\n",
@ -210,7 +210,7 @@ static int needs_to_stay(struct diff_queue_struct *q, int i,
*/ */
while (i < q->nr) { while (i < q->nr) {
struct diff_filepair *p = q->queue[i++]; struct diff_filepair *p = q->queue[i++];
if (!p->two->file_valid) if (!DIFF_FILE_VALID(p->two))
continue; /* removed is fine */ continue; /* removed is fine */
if (strcmp(p->one->path, it->path)) if (strcmp(p->one->path, it->path))
continue; /* not relevant */ continue; /* not relevant */
@ -247,12 +247,12 @@ void diff_detect_rename(int detect_rename,
for (i = 0; i < q->nr; i++) { for (i = 0; i < q->nr; i++) {
struct diff_filepair *p = q->queue[i]; struct diff_filepair *p = q->queue[i];
if (!p->one->file_valid) if (!DIFF_FILE_VALID(p->one))
if (!p->two->file_valid) if (!DIFF_FILE_VALID(p->two))
continue; /* ignore nonsense */ continue; /* ignore nonsense */
else else
diff_rename_pool_add(&created, p->two); diff_rename_pool_add(&created, p->two);
else if (!p->two->file_valid) else if (!DIFF_FILE_VALID(p->two))
diff_rename_pool_add(&deleted, p->one); diff_rename_pool_add(&deleted, p->one);
else if (1 < detect_rename) /* find copy, too */ else if (1 < detect_rename) /* find copy, too */
diff_rename_pool_add(&stay, p->one); diff_rename_pool_add(&stay, p->one);
@ -340,15 +340,15 @@ void diff_detect_rename(int detect_rename,
*/ */
for (i = 0; i < q->nr; i++) { for (i = 0; i < q->nr; i++) {
struct diff_filepair *dp, *p = q->queue[i]; struct diff_filepair *dp, *p = q->queue[i];
if (!p->one->file_valid) { if (!DIFF_FILE_VALID(p->one)) {
if (p->two->file_valid) { if (DIFF_FILE_VALID(p->two)) {
/* creation */ /* creation */
dp = diff_queue(&outq, p->one, p->two); dp = diff_queue(&outq, p->one, p->two);
dp->xfrm_work = 4; dp->xfrm_work = 4;
} }
/* otherwise it is a nonsense; just ignore it */ /* otherwise it is a nonsense; just ignore it */
} }
else if (!p->two->file_valid) { else if (!DIFF_FILE_VALID(p->two)) {
/* deletion */ /* deletion */
dp = diff_queue(&outq, p->one, p->two); dp = diff_queue(&outq, p->one, p->two);
dp->xfrm_work = 2; dp->xfrm_work = 2;
@ -374,14 +374,14 @@ void diff_detect_rename(int detect_rename,
/* Copy it out to q, removing duplicates. */ /* Copy it out to q, removing duplicates. */
for (i = 0; i < outq.nr; i++) { for (i = 0; i < outq.nr; i++) {
struct diff_filepair *p = outq.queue[i]; struct diff_filepair *p = outq.queue[i];
if (!p->one->file_valid) { if (!DIFF_FILE_VALID(p->one)) {
/* created */ /* created */
if (p->two->xfrm_flags & RENAME_DST_MATCHED) if (p->two->xfrm_flags & RENAME_DST_MATCHED)
; /* rename/copy created it already */ ; /* rename/copy created it already */
else else
diff_queue(q, p->one, p->two); diff_queue(q, p->one, p->two);
} }
else if (!p->two->file_valid) { else if (!DIFF_FILE_VALID(p->two)) {
/* deleted */ /* deleted */
if (p->one->xfrm_flags & RENAME_SRC_GONE) if (p->one->xfrm_flags & RENAME_SRC_GONE)
; /* rename/copy deleted it already */ ; /* rename/copy deleted it already */

View File

@ -26,7 +26,7 @@ struct diff_filespec {
* if false, use the name and read from * if false, use the name and read from
* the filesystem. * the filesystem.
*/ */
unsigned file_valid : 1; /* if false the file does not exist */ #define DIFF_FILE_VALID(spec) (((spec)->mode) != 0)
unsigned should_free : 1; /* data should be free()'ed */ unsigned should_free : 1; /* data should be free()'ed */
unsigned should_munmap : 1; /* data should be munmap()'ed */ unsigned should_munmap : 1; /* data should be munmap()'ed */
}; };

View File

@ -155,14 +155,14 @@ test_expect_success \
test "$newtree" = "$tree"' test "$newtree" = "$tree"'
cat >expected <<\EOF cat >expected <<\EOF
*100644->100644 blob f87290f8eb2cbbea7857214459a0739927eab154->0000000000000000000000000000000000000000 path0 :100644 100644 f87290f8eb2cbbea7857214459a0739927eab154 0000000000000000000000000000000000000000 path0 path0
*120000->120000 blob 15a98433ae33114b085f3eb3bb03b832b3180a01->0000000000000000000000000000000000000000 path0sym :120000 120000 15a98433ae33114b085f3eb3bb03b832b3180a01 0000000000000000000000000000000000000000 path0sym path0sym
*100644->100644 blob 3feff949ed00a62d9f7af97c15cd8a30595e7ac7->0000000000000000000000000000000000000000 path2/file2 :100644 100644 3feff949ed00a62d9f7af97c15cd8a30595e7ac7 0000000000000000000000000000000000000000 path2/file2 path2/file2
*120000->120000 blob d8ce161addc5173867a3c3c730924388daedbc38->0000000000000000000000000000000000000000 path2/file2sym :120000 120000 d8ce161addc5173867a3c3c730924388daedbc38 0000000000000000000000000000000000000000 path2/file2sym path2/file2sym
*100644->100644 blob 0aa34cae68d0878578ad119c86ca2b5ed5b28376->0000000000000000000000000000000000000000 path3/file3 :100644 100644 0aa34cae68d0878578ad119c86ca2b5ed5b28376 0000000000000000000000000000000000000000 path3/file3 path3/file3
*120000->120000 blob 8599103969b43aff7e430efea79ca4636466794f->0000000000000000000000000000000000000000 path3/file3sym :120000 120000 8599103969b43aff7e430efea79ca4636466794f 0000000000000000000000000000000000000000 path3/file3sym path3/file3sym
*100644->100644 blob 00fb5908cb97c2564a9783c0c64087333b3b464f->0000000000000000000000000000000000000000 path3/subp3/file3 :100644 100644 00fb5908cb97c2564a9783c0c64087333b3b464f 0000000000000000000000000000000000000000 path3/subp3/file3 path3/subp3/file3
*120000->120000 blob 6649a1ebe9e9f1c553b66f5a6e74136a07ccc57c->0000000000000000000000000000000000000000 path3/subp3/file3sym :120000 120000 6649a1ebe9e9f1c553b66f5a6e74136a07ccc57c 0000000000000000000000000000000000000000 path3/subp3/file3sym path3/subp3/file3sym
EOF EOF
test_expect_success \ test_expect_success \
'validate git-diff-files output for a know cache/work tree state.' \ 'validate git-diff-files output for a know cache/work tree state.' \

View File

@ -10,123 +10,122 @@ test_description='Test diff raw-output.
. ../lib-read-tree-m-3way.sh . ../lib-read-tree-m-3way.sh
cat >.test-plain-OA <<\EOF cat >.test-plain-OA <<\EOF
+100644 blob ccba72ad3888a3520b39efcf780b9ee64167535d AA :000000 100644 0000000000000000000000000000000000000000 ccba72ad3888a3520b39efcf780b9ee64167535d AA AA
+100644 blob 7e426fb079479fd67f6d81f984e4ec649a44bc25 AN :000000 100644 0000000000000000000000000000000000000000 7e426fb079479fd67f6d81f984e4ec649a44bc25 AN AN
-100644 blob bcc68ef997017466d5c9094bcf7692295f588c9a DD :100644 000000 bcc68ef997017466d5c9094bcf7692295f588c9a 0000000000000000000000000000000000000000 DD DD
+040000 tree 6d50f65d3bdab91c63444294d38f08aeff328e42 DF :000000 040000 0000000000000000000000000000000000000000 6d50f65d3bdab91c63444294d38f08aeff328e42 DF DF
-100644 blob 141c1f1642328e4bc46a7d801a71da392e66791e DM :100644 000000 141c1f1642328e4bc46a7d801a71da392e66791e 0000000000000000000000000000000000000000 DM DM
-100644 blob 35abde1506ddf806572ff4d407bd06885d0f8ee9 DN :100644 000000 35abde1506ddf806572ff4d407bd06885d0f8ee9 0000000000000000000000000000000000000000 DN DN
+100644 blob 1d41122ebdd7a640f29d3c9cc4f9d70094374762 LL :000000 100644 0000000000000000000000000000000000000000 1d41122ebdd7a640f29d3c9cc4f9d70094374762 LL LL
*100644->100644 blob 03f24c8c4700babccfd28b654e7e8eac402ad6cd->103d9f89b50b9aad03054b579be5e7aa665f2d57 MD :100644 100644 03f24c8c4700babccfd28b654e7e8eac402ad6cd 103d9f89b50b9aad03054b579be5e7aa665f2d57 MD MD
*100644->100644 blob b258508afb7ceb449981bd9d63d2d3e971bf8d34->b431b272d829ff3aa4d1a5085f4394ab4d3305b6 MM :100644 100644 b258508afb7ceb449981bd9d63d2d3e971bf8d34 b431b272d829ff3aa4d1a5085f4394ab4d3305b6 MM MM
*100644->100644 blob bd084b0c27c7b6cc34f11d6d0509a29be3caf970->a716d58de4a570e0038f5c307bd8db34daea021f MN :100644 100644 bd084b0c27c7b6cc34f11d6d0509a29be3caf970 a716d58de4a570e0038f5c307bd8db34daea021f MN MN
*100644->100644 blob 40c959f984c8b89a2b02520d17f00d717f024397->2ac547ae9614a00d1b28275de608131f7a0e259f SS :100644 100644 40c959f984c8b89a2b02520d17f00d717f024397 2ac547ae9614a00d1b28275de608131f7a0e259f SS SS
*100644->100644 blob 4ac13458899ab908ef3b1128fa378daefc88d356->4c86f9a85fbc5e6804ee2e17a797538fbe785bca TT :100644 100644 4ac13458899ab908ef3b1128fa378daefc88d356 4c86f9a85fbc5e6804ee2e17a797538fbe785bca TT TT
*040000->040000 tree 7d670fdcdb9929f6c7dac196ff78689cd1c566a1->5e5f22072bb39f6e12cf663a57cb634c76eefb49 Z :040000 040000 7d670fdcdb9929f6c7dac196ff78689cd1c566a1 5e5f22072bb39f6e12cf663a57cb634c76eefb49 Z Z
EOF EOF
cat >.test-recursive-OA <<\EOF cat >.test-recursive-OA <<\EOF
+100644 blob ccba72ad3888a3520b39efcf780b9ee64167535d AA :000000 100644 0000000000000000000000000000000000000000 ccba72ad3888a3520b39efcf780b9ee64167535d AA AA
+100644 blob 7e426fb079479fd67f6d81f984e4ec649a44bc25 AN :000000 100644 0000000000000000000000000000000000000000 7e426fb079479fd67f6d81f984e4ec649a44bc25 AN AN
-100644 blob bcc68ef997017466d5c9094bcf7692295f588c9a DD :100644 000000 bcc68ef997017466d5c9094bcf7692295f588c9a 0000000000000000000000000000000000000000 DD DD
+100644 blob 68a6d8b91da11045cf4aa3a5ab9f2a781c701249 DF/DF :000000 100644 0000000000000000000000000000000000000000 68a6d8b91da11045cf4aa3a5ab9f2a781c701249 DF/DF DF/DF
-100644 blob 141c1f1642328e4bc46a7d801a71da392e66791e DM :100644 000000 141c1f1642328e4bc46a7d801a71da392e66791e 0000000000000000000000000000000000000000 DM DM
-100644 blob 35abde1506ddf806572ff4d407bd06885d0f8ee9 DN :100644 000000 35abde1506ddf806572ff4d407bd06885d0f8ee9 0000000000000000000000000000000000000000 DN DN
+100644 blob 1d41122ebdd7a640f29d3c9cc4f9d70094374762 LL :000000 100644 0000000000000000000000000000000000000000 1d41122ebdd7a640f29d3c9cc4f9d70094374762 LL LL
*100644->100644 blob 03f24c8c4700babccfd28b654e7e8eac402ad6cd->103d9f89b50b9aad03054b579be5e7aa665f2d57 MD :100644 100644 03f24c8c4700babccfd28b654e7e8eac402ad6cd 103d9f89b50b9aad03054b579be5e7aa665f2d57 MD MD
*100644->100644 blob b258508afb7ceb449981bd9d63d2d3e971bf8d34->b431b272d829ff3aa4d1a5085f4394ab4d3305b6 MM :100644 100644 b258508afb7ceb449981bd9d63d2d3e971bf8d34 b431b272d829ff3aa4d1a5085f4394ab4d3305b6 MM MM
*100644->100644 blob bd084b0c27c7b6cc34f11d6d0509a29be3caf970->a716d58de4a570e0038f5c307bd8db34daea021f MN :100644 100644 bd084b0c27c7b6cc34f11d6d0509a29be3caf970 a716d58de4a570e0038f5c307bd8db34daea021f MN MN
*100644->100644 blob 40c959f984c8b89a2b02520d17f00d717f024397->2ac547ae9614a00d1b28275de608131f7a0e259f SS :100644 100644 40c959f984c8b89a2b02520d17f00d717f024397 2ac547ae9614a00d1b28275de608131f7a0e259f SS SS
*100644->100644 blob 4ac13458899ab908ef3b1128fa378daefc88d356->4c86f9a85fbc5e6804ee2e17a797538fbe785bca TT :100644 100644 4ac13458899ab908ef3b1128fa378daefc88d356 4c86f9a85fbc5e6804ee2e17a797538fbe785bca TT TT
+100644 blob 8acb8e9750e3f644bf323fcf3d338849db106c77 Z/AA :000000 100644 0000000000000000000000000000000000000000 8acb8e9750e3f644bf323fcf3d338849db106c77 Z/AA Z/AA
+100644 blob 087494262084cefee7ed484d20c8dc0580791272 Z/AN :000000 100644 0000000000000000000000000000000000000000 087494262084cefee7ed484d20c8dc0580791272 Z/AN Z/AN
-100644 blob 879007efae624d2b1307214b24a956f0a8d686a8 Z/DD :100644 000000 879007efae624d2b1307214b24a956f0a8d686a8 0000000000000000000000000000000000000000 Z/DD Z/DD
-100644 blob 9b541b2275c06e3a7b13f28badf5294e2ae63df4 Z/DM :100644 000000 9b541b2275c06e3a7b13f28badf5294e2ae63df4 0000000000000000000000000000000000000000 Z/DM Z/DM
-100644 blob beb5d38c55283d280685ea21a0e50cfcc0ca064a Z/DN :100644 000000 beb5d38c55283d280685ea21a0e50cfcc0ca064a 0000000000000000000000000000000000000000 Z/DN Z/DN
*100644->100644 blob d41fda41b7ec4de46b43cb7ea42a45001ae393d5->a79ac3be9377639e1c7d1edf1ae1b3a5f0ccd8a9 Z/MD :100644 100644 d41fda41b7ec4de46b43cb7ea42a45001ae393d5 a79ac3be9377639e1c7d1edf1ae1b3a5f0ccd8a9 Z/MD Z/MD
*100644->100644 blob 4ca22bae2527d3d9e1676498a0fba3b355bd1278->61422ba9c2c873416061a88cd40a59a35b576474 Z/MM :100644 100644 4ca22bae2527d3d9e1676498a0fba3b355bd1278 61422ba9c2c873416061a88cd40a59a35b576474 Z/MM Z/MM
*100644->100644 blob b16d7b25b869f2beb124efa53467d8a1550ad694->a5c544c21cfcb07eb80a4d89a5b7d1570002edfd Z/MN :100644 100644 b16d7b25b869f2beb124efa53467d8a1550ad694 a5c544c21cfcb07eb80a4d89a5b7d1570002edfd Z/MN Z/MN
EOF EOF
cat >.test-plain-OB <<\EOF cat >.test-plain-OB <<\EOF
+100644 blob 6aa2b5335b16431a0ef71e5c0a28be69183cf6a2 AA :000000 100644 0000000000000000000000000000000000000000 6aa2b5335b16431a0ef71e5c0a28be69183cf6a2 AA AA
-100644 blob bcc68ef997017466d5c9094bcf7692295f588c9a DD :100644 000000 bcc68ef997017466d5c9094bcf7692295f588c9a 0000000000000000000000000000000000000000 DD DD
+100644 blob 71420ab81e254145d26d6fc0cddee64c1acd4787 DF :000000 100644 0000000000000000000000000000000000000000 71420ab81e254145d26d6fc0cddee64c1acd4787 DF DF
*100644->100644 blob 141c1f1642328e4bc46a7d801a71da392e66791e->3c4d8de5fbad08572bab8e10eef8dbb264cf0231 DM :100644 100644 141c1f1642328e4bc46a7d801a71da392e66791e 3c4d8de5fbad08572bab8e10eef8dbb264cf0231 DM DM
+100644 blob 1d41122ebdd7a640f29d3c9cc4f9d70094374762 LL :000000 100644 0000000000000000000000000000000000000000 1d41122ebdd7a640f29d3c9cc4f9d70094374762 LL LL
-100644 blob 03f24c8c4700babccfd28b654e7e8eac402ad6cd MD :100644 000000 03f24c8c4700babccfd28b654e7e8eac402ad6cd 0000000000000000000000000000000000000000 MD MD
*100644->100644 blob b258508afb7ceb449981bd9d63d2d3e971bf8d34->19989d4559aae417fedee240ccf2ba315ea4dc2b MM :100644 100644 b258508afb7ceb449981bd9d63d2d3e971bf8d34 19989d4559aae417fedee240ccf2ba315ea4dc2b MM MM
+100644 blob 15885881ea69115351c09b38371f0348a3fb8c67 NA :000000 100644 0000000000000000000000000000000000000000 15885881ea69115351c09b38371f0348a3fb8c67 NA NA
-100644 blob a4e179e4291e5536a5e1c82e091052772d2c5a93 ND :100644 000000 a4e179e4291e5536a5e1c82e091052772d2c5a93 0000000000000000000000000000000000000000 ND ND
*100644->100644 blob c8f25781e8f1792e3e40b74225e20553041b5226->cdb9a8c3da571502ac30225e9c17beccb8387983 NM :100644 100644 c8f25781e8f1792e3e40b74225e20553041b5226 cdb9a8c3da571502ac30225e9c17beccb8387983 NM NM
*100644->100644 blob 40c959f984c8b89a2b02520d17f00d717f024397->2ac547ae9614a00d1b28275de608131f7a0e259f SS :100644 100644 40c959f984c8b89a2b02520d17f00d717f024397 2ac547ae9614a00d1b28275de608131f7a0e259f SS SS
*100644->100644 blob 4ac13458899ab908ef3b1128fa378daefc88d356->c4e4a12231b9fa79a0053cb6077fcb21bb5b135a TT :100644 100644 4ac13458899ab908ef3b1128fa378daefc88d356 c4e4a12231b9fa79a0053cb6077fcb21bb5b135a TT TT
*040000->040000 tree 7d670fdcdb9929f6c7dac196ff78689cd1c566a1->1ba523955d5160681af65cb776411f574c1e8155 Z :040000 040000 7d670fdcdb9929f6c7dac196ff78689cd1c566a1 1ba523955d5160681af65cb776411f574c1e8155 Z Z
EOF EOF
cat >.test-recursive-OB <<\EOF cat >.test-recursive-OB <<\EOF
+100644 blob 6aa2b5335b16431a0ef71e5c0a28be69183cf6a2 AA :000000 100644 0000000000000000000000000000000000000000 6aa2b5335b16431a0ef71e5c0a28be69183cf6a2 AA AA
-100644 blob bcc68ef997017466d5c9094bcf7692295f588c9a DD :100644 000000 bcc68ef997017466d5c9094bcf7692295f588c9a 0000000000000000000000000000000000000000 DD DD
+100644 blob 71420ab81e254145d26d6fc0cddee64c1acd4787 DF :000000 100644 0000000000000000000000000000000000000000 71420ab81e254145d26d6fc0cddee64c1acd4787 DF DF
*100644->100644 blob 141c1f1642328e4bc46a7d801a71da392e66791e->3c4d8de5fbad08572bab8e10eef8dbb264cf0231 DM :100644 100644 141c1f1642328e4bc46a7d801a71da392e66791e 3c4d8de5fbad08572bab8e10eef8dbb264cf0231 DM DM
+100644 blob 1d41122ebdd7a640f29d3c9cc4f9d70094374762 LL :000000 100644 0000000000000000000000000000000000000000 1d41122ebdd7a640f29d3c9cc4f9d70094374762 LL LL
-100644 blob 03f24c8c4700babccfd28b654e7e8eac402ad6cd MD :100644 000000 03f24c8c4700babccfd28b654e7e8eac402ad6cd 0000000000000000000000000000000000000000 MD MD
*100644->100644 blob b258508afb7ceb449981bd9d63d2d3e971bf8d34->19989d4559aae417fedee240ccf2ba315ea4dc2b MM :100644 100644 b258508afb7ceb449981bd9d63d2d3e971bf8d34 19989d4559aae417fedee240ccf2ba315ea4dc2b MM MM
+100644 blob 15885881ea69115351c09b38371f0348a3fb8c67 NA :000000 100644 0000000000000000000000000000000000000000 15885881ea69115351c09b38371f0348a3fb8c67 NA NA
-100644 blob a4e179e4291e5536a5e1c82e091052772d2c5a93 ND :100644 000000 a4e179e4291e5536a5e1c82e091052772d2c5a93 0000000000000000000000000000000000000000 ND ND
*100644->100644 blob c8f25781e8f1792e3e40b74225e20553041b5226->cdb9a8c3da571502ac30225e9c17beccb8387983 NM :100644 100644 c8f25781e8f1792e3e40b74225e20553041b5226 cdb9a8c3da571502ac30225e9c17beccb8387983 NM NM
*100644->100644 blob 40c959f984c8b89a2b02520d17f00d717f024397->2ac547ae9614a00d1b28275de608131f7a0e259f SS :100644 100644 40c959f984c8b89a2b02520d17f00d717f024397 2ac547ae9614a00d1b28275de608131f7a0e259f SS SS
*100644->100644 blob 4ac13458899ab908ef3b1128fa378daefc88d356->c4e4a12231b9fa79a0053cb6077fcb21bb5b135a TT :100644 100644 4ac13458899ab908ef3b1128fa378daefc88d356 c4e4a12231b9fa79a0053cb6077fcb21bb5b135a TT TT
+100644 blob 6c0b99286d0bce551ac4a7b3dff8b706edff3715 Z/AA :000000 100644 0000000000000000000000000000000000000000 6c0b99286d0bce551ac4a7b3dff8b706edff3715 Z/AA Z/AA
-100644 blob 879007efae624d2b1307214b24a956f0a8d686a8 Z/DD :100644 000000 879007efae624d2b1307214b24a956f0a8d686a8 0000000000000000000000000000000000000000 Z/DD Z/DD
*100644->100644 blob 9b541b2275c06e3a7b13f28badf5294e2ae63df4->d77371d15817fcaa57eeec27f770c505ba974ec1 Z/DM :100644 100644 9b541b2275c06e3a7b13f28badf5294e2ae63df4 d77371d15817fcaa57eeec27f770c505ba974ec1 Z/DM Z/DM
-100644 blob d41fda41b7ec4de46b43cb7ea42a45001ae393d5 Z/MD :100644 000000 d41fda41b7ec4de46b43cb7ea42a45001ae393d5 0000000000000000000000000000000000000000 Z/MD Z/MD
*100644->100644 blob 4ca22bae2527d3d9e1676498a0fba3b355bd1278->697aad7715a1e7306ca76290a3dd4208fbaeddfa Z/MM :100644 100644 4ca22bae2527d3d9e1676498a0fba3b355bd1278 697aad7715a1e7306ca76290a3dd4208fbaeddfa Z/MM Z/MM
+100644 blob d12979c22fff69c59ca9409e7a8fe3ee25eaee80 Z/NA :000000 100644 0000000000000000000000000000000000000000 d12979c22fff69c59ca9409e7a8fe3ee25eaee80 Z/NA Z/NA
-100644 blob a18393c636b98e9bd7296b8b437ea4992b72440c Z/ND :100644 000000 a18393c636b98e9bd7296b8b437ea4992b72440c 0000000000000000000000000000000000000000 Z/ND Z/ND
*100644->100644 blob 3fdbe17fd013303a2e981e1ca1c6cd6e72789087->7e09d6a3a14bd630913e8c75693cea32157b606d Z/NM :100644 100644 3fdbe17fd013303a2e981e1ca1c6cd6e72789087 7e09d6a3a14bd630913e8c75693cea32157b606d Z/NM Z/NM
EOF EOF
cat >.test-plain-AB <<\EOF cat >.test-plain-AB <<\EOF
*100644->100644 blob ccba72ad3888a3520b39efcf780b9ee64167535d->6aa2b5335b16431a0ef71e5c0a28be69183cf6a2 AA :100644 100644 ccba72ad3888a3520b39efcf780b9ee64167535d 6aa2b5335b16431a0ef71e5c0a28be69183cf6a2 AA AA
-100644 blob 7e426fb079479fd67f6d81f984e4ec649a44bc25 AN :100644 000000 7e426fb079479fd67f6d81f984e4ec649a44bc25 0000000000000000000000000000000000000000 AN AN
+100644 blob 71420ab81e254145d26d6fc0cddee64c1acd4787 DF :000000 100644 0000000000000000000000000000000000000000 71420ab81e254145d26d6fc0cddee64c1acd4787 DF DF
-040000 tree 6d50f65d3bdab91c63444294d38f08aeff328e42 DF :040000 000000 6d50f65d3bdab91c63444294d38f08aeff328e42 0000000000000000000000000000000000000000 DF DF
+100644 blob 3c4d8de5fbad08572bab8e10eef8dbb264cf0231 DM :000000 100644 0000000000000000000000000000000000000000 3c4d8de5fbad08572bab8e10eef8dbb264cf0231 DM DM
+100644 blob 35abde1506ddf806572ff4d407bd06885d0f8ee9 DN :000000 100644 0000000000000000000000000000000000000000 35abde1506ddf806572ff4d407bd06885d0f8ee9 DN DN
-100644 blob 103d9f89b50b9aad03054b579be5e7aa665f2d57 MD :100644 000000 103d9f89b50b9aad03054b579be5e7aa665f2d57 0000000000000000000000000000000000000000 MD MD
*100644->100644 blob b431b272d829ff3aa4d1a5085f4394ab4d3305b6->19989d4559aae417fedee240ccf2ba315ea4dc2b MM :100644 100644 b431b272d829ff3aa4d1a5085f4394ab4d3305b6 19989d4559aae417fedee240ccf2ba315ea4dc2b MM MM
*100644->100644 blob a716d58de4a570e0038f5c307bd8db34daea021f->bd084b0c27c7b6cc34f11d6d0509a29be3caf970 MN :100644 100644 a716d58de4a570e0038f5c307bd8db34daea021f bd084b0c27c7b6cc34f11d6d0509a29be3caf970 MN MN
+100644 blob 15885881ea69115351c09b38371f0348a3fb8c67 NA :000000 100644 0000000000000000000000000000000000000000 15885881ea69115351c09b38371f0348a3fb8c67 NA NA
-100644 blob a4e179e4291e5536a5e1c82e091052772d2c5a93 ND :100644 000000 a4e179e4291e5536a5e1c82e091052772d2c5a93 0000000000000000000000000000000000000000 ND ND
*100644->100644 blob c8f25781e8f1792e3e40b74225e20553041b5226->cdb9a8c3da571502ac30225e9c17beccb8387983 NM :100644 100644 c8f25781e8f1792e3e40b74225e20553041b5226 cdb9a8c3da571502ac30225e9c17beccb8387983 NM NM
*100644->100644 blob 4c86f9a85fbc5e6804ee2e17a797538fbe785bca->c4e4a12231b9fa79a0053cb6077fcb21bb5b135a TT :100644 100644 4c86f9a85fbc5e6804ee2e17a797538fbe785bca c4e4a12231b9fa79a0053cb6077fcb21bb5b135a TT TT
*040000->040000 tree 5e5f22072bb39f6e12cf663a57cb634c76eefb49->1ba523955d5160681af65cb776411f574c1e8155 Z :040000 040000 5e5f22072bb39f6e12cf663a57cb634c76eefb49 1ba523955d5160681af65cb776411f574c1e8155 Z Z
EOF EOF
cat >.test-recursive-AB <<\EOF cat >.test-recursive-AB <<\EOF
*100644->100644 blob ccba72ad3888a3520b39efcf780b9ee64167535d->6aa2b5335b16431a0ef71e5c0a28be69183cf6a2 AA :100644 100644 ccba72ad3888a3520b39efcf780b9ee64167535d 6aa2b5335b16431a0ef71e5c0a28be69183cf6a2 AA AA
-100644 blob 7e426fb079479fd67f6d81f984e4ec649a44bc25 AN :100644 000000 7e426fb079479fd67f6d81f984e4ec649a44bc25 0000000000000000000000000000000000000000 AN AN
+100644 blob 71420ab81e254145d26d6fc0cddee64c1acd4787 DF :000000 100644 0000000000000000000000000000000000000000 71420ab81e254145d26d6fc0cddee64c1acd4787 DF DF
-100644 blob 68a6d8b91da11045cf4aa3a5ab9f2a781c701249 DF/DF :100644 000000 68a6d8b91da11045cf4aa3a5ab9f2a781c701249 0000000000000000000000000000000000000000 DF/DF DF/DF
+100644 blob 3c4d8de5fbad08572bab8e10eef8dbb264cf0231 DM :000000 100644 0000000000000000000000000000000000000000 3c4d8de5fbad08572bab8e10eef8dbb264cf0231 DM DM
+100644 blob 35abde1506ddf806572ff4d407bd06885d0f8ee9 DN :000000 100644 0000000000000000000000000000000000000000 35abde1506ddf806572ff4d407bd06885d0f8ee9 DN DN
-100644 blob 103d9f89b50b9aad03054b579be5e7aa665f2d57 MD :100644 000000 103d9f89b50b9aad03054b579be5e7aa665f2d57 0000000000000000000000000000000000000000 MD MD
*100644->100644 blob b431b272d829ff3aa4d1a5085f4394ab4d3305b6->19989d4559aae417fedee240ccf2ba315ea4dc2b MM :100644 100644 b431b272d829ff3aa4d1a5085f4394ab4d3305b6 19989d4559aae417fedee240ccf2ba315ea4dc2b MM MM
*100644->100644 blob a716d58de4a570e0038f5c307bd8db34daea021f->bd084b0c27c7b6cc34f11d6d0509a29be3caf970 MN :100644 100644 a716d58de4a570e0038f5c307bd8db34daea021f bd084b0c27c7b6cc34f11d6d0509a29be3caf970 MN MN
+100644 blob 15885881ea69115351c09b38371f0348a3fb8c67 NA :000000 100644 0000000000000000000000000000000000000000 15885881ea69115351c09b38371f0348a3fb8c67 NA NA
-100644 blob a4e179e4291e5536a5e1c82e091052772d2c5a93 ND :100644 000000 a4e179e4291e5536a5e1c82e091052772d2c5a93 0000000000000000000000000000000000000000 ND ND
*100644->100644 blob c8f25781e8f1792e3e40b74225e20553041b5226->cdb9a8c3da571502ac30225e9c17beccb8387983 NM :100644 100644 c8f25781e8f1792e3e40b74225e20553041b5226 cdb9a8c3da571502ac30225e9c17beccb8387983 NM NM
*100644->100644 blob 4c86f9a85fbc5e6804ee2e17a797538fbe785bca->c4e4a12231b9fa79a0053cb6077fcb21bb5b135a TT :100644 100644 4c86f9a85fbc5e6804ee2e17a797538fbe785bca c4e4a12231b9fa79a0053cb6077fcb21bb5b135a TT TT
*100644->100644 blob 8acb8e9750e3f644bf323fcf3d338849db106c77->6c0b99286d0bce551ac4a7b3dff8b706edff3715 Z/AA :100644 100644 8acb8e9750e3f644bf323fcf3d338849db106c77 6c0b99286d0bce551ac4a7b3dff8b706edff3715 Z/AA Z/AA
-100644 blob 087494262084cefee7ed484d20c8dc0580791272 Z/AN :100644 000000 087494262084cefee7ed484d20c8dc0580791272 0000000000000000000000000000000000000000 Z/AN Z/AN
+100644 blob d77371d15817fcaa57eeec27f770c505ba974ec1 Z/DM :000000 100644 0000000000000000000000000000000000000000 d77371d15817fcaa57eeec27f770c505ba974ec1 Z/DM Z/DM
+100644 blob beb5d38c55283d280685ea21a0e50cfcc0ca064a Z/DN :000000 100644 0000000000000000000000000000000000000000 beb5d38c55283d280685ea21a0e50cfcc0ca064a Z/DN Z/DN
-100644 blob a79ac3be9377639e1c7d1edf1ae1b3a5f0ccd8a9 Z/MD :100644 000000 a79ac3be9377639e1c7d1edf1ae1b3a5f0ccd8a9 0000000000000000000000000000000000000000 Z/MD Z/MD
*100644->100644 blob 61422ba9c2c873416061a88cd40a59a35b576474->697aad7715a1e7306ca76290a3dd4208fbaeddfa Z/MM :100644 100644 61422ba9c2c873416061a88cd40a59a35b576474 697aad7715a1e7306ca76290a3dd4208fbaeddfa Z/MM Z/MM
*100644->100644 blob a5c544c21cfcb07eb80a4d89a5b7d1570002edfd->b16d7b25b869f2beb124efa53467d8a1550ad694 Z/MN :100644 100644 a5c544c21cfcb07eb80a4d89a5b7d1570002edfd b16d7b25b869f2beb124efa53467d8a1550ad694 Z/MN Z/MN
+100644 blob d12979c22fff69c59ca9409e7a8fe3ee25eaee80 Z/NA :000000 100644 0000000000000000000000000000000000000000 d12979c22fff69c59ca9409e7a8fe3ee25eaee80 Z/NA Z/NA
-100644 blob a18393c636b98e9bd7296b8b437ea4992b72440c Z/ND :100644 000000 a18393c636b98e9bd7296b8b437ea4992b72440c 0000000000000000000000000000000000000000 Z/ND Z/ND
*100644->100644 blob 3fdbe17fd013303a2e981e1ca1c6cd6e72789087->7e09d6a3a14bd630913e8c75693cea32157b606d Z/NM :100644 100644 3fdbe17fd013303a2e981e1ca1c6cd6e72789087 7e09d6a3a14bd630913e8c75693cea32157b606d Z/NM Z/NM
EOF EOF
x40='[0-9a-f][0-9a-f][0-9a-f][0-9a-f][0-9a-f]' x40='[0-9a-f][0-9a-f][0-9a-f][0-9a-f][0-9a-f]'
x40="$x40$x40$x40$x40$x40$x40$x40$x40" x40="$x40$x40$x40$x40$x40$x40$x40$x40"
z40='0000000000000000000000000000000000000000' z40='0000000000000000000000000000000000000000'
@ -135,7 +134,7 @@ cmp_diff_files_output () {
# object ID for the changed files because it wants you to look at the # object ID for the changed files because it wants you to look at the
# filesystem. # filesystem.
sed <"$2" >.test-tmp \ sed <"$2" >.test-tmp \
-e '/^+/d;/\^*/s/\( '$x40'->\)'$x40' /\1'$z40' /' && -e '/^:000000 /d;s/'$x40' /'$z40' /' &&
diff "$1" .test-tmp diff "$1" .test-tmp
} }

View File

@ -26,7 +26,7 @@ test_expect_success \
# both are slightly edited. So we say you copy-and-edit one, # both are slightly edited. So we say you copy-and-edit one,
# and rename-and-edit the other. # and rename-and-edit the other.
GIT_DIFF_OPTS=--unified=0 git-diff-cache -M $tree | GIT_DIFF_OPTS=--unified=0 git-diff-cache -M -p $tree |
sed -e 's/\([0-9][0-9]*\)/#/g' >current && sed -e 's/\([0-9][0-9]*\)/#/g' >current &&
cat >expected <<\EOF cat >expected <<\EOF
diff --git a/COPYING b/COPYING.# diff --git a/COPYING b/COPYING.#
@ -68,7 +68,7 @@ test_expect_success \
# both are slightly edited. So we say you edited one, # both are slightly edited. So we say you edited one,
# and copy-and-edit the other. # and copy-and-edit the other.
GIT_DIFF_OPTS=--unified=0 git-diff-cache -C $tree | GIT_DIFF_OPTS=--unified=0 git-diff-cache -C -p $tree |
sed -e 's/\([0-9][0-9]*\)/#/g' >current sed -e 's/\([0-9][0-9]*\)/#/g' >current
cat >expected <<\EOF cat >expected <<\EOF
diff --git a/COPYING b/COPYING.# diff --git a/COPYING b/COPYING.#
@ -108,7 +108,7 @@ test_expect_success \
# this is only possible because -C mode now reports the unmodified # this is only possible because -C mode now reports the unmodified
# file to the diff-core. # file to the diff-core.
GIT_DIFF_OPTS=--unified=0 git-diff-cache -C $tree | GIT_DIFF_OPTS=--unified=0 git-diff-cache -C -p $tree |
sed -e 's/\([0-9][0-9]*\)/#/g' >current sed -e 's/\([0-9][0-9]*\)/#/g' >current
cat >expected <<\EOF cat >expected <<\EOF
diff --git a/COPYING b/COPYING.# diff --git a/COPYING b/COPYING.#