summaryrefslogtreecommitdiff
path: root/src/missing_deps.h
diff options
context:
space:
mode:
authorTomasz Śniatowski <tsniatowski@vewd.com>2021-02-15 13:06:30 +0100
committerTomasz Śniatowski <tsniatowski@vewd.com>2021-02-22 23:48:58 +0100
commit3030254733f0baae1353b99e72e85babfbf5fbce (patch)
tree96c06dc3f48c930eb1caa3b468eb00bdbdb308c0 /src/missing_deps.h
parentd0489c3863f6b2a0d0b0e15880217da3fd4e6d8f (diff)
downloadninja-3030254733f0baae1353b99e72e85babfbf5fbce.tar.gz
Add a -t missingdeps tool to detect some classes of build flakes
The tool looks for targets that depend on a generated file, but do not properly specify a dependency on the generator. It needs to be run after a successful build, and will list all potential flakes that may have broken the build, but didn't due to accidental build step ordering. The search relies on the correctness of depfile and generator output information, but these are usually easier to get right than dependencies. The errors found can usually be verified as actual build flakes by trying to build the listed problematic files alone in a clean build directory. Such builds usually fail with a compile job lacking a generated file. There is some overlap between this tool and 'gn check', but not everyone uses gn, not everyone using gn uses gn check, and most importantly, gn check is more about modularity, and less about actual build-time deps without flakes. The tool needs to be run after a build completes and depfile data is collected. It may take several seconds to process, up to a dozen or two on a large, chromium-sized build.
Diffstat (limited to 'src/missing_deps.h')
-rw-r--r--src/missing_deps.h105
1 files changed, 105 insertions, 0 deletions
diff --git a/src/missing_deps.h b/src/missing_deps.h
new file mode 100644
index 0000000..211a865
--- /dev/null
+++ b/src/missing_deps.h
@@ -0,0 +1,105 @@
+// Copyright 2019 Google Inc. All Rights Reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef NINJA_MISSING_DEPS_H_
+#define NINJA_MISSING_DEPS_H_
+
+#include <map>
+#include <set>
+#include <string>
+
+#if __cplusplus >= 201103L
+#include <unordered_map>
+#endif
+
+struct DepsLog;
+struct DiskInterface;
+struct Edge;
+struct Node;
+struct Rule;
+struct State;
+
+class MissingDependencyScannerDelegate {
+ public:
+ virtual ~MissingDependencyScannerDelegate();
+ virtual void OnMissingDep(Node* node, const std::string& path,
+ const Rule& generator) = 0;
+};
+
+class MissingDependencyPrinter : public MissingDependencyScannerDelegate {
+ void OnMissingDep(Node* node, const std::string& path, const Rule& generator);
+ void OnStats(int nodes_processed, int nodes_missing_deps,
+ int missing_dep_path_count, int generated_nodes,
+ int generator_rules);
+};
+
+struct EdgePair {
+ EdgePair(Edge* from, Edge* to) : from_(from), to_(to) {}
+ bool operator==(const EdgePair& other) const {
+ return from_ == other.from_ && to_ == other.to_;
+ }
+ bool operator<(const EdgePair& other) const {
+ return (from_ < other.from_) ||
+ ((from_ == other.from_) && (to_ < other.to_));
+ }
+ Edge* from_;
+ Edge* to_;
+};
+
+#if __cplusplus >= 201103L
+namespace std {
+template <>
+struct hash<EdgePair> {
+ size_t operator()(const EdgePair& k) const {
+ uintptr_t uint_from = uintptr_t(k.from_);
+ uintptr_t uint_to = uintptr_t(k.to_);
+ return hash<uintptr_t>()(uint_from ^ (uint_to >> 3));
+ }
+};
+} // namespace std
+#endif // __cplusplus >= 201103L
+
+struct MissingDependencyScanner {
+ public:
+ MissingDependencyScanner(MissingDependencyScannerDelegate* delegate,
+ DepsLog* deps_log, State* state,
+ DiskInterface* disk_interface);
+ void ProcessNode(Node* node);
+ void PrintStats();
+ bool HadMissingDeps() { return !nodes_missing_deps_.empty(); }
+
+ void ProcessNodeDeps(Node* node, Node** dep_nodes, int dep_nodes_count);
+
+ bool PathExistsBetween(Edge* from, Edge* to);
+
+ MissingDependencyScannerDelegate* delegate_;
+ DepsLog* deps_log_;
+ State* state_;
+ DiskInterface* disk_interface_;
+ std::set<Node*> seen_;
+ std::set<Node*> nodes_missing_deps_;
+ std::set<Node*> generated_nodes_;
+ std::set<const Rule*> generator_rules_;
+ int missing_dep_path_count_;
+
+ private:
+#if __cplusplus >= 201103L
+ using EdgeAdjacencyMap = std::unordered_map<EdgePair, bool>;
+#else
+ typedef std::map<EdgePair, bool> EdgeAdjacencyMap;
+#endif
+ EdgeAdjacencyMap edge_adjacency_map_;
+};
+
+#endif // NINJA_MISSING_DEPS_H_