mirror of
https://github.com/verilator/verilator.git
synced 2025-04-06 04:32: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<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();
|
||||||
|
@ -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'
|
||||||
|
@ -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
Loading…
Reference in New Issue
Block a user