Fix multiple edge timing controls in class methods (#4318) (#4320) (#4344)

Multiple edge timing controls in class methods would cause compilation errors on
the generated C++ code. This is because the `SenExprBuilder` used for these
would get recreated per timing control, resulting in duplicate variable names.
The fix is to have a single `SenExprBuilder` per scope.
This commit is contained in:
Krzysztof Bieganski 2023-07-07 14:19:49 +02:00 committed by GitHub
parent da043ca16d
commit d5d1cc47e0
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 735 additions and 112 deletions

View File

@ -232,7 +232,13 @@ public:
} }
std::vector<AstNodeStmt*> getAndClearInits() { return std::move(m_inits); } std::vector<AstNodeStmt*> getAndClearInits() { return std::move(m_inits); }
std::vector<AstVar*> getAndClearLocals() { return std::move(m_locals); }
std::vector<AstVar*> getAndClearLocals() {
// With m_locals empty, m_prev and m_curr are no longer valid
m_prev.clear();
m_curr.clear();
return std::move(m_locals);
}
std::vector<AstNodeStmt*> getAndClearPreUpdates() { std::vector<AstNodeStmt*> getAndClearPreUpdates() {
m_hasPreUpdate.clear(); m_hasPreUpdate.clear();

View File

@ -299,6 +299,7 @@ private:
// Other // Other
SenTreeFinder m_finder{m_netlistp}; // Sentree finder and uniquifier SenTreeFinder m_finder{m_netlistp}; // Sentree finder and uniquifier
SenExprBuilder* m_senExprBuilderp = nullptr; // Sens expression builder for current m_scope
// METHODS // METHODS
// Find net delay on the LHS of an assignment // Find net delay on the LHS of an assignment
@ -571,7 +572,12 @@ private:
void visit(AstScope* nodep) override { void visit(AstScope* nodep) override {
VL_RESTORER(m_scopep); VL_RESTORER(m_scopep);
m_scopep = nodep; m_scopep = nodep;
iterateChildren(nodep); SenExprBuilder senExprBuilder{m_scopep};
{ // Restore m_senExprBuilderp before destroying senExprBuilder
VL_RESTORER(m_senExprBuilderp);
m_senExprBuilderp = &senExprBuilder;
iterateChildren(nodep);
}
} }
void visit(AstActive* nodep) override { void visit(AstActive* nodep) override {
m_activep = nodep; m_activep = nodep;
@ -727,14 +733,14 @@ private:
= new AstCAwait{flp, evalMethodp, getCreateDynamicTriggerSenTree()}; = new AstCAwait{flp, evalMethodp, getCreateDynamicTriggerSenTree()};
awaitEvalp->dtypeSetVoid(); awaitEvalp->dtypeSetVoid();
// Construct the sen expression for this sentree // Construct the sen expression for this sentree
SenExprBuilder senExprBuilder{m_scopep}; UASSERT_OBJ(m_senExprBuilderp, nodep, "No SenExprBuilder for this scope");
auto* const assignp = new AstAssign{flp, new AstVarRef{flp, trigvscp, VAccess::WRITE}, auto* const assignp = new AstAssign{flp, new AstVarRef{flp, trigvscp, VAccess::WRITE},
senExprBuilder.build(sensesp).first}; m_senExprBuilderp->build(sensesp).first};
// Put all the locals and inits before the trigger eval loop // Put all the locals and inits before the trigger eval loop
for (AstVar* const varp : senExprBuilder.getAndClearLocals()) { for (AstVar* const varp : m_senExprBuilderp->getAndClearLocals()) {
nodep->addHereThisAsNext(varp); nodep->addHereThisAsNext(varp);
} }
for (AstNodeStmt* const stmtp : senExprBuilder.getAndClearInits()) { for (AstNodeStmt* const stmtp : m_senExprBuilderp->getAndClearInits()) {
nodep->addHereThisAsNext(stmtp); nodep->addHereThisAsNext(stmtp);
} }
// Create the trigger eval loop, which will await the evaluation step and check the // Create the trigger eval loop, which will await the evaluation step and check the
@ -743,7 +749,7 @@ private:
flp, new AstLogNot{flp, new AstVarRef{flp, trigvscp, VAccess::READ}}, flp, new AstLogNot{flp, new AstVarRef{flp, trigvscp, VAccess::READ}},
awaitEvalp->makeStmt()}; awaitEvalp->makeStmt()};
// Put pre updates before the trigger check and assignment // Put pre updates before the trigger check and assignment
for (AstNodeStmt* const stmtp : senExprBuilder.getAndClearPreUpdates()) { for (AstNodeStmt* const stmtp : m_senExprBuilderp->getAndClearPreUpdates()) {
loopp->addStmtsp(stmtp); loopp->addStmtsp(stmtp);
} }
// Then the trigger check and assignment // Then the trigger check and assignment
@ -756,7 +762,7 @@ private:
loopp->addStmtsp(awaitPostUpdatep->makeStmt()); loopp->addStmtsp(awaitPostUpdatep->makeStmt());
} }
// Put the post updates at the end of the loop // Put the post updates at the end of the loop
for (AstNodeStmt* const stmtp : senExprBuilder.getAndClearPostUpdates()) { for (AstNodeStmt* const stmtp : m_senExprBuilderp->getAndClearPostUpdates()) {
loopp->addStmtsp(stmtp); loopp->addStmtsp(stmtp);
} }
// Finally, await the resumption step in 'act' // Finally, await the resumption step in 'act'

View File

@ -80,9 +80,32 @@ module t;
endtask endtask
endclass endclass
class ClkClass;
logic clk;
int count;
function new;
clk = 0;
count = 0;
endfunction
task flip;
clk = ~clk;
endtask;
task count_5;
@(posedge clk) count++;
@(posedge clk) count++;
@(posedge clk) count++;
@(posedge clk) count++;
@(posedge clk) count++;
endtask
endclass
EventClass ec = new; EventClass ec = new;
WaitClass wc = new; WaitClass wc = new;
LocalWaitClass lc = new; LocalWaitClass lc = new;
ClkClass cc = new;
initial begin initial begin
@ec.e; @ec.e;
@ -105,8 +128,13 @@ module t;
`WRITE_VERBOSE(("Event in class triggered at time %0t!\n", $time)); `WRITE_VERBOSE(("Event in class triggered at time %0t!\n", $time));
end end
always #5 cc.flip;
initial cc.count_5;
initial begin initial begin
#80 #80
if (cc.count != 5) $stop;
if (ec.trig_count != 3) $stop; if (ec.trig_count != 3) $stop;
if (!wc.ok) $stop; if (!wc.ok) $stop;
if (!lc.ok) $stop; if (!lc.ok) $stop;

File diff suppressed because it is too large Load Diff