On Thu, Jan 30, 2020 at 7:55 AM Derrick Stolee <stolee@xxxxxxxxx> wrote: > > However, on the positive side, it does make the code much faster. For > > the following simple shell loop in an empty repository: > > > > for depth in $(seq 10 25) > > do > > dirs=$(for i in $(seq 1 $depth) ; do printf 'dir/' ; done) > > rm -rf dir > > mkdir -p $dirs > > >$dirs/untracked-file > > /usr/bin/time --format="$depth: %e" git status --ignored >/dev/null > > done > > > > I saw the following timings, in seconds (note that the numbers are a > > little noisy from run-to-run, but the trend is very clear with every > > run): > > > > 10: 0.03 > > 11: 0.05 > > 12: 0.08 > > 13: 0.19 > > 14: 0.29 > > 15: 0.50 > > 16: 1.05 > > 17: 2.11 > > 18: 4.11 > > 19: 8.60 > > 20: 17.55 > > 21: 33.87 > > 22: 68.71 > > 23: 140.05 > > 24: 274.45 > > 25: 551.15 > > Are these timings on Linux? I imagine that the timings will increase > much more quickly on Windows. Yes, on Linux, with an SSD for the hard drive in this case (though I suspect OS caching of the directories would probably eliminate any differences between an SSD and a spinny disk since the same directories are visited so many times). > > After this fix, those drop to: > > > > 10: 0.00 > ... > > 25: 0.00 > > Nice. I wonder if presenting these 0.00 values as a table is worth > the space? At least the effect is dramatic. I first considered a table, but then noted it didn't match the code snippet I provided and was worried I'd have to spend more time explaining how I post-processed the output from two runs than we'd gain from compressing the number of lines of the commit message. Assuming reader time was more valuable, I opted to just keep the two snippets of output. > > WARNING: This change breaks t7063. I don't know whether that is to be expected > > (I now intentionally visit untracked directories differently so naturally the > > untracked cache should change), or if I've broken something. I'm hoping to get > > an untracked cache expert to chime in... > > I suppose that when the untracked cache is enabled, your expectation that we > do not need to recurse into an untracked directory is incorrect: we actually > want to explore that directory. Is there a mode we can check to see if we > are REALLY REALLY collecting _all_ untracked paths? Perhaps we need to create > one? I don't think I made any significant changes about using the untracked cache versus traversing; the primary differences should be that I traverse each directory once instead of 2^N times. However, the previous code would traverse with both check_only=0 and check_only=1, and to avoid the whole 2^N thing I only traverse once. That fundamentally means I only won't traverse with both settings of that flag. The output in t7063 seems to suggest to me that the check_only flag matters to what the untracked-cache stores ("check_only" literally appears as part of the expected output), and the output also suggests that the untracked-cache is recording when entries are visited multiple times somehow. Or maybe I'm just totally misunderstanding the expected output in t7063. I really have no clue about that stuff. > I'm CC'ing Kevin Willford because he is more familiar with the Git index > than me, and perhaps the untracked cache in particular. Getting another set of eyes, even if they only know enough to provide hunches or guesses, would be very welcome. > > Signed-off-by: Elijah Newren <newren@xxxxxxxxx> > > --- > > dir.c | 151 ++++++++++++++++++++++++++++++++++++++++------------------ > > 1 file changed, 105 insertions(+), 46 deletions(-) > > > > diff --git a/dir.c b/dir.c > > index ef3307718a..aaf038a9c4 100644 > > --- a/dir.c > > +++ b/dir.c > > @@ -1659,7 +1659,13 @@ static enum path_treatment treat_directory(struct dir_struct *dir, > > const char *dirname, int len, int baselen, int excluded, > > const struct pathspec *pathspec) > > { > > - int nested_repo; > > + /* > > + * WARNING: From this function, you can return path_recurse or you > > + * can call read_directory_recursive() (or neither), but > > + * you CAN'T DO BOTH. > > + */ > > + enum path_treatment state; > > + int nested_repo, old_ignored_nr, stop_early; > > > > /* The "len-1" is to strip the final '/' */ > > switch (directory_exists_in_index(istate, dirname, len-1)) { > > @@ -1713,18 +1719,101 @@ static enum path_treatment treat_directory(struct dir_struct *dir, > > > > /* This is the "show_other_directories" case */ > > > > - if (!(dir->flags & DIR_HIDE_EMPTY_DIRECTORIES)) > > + /* > > + * We only need to recurse into untracked/ignored directories if > > + * either of the following bits is set: > > + * - DIR_SHOW_IGNORED_TOO (because then we need to determine if > > + * there are ignored directories below) > > + * - DIR_HIDE_EMPTY_DIRECTORIES (because we have to determine if > > + * the directory is empty) > > Perhaps here is where you could also have a DIR_LIST_ALL_UNTRACKED > flag to ensure the untracked cache loads all untracked paths? Do you mean DIR_KEEP_UNTRACKED_CONTENTS (which is documented in dir.h as only having meaning when DIR_SHOW_IGNORED_TOO is also set, and thus caused me to not list it separately)? Speaking of DIR_KEEP_UNTRACKED_CONTENTS, though, its handling as a post-processing step in read_directory() is now inconsistent with how we handle squashing a directory full of ignores into just marking the containing directory as ignored. I think I should move the read_directory() logic for DIR_KEEP_UNTRACKED_CONTENTS to treat_directory() and use another counter similar to old_ignored_nr. It should be more efficient that way, too. > > > + */ > > + if (!(dir->flags & (DIR_SHOW_IGNORED_TOO | DIR_HIDE_EMPTY_DIRECTORIES))) > > return excluded ? path_excluded : path_untracked; > > > > + /* > > + * If we only want to determine if dirname is empty, then we can > > + * stop at the first file we find underneath that directory rather > > + * than continuing to recurse beyond it. If DIR_SHOW_IGNORED_TOO > > + * is set, then we want MORE than just determining if dirname is > > + * empty. > > + */ > > + stop_early = ((dir->flags & DIR_HIDE_EMPTY_DIRECTORIES) && > > + !(dir->flags & DIR_SHOW_IGNORED_TOO)); > > + > > + /* > > + * If /every/ file within an untracked directory is ignored, then > > + * we want to treat the directory as ignored (for e.g. status > > + * --porcelain), without listing the individual ignored files > > + * underneath. To do so, we'll save the current ignored_nr, and > > + * pop all the ones added after it if it turns out the entire > > + * directory is ignored. > > Here is a question for an untracked cache expert: Do we store ignored > paths in the untracked cache? According to 0dcb8d7fe0ec ("untracked cache: record .gitignore information and dir hierarchy", 2015-03-08), no: This cached output is about untracked files only, not ignored files because the number of tracked files is usually small, so small cache overhead, while the number of ignored files could go really high (e.g. *.o files mixing with source code). ...unless, of course, someone came along later and changed the design goals. [...] > This is certainly a substantial change, and I'm not able to read it > carefully right now. I hope to return to it soon, but hopefully I've > pointed out some places that may lead you to resolve your untracked > cache issues. Yeah, it's pretty hard to reason about; personally I needed lots of dumps of state during traversals just to partially make sense of it. I had dumps of output from both before and after my changes printing out return values of treat_directory() and paths and a bunch of other stuff and was doing lots of comparisons (and repeatedly did this for many, many different testcases with different toplevel git commands). It was particularly annoying that the old stuff would traverse everything 2^N times, half the time with check_only on and half the time with it off. It would return different state values for the same path from different calls, often depending on the side effects of dir.entries having had more entries added by the first recursion to get the right output, despite the fact that the "wrong" state was returned by treat_directory() for later visits to the same path (e.g. path_untracked returned for the first time it was visited, then path_none later, and it was a case where path_untracked was correct in my view). Despite those difficulties, having an extra set of eyes try to reason about it and pointing out anything that looks amiss or even that just looks hard to understand would be very welcome.