Fix infinite recursion due to recursive functions/tasks (#5398)

This commit is contained in:
Krzysztof Bieganski 2024-08-26 18:18:52 +02:00 committed by GitHub
parent c4cb26fa9a
commit b1927e4fb5
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 73 additions and 11 deletions

View File

@ -76,7 +76,12 @@ bool AstNodeFTaskRef::isPure() {
// cached. // cached.
return false; return false;
} else { } else {
if (!m_purity.isCached()) m_purity.set(this->getPurityRecurse()); if (!m_purity.isCached()) {
m_purity.set(true); // To prevent infinite recursion, set to true before getting
// the actual purity. If there are impure statements in the
// task/function, they'll taint this call anyway.
m_purity.set(this->getPurityRecurse());
}
return m_purity.get(); return m_purity.get();
} }
} }

View File

@ -33,9 +33,24 @@ class HasherVisitor final : public VNVisitorConst {
// STATE // STATE
V3Hash m_hash; // Hash value accumulator V3Hash m_hash; // Hash value accumulator
const bool m_cacheInUser4; // Use user4 to cache each V3Hash? const bool m_cacheInUser4; // Use user4 to cache each V3Hash?
std::set<AstNode*> m_visited; // Keeps track of some visited nodes to prevent
// infinite recursion
// METHODS // METHODS
void guardRecursion(AstNode* const nodep, std::function<void()>&& f) {
// Guard against infinite recursion if there's no caching
// Otherwise caching does the same but faster
if (!m_cacheInUser4) {
if (m_visited.find(nodep) != m_visited.end()) {
m_hash += V3Hash{nodep->name()};
return;
}
m_visited.insert(nodep);
}
f();
}
V3Hash hashNodeAndIterate(AstNode* nodep, bool hashDType, bool hashChildren, V3Hash hashNodeAndIterate(AstNode* nodep, bool hashDType, bool hashChildren,
std::function<void()>&& f) { std::function<void()>&& f) {
// See comments in visit(AstCFunc) about this breaking recursion // See comments in visit(AstCFunc) about this breaking recursion
@ -410,15 +425,17 @@ class HasherVisitor final : public VNVisitorConst {
}); });
} }
void visit(AstCFunc* nodep) override { void visit(AstCFunc* nodep) override {
guardRecursion(nodep, [this, nodep]() { //
m_hash += hashNodeAndIterate(nodep, HASH_DTYPE, HASH_CHILDREN, [this, nodep]() { // m_hash += hashNodeAndIterate(nodep, HASH_DTYPE, HASH_CHILDREN, [this, nodep]() { //
// We might be in a recursive function, if so on *second* call // We might be in a recursive function, if so on *second* call
// here we need to break what would be an infinite loop. // here we need to break what would be an infinite loop.
nodep->user4(V3Hash{1}.value()); // Set this "first" call if (m_cacheInUser4) nodep->user4(V3Hash{1}.value()); // Set this "first" call
// So that a second call will then exit hashNodeAndIterate // So that a second call will then exit hashNodeAndIterate
// Having a constant in the hash just means the recursion will // Having a constant in the hash just means the recursion will
// end, it shouldn't change the CFunc having a unique hash itself. // end, it shouldn't change the CFunc having a unique hash itself.
m_hash += nodep->isLoose(); m_hash += nodep->isLoose();
}); });
});
} }
void visit(AstVar* nodep) override { void visit(AstVar* nodep) override {
m_hash += hashNodeAndIterate(nodep, HASH_DTYPE, HASH_CHILDREN, [this, nodep]() { m_hash += hashNodeAndIterate(nodep, HASH_DTYPE, HASH_CHILDREN, [this, nodep]() {
@ -469,8 +486,12 @@ class HasherVisitor final : public VNVisitorConst {
[this, nodep]() { m_hash += nodep->name(); }); [this, nodep]() { m_hash += nodep->name(); });
} }
void visit(AstNodeFTask* nodep) override { void visit(AstNodeFTask* nodep) override {
guardRecursion(nodep, [this, nodep]() { //
m_hash += hashNodeAndIterate(nodep, HASH_DTYPE, HASH_CHILDREN, [this, nodep]() { // m_hash += hashNodeAndIterate(nodep, HASH_DTYPE, HASH_CHILDREN, [this, nodep]() { //
m_hash += nodep->name(); m_hash += nodep->name();
// See comments in AstCFunc
if (m_cacheInUser4) nodep->user4(V3Hash{1}.value());
});
}); });
} }
void visit(AstModport* nodep) override { void visit(AstModport* nodep) override {

View File

@ -0,0 +1,18 @@
#!/usr/bin/env perl
if (!$::Driver) { use FindBin; exec("$FindBin::Bin/bootstrap.pl", @ARGV, $0); die; }
# DESCRIPTION: Verilator: Verilog Test driver/expect definition
#
# Copyright 2023 by Wilson Snyder. This program is free software; you
# can redistribute it and/or modify it under the terms of either the GNU
# Lesser General Public License Version 3 or the Perl Artistic License
# Version 2.0.
# SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
scenarios(vlt => 1);
lint(
verilator_flags2 => ["--no-unlimited-stack"],
);
ok(1);
1;

View File

@ -0,0 +1,18 @@
// DESCRIPTION: Verilator: Verilog Test module
//
// This file ONLY is placed under the Creative Commons Public Domain, for
// any use, without warranty, 2024 by Antmicro.
// SPDX-License-Identifier: CC0-1.0
class cls;
task t; t; endtask
task pre_randomize;
t;
endtask
endclass
module t;
cls obj;
task static t;
int _ = obj.randomize() with {1 == 1;};
endtask
endmodule