diff.c 7.7 KB
Newer Older
1 2 3 4 5
/*
 * Copyright (C) 2005 Junio C Hamano
 */
#include <sys/types.h>
#include <sys/wait.h>
6
#include <signal.h>
7 8 9
#include "cache.h"
#include "diff.h"

10
static char *diff_opts = "-pu";
11

12
static const char *external_diff(void)
13
{
14 15 16 17 18 19
	static char *external_diff_cmd = NULL;
	static int done_preparing = 0;

	if (done_preparing)
		return external_diff_cmd;

20 21 22 23 24 25 26 27
	/*
	 * Default values above are meant to match the
	 * Linux kernel development style.  Examples of
	 * alternative styles you can specify via environment
	 * variables are:
	 *
	 * GIT_DIFF_OPTS="-c";
	 */
28 29 30 31
	if (getenv("GIT_EXTERNAL_DIFF"))
		external_diff_cmd = getenv("GIT_EXTERNAL_DIFF");

	/* In case external diff fails... */
32
	diff_opts = getenv("GIT_DIFF_OPTS") ? : diff_opts;
33 34 35

	done_preparing = 1;
	return external_diff_cmd;
36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59
}

/* Help to copy the thing properly quoted for the shell safety.
 * any single quote is replaced with '\'', and the caller is
 * expected to enclose the result within a single quote pair.
 *
 * E.g.
 *  original     sq_expand     result
 *  name     ==> name      ==> 'name'
 *  a b      ==> a b       ==> 'a b'
 *  a'b      ==> a'\''b    ==> 'a'\''b'
 */
static char *sq_expand(const char *src)
{
	static char *buf = NULL;
	int cnt, c;
	const char *cp;
	char *bp;

	/* count bytes needed to store the quoted string. */ 
	for (cnt = 1, cp = src; *cp; cnt++, cp++)
		if (*cp == '\'')
			cnt += 3;

60
	buf = xmalloc(cnt);
61 62 63 64 65 66 67 68 69 70 71 72 73
	bp = buf;
	while ((c = *src++)) {
		if (c != '\'')
			*bp++ = c;
		else {
			bp = strcpy(bp, "'\\''");
			bp += 4;
		}
	}
	*bp = 0;
	return buf;
}

74 75 76 77 78 79 80 81 82
static struct diff_tempfile {
	const char *name;
	char hex[41];
	char mode[10];
	char tmp_path[50];
} diff_temp[2];

static void builtin_diff(const char *name,
			 struct diff_tempfile *temp)
83
{
84
	int i, next_at;
85
	const char *diff_cmd = "diff -L'%s%s' -L'%s%s'";
86 87 88 89
	const char *diff_arg  = "'%s' '%s'";
	const char *input_name_sq[2];
	const char *path0[2];
	const char *path1[2];
90
	const char *name_sq = sq_expand(name);
91 92
	char *cmd;
	
93 94
	/* diff_cmd and diff_arg have 6 %s in total which makes
	 * the sum of these strings 12 bytes larger than required.
95
	 * we use 2 spaces around diff-opts, and we need to count
96
	 * terminating NUL, so we subtract 9 here.
97
	 */
98
	int cmd_size = (strlen(diff_cmd) + strlen(diff_opts) +
99
			strlen(diff_arg) - 9);
100 101 102 103 104 105
	for (i = 0; i < 2; i++) {
		input_name_sq[i] = sq_expand(temp[i].name);
		if (!strcmp(temp[i].name, "/dev/null")) {
			path0[i] = "/dev/null";
			path1[i] = "";
		} else {
106
			path0[i] = i ? "b/" : "a/";
107 108 109
			path1[i] = name_sq;
		}
		cmd_size += (strlen(path0[i]) + strlen(path1[i]) +
110
			     strlen(input_name_sq[i]));
111
	}
112

113 114 115
	cmd = xmalloc(cmd_size);

	next_at = 0;
116
	next_at += snprintf(cmd+next_at, cmd_size-next_at,
117
			    diff_cmd,
118
			    path0[0], path1[0], path0[1], path1[1]);
119 120 121
	next_at += snprintf(cmd+next_at, cmd_size-next_at,
			    " %s ", diff_opts);
	next_at += snprintf(cmd+next_at, cmd_size-next_at,
122 123
			    diff_arg, input_name_sq[0], input_name_sq[1]);

124 125 126 127 128 129 130 131
	if (!path1[0][0])
		printf("Created: %s (mode:%s)\n", name, temp[1].mode);
	else if (!path1[1][0])
		printf("Deleted: %s\n", name);
	else if (strcmp(temp[0].mode, temp[1].mode))
		printf("Mode changed: %s (%s->%s)\n", name,
		       temp[0].mode, temp[1].mode);
	fflush(NULL);
132
	execlp("/bin/sh","sh", "-c", cmd, NULL);
133 134
}

135 136 137
static void prepare_temp_file(const char *name,
			      struct diff_tempfile *temp,
			      struct diff_spec *one)
