Support IEEE-compliant intra-assign delays (#3711) (#5441)

This commit is contained in:
Krzysztof Bieganski 2024-09-07 00:13:52 +02:00 committed by GitHub
parent d6923c8571
commit afb8428db4
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
14 changed files with 561 additions and 11334 deletions

View File

@ -3611,6 +3611,7 @@ public:
}); });
} }
bool brokeLhsMustBeLvalue() const override { return true; } bool brokeLhsMustBeLvalue() const override { return true; }
AstDelay* getLhsNetDelay() const;
AstAlways* convertToAlways(); AstAlways* convertToAlways();
}; };

View File

@ -2899,6 +2899,15 @@ void AstCUse::dumpJson(std::ostream& str) const {
dumpJsonGen(str); dumpJsonGen(str);
} }
static AstDelay* getLhsNetDelayRecurse(const AstNodeExpr* const nodep) {
if (const AstNodeVarRef* const refp = VN_CAST(nodep, NodeVarRef)) {
if (refp->varp()->delayp()) return refp->varp()->delayp();
} else if (const AstNodeSel* const selp = VN_CAST(nodep, NodeSel)) {
return getLhsNetDelayRecurse(selp->fromp());
}
return nullptr;
}
AstDelay* AstAssignW::getLhsNetDelay() const { return getLhsNetDelayRecurse(lhsp()); }
AstAlways* AstAssignW::convertToAlways() { AstAlways* AstAssignW::convertToAlways() {
const bool hasTimingControl = isTimingControl(); const bool hasTimingControl = isTimingControl();
AstNodeExpr* const lhs1p = lhsp()->unlinkFrBack(); AstNodeExpr* const lhs1p = lhsp()->unlinkFrBack();

View File

@ -226,8 +226,7 @@ AstVar* V3ParseGrammar::createVariable(FileLine* fileline, const string& name,
nodep->ansi(m_pinAnsi); nodep->ansi(m_pinAnsi);
nodep->declTyped(m_varDeclTyped); nodep->declTyped(m_varDeclTyped);
nodep->lifetime(m_varLifetime); nodep->lifetime(m_varLifetime);
nodep->delayp(m_netDelayp); nodep->delayp(getNetDelay());
m_netDelayp = nullptr;
if (GRAMMARP->m_varDecl != VVarType::UNKNOWN) nodep->combineType(GRAMMARP->m_varDecl); if (GRAMMARP->m_varDecl != VVarType::UNKNOWN) nodep->combineType(GRAMMARP->m_varDecl);
if (GRAMMARP->m_varIO != VDirection::NONE) { if (GRAMMARP->m_varIO != VDirection::NONE) {
nodep->declDirection(GRAMMARP->m_varIO); nodep->declDirection(GRAMMARP->m_varIO);

View File

@ -300,14 +300,8 @@ void transformForks(AstNetlist* const netlistp) {
// If not a fork..join, copy. All write refs should've been handled by V3Fork // If not a fork..join, copy. All write refs should've been handled by V3Fork
bool passByValue = !m_forkp->joinType().join(); bool passByValue = !m_forkp->joinType().join();
if (!varp->isFuncLocal()) { if (!varp->isFuncLocal()) {
if (VString::startsWith(varp->name(), "__Vintra")) { // Not func local. Its lifetime is longer than the forked process. Skip
// Pass it by value to the new function, as otherwise there are issues with return;
// -flocalize (see t_timing_intra_assign)
passByValue = true;
} else {
// Not func local. Its lifetime is longer than the forked process. Skip
return;
}
} else if (!varp->user1()) { } else if (!varp->user1()) {
// Not declared before the fork. It cannot outlive the forked process // Not declared before the fork. It cannot outlive the forked process
return; return;

View File

@ -445,6 +445,8 @@ class TimingControlVisitor final : public VNVisitor {
// NODE STATE // NODE STATE
// Ast{Always,NodeCCall,Fork,NodeAssign}::user1() -> bool. Set true if the node has // Ast{Always,NodeCCall,Fork,NodeAssign}::user1() -> bool. Set true if the node has
// been processed. // been processed.
// AstAssignW::user1() -> bool. Set true if the assignment
// represents the net delay
// AstSenTree::user1() -> AstVarScope*. Trigger scheduler assigned // AstSenTree::user1() -> AstVarScope*. Trigger scheduler assigned
// to this sentree // to this sentree
// Ast{NodeProcedure,CFunc,Begin}::user2() -> bool. Set true if process/task // Ast{NodeProcedure,CFunc,Begin}::user2() -> bool. Set true if process/task
@ -471,7 +473,9 @@ class TimingControlVisitor final : public VNVisitor {
// Unique names // Unique names
V3UniqueNames m_dlyforkNames{"__Vdlyfork"}; // Names for temp AssignW vars V3UniqueNames m_dlyforkNames{"__Vdlyfork"}; // Names for temp AssignW vars
V3UniqueNames m_contAssignVarNames{"__VassignWtmp"}; // Names for temp AssignW vars V3UniqueNames m_contAsgnTmpNames{"__VassignWtmp"}; // Names for temp AssignW vars
V3UniqueNames m_contAsgnGenNames{"__VassignWgen"}; // Continuous assign generation name
// generator
V3UniqueNames m_intraValueNames{"__Vintraval"}; // Intra assign delay value var names V3UniqueNames m_intraValueNames{"__Vintraval"}; // Intra assign delay value var names
V3UniqueNames m_intraIndexNames{"__Vintraidx"}; // Intra assign delay index var names V3UniqueNames m_intraIndexNames{"__Vintraidx"}; // Intra assign delay index var names
V3UniqueNames m_intraLsbNames{"__Vintralsb"}; // Intra assign delay LSB var names V3UniqueNames m_intraLsbNames{"__Vintralsb"}; // Intra assign delay LSB var names
@ -493,47 +497,24 @@ class TimingControlVisitor final : public VNVisitor {
SenExprBuilder* m_senExprBuilderp = nullptr; // Sens expression builder for current m_scope SenExprBuilder* m_senExprBuilderp = nullptr; // Sens expression builder for current m_scope
// METHODS // METHODS
// Find net delay on the LHS of an assignment
AstDelay* getLhsNetDelayRecurse(const AstNodeExpr* const nodep) const {
if (const AstNodeVarRef* const refp = VN_CAST(nodep, NodeVarRef)) {
if (refp->varp()->delayp()) return refp->varp()->delayp()->unlinkFrBack();
} else if (const AstSel* const selp = VN_CAST(nodep, Sel)) {
return getLhsNetDelayRecurse(selp->fromp());
}
return nullptr;
}
// Transform an assignment with an intra timing control into a timing control with the // Transform an assignment with an intra timing control into a timing control with the
// assignment under it // assignment under it
AstNode* factorOutTimingControl(AstNodeAssign* nodep) const { AstNode* factorOutTimingControl(AstNodeAssign* nodep) const {
AstNode* stmtp = nodep; AstNode* stmtp = nodep;
AstDelay* delayp = getLhsNetDelayRecurse(nodep->lhsp());
FileLine* const flp = nodep->fileline();
AstNode* const controlp = nodep->timingControlp(); AstNode* const controlp = nodep->timingControlp();
if (controlp) { if (AstDelay* const delayp = VN_CAST(controlp, Delay)) {
controlp->unlinkFrBack(); stmtp->replaceWith(delayp->unlinkFrBack());
if (auto* const assignDelayp = VN_CAST(controlp, Delay)) {
if (delayp) {
delayp->lhsp(new AstAdd{flp, delayp->lhsp()->unlinkFrBack(),
assignDelayp->lhsp()->unlinkFrBack()});
VL_DO_DANGLING(assignDelayp->deleteTree(), nodep);
} else {
delayp = assignDelayp;
}
}
}
if (delayp) {
stmtp->replaceWith(delayp);
delayp->addStmtsp(stmtp); delayp->addStmtsp(stmtp);
stmtp = delayp; stmtp = delayp;
} } else if (AstSenTree* const sensesp = VN_CAST(controlp, SenTree)) {
if (auto* const sensesp = VN_CAST(controlp, SenTree)) { AstEventControl* const eventControlp
auto* const eventControlp = new AstEventControl{flp, sensesp, nullptr}; = new AstEventControl{sensesp->fileline(), sensesp->unlinkFrBack(), nullptr};
stmtp->replaceWith(eventControlp); stmtp->replaceWith(eventControlp);
eventControlp->addStmtsp(stmtp); eventControlp->addStmtsp(stmtp);
stmtp = eventControlp; stmtp = eventControlp;
} else if (auto* const beginp = VN_CAST(controlp, Begin)) { } else if (AstBegin* const beginp = VN_CAST(controlp, Begin)) {
// Begin from V3AssertPre // Begin from V3AssertPre
stmtp->replaceWith(beginp); stmtp->replaceWith(beginp->unlinkFrBack());
beginp->addStmtsp(stmtp); beginp->addStmtsp(stmtp);
stmtp = beginp; stmtp = beginp;
} }
@ -1062,58 +1043,106 @@ class TimingControlVisitor final : public VNVisitor {
controlp->addHereThisAsNext( controlp->addHereThisAsNext(
new AstAssign{flp, new AstVarRef{flp, newvscp, VAccess::WRITE}, valuep}); new AstAssign{flp, new AstVarRef{flp, newvscp, VAccess::WRITE}, valuep});
}; };
// Create the intermediate select vars. Note: because 'foreach' proceeds in // NBAs with delays evaluate LHS indices immediately
// pre-order, and we replace indices in selects with variables, we cannot if (inAssignDly) {
// reach another select under the index position. This is exactly what // Create the intermediate select vars. Note: because 'foreach' proceeds in pre-order,
// we want as only the top level selects are LValues. As an example, // and we replace indices in selects with variables, we cannot reach another select
// this transforms 'x[a[i]][b[j]] = y' // under the index position. This is exactly what we want as only the top level selects
// into 't1 = a[i]; t0 = b[j]; x[t1][t0] = y'. // are LValues. As an example, this transforms 'x[a[i]][b[j]] = y' into 't1 = a[i]; t0
nodep->lhsp()->foreach([&](AstSel* selp) { // = b[j]; x[t1][t0] = y'.
if (VN_IS(selp->lsbp(), Const)) return; nodep->lhsp()->foreach([&](AstSel* selp) {
replaceWithIntermediate(selp->lsbp(), m_intraLsbNames.get(nodep)); if (VN_IS(selp->lsbp(), Const)) return;
// widthp should be const replaceWithIntermediate(selp->lsbp(), m_intraLsbNames.get(nodep));
}); // widthp should be const
nodep->lhsp()->foreach([&](AstNodeSel* selp) { });
if (VN_IS(selp->bitp(), Const)) return; nodep->lhsp()->foreach([&](AstNodeSel* selp) {
replaceWithIntermediate(selp->bitp(), m_intraIndexNames.get(nodep)); if (VN_IS(selp->bitp(), Const)) return;
}); replaceWithIntermediate(selp->bitp(), m_intraIndexNames.get(nodep));
});
}
// Replace the RHS with an intermediate value var // Replace the RHS with an intermediate value var
replaceWithIntermediate(nodep->rhsp(), m_intraValueNames.get(nodep)); replaceWithIntermediate(nodep->rhsp(), m_intraValueNames.get(nodep));
} }
void visit(AstAssignW* nodep) override { void visit(AstAssignW* nodep) override {
AstDelay* const netDelayp = getLhsNetDelayRecurse(nodep->lhsp());
if (!netDelayp && !nodep->timingControlp()) return;
// This assignment will be converted to an always. In some cases this may generate an
// UNOPTFLAT, e.g.: assign #1 clk = ~clk. We create a temp var for the LHS of this
// assign, to disable the UNOPTFLAT warning for it.
// TODO: Find a way to do this without introducing this var. Perhaps make
// V3SchedAcyclic recognize awaits and prevent it from treating this kind of logic as
// cyclic
AstNodeExpr* const lhsp = nodep->lhsp()->unlinkFrBack();
std::string varname;
if (auto* const refp = VN_CAST(lhsp, VarRef)) {
varname = m_contAssignVarNames.get(refp->name());
} else {
varname = m_contAssignVarNames.get(lhsp);
}
auto* const tempvscp = m_scopep->createTemp(varname, lhsp->dtypep());
tempvscp->varp()->delayp(netDelayp);
FileLine* const flp = nodep->fileline(); FileLine* const flp = nodep->fileline();
// Get the net delay unless this assignment was created for handling the net delay (user1)
AstDelay* const netDelayp = nodep->user1() ? nullptr : nodep->getLhsNetDelay();
if (netDelayp) {
if (nodep->timingControlp()) {
// If this assignment has a delay, create another one to handle the net delay
AstVarScope* const newvscp
= createTemp(flp, m_contAsgnTmpNames.get(nodep), nodep->dtypep());
AstAssignW* assignp = new AstAssignW{
nodep->fileline(), nodep->lhsp()->unlinkFrBack(),
new AstVarRef{flp, newvscp, VAccess::READ}, netDelayp->cloneTree(false)};
assignp->user1(true);
nodep->addNextHere(assignp);
nodep->lhsp(new AstVarRef{flp, newvscp, VAccess::WRITE});
} else {
// Else just use this one with the net delay
nodep->timingControlp(netDelayp->cloneTree(false));
}
}
if (!nodep->timingControlp()) return;
// There will be some circular logic here, suppress the warning for newly created vars
flp->modifyWarnOff(V3ErrorCode::UNOPTFLAT, true); flp->modifyWarnOff(V3ErrorCode::UNOPTFLAT, true);
tempvscp->fileline(flp); // Also suppress the warning for the LHS var, for cases like `assign #1 clk = ~clk;`
tempvscp->varp()->fileline(flp); // TODO: Restore the warning for other, non-delayed drivers
// Remap the LHS to the new temp var nodep->lhsp()->foreach([](AstVarRef* refp) {
nodep->lhsp(new AstVarRef{flp, tempvscp, VAccess::WRITE}); if (refp->access().isWriteOrRW()) {
refp->varp()->fileline()->modifyWarnOff(V3ErrorCode::UNOPTFLAT, true);
}
});
// Convert it to an always; the new assign with intra delay will be handled by // Convert it to an always; the new assign with intra delay will be handled by
// visit(AstNodeAssign*) // visit(AstNodeAssign*)
AstAlways* const alwaysp = nodep->convertToAlways(); AstAlways* const alwaysp = nodep->convertToAlways();
visit(alwaysp); visit(alwaysp); // Visit now as we need to do some post-processing
// Put the LHS back in the AssignW; put the temp var on the RHS VL_DO_DANGLING(nodep->deleteTree(), nodep);
nodep->lhsp(lhsp); // IEEE 1800-2023 10.3.3 - if the RHS value differs from the currently scheduled value to
nodep->rhsp(new AstVarRef{flp, tempvscp, VAccess::READ}); // be assigned, the currently scheduled assignment is descheduled. To keep track if an
// Put the AssignW right after the always. Different order can produce UNOPTFLAT on the LHS // assignment should be descheduled, each scheduled assignment event has a 'generation',
// var // and if at assignment time its generation differs from the current generation, it won't
alwaysp->addNextHere(nodep); // be performed
AstFork* const forkp = VN_AS(alwaysp->stmtsp(), Fork);
UASSERT_OBJ(forkp, alwaysp, "Fork should be there from convertToAlways()");
AstBegin* const beginp = VN_AS(forkp->stmtsp(), Begin);
UASSERT_OBJ(beginp, alwaysp, "Begin should be there from convertToAlways()");
AstAssign* const preAssignp = VN_AS(beginp->stmtsp(), Assign);
UASSERT_OBJ(preAssignp, alwaysp, "Pre-assign should be there from convertToAlways()");
AstAssign* const postAssignp = VN_AS(preAssignp->nextp()->nextp(), Assign);
UASSERT_OBJ(postAssignp, alwaysp, "Post-assign should be there from convertToAlways()");
// Increment generation and copy it to a local
AstVarScope* const generationVarp
= createTemp(flp, m_contAsgnGenNames.get(alwaysp), alwaysp->findUInt64DType());
AstVarScope* const genLocalVarp
= createTemp(flp, generationVarp->varp()->name() + "__local",
alwaysp->findUInt64DType(), preAssignp);
preAssignp->addHereThisAsNext(
new AstAssign{flp, new AstVarRef{flp, generationVarp, VAccess::WRITE},
new AstAdd{flp, new AstVarRef{flp, generationVarp, VAccess::READ},
new AstConst{flp, 1}}});
preAssignp->addHereThisAsNext(
new AstAssign{flp, new AstVarRef{flp, genLocalVarp, VAccess::WRITE},
new AstVarRef{flp, generationVarp, VAccess::READ}});
// If the current generation is same as the one saved in the local var, assign
beginp->addStmtsp(
new AstIf{flp,
new AstEq{flp, new AstVarRef{flp, generationVarp, VAccess::READ},
new AstVarRef{flp, genLocalVarp, VAccess::READ}},
postAssignp->unlinkFrBack()});
// Save scheduled RHS value before delay
AstVarScope* const tmpVarp
= createTemp(flp, m_contAsgnTmpNames.get(alwaysp), preAssignp->rhsp()->dtypep());
AstVarRef* const tmpAssignRhsp = VN_AS(preAssignp->lhsp(), VarRef)->cloneTree(false);
tmpAssignRhsp->access(VAccess::WRITE);
preAssignp->addNextHere(
new AstAssign{flp, new AstVarRef{flp, tmpVarp, VAccess::WRITE}, tmpAssignRhsp});
// If the RHS is different from the currently scheduled value, schedule the new assignment
// The generation will increase, effectively 'descheduling' the previous assignment.
alwaysp->addStmtsp(new AstIf{flp,
new AstNeq{flp, preAssignp->rhsp()->cloneTree(false),
new AstVarRef{flp, tmpVarp, VAccess::READ}},
forkp->unlinkFrBack()});
} }
void visit(AstDisableFork* nodep) override { void visit(AstDisableFork* nodep) override {
if (hasFlags(m_procp, T_HAS_PROC)) return; if (hasFlags(m_procp, T_HAS_PROC)) return;

View File

@ -1247,7 +1247,10 @@ class TristateVisitor final : public TristateBaseVisitor {
VL_RESTORER(m_alhs); VL_RESTORER(m_alhs);
VL_RESTORER(m_currentStrength); VL_RESTORER(m_currentStrength);
if (m_graphing) { if (m_graphing) {
if (AstAssignW* assignWp = VN_CAST(nodep, AssignW)) addToAssignmentList(assignWp); if (AstAssignW* assignWp = VN_CAST(nodep, AssignW)) {
if (assignWp->timingControlp() || assignWp->getLhsNetDelay()) return;
addToAssignmentList(assignWp);
}
if (nodep->user2() & U2_GRAPHING) return; if (nodep->user2() & U2_GRAPHING) return;
VL_RESTORER(m_logicp); VL_RESTORER(m_logicp);

View File

@ -83,7 +83,8 @@ public:
AstCase* m_caseAttrp = nullptr; // Current case statement for attribute adding AstCase* m_caseAttrp = nullptr; // Current case statement for attribute adding
AstNodeDType* m_varDTypep = nullptr; // Pointer to data type for next signal declaration AstNodeDType* m_varDTypep = nullptr; // Pointer to data type for next signal declaration
AstNodeDType* m_memDTypep = nullptr; // Pointer to data type for next member declaration AstNodeDType* m_memDTypep = nullptr; // Pointer to data type for next member declaration
AstDelay* m_netDelayp = nullptr; // Pointer to delay for next signal declaration std::unique_ptr<AstDelay> m_netDelayp = nullptr; // Pointer to delay for next signal
// declaration
AstStrengthSpec* m_netStrengthp = nullptr; // Pointer to strength for next net declaration AstStrengthSpec* m_netStrengthp = nullptr; // Pointer to strength for next net declaration
FileLine* m_instModuleFl = nullptr; // Fileline of module referenced for instantiations FileLine* m_instModuleFl = nullptr; // Fileline of module referenced for instantiations
AstPin* m_instParamp = nullptr; // Parameters for instantiations AstPin* m_instParamp = nullptr; // Parameters for instantiations
@ -211,7 +212,8 @@ public:
if (m_varDTypep) VL_DO_CLEAR(m_varDTypep->deleteTree(), m_varDTypep = nullptr); if (m_varDTypep) VL_DO_CLEAR(m_varDTypep->deleteTree(), m_varDTypep = nullptr);
m_varDTypep = dtypep; m_varDTypep = dtypep;
} }
void setNetDelay(AstDelay* netDelayp) { m_netDelayp = netDelayp; } void setNetDelay(AstDelay* netDelayp) { m_netDelayp.reset(netDelayp); }
AstDelay* getNetDelay() { return m_netDelayp.release(); }
void setNetStrength(AstStrengthSpec* netStrengthp) { m_netStrengthp = netStrengthp; } void setNetStrength(AstStrengthSpec* netStrengthp) { m_netStrengthp = netStrengthp; }
void pinPush() { void pinPush() {
m_pinStack.push(m_pinNum); m_pinStack.push(m_pinNum);
@ -3045,8 +3047,10 @@ netSig<varp>: // IEEE: net_decl_assignment - one element from
netId sigAttrListE netId sigAttrListE
{ $$ = VARDONEA($<fl>1, *$1, nullptr, $2); } { $$ = VARDONEA($<fl>1, *$1, nullptr, $2); }
| netId sigAttrListE '=' expr | netId sigAttrListE '=' expr
{ $$ = VARDONEA($<fl>1, *$1, nullptr, $2); { AstDelay* const delayp = GRAMMARP->getNetDelay();
auto* const assignp = new AstAssignW{$3, new AstParseRef{$<fl>1, VParseRefExp::PX_TEXT, *$1}, $4}; AstAssignW* const assignp = new AstAssignW{$3, new AstParseRef{$<fl>1, VParseRefExp::PX_TEXT, *$1}, $4, delayp};
$$ = VARDONEA($<fl>1, *$1, nullptr, $2);
if (delayp) GRAMMARP->setNetDelay(delayp->cloneTree(false));
if (GRAMMARP->m_netStrengthp) assignp->strengthSpecp(GRAMMARP->m_netStrengthp->cloneTree(false)); if (GRAMMARP->m_netStrengthp) assignp->strengthSpecp(GRAMMARP->m_netStrengthp->cloneTree(false));
AstNode::addNext<AstNode, AstNode>($$, assignp); } AstNode::addNext<AstNode, AstNode>($$, assignp); }
| netId variable_dimensionList sigAttrListE | netId variable_dimensionList sigAttrListE

View File

@ -1,15 +1,35 @@
%Warning-STMTDLY: t/t_net_delay.v:16:14: Ignoring delay on this statement due to --no-timing %Warning-STMTDLY: t/t_net_delay.v:14:11: Ignoring delay on this statement due to --no-timing
: ... note: In instance 't' : ... note: In instance 't'
16 | wire[3:0] #4 val1 = half_cyc; 14 | always #2 clk = ~clk;
| ^ | ^
... For warning description see https://verilator.org/warn/STMTDLY?v=latest ... For warning description see https://verilator.org/warn/STMTDLY?v=latest
... Use "/* verilator lint_off STMTDLY */" and lint_on around source to disable this message. ... Use "/* verilator lint_off STMTDLY */" and lint_on around source to disable this message.
%Warning-STMTDLY: t/t_net_delay.v:17:14: Ignoring delay on this statement due to --no-timing %Warning-STMTDLY: t/t_net_delay.v:20:14: Ignoring delay on this statement due to --no-timing
: ... note: In instance 't' : ... note: In instance 't'
17 | wire[3:0] #4 val2; 20 | wire[3:0] #3 val1;
| ^ | ^
%Warning-ASSIGNDLY: t/t_net_delay.v:20:11: Ignoring timing control on this assignment/primitive due to --no-timing %Warning-STMTDLY: t/t_net_delay.v:21:14: Ignoring delay on this statement due to --no-timing
: ... note: In instance 't'
21 | wire[3:0] #3 val2;
| ^
%Warning-ASSIGNDLY: t/t_net_delay.v:22:14: Ignoring timing control on this assignment/primitive due to --no-timing
: ... note: In instance 't' : ... note: In instance 't'
20 | assign #4 val2 = half_cyc; 22 | wire[3:0] #5 val3 = cyc;
| ^
%Warning-STMTDLY: t/t_net_delay.v:23:14: Ignoring delay on this statement due to --no-timing
: ... note: In instance 't'
23 | wire[3:0] #5 val4;
| ^
%Warning-ASSIGNDLY: t/t_net_delay.v:24:14: Ignoring timing control on this assignment/primitive due to --no-timing
: ... note: In instance 't'
24 | wire[3:0] #3 val5 = x, val6 = cyc;
| ^
%Warning-ASSIGNDLY: t/t_net_delay.v:27:11: Ignoring timing control on this assignment/primitive due to --no-timing
: ... note: In instance 't'
27 | assign #3 val2 = cyc;
| ^ | ^
%Warning-STMTDLY: t/t_net_delay.v:39:26: Ignoring delay on this statement due to --no-timing
: ... note: In instance 't'
39 | always @(posedge clk) #1 begin
| ^
%Error: Exiting due to %Error: Exiting due to

View File

@ -7,30 +7,47 @@
`define stop $stop `define stop $stop
`define checkh(gotv,expv) do if ((gotv) !== (expv)) begin $write("%%Error: %s:%0d: got='h%x exp='h%x\n", `__FILE__,`__LINE__, (gotv), (expv)); `stop; end while(0) `define checkh(gotv,expv) do if ((gotv) !== (expv)) begin $write("%%Error: %s:%0d: got='h%x exp='h%x\n", `__FILE__,`__LINE__, (gotv), (expv)); `stop; end while(0)
module t (/*AUTOARG*/ module t;
// Inputs // verilator lint_off UNOPTFLAT
clk logic clk = 0;
); // verilator lint_on UNOPTFLAT
input clk; always #2 clk = ~clk;
wire[3:0] #4 val1 = half_cyc; // verilator lint_off UNDRIVEN
wire[3:0] #4 val2; wire[3:0] x;
reg[3:0] half_cyc = 0; // verilator lint_on UNDRIVEN
reg[3:0] cyc = 0;
wire[3:0] #3 val1;
wire[3:0] #3 val2;
wire[3:0] #5 val3 = cyc;
wire[3:0] #5 val4;
wire[3:0] #3 val5 = x, val6 = cyc;
assign #4 val2 = half_cyc; assign val1 = cyc;
assign #3 val2 = cyc;
assign val4 = cyc;
assign val5 = cyc;
always @(clk) begin always @(posedge clk) begin
if ($time > 0) half_cyc <= half_cyc + 1; if ($time > 0) cyc <= cyc + 1;
`ifdef TEST_VERBOSE if (cyc == 15) begin
$strobe("[%0t] half_cyc=%0d, val1=%0d, val2=%0d", $time, half_cyc, val1, val2);
`endif
if (half_cyc >= 7) begin
`checkh(val1, half_cyc - 3);
`checkh(val2, half_cyc - 7);
end
if (half_cyc == 15) begin
$write("*-* All Finished *-*\n"); $write("*-* All Finished *-*\n");
$finish; $finish;
end end
end end
always @(posedge clk) #1 begin
`ifdef TEST_VERBOSE
$display("[%0t] cyc=%0d val1=%0d val2=%0d val3=%0d val4=%0d val5=%0d val6=%0d",
$time, cyc, val1, val2, val3, val4, val5, val6);
`endif
if (cyc >= 3) begin
`checkh(val1, cyc - 1);
`checkh(val2, cyc - 2);
`checkh(val3, 0);
`checkh(val4, 0);
`checkh(val5, cyc);
`checkh(val6, cyc - 1);
end
end
endmodule endmodule

File diff suppressed because it is too large Load Diff

View File

@ -1,15 +1,25 @@
val[0]=0 val[1]=0 val[2]=0 [0] val[0]=0 val[1]=0 val[2]=0 net[0]=0 net[1]=0
val[0]=1 val[1]=0 val[2]=0 [10] val[0]=1 val[1]=0 val[2]=0 net[0]=0 net[1]=15
val[0]=2 val[1]=0 val[2]=15 [14] val[0]=1 val[1]=1 val[2]=0 net[0]=0 net[1]=15
val[0]=3 val[1]=0 val[2]=15 [16] val[0]=1 val[1]=1 val[2]=1 net[0]=0 net[1]=15
val[0]=4 val[1]=1 val[2]=14 [20] val[0]=2 val[1]=1 val[2]=1 net[0]=1 net[1]=14
val[0]=5 val[1]=1 val[2]=14 [24] val[0]=2 val[1]=2 val[2]=1 net[0]=1 net[1]=14
val[0]=6 val[1]=2 val[2]=13 [25] val[0]=3 val[1]=2 val[2]=1 net[0]=1 net[1]=14
val[0]=7 val[1]=4 val[2]=11 [29] val[0]=3 val[1]=3 val[2]=1 net[0]=2 net[1]=13
val[0]=8 val[1]=7 val[2]=8 [30] val[0]=4 val[1]=3 val[2]=1 net[0]=2 net[1]=13
val[0]=9 val[1]=7 val[2]=8 [34] val[0]=4 val[1]=4 val[2]=1 net[0]=3 net[1]=12
val[0]=10 val[1]=7 val[2]=8 [35] val[0]=5 val[1]=4 val[2]=1 net[0]=3 net[1]=12
val[0]=11 val[1]=7 val[2]=8 [39] val[0]=5 val[1]=5 val[2]=1 net[0]=4 net[1]=11
val[0]=12 val[1]=7 val[2]=8 [40] val[0]=6 val[1]=5 val[2]=1 net[0]=4 net[1]=11
val[0]=13 val[1]=7 val[2]=8 [44] val[0]=6 val[1]=6 val[2]=1 net[0]=5 net[1]=10
[46] val[0]=6 val[1]=6 val[2]=6 net[0]=5 net[1]=10
[50] val[0]=7 val[1]=6 val[2]=6 net[0]=6 net[1]=9
[54] val[0]=7 val[1]=7 val[2]=6 net[0]=6 net[1]=9
[56] val[0]=7 val[1]=7 val[2]=7 net[0]=6 net[1]=9
[75] val[0]=8 val[1]=7 val[2]=7 net[0]=7 net[1]=8
[76] val[0]=9 val[1]=7 val[2]=7 net[0]=7 net[1]=8
[78] val[0]=10 val[1]=7 val[2]=7 net[0]=7 net[1]=8
[79] val[0]=11 val[1]=7 val[2]=7 net[0]=7 net[1]=8
[80] val[0]=12 val[1]=7 val[2]=7 net[0]=7 net[1]=8
[82] val[0]=13 val[1]=7 val[2]=7 net[0]=7 net[1]=8
*-* All Finished *-* *-* All Finished *-*

View File

@ -11,7 +11,7 @@ if (!$::Driver) { use FindBin; exec("$FindBin::Bin/bootstrap.pl", @ARGV, $0); di
scenarios(simulator => 1); scenarios(simulator => 1);
compile( compile(
verilator_flags2 => ["--exe --main --timing -Wno-UNOPTFLAT"], verilator_flags2 => ["--binary"],
); );
execute( execute(
@ -19,7 +19,7 @@ execute(
); );
compile( compile(
verilator_flags2 => ["--exe --main --timing -Wno-UNOPTFLAT -fno-localize"], verilator_flags2 => ["--binary -fno-localize"],
); );
execute( execute(

View File

@ -6,28 +6,35 @@
module t; module t;
logic[3:0] val[3]; logic[3:0] val[3];
wire[3:0] #5 net[2];
logic[1:0] idx1 = 0; logic[1:0] idx1 = 0;
logic[1:0] idx2 = 0; logic[1:0] idx2 = 0;
logic[0:0] idx3 = 0; logic[0:0] idx3 = 0;
event e; event e;
always @val[0] $write("val[0]=%0d val[1]=%0d val[2]=%0d\n", val[0], val[1], val[2]); always @val $write("[%0t] val[0]=%0d val[1]=%0d val[2]=%0d net[0]=%0d net[1]=%0d\n",
$time, val[0], val[1], val[2], net[0], net[1]);
assign #10 {val[1], val[2]} = {val[0], 4'hf-val[0]}; assign {net[0], net[1]} = {val[1], 4'hf-val[1]};
assign #4 val[1] = val[0];
assign #6 val[2] = val[0];
always #10 begin // always so we can use NBA always #10 begin // always so we can use NBA
val[0] = 1; val[0] = 1;
#10 val[0] = 2; #10 val[0] = 2;
fork #5 val[0] = 3; join_none fork #5 val[0] = 3; join_none
val[0] = #10 val[0] + 2; val[0] = #10 val[0] + 2;
val[idx1] <= #10 val[idx1] + 2; val[0] <= #10 val[idx1] + 2;
fork begin #5 fork begin #5
val[0] = 5; val[0] = 5;
idx1 = 2; idx1 = 0;
idx2 = 3; idx2 = 0;
idx3 = 1; idx3 = 0;
#40 ->e; #40 ->e;
end join_none end join_none
idx1 = 2;
idx2 = 3;
idx3 = 1;
val[idx1][idx2[idx3+:2]] = #20 1; val[idx1][idx2[idx3+:2]] = #20 1;
@e val[0] = 8; @e val[0] = 8;
fork begin fork begin

File diff suppressed because it is too large Load Diff