Make path-limiting be incremental when possible.
authorLinus Torvalds <torvalds@osdl.org>
Fri, 31 Mar 2006 01:05:25 +0000 (17:05 -0800)
committerJunio C Hamano <junkio@cox.net>
Sat, 1 Apr 2006 00:24:48 +0000 (16:24 -0800)
This makes git-rev-list able to do path-limiting without having to parse
all of history before it starts showing the results.

This makes things like "git log -- pathname" much more pleasant to use.

This is actually a pretty small patch, and the biggest part of it is
purely cleanups (turning the "goto next" statements into "continue"), but
it's conceptually a lot bigger than it looks.

What it does is that if you do a path-limited revision list, and you do
_not_ ask for pseudo-parenthood information, it won't do all the
path-limiting up-front, but instead do it incrementally in
"get_revision()".

This is an absolutely huge deal for anything like "git log -- <pathname>",
but also for some things that we don't do yet - like the "find where
things changed" logic I've described elsewhere, where we want to find the
previous revision that changed a file.

The reason I put "RFC" in the subject line is that while I've validated it
various ways, like doing

git-rev-list HEAD -- drivers/char/ | md5sum

before-and-after on the kernel archive, it's "git-rev-list" after all. In
other words, it's that really really subtle and complex central piece of
software. So while I think this is important and should go in asap, I also
think it should get lots of testing and eyeballs looking at the code.

Btw, don't even bother testing this with the git archive. git itself is so
small that parsing the whole revision history for it takes about a second
even with path limiting. The thing that _really_ shows this off is doing

git log drivers/

on the kernel archive, or even better, on the _historic_ kernel archive.

With this change, the response is instantaneous (although seeking to the
end of the result will obviously take as long as it ever did). Before this
change, the command would think about the result for tens of seconds - or
even minutes, in the case of the bigger old kernel archive - before
starting to output the results.

NOTE NOTE NOTE! Using path limiting with things like "gitk", which uses
the "--parents" flag to actually generate a pseudo-history of the
resulting commits won't actually see the improvement in interactivity,
since that forces git-rev-list to do the whole-history thing after all.

MAYBE we can fix that too at some point, but I won't promise anything.

Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Signed-off-by: Junio C Hamano <junkio@cox.net>
revision.c

index 1224a2de624047eebd24969852af6833dbea2874..a8a54b658029606839853daf1cee380833486dd4 100644 (file)
@@ -711,7 +711,13 @@ int setup_revisions(int argc, const char **argv, struct rev_info *revs, const ch
        if (revs->prune_data) {
                diff_tree_setup_paths(revs->prune_data);
                revs->prune_fn = try_to_simplify_commit;
-               revs->limited = 1;
+
+               /*
+                * If we fix up parent data, we currently cannot
+                * do that on-the-fly.
+                */
+               if (revs->parents)
+                       revs->limited = 1;
        }
 
        return left;
@@ -773,41 +779,34 @@ struct commit *get_revision(struct rev_info *revs)
        do {
                struct commit *commit = revs->commits->item;
 
+               revs->commits = revs->commits->next;
+
+               /*
+                * If we haven't done the list limiting, we need to look at
+                * the parents here
+                */
+               if (!revs->limited)
+                       add_parents_to_list(revs, commit, &revs->commits);
                if (commit->object.flags & SHOWN)
-                       goto next;
+                       continue;
                if (!(commit->object.flags & BOUNDARY) &&
                    (commit->object.flags & UNINTERESTING))
-                       goto next;
+                       continue;
                if (revs->min_age != -1 && (commit->date > revs->min_age))
-                       goto next;
+                       continue;
                if (revs->max_age != -1 && (commit->date < revs->max_age))
                        return NULL;
                if (revs->no_merges &&
                    commit->parents && commit->parents->next)
-                       goto next;
+                       continue;
                if (revs->prune_fn && revs->dense) {
                        if (!(commit->object.flags & TREECHANGE))
-                               goto next;
-                       rewrite_parents(commit);
-               }
-               /* More to go? */
-               if (revs->max_count) {
-                       if (commit->object.flags & BOUNDARY) {
-                               /* this is already uninteresting,
-                                * so there is no point popping its
-                                * parents into the list.
-                                */
-                               struct commit_list *it = revs->commits;
-                               revs->commits = it->next;
-                               free(it);
-                       }
-                       else
-                               pop_most_recent_commit(&revs->commits, SEEN);
+                               continue;
+                       if (revs->parents)
+                               rewrite_parents(commit);
                }
                commit->object.flags |= SHOWN;
                return commit;
-next:
-               pop_most_recent_commit(&revs->commits, SEEN);
        } while (revs->commits);
        return NULL;
 }