Skip to content

Commit 750f7b6

Browse files
torvaldsgitster
authored andcommitted
Finally implement "git log --follow"
Ok, I've really held off doing this too damn long, because I'm lazy, and I was always hoping that somebody else would do it. But no, people keep asking for it, but nobody actually did anything, so I decided I might as well bite the bullet, and instead of telling people they could add a "--follow" flag to "git log" to do what they want to do, I decided that it looks like I just have to do it for them.. The code wasn't actually that complicated, in that the diffstat for this patch literally says "70 insertions(+), 1 deletions(-)", but I will have to admit that in order to get to this fairly simple patch, you did have to know and understand the internal git diff generation machinery pretty well, and had to really be able to follow how commit generation interacts with generating patches and generating the log. So I suspect that while I was right that it wasn't that hard, I might have been expecting too much of random people - this patch does seem to be firmly in the core "Linus or Junio" territory. To make a long story short: I'm sorry for it taking so long until I just did it. I'm not going to guarantee that this works for everybody, but you really can just look at the patch, and after the appropriate appreciative noises ("Ooh, aah") over how clever I am, you can then just notice that the code itself isn't really that complicated. All the real new code is in the new "try_to_follow_renames()" function. It really isn't rocket science: we notice that the pathname we were looking at went away, so we start a full tree diff and try to see if we can instead make that pathname be a rename or a copy from some other previous pathname. And if we can, we just continue, except we show *that* particular diff, and ever after we use the _previous_ pathname. One thing to look out for: the "rename detection" is considered to be a singular event in the _linear_ "git log" output! That's what people want to do, but I just wanted to point out that this patch is *not* carrying around a "commit,pathname" kind of pair and it's *not* going to be able to notice the file coming from multiple *different* files in earlier history. IOW, if you use "git log --follow", then you get the stupid CVS/SVN kind of "files have single identities" kind of semantics, and git log will just pick the identity based on the normal move/copy heuristics _as_if_ the history could be linearized. Put another way: I think the model is broken, but given the broken model, I think this patch does just about as well as you can do. If you have merges with the same "file" having different filenames over the two branches, git will just end up picking _one_ of the pathnames at the point where the newer one goes away. It never looks at multiple pathnames in parallel. And if you understood all that, you probably didn't need it explained, and if you didn't understand the above blathering, it doesn't really mtter to you. What matters to you is that you can now do git log -p --follow builtin-rev-list.c and it will find the point where the old "rev-list.c" got renamed to "builtin-rev-list.c" and show it as such. Signed-off-by: Linus Torvalds <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent 4d9b580 commit 750f7b6

File tree

5 files changed

+70
-1
lines changed

5 files changed

+70
-1
lines changed

builtin-log.c

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,11 @@ static void cmd_log_init(int argc, const char **argv, const char *prefix,
5858
argc = setup_revisions(argc, argv, rev, "HEAD");
5959
if (rev->diffopt.pickaxe || rev->diffopt.filter)
6060
rev->always_show_header = 0;
61+
if (rev->diffopt.follow_renames) {
62+
rev->always_show_header = 0;
63+
if (rev->diffopt.nr_paths != 1)
64+
usage("git logs can only follow renames on one pathname at a time");
65+
}
6166
for (i = 1; i < argc; i++) {
6267
const char *arg = argv[i];
6368
if (!strcmp(arg, "--decorate")) {

diff.c

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -2210,6 +2210,8 @@ int diff_opt_parse(struct diff_options *options, const char **av, int ac)
22102210
}
22112211
else if (!strcmp(arg, "--find-copies-harder"))
22122212
options->find_copies_harder = 1;
2213+
else if (!strcmp(arg, "--follow"))
2214+
options->follow_renames = 1;
22132215
else if (!strcmp(arg, "--abbrev"))
22142216
options->abbrev = DEFAULT_ABBREV;
22152217
else if (!prefixcmp(arg, "--abbrev=")) {

diff.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,7 @@ struct diff_options {
5555
full_index:1,
5656
silent_on_remove:1,
5757
find_copies_harder:1,
58+
follow_renames:1,
5859
color_diff:1,
5960
color_diff_words:1,
6061
has_changes:1,

revision.c

Lines changed: 3 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1230,7 +1230,9 @@ int setup_revisions(int argc, const char **argv, struct rev_info *revs, const ch
12301230

12311231
if (revs->prune_data) {
12321232
diff_tree_setup_paths(revs->prune_data, &revs->pruning);
1233-
revs->prune_fn = try_to_simplify_commit;
1233+
/* Can't prune commits with rename following: the paths change.. */
1234+
if (!revs->diffopt.follow_renames)
1235+
revs->prune_fn = try_to_simplify_commit;
12341236
if (!revs->full_diff)
12351237
diff_tree_setup_paths(revs->prune_data, &revs->diffopt);
12361238
}

tree-diff.c

Lines changed: 59 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
*/
44
#include "cache.h"
55
#include "diff.h"
6+
#include "diffcore.h"
67
#include "tree.h"
78

89
static char *malloc_base(const char *base, int baselen, const char *path, int pathlen)
@@ -290,6 +291,59 @@ int diff_tree(struct tree_desc *t1, struct tree_desc *t2, const char *base, stru
290291
return 0;
291292
}
292293

294+
/*
295+
* Does it look like the resulting diff might be due to a rename?
296+
* - single entry
297+
* - not a valid previous file
298+
*/
299+
static inline int diff_might_be_rename(void)
300+
{
301+
return diff_queued_diff.nr == 1 &&
302+
!DIFF_FILE_VALID(diff_queued_diff.queue[0]->one);
303+
}
304+
305+
static void try_to_follow_renames(struct tree_desc *t1, struct tree_desc *t2, const char *base, struct diff_options *opt)
306+
{
307+
struct diff_options diff_opts;
308+
const char *paths[2];
309+
int i;
310+
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];
316+
paths[0] = NULL;
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);
322+
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];
326+
327+
/*
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
330+
* the future!
331+
*/
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);
336+
break;
337+
}
338+
}
339+
340+
/*
341+
* Then, ignore any but the first entry! It might be the old one,
342+
* or it might be the rename/copy we found
343+
*/
344+
diff_queued_diff.nr = 1;
345+
}
346+
293347
int diff_tree_sha1(const unsigned char *old, const unsigned char *new, const char *base, struct diff_options *opt)
294348
{
295349
void *tree1, *tree2;
@@ -306,6 +360,11 @@ int diff_tree_sha1(const unsigned char *old, const unsigned char *new, const cha
306360
init_tree_desc(&t1, tree1, size1);
307361
init_tree_desc(&t2, tree2, size2);
308362
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);
367+
}
309368
free(tree1);
310369
free(tree2);
311370
return retval;

0 commit comments

Comments
 (0)