diff options
author | Nico Weber <nicolasweber@gmx.de> | 2014-12-08 00:02:35 (GMT) |
---|---|---|
committer | Nico Weber <nicolasweber@gmx.de> | 2014-12-08 00:20:24 (GMT) |
commit | 015c818cbe85093316115c5a510274eccdb4180b (patch) | |
tree | c614fbaf19e47e84aa625b5ccc4533677d8b65a0 /src/canon_perftest.cc | |
parent | a9dac57ecad0b8ff4822c39446e9bf35afc0a615 (diff) | |
download | Ninja-015c818cbe85093316115c5a510274eccdb4180b.zip Ninja-015c818cbe85093316115c5a510274eccdb4180b.tar.gz Ninja-015c818cbe85093316115c5a510274eccdb4180b.tar.bz2 |
Let DependencyScan::RecomputeDirty() work correclty with cyclic graphs.
RecomputDirty(edge) currently works roughly like this:
RecomputeDirty(edge):
LoadDeps(edge)
for in in edge.inputs:
if in.StatIfNecessary():
RecomputeDirty(in.in_edge) # recurse into inputs
for out in edge.outputs:
out.StatIfNecessary() # mark outputs as visited
It uses the stat state of each node to mark nodes as visited and doesn't
traverse across nodes that have been visited already. For cyclic graphs
with an edge with multiple outputs on the cycle, nothing prevents an
edge to be visited more than once if the cycle is entered through an
output that isn't on the cycle. In other words, RecomputeDirty() for
the same edge can be on the call stack more than once. This is bad for
at least two reasons:
1. Deps are added multiple times, making the graph confusing to reason
about.
2. LoadDeps() will insert into the inputs_ of an edge that's iterated
over in a callframe higher up. This can invalidate the iterator,
which causes a crash when the callframe with the loop over the
now-invalidated iterator resumes.
To fix this, let RecomputeDirty() mark all outputs of an edge as visited
as the first thing it does. This way, even if the edge is on a cycle
with several outputs, each output is already marked and no edge will
have its deps loaded more than once.
Fixes the crashes in #875. (In practice, it turns the crashes into
"stuck [this is a bug]" messages for now, due to the example without
duplicate rules in #867)
Diffstat (limited to 'src/canon_perftest.cc')
0 files changed, 0 insertions, 0 deletions