138
{
139 140 141 142
	static unsigned char null_sha1[20] = { 0, };

	if (!one->file_valid) {
	not_a_valid_file:
143 144 145
		/* A '-' entry produces this for file-2, and
		 * a '+' entry produces this for file-1.
		 */
146 147 148
		temp->name = "/dev/null";
		strcpy(temp->hex, ".");
		strcpy(temp->mode, ".");
149 150
		return;
	}
151 152 153 154 155 156 157 158 159 160 161 162 163 164 165

	if (one->sha1_valid &&
	    !memcmp(one->u.sha1, null_sha1, sizeof(null_sha1))) {
		one->sha1_valid = 0;
		one->u.name = name;
	}

	if (!one->sha1_valid) {
		struct stat st;
		temp->name = one->u.name;
		if (stat(temp->name, &st) < 0) {
			if (errno == ENOENT)
				goto not_a_valid_file;
			die("stat(%s): %s", temp->name, strerror(errno));
		}
166
		strcpy(temp->hex, sha1_to_hex(null_sha1));
167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206
		sprintf(temp->mode, "%06o",
			S_IFREG |ce_permissions(st.st_mode));
	}
	else {
		int fd;
		void *blob;
		char type[20];
		unsigned long size;

		blob = read_sha1_file(one->u.sha1, type, &size);
		if (!blob || strcmp(type, "blob"))
			die("unable to read blob object for %s (%s)",
			    name, sha1_to_hex(one->u.sha1));

		strcpy(temp->tmp_path, ".diff_XXXXXX");
		fd = mkstemp(temp->tmp_path);
		if (fd < 0)
			die("unable to create temp-file");
		if (write(fd, blob, size) != size)
			die("unable to write temp-file");
		close(fd);
		free(blob);
		temp->name = temp->tmp_path;
		strcpy(temp->hex, sha1_to_hex(one->u.sha1));
		temp->hex[40] = 0;
		sprintf(temp->mode, "%06o", one->mode);
	}
}

static void remove_tempfile(void)
{
	int i;

	for (i = 0; i < 2; i++)
		if (diff_temp[i].name == diff_temp[i].tmp_path) {
			unlink(diff_temp[i].name);
			diff_temp[i].name = NULL;
		}
}

207 208 209 210 211
static void remove_tempfile_on_signal(int signo)
{
	remove_tempfile();
}

212 213 214 215 216 217 218 219 220 221 222
/* An external diff command takes:
 *
 * diff-cmd name infile1 infile1-sha1 infile1-mode \
 *               infile2 infile2-sha1 infile2-mode.
 *
 */
void run_external_diff(const char *name,
		       struct diff_spec *one,
		       struct diff_spec *two)
{
	struct diff_tempfile *temp = diff_temp;
223 224
	pid_t pid;
	int status;
225 226
	static int atexit_asked = 0;

227 228 229 230 231 232 233 234 235
	if (one && two) {
		prepare_temp_file(name, &temp[0], one);
		prepare_temp_file(name, &temp[1], two);
		if (! atexit_asked &&
		    (temp[0].name == temp[0].tmp_path ||
		     temp[1].name == temp[1].tmp_path)) {
			atexit_asked = 1;
			atexit(remove_tempfile);
		}
236
		signal(SIGINT, remove_tempfile_on_signal);
237 238 239 240 241 242 243 244
	}

	fflush(NULL);
	pid = fork();
	if (pid < 0)
		die("unable to fork");
	if (!pid) {
		const char *pgm = external_diff();
245 246 247 248 249 250 251 252 253 254
		if (pgm) {
			if (one && two)
				execlp(pgm, pgm,
				       name,
				       temp[0].name, temp[0].hex, temp[0].mode,
				       temp[1].name, temp[1].hex, temp[1].mode,
				       NULL);
			else
				execlp(pgm, pgm, name, NULL);
		}
255 256 257
		/*
		 * otherwise we use the built-in one.
		 */
258 259 260 261
		if (one && two)
			builtin_diff(name, temp);
		else
			printf("* Unmerged path %s\n", name);
262 263
		exit(0);
	}
264 265 266 267 268 269 270 271 272 273 274
	if (waitpid(pid, &status, 0) < 0 || !WIFEXITED(status)) {
		/* We do not check the exit status because typically
		 * diff exits non-zero if files are different, and
		 * we are not interested in knowing that.  We *knew*
		 * they are different and that's why we ran diff
		 * in the first place!  However if it dies by a signal,
		 * we stop processing immediately.
		 */
		remove_tempfile();
		die("external diff died unexpectedly.\n");
	}
275 276 277
	remove_tempfile();
}

278 279 280
void diff_addremove(int addremove, unsigned mode,
		    const unsigned char *sha1,
		    const char *base, const char *path)
281
{
282
	char concatpath[PATH_MAX];
283 284
	struct diff_spec spec[2], *one, *two;

285 286
	memcpy(spec[0].u.sha1, sha1, 20);
	spec[0].mode = mode;
287 288 289
	spec[0].sha1_valid = spec[0].file_valid = 1;
	spec[1].file_valid = 0;

290
	if (addremove == '+') {
291 292 293 294
		one = spec + 1; two = spec;
	} else {
		one = spec; two = one + 1;
	}
295 296 297 298 299 300
	
	if (path) {
		strcpy(concatpath, base);
		strcat(concatpath, path);
	}
	run_external_diff(path ? concatpath : base, one, two);
301 302
}

303 304 305 306 307 308 309 310 311 312 313
void diff_change(unsigned old_mode, unsigned new_mode,
		 const unsigned char *old_sha1,
		 const unsigned char *new_sha1,
		 const char *base, const char *path) {
	char concatpath[PATH_MAX];
	struct diff_spec spec[2];

	memcpy(spec[0].u.sha1, old_sha1, 20);
	spec[0].mode = old_mode;
	memcpy(spec[1].u.sha1, new_sha1, 20);
	spec[1].mode = new_mode;
314
	spec[0].sha1_valid = spec[0].file_valid = 1;
315
	spec[1].sha1_valid = spec[1].file_valid = 1;
316

317 318 319
	if (path) {
		strcpy(concatpath, base);
		strcat(concatpath, path);
320
	}
321 322
	run_external_diff(path ? concatpath : base, &spec[0], &spec[1]);
}
323

324 325 326
void diff_unmerge(const char *path)
{
	run_external_diff(path, NULL, NULL);
327
}