2 * Helper functions for tree diff generation
9 static char *malloc_base(const char *base
, int baselen
, const char *path
, int pathlen
)
11 char *newbase
= xmalloc(baselen
+ pathlen
+ 2);
12 memcpy(newbase
, base
, baselen
);
13 memcpy(newbase
+ baselen
, path
, pathlen
);
14 memcpy(newbase
+ baselen
+ pathlen
, "/", 2);
18 static void show_entry(struct diff_options
*opt
, const char *prefix
, struct tree_desc
*desc
,
19 const char *base
, int baselen
);
21 static int compare_tree_entry(struct tree_desc
*t1
, struct tree_desc
*t2
, const char *base
, int baselen
, struct diff_options
*opt
)
23 unsigned mode1
, mode2
;
24 const char *path1
, *path2
;
25 const unsigned char *sha1
, *sha2
;
26 int cmp
, pathlen1
, pathlen2
;
28 sha1
= tree_entry_extract(t1
, &path1
, &mode1
);
29 sha2
= tree_entry_extract(t2
, &path2
, &mode2
);
31 pathlen1
= tree_entry_len(path1
, sha1
);
32 pathlen2
= tree_entry_len(path2
, sha2
);
33 cmp
= base_name_compare(path1
, pathlen1
, mode1
, path2
, pathlen2
, mode2
);
35 show_entry(opt
, "-", t1
, base
, baselen
);
39 show_entry(opt
, "+", t2
, base
, baselen
);
42 if (!opt
->find_copies_harder
&& !hashcmp(sha1
, sha2
) && mode1
== mode2
)
46 * If the filemode has changed to/from a directory from/to a regular
47 * file, we need to consider it a remove and an add.
49 if (S_ISDIR(mode1
) != S_ISDIR(mode2
)) {
50 show_entry(opt
, "-", t1
, base
, baselen
);
51 show_entry(opt
, "+", t2
, base
, baselen
);
55 if (opt
->recursive
&& S_ISDIR(mode1
)) {
57 char *newbase
= malloc_base(base
, baselen
, path1
, pathlen1
);
58 if (opt
->tree_in_recursive
)
59 opt
->change(opt
, mode1
, mode2
,
60 sha1
, sha2
, base
, path1
);
61 retval
= diff_tree_sha1(sha1
, sha2
, newbase
, opt
);
66 opt
->change(opt
, mode1
, mode2
, sha1
, sha2
, base
, path1
);
71 * Is a tree entry interesting given the pathspec we have?
74 * - 2 for "yes, and all subsequent entries will be"
77 * - negative for "no, and no subsequent entries will be either"
79 static int tree_entry_interesting(struct tree_desc
*desc
, const char *base
, int baselen
, struct diff_options
*opt
)
82 const unsigned char *sha1
;
86 int never_interesting
= -1;
91 sha1
= tree_entry_extract(desc
, &path
, &mode
);
93 pathlen
= tree_entry_len(path
, sha1
);
95 for (i
= 0; i
< opt
->nr_paths
; i
++) {
96 const char *match
= opt
->paths
[i
];
97 int matchlen
= opt
->pathlens
[i
];
98 int m
= -1; /* signals that we haven't called strncmp() */
100 if (baselen
>= matchlen
) {
101 /* If it doesn't match, move along... */
102 if (strncmp(base
, match
, matchlen
))
106 * The base is a subdirectory of a path which
107 * was specified, so all of them are interesting.
112 /* Does the base match? */
113 if (strncmp(base
, match
, baselen
))
119 if (never_interesting
) {
121 * We have not seen any match that sorts later
122 * than the current path.
126 * Does match sort strictly earlier than path
127 * with their common parts?
129 m
= strncmp(match
, path
,
130 (matchlen
< pathlen
) ? matchlen
: pathlen
);
135 * If we come here even once, that means there is at
136 * least one pathspec that would sort equal to or
137 * later than the path we are currently looking at.
138 * In other words, if we have never reached this point
139 * after iterating all pathspecs, it means all
140 * pathspecs are either outside of base, or inside the
141 * base but sorts strictly earlier than the current
142 * one. In either case, they will never match the
143 * subsequent entries. In such a case, we initialized
144 * the variable to -1 and that is what will be
145 * returned, allowing the caller to terminate early.
147 never_interesting
= 0;
150 if (pathlen
> matchlen
)
153 if (matchlen
> pathlen
) {
154 if (match
[pathlen
] != '/')
162 * we cheated and did not do strncmp(), so we do
165 m
= strncmp(match
, path
, pathlen
);
168 * If common part matched earlier then it is a hit,
169 * because we rejected the case where path is not a
170 * leading directory and is shorter than match.
175 return never_interesting
; /* No matches */
178 /* A whole sub-tree went away or appeared */
179 static void show_tree(struct diff_options
*opt
, const char *prefix
, struct tree_desc
*desc
, const char *base
, int baselen
)
181 int all_interesting
= 0;
188 show
= tree_entry_interesting(desc
, base
, baselen
,
196 show_entry(opt
, prefix
, desc
, base
, baselen
);
197 update_tree_entry(desc
);
201 /* A file entry went away or appeared */
202 static void show_entry(struct diff_options
*opt
, const char *prefix
, struct tree_desc
*desc
,
203 const char *base
, int baselen
)
207 const unsigned char *sha1
= tree_entry_extract(desc
, &path
, &mode
);
209 if (opt
->recursive
&& S_ISDIR(mode
)) {
210 enum object_type type
;
211 int pathlen
= tree_entry_len(path
, sha1
);
212 char *newbase
= malloc_base(base
, baselen
, path
, pathlen
);
213 struct tree_desc inner
;
217 tree
= read_sha1_file(sha1
, &type
, &size
);
218 if (!tree
|| type
!= OBJ_TREE
)
219 die("corrupt tree sha %s", sha1_to_hex(sha1
));
221 init_tree_desc(&inner
, tree
, size
);
222 show_tree(opt
, prefix
, &inner
, newbase
, baselen
+ 1 + pathlen
);
227 opt
->add_remove(opt
, prefix
[0], mode
, sha1
, base
, path
);
231 static void skip_uninteresting(struct tree_desc
*t
, const char *base
, int baselen
, struct diff_options
*opt
)
233 int all_interesting
= 0;
240 show
= tree_entry_interesting(t
, base
, baselen
, opt
);
245 update_tree_entry(t
);
255 int diff_tree(struct tree_desc
*t1
, struct tree_desc
*t2
, const char *base
, struct diff_options
*opt
)
257 int baselen
= strlen(base
);
260 if (opt
->quiet
&& opt
->has_changes
)
263 skip_uninteresting(t1
, base
, baselen
, opt
);
264 skip_uninteresting(t2
, base
, baselen
, opt
);
269 show_entry(opt
, "+", t2
, base
, baselen
);
270 update_tree_entry(t2
);
274 show_entry(opt
, "-", t1
, base
, baselen
);
275 update_tree_entry(t1
);
278 switch (compare_tree_entry(t1
, t2
, base
, baselen
, opt
)) {
280 update_tree_entry(t1
);
283 update_tree_entry(t1
);
286 update_tree_entry(t2
);
289 die("git-diff-tree: internal error");
295 * Does it look like the resulting diff might be due to a rename?
297 * - not a valid previous file
299 static inline int diff_might_be_rename(void)
301 return diff_queued_diff
.nr
== 1 &&
302 !DIFF_FILE_VALID(diff_queued_diff
.queue
[0]->one
);
305 static void try_to_follow_renames(struct tree_desc
*t1
, struct tree_desc
*t2
, const char *base
, struct diff_options
*opt
)
307 struct diff_options diff_opts
;
308 const char *paths
[2];
311 diff_setup(&diff_opts
);
312 diff_opts
.recursive
= 1;
313 diff_opts
.detect_rename
= DIFF_DETECT_RENAME
;
314 diff_opts
.output_format
= DIFF_FORMAT_NO_OUTPUT
;
315 diff_opts
.single_follow
= opt
->paths
[0];
317 diff_tree_setup_paths(paths
, &diff_opts
);
318 if (diff_setup_done(&diff_opts
) < 0)
319 die("unable to set up diff options to follow renames");
320 diff_tree(t1
, t2
, base
, &diff_opts
);
321 diffcore_std(&diff_opts
);
323 /* NOTE! Ignore the first diff! That was the old one! */
324 for (i
= 1; i
< diff_queued_diff
.nr
; i
++) {
325 struct diff_filepair
*p
= diff_queued_diff
.queue
[i
];
328 * Found a source? Not only do we use that for the new
329 * diff_queued_diff, we also use that as the path in
332 if ((p
->status
== 'R' || p
->status
== 'C') && !strcmp(p
->two
->path
, opt
->paths
[0])) {
333 diff_queued_diff
.queue
[0] = p
;
334 opt
->paths
[0] = xstrdup(p
->one
->path
);
335 diff_tree_setup_paths(opt
->paths
, opt
);
341 * Then, ignore any but the first entry! It might be the old one,
342 * or it might be the rename/copy we found
344 diff_queued_diff
.nr
= 1;
347 int diff_tree_sha1(const unsigned char *old
, const unsigned char *new, const char *base
, struct diff_options
*opt
)
350 struct tree_desc t1
, t2
;
351 unsigned long size1
, size2
;
354 tree1
= read_object_with_reference(old
, tree_type
, &size1
, NULL
);
356 die("unable to read source tree (%s)", sha1_to_hex(old
));
357 tree2
= read_object_with_reference(new, tree_type
, &size2
, NULL
);
359 die("unable to read destination tree (%s)", sha1_to_hex(new));
360 init_tree_desc(&t1
, tree1
, size1
);
361 init_tree_desc(&t2
, tree2
, size2
);
362 retval
= diff_tree(&t1
, &t2
, base
, opt
);
363 if (opt
->follow_renames
&& diff_might_be_rename()) {
364 init_tree_desc(&t1
, tree1
, size1
);
365 init_tree_desc(&t2
, tree2
, size2
);
366 try_to_follow_renames(&t1
, &t2
, base
, opt
);
373 int diff_root_tree_sha1(const unsigned char *new, const char *base
, struct diff_options
*opt
)
378 struct tree_desc empty
, real
;
380 tree
= read_object_with_reference(new, tree_type
, &size
, NULL
);
382 die("unable to read root tree (%s)", sha1_to_hex(new));
383 init_tree_desc(&real
, tree
, size
);
385 init_tree_desc(&empty
, "", 0);
386 retval
= diff_tree(&empty
, &real
, base
, opt
);
391 static int count_paths(const char **paths
)
399 void diff_tree_release_paths(struct diff_options
*opt
)
404 void diff_tree_setup_paths(const char **p
, struct diff_options
*opt
)
407 opt
->pathlens
= NULL
;
414 opt
->nr_paths
= count_paths(p
);
415 if (opt
->nr_paths
== 0) {
416 opt
->pathlens
= NULL
;
419 opt
->pathlens
= xmalloc(opt
->nr_paths
* sizeof(int));
420 for (i
=0; i
< opt
->nr_paths
; i
++)
421 opt
->pathlens
[i
] = strlen(p
[i
]);