resolver: Fix bug in dep graph SortGlobals()

If we encounter a global twice, the stack vector was not popped, which could trigger false cyclic-dependency errors. Because the cyclic dependency did not actually exist, later logic could break, expecting to find a global->global edge which did not exist.

Fixed: chromium:1273451
Change-Id: I9d99f1aeeaea042d9ed847a878c4717803122240
Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/70980
Reviewed-by: David Neto <dneto@google.com>
Kokoro: Kokoro <noreply+kokoro@google.com>
Commit-Queue: Ben Clayton <bclayton@google.com>
This commit is contained in:
Ben Clayton 2021-11-25 18:07:57 +00:00 committed by Tint LUCI CQ
parent 0079a597e6
commit 91785d277b
5 changed files with 77 additions and 21 deletions

View File

@ -305,8 +305,8 @@ class DependencyScanner {
auto global_it = globals_.find(ident->symbol); auto global_it = globals_.find(ident->symbol);
if (global_it != globals_.end() && if (global_it != globals_.end() &&
node == global_it->second->node) { node == global_it->second->node) {
ResolveGlobalDependency(ident, ident->symbol, "identifier", AddGlobalDependency(ident, ident->symbol, "identifier",
"references"); "references");
} else { } else {
graph_.resolved_symbols.emplace(ident, node); graph_.resolved_symbols.emplace(ident, node);
} }
@ -314,9 +314,9 @@ class DependencyScanner {
if (auto* call = expr->As<ast::CallExpression>()) { if (auto* call = expr->As<ast::CallExpression>()) {
if (call->target.name) { if (call->target.name) {
if (!IsIntrinsic(call->target.name->symbol)) { if (!IsIntrinsic(call->target.name->symbol)) {
ResolveGlobalDependency(call->target.name, AddGlobalDependency(call->target.name,
call->target.name->symbol, "function", call->target.name->symbol, "function",
"calls"); "calls");
graph_.resolved_symbols.emplace( graph_.resolved_symbols.emplace(
call, call,
utils::Lookup(graph_.resolved_symbols, call->target.name)); utils::Lookup(graph_.resolved_symbols, call->target.name));
@ -357,7 +357,7 @@ class DependencyScanner {
return; return;
} }
if (auto* tn = ty->As<ast::TypeName>()) { if (auto* tn = ty->As<ast::TypeName>()) {
ResolveGlobalDependency(tn, tn->name, "type", "references"); AddGlobalDependency(tn, tn->name, "type", "references");
return; return;
} }
if (auto* vec = ty->As<ast::Vector>()) { if (auto* vec = ty->As<ast::Vector>()) {
@ -415,10 +415,10 @@ class DependencyScanner {
} }
/// Adds the dependency to the currently processed global /// Adds the dependency to the currently processed global
void ResolveGlobalDependency(const ast::Node* from, void AddGlobalDependency(const ast::Node* from,
Symbol to, Symbol to,
const char* use, const char* use,
const char* action) { const char* action) {
auto global_it = globals_.find(to); auto global_it = globals_.find(to);
if (global_it != globals_.end()) { if (global_it != globals_.end()) {
auto* global = global_it->second; auto* global = global_it->second;
@ -482,9 +482,8 @@ struct DependencyAnalysis {
// Sort the globals into dependency order // Sort the globals into dependency order
SortGlobals(); SortGlobals();
#if TINT_DUMP_DEPENDENCY_GRAPH // Dump the dependency graph if TINT_DUMP_DEPENDENCY_GRAPH is non-zero
DumpDependencyGraph(); DumpDependencyGraph();
#endif
if (!allow_out_of_order_decls) { if (!allow_out_of_order_decls) {
// Prevent out-of-order declarations. // Prevent out-of-order declarations.
@ -630,16 +629,26 @@ struct DependencyAnalysis {
return false; return false;
} }
if (sorted_.contains(g->node)) { if (sorted_.contains(g->node)) {
return false; // Visited this global already. // Visited this global already.
// stack was pushed, but exit() will not be called when we return
// false, so pop here.
stack.pop_back();
return false;
} }
return true; return true;
}, },
[&](const Global* g) { // Exit [&](const Global* g) { // Exit. Only called if Enter returned true.
sorted_.add(g->node); sorted_.add(g->node);
stack.pop_back(); stack.pop_back();
}); });
sorted_.add(global->node); sorted_.add(global->node);
if (!stack.empty()) {
// Each stack.push() must have a corresponding stack.pop_back().
TINT_ICE(Resolver, diagnostics_)
<< "stack not empty after returning from TraverseDependencies()";
}
} }
} }
@ -720,25 +729,28 @@ struct DependencyAnalysis {
} }
} }
#if TINT_DUMP_DEPENDENCY_GRAPH
void DumpDependencyGraph() { void DumpDependencyGraph() {
#if TINT_DUMP_DEPENDENCY_GRAPH == 0
if ((true)) {
return;
}
#endif // TINT_DUMP_DEPENDENCY_GRAPH
printf("=========================\n"); printf("=========================\n");
printf("------ declaration ------ \n"); printf("------ declaration ------ \n");
for (auto* global : declaration_order_) { for (auto* global : declaration_order_) {
printf("%s\n", NameOf(global->node).c_str()); printf("%s\n", NameOf(global->node).c_str());
} }
printf("------ dependencies ------ \n"); printf("------ dependencies ------ \n");
for (auto* node : sorted) { for (auto* node : sorted_) {
auto symbol = SymbolOf(node); auto symbol = SymbolOf(node);
auto* global = globals.at(symbol); auto* global = globals_.at(symbol);
printf("%s depends on:\n", symbols.NameFor(symbol).c_str()); printf("%s depends on:\n", symbols_.NameFor(symbol).c_str());
for (auto& dep : global->deps) { for (auto* dep : global->deps) {
printf(" %s\n", NameOf(dep.global->node).c_str()); printf(" %s\n", NameOf(dep->node).c_str());
} }
} }
printf("=========================\n"); printf("=========================\n");
} }
#endif // TINT_DUMP_DEPENDENCY_GRAPH
/// Program symbols /// Program symbols
const SymbolTable& symbols_; const SymbolTable& symbols_;

View File

@ -1339,6 +1339,19 @@ TEST_F(ResolverDependencyGraphTraversalTest, InferredType) {
Build(); Build();
} }
// Reproduces an unbalanced stack push / pop bug in
// DependencyAnalysis::SortGlobals(), found by clusterfuzz.
// See: crbug.com/chromium/1273451
TEST_F(ResolverDependencyGraphTraversalTest, chromium_1273451) {
Structure("A", {Member("a", ty.i32())});
Structure("B", {Member("b", ty.i32())});
Func("f", {Param("a", ty.type_name("A"))}, ty.type_name("B"),
{
Return(Construct(ty.type_name("B"))),
});
Build();
}
} // namespace ast_traversal } // namespace ast_traversal
} // namespace } // namespace

