Stefan Beller wrote: > Suppose I have a superproject 'super', with two submodules 'super/sub' > and 'super/sub1'. 'super/sub' itself contains a submodule > 'super/sub/subsub'. Now suppose I run, from within 'super': > > echo hi >sub/subsub/stray-file > echo hi >sub1/stray-file > > Currently we get would see the following output in git-status: > > git status --short > m sub > ? sub1 > > With this patch applied, the untracked file in the nested submodule is > turned into an untracked file on the 'super' level as well. s/turned into/displayed as/ > git status --short > ? sub > ? sub1 > > This doesn't change the output of 'git status --porcelain=1' for nested > submodules, because its output is always ' M' for either untracked files > or local modifications no matter the nesting level of the submodule. > > 'git status --porcelain=2' is affected by this change in a nested > submodule, though. Without this patch it would report the direct submodule > as modified and having no untracked files. With this patch it would report > untracked files. Chalk this up as a bug fix. I think that's reasonable, and the documentation does a good job of describing it. Does this have any effect on the default mode of 'git status' (without --short or --porcelain)? [...] > --- a/Documentation/git-status.txt > +++ b/Documentation/git-status.txt > @@ -187,6 +187,8 @@ Submodules have more state and instead report > m the submodule has modified content > ? the submodule has untracked files > > +Note that 'm' and '?' are applied recursively, e.g. if a nested submodule > +in a submodule contains an untracked file, this is reported as '?' as well. Language nits: * Can simplify by leaving out "Note that ". * s/, e\.g\./. For example,/ Should this say a brief word about rationale? The obvious way to describe it would involve "git add --recurse-submodules", which alas doesn't exist yet. But we could say this is reported as '?' as well, since the change cannot be added using "git add -u" from within any of the submodules. [...] > --- a/submodule.c > +++ b/submodule.c > @@ -1078,8 +1078,27 @@ unsigned is_submodule_modified(const char *path, int ignore_untracked) > /* regular untracked files */ > if (buf.buf[0] == '?') > dirty_submodule |= DIRTY_SUBMODULE_UNTRACKED; > - else > - dirty_submodule |= DIRTY_SUBMODULE_MODIFIED; > + > + if (buf.buf[0] == 'u' || > + buf.buf[0] == '1' || > + buf.buf[0] == '2') { > + /* > + * T XY SSSS: > + * T = line type, XY = status, SSSS = submodule state > + */ > + if (buf.len < 1 + 1 + 2 + 1 + 4) optional: Can shorten: /* T = line type, XY = status, SSSS = submodule state */ if (buf.len < strlen("T XY SSSS")) ... That produces the same code at run time because compilers are able to inline the strlen of a constant. What you already have also seems sensible, though. [...] > + die("BUG: invalid status --porcelain=2 line %s", > + buf.buf); > + > + /* regular unmerged and renamed files */ > + if (buf.buf[5] == 'S' && buf.buf[8] == 'U') > + /* nested untracked file */ > + dirty_submodule |= DIRTY_SUBMODULE_UNTRACKED; > + > + if (memcmp(buf.buf + 5, "S..U", 4)) > + /* other change */ > + dirty_submodule |= DIRTY_SUBMODULE_MODIFIED; sanity check: What does this do for a "2" line indicating a sub-submodule that has been renamed that contains an untracked file? Do we need to rely on some other indication to show this as a change? Enumerating some more cases, since I keep finding myself getting lost: - if the HEAD commit of "sub" changes, we show this as " M sub". What should we show if the HEAD commit of "sub/subsub" changes? I think this should be " m". - if "sub" is renamed, we show this as "R sub -> newname". What should we show if "sub/subsub" is renamed? It is tempting to show this as " m". - if "sub" is deleted, we show this as "D sub". What should we show if "sub/subsub" is deleted? I think this is " m". It keeps getting more complicated, but I think this is making sense. Thanks and hope that helps, Jonathan