On Tue, Feb 12, 2013 at 09:48:18PM +0100, Karsten Blees wrote: > However, the difference between git status -uall and -uno was always > about 1.3 s in all fscache versions, even though > opendir/readdir/closedir was served entirely from the cache. I added > a bit of performance tracing to find the cause, and I think most of > the time spent in wt_status_collect_untracked can be eliminated: > > 1.) 0.939 s is spent in dir.c/excluded (i.e. checking > .gitignore). This check is done for *every* file in the working > copy, including files in the index. Checking the index first could > eliminate most of that, i.e.: > > (Note: patches are for discussion only, I'm aware that they may have > unintended side effects...) > > @@ -1097,6 +1097,8 @@ static enum path_treatment treat_path(struct dir_struct *dir, > return path_ignored; > strbuf_setlen(path, baselen); > strbuf_addstr(path, de->d_name); > + if (cache_name_exists(path->buf, path->len, ignore_case)) > + return path_ignored; > if (simplify_away(path->buf, path->len, simplify)) > return path_ignored; The below patch passes the test suite for me and still does the same thing. On my Linux box, running "git status" on gentoo-x86.git with this patch saves 0.05s (0.548s without the patch, 0.505s with the patch, best number of 20 runs). And I just realized gentoo-x86.git does not have any .gitignore. On webkit.git, it cuts "git status" time from 1.121s down to 0.762s. Unless I'm mistaken, "git add" should have the same benefit on normal case too. Good finding! -- 8< -- diff --git a/dir.c b/dir.c index 57394e4..4b4cf60 100644 --- a/dir.c +++ b/dir.c @@ -1244,7 +1244,19 @@ static enum path_treatment treat_one_path(struct dir_struct *dir, const struct path_simplify *simplify, int dtype, struct dirent *de) { - int exclude = is_excluded(dir, path->buf, &dtype); + int exclude; + + if (dtype == DT_UNKNOWN) + dtype = get_dtype(de, path->buf, path->len); + + if (!(dir->flags & DIR_SHOW_IGNORED) && + !(dir->flags & DIR_COLLECT_IGNORED) && + dtype == DT_REG && + cache_name_exists(path->buf, path->len, ignore_case)) + return path_ignored; + + exclude = is_excluded(dir, path->buf, &dtype); + if (exclude && (dir->flags & DIR_COLLECT_IGNORED) && exclude_matches_pathspec(path->buf, path->len, simplify)) dir_add_ignored(dir, path->buf, path->len); @@ -1256,9 +1268,6 @@ static enum path_treatment treat_one_path(struct dir_struct *dir, if (exclude && !(dir->flags & DIR_SHOW_IGNORED)) return path_ignored; - if (dtype == DT_UNKNOWN) - dtype = get_dtype(de, path->buf, path->len); - switch (dtype) { default: return path_ignored; -- 8< -- -- To unsubscribe from this list: send the line "unsubscribe git" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html