View File

@ -71,6 +71,9 @@ struct UniqueVector {
/// @returns the element at the index `i` /// @returns the element at the index `i`
const T& operator[](size_t i) const { return vector[i]; } const T& operator[](size_t i) const { return vector[i]; }
/// @returns true if the vector is empty
size_t empty() const { return vector.empty(); }
/// @returns the number of items in the vector /// @returns the number of items in the vector
size_t size() const { return vector.size(); } size_t size() const { return vector.size(); }

View File

@ -24,12 +24,14 @@ namespace {
TEST(UniqueVectorTest, Empty) { TEST(UniqueVectorTest, Empty) {
UniqueVector<int> unique_vec; UniqueVector<int> unique_vec;
EXPECT_EQ(unique_vec.size(), 0u); EXPECT_EQ(unique_vec.size(), 0u);
EXPECT_EQ(unique_vec.empty(), true);
EXPECT_EQ(unique_vec.begin(), unique_vec.end()); EXPECT_EQ(unique_vec.begin(), unique_vec.end());
} }
TEST(UniqueVectorTest, MoveConstructor) { TEST(UniqueVectorTest, MoveConstructor) {
UniqueVector<int> unique_vec(std::vector<int>{0, 3, 2, 1, 2}); UniqueVector<int> unique_vec(std::vector<int>{0, 3, 2, 1, 2});
EXPECT_EQ(unique_vec.size(), 4u); EXPECT_EQ(unique_vec.size(), 4u);
EXPECT_EQ(unique_vec.empty(), false);
EXPECT_EQ(unique_vec[0], 0); EXPECT_EQ(unique_vec[0], 0);
EXPECT_EQ(unique_vec[1], 3); EXPECT_EQ(unique_vec[1], 3);
EXPECT_EQ(unique_vec[2], 2); EXPECT_EQ(unique_vec[2], 2);
@ -42,6 +44,7 @@ TEST(UniqueVectorTest, AddUnique) {
unique_vec.add(1); unique_vec.add(1);
unique_vec.add(2); unique_vec.add(2);
EXPECT_EQ(unique_vec.size(), 3u); EXPECT_EQ(unique_vec.size(), 3u);
EXPECT_EQ(unique_vec.empty(), false);
int i = 0; int i = 0;
for (auto n : unique_vec) { for (auto n : unique_vec) {
EXPECT_EQ(n, i); EXPECT_EQ(n, i);
@ -65,6 +68,7 @@ TEST(UniqueVectorTest, AddDuplicates) {
unique_vec.add(1); unique_vec.add(1);
unique_vec.add(2); unique_vec.add(2);
EXPECT_EQ(unique_vec.size(), 3u); EXPECT_EQ(unique_vec.size(), 3u);
EXPECT_EQ(unique_vec.empty(), false);
int i = 0; int i = 0;
for (auto n : unique_vec) { for (auto n : unique_vec) {
EXPECT_EQ(n, i); EXPECT_EQ(n, i);
@ -90,6 +94,7 @@ TEST(UniqueVectorTest, AsVector) {
const std::vector<int>& vec = unique_vec; const std::vector<int>& vec = unique_vec;
EXPECT_EQ(vec.size(), 3u); EXPECT_EQ(vec.size(), 3u);
EXPECT_EQ(unique_vec.empty(), false);
int i = 0; int i = 0;
for (auto n : vec) { for (auto n : vec) {
EXPECT_EQ(n, i); EXPECT_EQ(n, i);
@ -109,18 +114,30 @@ TEST(UniqueVectorTest, PopBack) {
EXPECT_EQ(unique_vec.pop_back(), 1); EXPECT_EQ(unique_vec.pop_back(), 1);
EXPECT_EQ(unique_vec.size(), 2u); EXPECT_EQ(unique_vec.size(), 2u);
EXPECT_EQ(unique_vec.empty(), false);
EXPECT_EQ(unique_vec[0], 0); EXPECT_EQ(unique_vec[0], 0);
EXPECT_EQ(unique_vec[1], 2); EXPECT_EQ(unique_vec[1], 2);
EXPECT_EQ(unique_vec.pop_back(), 2); EXPECT_EQ(unique_vec.pop_back(), 2);
EXPECT_EQ(unique_vec.size(), 1u); EXPECT_EQ(unique_vec.size(), 1u);
EXPECT_EQ(unique_vec.empty(), false);
EXPECT_EQ(unique_vec[0], 0); EXPECT_EQ(unique_vec[0], 0);
unique_vec.add(1); unique_vec.add(1);
EXPECT_EQ(unique_vec.size(), 2u); EXPECT_EQ(unique_vec.size(), 2u);
EXPECT_EQ(unique_vec.empty(), false);
EXPECT_EQ(unique_vec[0], 0); EXPECT_EQ(unique_vec[0], 0);
EXPECT_EQ(unique_vec[1], 1); EXPECT_EQ(unique_vec[1], 1);
EXPECT_EQ(unique_vec.pop_back(), 1);
EXPECT_EQ(unique_vec.size(), 1u);
EXPECT_EQ(unique_vec.empty(), false);
EXPECT_EQ(unique_vec[0], 0);
EXPECT_EQ(unique_vec.pop_back(), 0);
EXPECT_EQ(unique_vec.size(), 0u);
EXPECT_EQ(unique_vec.empty(), true);
} }
} // namespace } // namespace

View File

@ -0,0 +1,11 @@
struct A {
a : i32;
};
struct B {
b : i32;
};
fn f(a : A) -> B {
return B();
}