diff options
| author | Junio C Hamano <gitster@pobox.com> | 2014-06-03 12:06:40 -0700 | 
|---|---|---|
| committer | Junio C Hamano <gitster@pobox.com> | 2014-06-03 12:06:40 -0700 | 
| commit | 8eaf517835d0534767d6a54d12d072ce30276ad9 (patch) | |
| tree | 19368a64c5b367f601fb6ddd77d4c6e903a2df10 /combine-diff.c | |
| parent | f008cef4abb2a4db766b4a152b304aca91a0101a (diff) | |
| parent | 22f4c27e68f448d5fce316a73ea3f7bab6aa1268 (diff) | |
| download | git-8eaf517835d0534767d6a54d12d072ce30276ad9.tar.gz | |
Merge branch 'ks/tree-diff-nway'
Instead of running N pair-wise diff-trees when inspecting a
N-parent merge, find the set of paths that were touched by walking
N+1 trees in parallel.  These set of paths can then be turned into
N pair-wise diff-tree results to be processed through rename
detections and such.  And N=2 case nicely degenerates to the usual
2-way diff-tree, which is very nice.
* ks/tree-diff-nway:
  mingw: activate alloca
  combine-diff: speed it up, by using multiparent diff tree-walker directly
  tree-diff: rework diff_tree() to generate diffs for multiparent cases as well
  Portable alloca for Git
  tree-diff: reuse base str(buf) memory on sub-tree recursion
  tree-diff: no need to call "full" diff_tree_sha1 from show_path()
  tree-diff: rework diff_tree interface to be sha1 based
  tree-diff: diff_tree() should now be static
  tree-diff: remove special-case diff-emitting code for empty-tree cases
  tree-diff: simplify tree_entry_pathcmp
  tree-diff: show_path prototype is not needed anymore
  tree-diff: rename compare_tree_entry -> tree_entry_pathcmp
  tree-diff: move all action-taking code out of compare_tree_entry()
  tree-diff: don't assume compare_tree_entry() returns -1,0,1
  tree-diff: consolidate code for emitting diffs and recursion in one place
  tree-diff: show_tree() is not needed
  tree-diff: no need to pass match to skip_uninteresting()
  tree-diff: no need to manually verify that there is no mode change for a path
  combine-diff: move changed-paths scanning logic into its own function
  combine-diff: move show_log_first logic/action out of paths scanning
