mirror of
https://github.com/verilator/verilator.git
synced 2025-04-05 12:12:39 +00:00
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:
parent
da043ca16d
commit
d5d1cc47e0
@ -232,7 +232,13 @@ public:
|
||||
}
|
||||
|
||||
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() {
|
||||
m_hasPreUpdate.clear();
|
||||
|
@ -299,6 +299,7 @@ private:
|
||||
|
||||
// Other
|
||||
SenTreeFinder m_finder{m_netlistp}; // Sentree finder and uniquifier
|
||||
SenExprBuilder* m_senExprBuilderp = nullptr; // Sens expression builder for current m_scope
|
||||
|
||||
// METHODS
|
||||
// Find net delay on the LHS of an assignment
|
||||
@ -571,7 +572,12 @@ private:
|
||||
void visit(AstScope* nodep) override {
|
||||
VL_RESTORER(m_scopep);
|
||||
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 {
|
||||
m_activep = nodep;
|
||||
@ -727,14 +733,14 @@ private:
|
||||
= new AstCAwait{flp, evalMethodp, getCreateDynamicTriggerSenTree()};
|
||||
awaitEvalp->dtypeSetVoid();
|
||||
// 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},
|
||||
senExprBuilder.build(sensesp).first};
|
||||
m_senExprBuilderp->build(sensesp).first};
|
||||
// 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);
|
||||
}
|
||||
for (AstNodeStmt* const stmtp : senExprBuilder.getAndClearInits()) {
|
||||
for (AstNodeStmt* const stmtp : m_senExprBuilderp->getAndClearInits()) {
|
||||
nodep->addHereThisAsNext(stmtp);
|
||||
}
|
||||
// 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}},
|
||||
awaitEvalp->makeStmt()};
|
||||
// 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);
|
||||
}
|
||||
// Then the trigger check and assignment
|
||||
@ -756,7 +762,7 @@ private:
|
||||
loopp->addStmtsp(awaitPostUpdatep->makeStmt());
|
||||
}
|
||||
// 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);
|
||||
}
|
||||
// Finally, await the resumption step in 'act'
|
||||
|
@ -80,9 +80,32 @@ module t;
|
||||
endtask
|
||||
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;
|
||||
WaitClass wc = new;
|
||||
LocalWaitClass lc = new;
|
||||
ClkClass cc = new;
|
||||
|
||||
initial begin
|
||||
@ec.e;
|
||||
@ -105,8 +128,13 @@ module t;
|
||||
`WRITE_VERBOSE(("Event in class triggered at time %0t!\n", $time));
|
||||
end
|
||||
|
||||
always #5 cc.flip;
|
||||
|
||||
initial cc.count_5;
|
||||
|
||||
initial begin
|
||||
#80
|
||||
if (cc.count != 5) $stop;
|
||||
if (ec.trig_count != 3) $stop;
|
||||
if (!wc.ok) $stop;
|
||||
if (!lc.ok) $stop;
|
||||
|
File diff suppressed because it is too large
Load Diff
Loading…
Reference in New Issue
Block a user