2007-07-15 01:14:45 +02:00
|
|
|
#include "builtin.h"
|
2006-12-06 16:26:06 +01:00
|
|
|
#include "cache.h"
|
|
|
|
#include "xdiff/xdiff.h"
|
2006-12-20 17:37:07 +01:00
|
|
|
#include "xdiff-interface.h"
|
2006-12-06 16:26:06 +01:00
|
|
|
|
|
|
|
static const char merge_file_usage[] =
|
2008-08-28 10:10:04 +02:00
|
|
|
"git merge-file [-p | --stdout] [--diff3] [-q | --quiet] [-L name1 [-L orig [-L name2]]] file1 orig_file file2";
|
2006-12-06 16:26:06 +01:00
|
|
|
|
2007-07-15 01:14:45 +02:00
|
|
|
int cmd_merge_file(int argc, const char **argv, const char *prefix)
|
2006-12-06 16:26:06 +01:00
|
|
|
{
|
2007-07-15 01:14:45 +02:00
|
|
|
const char *names[3];
|
2006-12-06 16:26:06 +01:00
|
|
|
mmfile_t mmfs[3];
|
|
|
|
mmbuffer_t result = {NULL, 0};
|
|
|
|
xpparam_t xpp = {XDF_NEED_MINIMAL};
|
2006-12-06 16:45:42 +01:00
|
|
|
int ret = 0, i = 0, to_stdout = 0;
|
2008-08-28 10:10:04 +02:00
|
|
|
int merge_level = XDL_MERGE_ZEALOUS_ALNUM;
|
|
|
|
int merge_style = 0;
|
2008-08-29 19:49:56 +02:00
|
|
|
int nongit;
|
|
|
|
|
|
|
|
prefix = setup_git_directory_gently(&nongit);
|
|
|
|
if (!nongit) {
|
|
|
|
/* Read the configuration file */
|
|
|
|
git_config(git_xmerge_config, NULL);
|
|
|
|
if (0 <= git_xmerge_style)
|
|
|
|
merge_style = git_xmerge_style;
|
|
|
|
}
|
2006-12-06 16:26:06 +01:00
|
|
|
|
|
|
|
while (argc > 4) {
|
2006-12-06 16:45:42 +01:00
|
|
|
if (!strcmp(argv[1], "-L") && i < 3) {
|
2006-12-06 16:26:06 +01:00
|
|
|
names[i++] = argv[2];
|
2006-12-06 16:45:42 +01:00
|
|
|
argc--;
|
|
|
|
argv++;
|
|
|
|
} else if (!strcmp(argv[1], "-p") ||
|
|
|
|
!strcmp(argv[1], "--stdout"))
|
|
|
|
to_stdout = 1;
|
|
|
|
else if (!strcmp(argv[1], "-q") ||
|
|
|
|
!strcmp(argv[1], "--quiet"))
|
|
|
|
freopen("/dev/null", "w", stderr);
|
xmerge.c: "diff3 -m" style clips merge reduction level to EAGER or less
When showing a conflicting merge result, and "--diff3 -m" style is asked
for, this patch makes sure that the merge reduction level does not exceed
XDL_MERGE_EAGER. This is because "diff3 -m" style output would not make
sense for anything more aggressive than XDL_MERGE_EAGER, because of the
way how the merge reduction works.
"git merge-file" no longer has to force MERGE_EAGER when "--diff3" is
asked for because of this change.
Suppose a common ancestor (shared preimage) is modified to postimage #1
and #2 (each letter represents one line):
#####
postimage#1: 1234ABCDE789
| /
| /
preimage: 123456789
| \
postimage#2: 1234AXYE789
####
XDL_MERGE_MINIMAL and XDL_MERGE_EAGER would:
(1) find the s/56/ABCDE/ done on one side and s/56/AXYE/ done on the
other side,
(2) notice that they touch an overlapping area, and
(3) mark it as a conflict, "ABCDE vs AXYE".
The difference between the two algorithms is that EAGER drops the hunk
altogether if the postimages match (i.e. both sides modified the same
way), while MINIMAL keeps it. There is no other operation performed to
the hunk. As the result, lines marked with "#" in the above picure will
be in the RCS merge style output like this (letters <, = and > represent
conflict marker lines):
output: 1234<ABCDE=AXYE>789 ; with MINIMAL/EAGER
The part from the preimage that corresponds to these conflicting changes
is "56", which is what "diff3 -m" style output adds to it:
output: 1234<ABCDE|56=AXYE>789 ; in "diff3 -m" style
Now, XDL_MERGE_ZEALOUS looks at the differences between the changes two
postimages made in order to reduce the number of lines in the conflicting
regions. It notices that both sides start their new contents with "A",
and excludes it from the output (it also excludes "E" for the same
reason). The conflict that used to be "ABCDE vs AXYE" is now "BCD vs XY":
output: 1234A<BCD=XY>E789 ; with ZEALOUS
There could even be matching parts between two postimages in the middle.
Instead of one side rewriting the shared "56" to "ABCDE" and the other
side to "AXYE", imagine the case where the postimages are "ABCDE" and
"AXCYE", in which case instead of having one conflicted hunk "BCD vs XY",
you would have two conflicting hunks "B vs X" and "D vs Y".
In either case, once you reduce "ABCDE vs AXYE" to "BCD vs XY" (or "ABCDE
vs AXCYE" to "B vs X" and "D vs Y"), there is no part from the preimage
that corresponds to the conflicting change made in both postimages
anymore. In other words, conflict reduced by ZEALOUS algorithm cannot be
expressed in "diff3 -m" style. Representing the last illustration like
this is misleading to say the least:
output: 1234A<BCD|56=XY>E789 ; broken "diff3 -m" style
because the preimage was not ...4A56E... to begin with. "A" and "E" are
common only between the postimages.
Even worse, once a single conflicting hunk is split into multiple ones
(recall the example of breaking "ABCDE vs AXCYE" to "B vs X" and "D vs
Y"), there is no sane way to distribute the preimage text across split
conflicting hunks.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2008-08-29 17:22:55 +02:00
|
|
|
else if (!strcmp(argv[1], "--diff3"))
|
2008-08-28 10:10:04 +02:00
|
|
|
merge_style = XDL_MERGE_DIFF3;
|
2006-12-06 16:45:42 +01:00
|
|
|
else
|
|
|
|
usage(merge_file_usage);
|
|
|
|
argc--;
|
|
|
|
argv++;
|
2006-12-06 16:26:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (argc != 4)
|
|
|
|
usage(merge_file_usage);
|
|
|
|
|
|
|
|
for (; i < 3; i++)
|
|
|
|
names[i] = argv[i + 1];
|
|
|
|
|
2007-06-05 04:37:13 +02:00
|
|
|
for (i = 0; i < 3; i++) {
|
2006-12-20 17:37:07 +01:00
|
|
|
if (read_mmfile(mmfs + i, argv[i + 1]))
|
2006-12-06 16:26:06 +01:00
|
|
|
return -1;
|
2007-06-05 04:37:13 +02:00
|
|
|
if (buffer_is_binary(mmfs[i].ptr, mmfs[i].size))
|
|
|
|
return error("Cannot merge binary files: %s\n",
|
|
|
|
argv[i + 1]);
|
|
|
|
}
|
2006-12-06 16:26:06 +01:00
|
|
|
|
|
|
|
ret = xdl_merge(mmfs + 1, mmfs + 0, names[0], mmfs + 2, names[2],
|
2008-08-28 10:10:04 +02:00
|
|
|
&xpp, merge_level | merge_style, &result);
|
2006-12-06 16:26:06 +01:00
|
|
|
|
|
|
|
for (i = 0; i < 3; i++)
|
|
|
|
free(mmfs[i].ptr);
|
|
|
|
|
|
|
|
if (ret >= 0) {
|
2007-07-15 01:14:45 +02:00
|
|
|
const char *filename = argv[1];
|
2006-12-06 16:45:42 +01:00
|
|
|
FILE *f = to_stdout ? stdout : fopen(filename, "wb");
|
2006-12-06 16:26:06 +01:00
|
|
|
|
|
|
|
if (!f)
|
|
|
|
ret = error("Could not open %s for writing", filename);
|
2008-03-13 16:19:35 +01:00
|
|
|
else if (result.size &&
|
|
|
|
fwrite(result.ptr, result.size, 1, f) != 1)
|
2006-12-06 16:26:06 +01:00
|
|
|
ret = error("Could not write to %s", filename);
|
|
|
|
else if (fclose(f))
|
|
|
|
ret = error("Could not close %s", filename);
|
|
|
|
free(result.ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|