Diffstat (limited to 'combine-diff.c')
| -rw-r--r-- | combine-diff.c | 168 | 
1 files changed, 138 insertions, 30 deletions
| diff --git a/combine-diff.c b/combine-diff.c index 24ca7e2334..12764fb733 100644 --- a/combine-diff.c +++ b/combine-diff.c @@ -1301,6 +1301,81 @@ static const char *path_path(void *obj)  	return path->path;  } + +/* find set of paths that every parent touches */ +static struct combine_diff_path *find_paths_generic(const unsigned char *sha1, +	const struct sha1_array *parents, struct diff_options *opt) +{ +	struct combine_diff_path *paths = NULL; +	int i, num_parent = parents->nr; + +	int output_format = opt->output_format; +	const char *orderfile = opt->orderfile; + +	opt->output_format = DIFF_FORMAT_NO_OUTPUT; +	/* tell diff_tree to emit paths in sorted (=tree) order */ +	opt->orderfile = NULL; + +	/* D(A,P1...Pn) = D(A,P1) ^ ... ^ D(A,Pn)  (wrt paths) */ +	for (i = 0; i < num_parent; i++) { +		/* +		 * show stat against the first parent even when doing +		 * combined diff. +		 */ +		int stat_opt = (output_format & +				(DIFF_FORMAT_NUMSTAT|DIFF_FORMAT_DIFFSTAT)); +		if (i == 0 && stat_opt) +			opt->output_format = stat_opt; +		else +			opt->output_format = DIFF_FORMAT_NO_OUTPUT; +		diff_tree_sha1(parents->sha1[i], sha1, "", opt); +		diffcore_std(opt); +		paths = intersect_paths(paths, i, num_parent); + +		/* if showing diff, show it in requested order */ +		if (opt->output_format != DIFF_FORMAT_NO_OUTPUT && +		    orderfile) { +			diffcore_order(orderfile); +		} + +		diff_flush(opt); +	} + +	opt->output_format = output_format; +	opt->orderfile = orderfile; +	return paths; +} + + +/* + * find set of paths that everybody touches, assuming diff is run without + * rename/copy detection, etc, comparing all trees simultaneously (= faster). + */ +static struct combine_diff_path *find_paths_multitree( +	const unsigned char *sha1, const struct sha1_array *parents, +	struct diff_options *opt) +{ +	int i, nparent = parents->nr; +	const unsigned char **parents_sha1; +	struct combine_diff_path paths_head; +	struct strbuf base; + +	parents_sha1 = xmalloc(nparent * sizeof(parents_sha1[0])); +	for (i = 0; i < nparent; i++) +		parents_sha1[i] = parents->sha1[i]; + +	/* fake list head, so worker can assume it is non-NULL */ +	paths_head.next = NULL; + +	strbuf_init(&base, PATH_MAX); +	diff_tree_paths(&paths_head, sha1, parents_sha1, nparent, &base, opt); + +	strbuf_release(&base); +	free(parents_sha1); +	return paths_head.next; +} + +  void diff_tree_combined(const unsigned char *sha1,  			const struct sha1_array *parents,  			int dense, @@ -1308,49 +1383,82 @@ void diff_tree_combined(const unsigned char *sha1,  {  	struct diff_options *opt = &rev->diffopt;  	struct diff_options diffopts; -	struct combine_diff_path *p, *paths = NULL; +	struct combine_diff_path *p, *paths;  	int i, num_paths, needsep, show_log_first, num_parent = parents->nr; +	int need_generic_pathscan; + +	/* nothing to do, if no parents */ +	if (!num_parent) +		return; + +	show_log_first = !!rev->loginfo && !rev->no_commit_id; +	needsep = 0; +	if (show_log_first) { +		show_log(rev); + +		if (rev->verbose_header && opt->output_format) +			printf("%s%c", diff_line_prefix(opt), +			       opt->line_termination); +	}  	diffopts = *opt;  	copy_pathspec(&diffopts.pathspec, &opt->pathspec); -	diffopts.output_format = DIFF_FORMAT_NO_OUTPUT;  	DIFF_OPT_SET(&diffopts, RECURSIVE);  	DIFF_OPT_CLR(&diffopts, ALLOW_EXTERNAL); -	/* tell diff_tree to emit paths in sorted (=tree) order */ -	diffopts.orderfile = NULL; -	show_log_first = !!rev->loginfo && !rev->no_commit_id; -	needsep = 0; -	/* find set of paths that everybody touches */ -	for (i = 0; i < num_parent; i++) { -		/* show stat against the first parent even +	/* find set of paths that everybody touches +	 * +	 * NOTE +	 * +	 * Diffcore transformations are bound to diff_filespec and logic +	 * comparing two entries - i.e. they do not apply directly to combine +	 * diff. +	 * +	 * If some of such transformations is requested - we launch generic +	 * path scanning, which works significantly slower compared to +	 * simultaneous all-trees-in-one-go scan in find_paths_multitree(). +	 * +	 * TODO some of the filters could be ported to work on +	 * combine_diff_paths - i.e. all functionality that skips paths, so in +	 * theory, we could end up having only multitree path scanning. +	 * +	 * NOTE please keep this semantically in sync with diffcore_std() +	 */ +	need_generic_pathscan = opt->skip_stat_unmatch	|| +			DIFF_OPT_TST(opt, FOLLOW_RENAMES)	|| +			opt->break_opt != -1	|| +			opt->detect_rename	|| +			opt->pickaxe		|| +			opt->filter; + + +	if (need_generic_pathscan) { +		/* +		 * NOTE generic case also handles --stat, as it computes +		 * diff(sha1,parent_i) for all i to do the job, specifically +		 * for parent0. +		 */ +		paths = find_paths_generic(sha1, parents, &diffopts); +	} +	else { +		int stat_opt; +		paths = find_paths_multitree(sha1, parents, &diffopts); + +		/* +		 * show stat against the first parent even  		 * when doing combined diff.  		 */ -		int stat_opt = (opt->output_format & +		stat_opt = (opt->output_format &  				(DIFF_FORMAT_NUMSTAT|DIFF_FORMAT_DIFFSTAT)); -		if (i == 0 && stat_opt) +		if (stat_opt) {  			diffopts.output_format = stat_opt; -		else -			diffopts.output_format = DIFF_FORMAT_NO_OUTPUT; -		diff_tree_sha1(parents->sha1[i], sha1, "", &diffopts); -		diffcore_std(&diffopts); -		paths = intersect_paths(paths, i, num_parent); -		if (show_log_first && i == 0) { -			show_log(rev); - -			if (rev->verbose_header && opt->output_format) -				printf("%s%c", diff_line_prefix(opt), -				       opt->line_termination); +			diff_tree_sha1(parents->sha1[0], sha1, "", &diffopts); +			diffcore_std(&diffopts); +			if (opt->orderfile) +				diffcore_order(opt->orderfile); +			diff_flush(&diffopts);  		} - -		/* if showing diff, show it in requested order */ -		if (diffopts.output_format != DIFF_FORMAT_NO_OUTPUT && -		    opt->orderfile) { -			diffcore_order(opt->orderfile); -		} - -		diff_flush(&diffopts);  	}  	/* find out number of surviving paths */ | 
