diff --git a/docs/guide/warnings.rst b/docs/guide/warnings.rst index 6e4bb3b4abc..5f67d1313a8 100644 --- a/docs/guide/warnings.rst +++ b/docs/guide/warnings.rst @@ -210,22 +210,33 @@ List Of Warnings .. option:: BLKLOOPINIT - .. TODO better example - - This indicates that the initialization of an array needs to use - non-delayed assignments. This is done in the interest of speed; if - delayed assignments were used, the simulator would have to copy large - arrays every cycle. (In smaller loops, loop unrolling allows the - delayed assignment to work, though it's a bit slower than a non-delayed - assignment.) Here's an example + Indicates certain constructs where non-blocking assignments to unpacked + arrays (memories) are not supported inside loops. These typically appear in + initialization/reset code: .. code-block:: sv always @(posedge clk) if (~reset_l) for (i=0; i<`ARRAY_SIZE; i++) - array[i] = 0; // Non-delayed for verilator + array[i] <= 0; // Non-blocking assignment inside loop + else + array[address] <= data; + + While this is supported in typical synthesizeable code (including the + example above), some complicated cases are not supported. Namely: + + 1. If the above loop is inside a suspendable process or fork statement. + + 2. If the variable is also the target of a '<=' non-blocking assignment + in a suspendable process or fork statement (in addition to a synthesizable + loop). + + 3. If the element type of the array is a compound type. + It might slightly improve run-time performance if you change the + non-blocking assignment inside the loop into a blocking assignment + (that is: use '=' instead of '<='), if possible. This message is only seen on large or complicated loops because Verilator generally unrolls small loops. You may want to try increasing diff --git a/include/verilated_types.h b/include/verilated_types.h index 21371bd47ff..c2773f357ec 100644 --- a/include/verilated_types.h +++ b/include/verilated_types.h @@ -415,8 +415,20 @@ class VlWriteMem final { static int _vl_cmp_w(int words, WDataInP const lwp, WDataInP const rwp) VL_PURE; +template +struct VlWide; + +// Type trait to check if a type is VlWide +template +struct VlIsVlWide : public std::false_type {}; + +template +struct VlIsVlWide> : public std::true_type {}; + template struct VlWide final { + static constexpr size_t Words = T_Words; + // MEMBERS // This should be the only data member, otherwise generated static initializers need updating EData m_storage[T_Words]; // Contents of the packed array @@ -1511,6 +1523,181 @@ std::string VL_TO_STRING(const VlUnpacked& obj) { return obj.to_string(); } +//=================================================================== +// Helper to apply the given indices to a target expression + +template +struct VlApplyIndices final { + VL_ATTR_ALWINLINE + static auto& apply(T_Target& target, const size_t* indicesp) { + return VlApplyIndices::apply( + target[indicesp[Curr]], indicesp); + } +}; + +template +struct VlApplyIndices final { + VL_ATTR_ALWINLINE + static T_Target& apply(T_Target& target, const size_t*) { return target; } +}; + +//=================================================================== +// Commit queue for NBAs - currently only for unpacked arrays +// +// This data-structure is used to handle non-blocking assignments +// that might execute a variable number of times in a single +// evaluation. It has 2 operations: +// - 'enqueue' will add an update to the queue +// - 'commit' will apply all enqueued updates to the target variable, +// in the order they were enqueued. This ensures the last NBA +// takes effect as it is expected. +// There are 2 specializations of this class below: +// - A version when a partial element update is not required, +// e.g, to handle: +// logic [31:0] array[N]; +// for (int i = 0 ; i < N ; ++i) array[i] <= x; +// Here 'enqueue' takes the RHS ('x'), and the array indices ('i') +// as arguments. +// - A different version when a partial element update is required, +// e.g. for: +// logic [31:0] array[N]; +// for (int i = 0 ; i < N ; ++i) array[i][3:1] <= y; +// Here 'enqueue' takes one additional argument, which is a bitmask +// derived from the bit selects (_[3:1]), which masks the bits that +// need to be updated, and additionally the RHS is widened to a full +// element size, with the bits inserted into the masked region. +template +class VlNBACommitQueue; + +// Specialization for whole element updates only +template +class VlNBACommitQueue final { + // TYPES + struct Entry final { + T_Element value; + size_t indices[T_Rank]; + }; + + // STATE + std::vector m_pending; // Pending updates, in program order + +public: + // CONSTRUCTOR + VlNBACommitQueue() = default; + ~VlNBACommitQueue() = default; + + // METHODS + template + void enqueue(const T_Element& value, Args... indices) { + m_pending.emplace_back(Entry{value, {indices...}}); + } + + // Note: T_Commit might be different from T_Target. Specifically, when the signal is a + // top-level IO port, T_Commit will be a native C array, while T_Target, will be a VlUnpacked + template + void commit(T_Commit& target) { + if (m_pending.empty()) return; + for (const Entry& entry : m_pending) { + VlApplyIndices<0, T_Rank, T_Commit>::apply(target, entry.indices) = entry.value; + } + m_pending.clear(); + } +}; + +// With partial element updates +template +class VlNBACommitQueue final { + // TYPES + struct Entry final { + T_Element value; + T_Element mask; + size_t indices[T_Rank]; + }; + + // STATE + std::vector m_pending; // Pending updates, in program order + + // STATIC METHODS + + // Binary & | ~ for elements to use for masking in partial updates. Sorry for the templates. + template + VL_ATTR_ALWINLINE static typename std::enable_if::value, T>::type + bAnd(const T& a, const T& b) { + return a & b; + } + + template + VL_ATTR_ALWINLINE static typename std::enable_if::value, T>::type + bAnd(const T& a, const T& b) { + T result; + for (size_t i = 0; i < T::Words; ++i) { + result.m_storage[i] = a.m_storage[i] & b.m_storage[i]; + } + return result; + } + + template + VL_ATTR_ALWINLINE static typename std::enable_if::value, T>::type + bOr(const T& a, const T& b) { + return a | b; + } + + template + VL_ATTR_ALWINLINE static typename std::enable_if::value, T>::type // + bOr(const T& a, const T& b) { + T result; + for (size_t i = 0; i < T::Words; ++i) { + result.m_storage[i] = a.m_storage[i] | b.m_storage[i]; + } + return result; + } + + template + VL_ATTR_ALWINLINE static typename std::enable_if::value, T>::type + bNot(const T& a) { + return ~a; + } + + template + VL_ATTR_ALWINLINE static typename std::enable_if::value, T>::type + bNot(const T& a) { + T result; + for (size_t i = 0; i < T::Words; ++i) result.m_storage[i] = ~a.m_storage[i]; + return result; + } + +public: + // CONSTRUCTOR + VlNBACommitQueue() = default; + ~VlNBACommitQueue() = default; + + // METHODS + template + void enqueue(const T_Element& value, const T_Element& mask, Args... indices) { + m_pending.emplace_back(Entry{value, mask, {indices...}}); + } + + // Note: T_Commit might be different from T_Target. Specifically, when the signal is a + // top-level IO port, T_Commit will be a native C array, while T_Target, will be a VlUnpacked + template + void commit(T_Commit& target) { + if (m_pending.empty()) return; + for (const Entry& entry : m_pending) { // + auto& ref = VlApplyIndices<0, T_Rank, T_Commit>::apply(target, entry.indices); + // Maybe inefficient, but it works for now ... + const auto oldValue = ref; + ref = bOr(bAnd(entry.value, entry.mask), bAnd(oldValue, bNot(entry.mask))); + } + m_pending.clear(); + } +}; + //=================================================================== // Object that VlDeleter is capable of deleting diff --git a/src/V3AstNodeDType.h b/src/V3AstNodeDType.h index 4e781995e07..80379e6c7b9 100644 --- a/src/V3AstNodeDType.h +++ b/src/V3AstNodeDType.h @@ -959,6 +959,30 @@ class AstMemberDType final : public AstNodeDType { return false; } }; +class AstNBACommitQueueDType final : public AstNodeDType { + // @astgen ptr := m_subDTypep : AstNodeDType // Type of the corresponding variable + const bool m_partial; // Partial element update required + +public: + AstNBACommitQueueDType(FileLine* fl, AstNodeDType* subDTypep, bool partial) + : ASTGEN_SUPER_NBACommitQueueDType(fl) + , m_partial{partial} + , m_subDTypep{subDTypep} { + dtypep(this); + } + ASTGEN_MEMBERS_AstNBACommitQueueDType; + + AstNodeDType* subDTypep() const { return m_subDTypep; } + bool partial() const { return m_partial; } + bool similarDType(const AstNodeDType* samep) const override { return this == samep; } + AstBasicDType* basicp() const override { return nullptr; } + AstNodeDType* skipRefp() const override { return (AstNodeDType*)this; } + AstNodeDType* skipRefToConstp() const override { return (AstNodeDType*)this; } + AstNodeDType* skipRefToEnump() const override { return (AstNodeDType*)this; } + int widthAlignBytes() const override { return 1; } + int widthTotalBytes() const override { return 24; } + bool isCompound() const override { return true; } +}; class AstParamTypeDType final : public AstNodeDType { // Parents: MODULE // A parameter type statement; much like a var or typedef diff --git a/src/V3AstNodes.cpp b/src/V3AstNodes.cpp index 66369177752..610064ebde0 100644 --- a/src/V3AstNodes.cpp +++ b/src/V3AstNodes.cpp @@ -788,6 +788,22 @@ AstNodeDType::CTypeRecursed AstNodeDType::cTypeRecurse(bool compound, bool packe info.m_type = "VlUnpacked<" + sub.m_type; info.m_type += ", " + cvtToStr(adtypep->declRange().elements()); info.m_type += ">"; + } else if (const auto* const adtypep = VN_CAST(dtypep, NBACommitQueueDType)) { + UASSERT_OBJ(!packed, this, "Unsupported type for packed struct or union"); + compound = true; + const CTypeRecursed sub = adtypep->subDTypep()->cTypeRecurse(compound, false); + AstNodeDType* eDTypep = adtypep->subDTypep(); + unsigned rank = 0; + while (AstUnpackArrayDType* const uaDTypep = VN_CAST(eDTypep, UnpackArrayDType)) { + eDTypep = uaDTypep; + ++rank; + } + info.m_type = "VlNBACommitQueue<"; + info.m_type += sub.m_type; + info.m_type += ", " + adtypep->partial() ? ", true" : ", false"; + info.m_type += ", " + eDTypep->cTypeRecurse(compound, false).m_type; + info.m_type += ", " + std::to_string(rank); + info.m_type += ">"; } else if (packed && (VN_IS(dtypep, PackArrayDType))) { const AstPackArrayDType* const adtypep = VN_CAST(dtypep, PackArrayDType); const CTypeRecursed sub = adtypep->subDTypep()->cTypeRecurse(false, true); @@ -2683,6 +2699,7 @@ void AstCMethodHard::setPurity() { {"commit", false}, {"delay", false}, {"done", false}, + {"enqueue", false}, {"erase", false}, {"evaluate", false}, {"evaluation", false}, diff --git a/src/V3Delayed.cpp b/src/V3Delayed.cpp index bdbb043871d..1dfa63fab60 100644 --- a/src/V3Delayed.cpp +++ b/src/V3Delayed.cpp @@ -28,7 +28,7 @@ // - Add new "Post-scheduled" logic: // a = __Vdly__a; // -// An array LHS: +// An array LHS, that is not inside a loop: // a[idxa][idxb] <= RHS // is converted: // - Add new "Pre_scheduled" logic: @@ -44,6 +44,23 @@ // Any AstAssignDly in a suspendable process or fork also uses the // '__VdlySet' flag based scheme, like arrays, with some modifications. // +// An array LHS, which is updated inside a loop (that is, we don't know +// statically how many updates there might be): +// a[idxa][idxb] <= RHS +// is converted to: +// - In the original logic, replace the AstAssignDelay with: +// __VdlyDim0__a = idxa; +// __VdlyDim1__a = idxb; +// __VdlyVal__a = RHS; +// __VdlyCommitQueue.enqueue(RHS, idxa, idxb); +// - Add new "Post-scheduled" logic: +// __VdlyCommitQueue.commit(array-on-LHS); +// +// Note that it is necessary to use the same commit queue for all NBAs +// targeting the same array, if any NBAs require a commit queue. Hence +// we can only decide whether to use a commit queue or not after having +// examined all NBAs. +// //************************************************************************* #include "V3PchAstNoMT.h" // VL_MT_DISABLED_CODE_UNIT @@ -90,6 +107,7 @@ class DelayedVisitor final : public VNVisitor { // AstAlwaysPost::user1() -> AstIf*. Last IF (__VdlySet__) created under this AlwaysPost // // Cleared each scope/active: + // AstVarScope::user2() -> AstVarScope*. Commit queue for this variable // AstVarRef::user2() -> bool. Set true if already processed // AstAssignDly::user2() -> AstVarScope*. __VdlySet__ created for this assign // AstAlwaysPost::user2() -> AstVarScope*. __VdlySet__ last referenced in IF @@ -108,6 +126,12 @@ class DelayedVisitor final : public VNVisitor { AstAlwaysPost* postp = nullptr; // First reference encountered to the VarScope const AstNodeVarRef* firstRefp = nullptr; + // If the implementation cannot be deferred, this is the location why + FileLine* nonDeferredFlp = nullptr; + // Variable requires a commit queue due to dynamic NBA + bool needsCommitQueue = false; + // Array element is partially updated by at least some NBA + bool hasPartialUpdate = false; }; AstUser1Allocator m_vscpAux; @@ -116,7 +140,7 @@ class DelayedVisitor final : public VNVisitor { std::set m_timingDomains; // Timing resume domains // Table of new var names created under module std::map, AstVar*> m_modVarMap; - VDouble0 m_statSharedSet; // Statistic tracking + AstNode* m_insertionPointp = nullptr; // Where to insert new statements // STATE - for current visit position (use VL_RESTORER) AstActive* m_activep = nullptr; // Current activate @@ -132,6 +156,11 @@ class DelayedVisitor final : public VNVisitor { std::deque m_deferred; // Deferred AstAssignDly instances to lower at the end AstVarScope* m_setVscp = nullptr; // The last used set flag, for reuse + // STATE - Statistic tracking + VDouble0 m_statSharedSet; // "VdlySet" variables eliminated by reuse + VDouble0 m_commitQueuesWhole; // Variables needing a commit queue without partial updates + VDouble0 m_commitQueuesPartial; // Variables needing a commit queue with partial updates + // METHODS const AstNode* containingAssignment(const AstNode* nodep) { @@ -282,6 +311,13 @@ class DelayedVisitor final : public VNVisitor { return nodep; } + // Insert 'stmtp; after 'm_insertionPoint', then set 'm_insertionPoint' to the given statement + // so the next insertion goes after the inserted statement. + void insert(AstNode* stmtp) { + m_insertionPointp->addNextHere(stmtp); + m_insertionPointp = stmtp; + } + // Process the given AstAssignDly. Returns 'true' if the conversion is complete and 'nodep' can // be deleted. Returns 'false' if the 'nodep' must be retained for later processing. bool processAssignDly(AstAssignDly* nodep) { @@ -289,12 +325,10 @@ class DelayedVisitor final : public VNVisitor { const bool consecutive = m_nextDlyp == nodep; m_nextDlyp = VN_CAST(nodep->nextp(), AssignDly); - // Insertion point/helper for adding new statements in code order - AstNode* insertionPointp = nodep; - const auto insert = [&insertionPointp](AstNode* stmtp) { - insertionPointp->addNextHere(stmtp); - insertionPointp = stmtp; - }; + // Insertion point + UASSERT_OBJ(!m_insertionPointp, nodep, "Insertion point should be null"); + VL_RESTORER(m_insertionPointp); + m_insertionPointp = nodep; // Deconstruct the LHS LhsComponents lhsComponents = deconstructLhs(nodep->lhsp()->unlinkFrBack()); @@ -302,6 +336,9 @@ class DelayedVisitor final : public VNVisitor { // The referenced variable AstVarScope* const vscp = lhsComponents.refp->varScopep(); + // Location of the AstAssignDly + FileLine* const flp = nodep->fileline(); + // Name suffix for signals constructed below const std::string baseName = "__" + vscp->varp()->shortName() + "__v" + std::to_string(m_scopeVecMap[vscp]++); @@ -317,7 +354,6 @@ class DelayedVisitor final : public VNVisitor { if (VN_IS(exprp, Const)) return exprp; const std::string realName = "__" + name + baseName; AstVarScope* const tmpVscp = createNewVarScope(vscp, realName, exprp->dtypep()); - FileLine* const flp = exprp->fileline(); insert(new AstAssign{flp, new AstVarRef{flp, tmpVscp, VAccess::WRITE}, exprp}); return new AstVarRef{flp, tmpVscp, VAccess::READ}; }; @@ -337,9 +373,17 @@ class DelayedVisitor final : public VNVisitor { } if (m_inSuspendableOrFork) { - // Currently we convert all NBAs in suspendable blocks immediately - // TODO: error check that deferrence was not required - FileLine* const flp = nodep->fileline(); + if (m_inLoop && !lhsComponents.arrIdxps.empty()) { + nodep->v3warn(BLKLOOPINIT, + "Unsupported: Non-blocking assignment to array inside loop " + "in suspendable process or fork"); + return true; + } + + // Currently we convert all NBAs in suspendable blocks immediately. + if (!m_vscpAux(vscp).nonDeferredFlp) { + m_vscpAux(vscp).nonDeferredFlp = nodep->fileline(); + } // Get/Create 'Post' ordered block to commit the delayed value AstAlwaysPost* postp = m_vscpAux(vscp).suspPostp; @@ -371,6 +415,20 @@ class DelayedVisitor final : public VNVisitor { // The original AstAssignDly ('nodep') can be deleted return true; } else { + if (m_inLoop) { + UASSERT_OBJ(!lhsComponents.arrIdxps.empty(), nodep, "Should be an array"); + const AstBasicDType* const basicp = vscp->dtypep()->basicp(); + if (!basicp + || !(basicp->isIntegralOrPacked() || basicp->isDouble() + || basicp->isString())) { + nodep->v3warn(BLKLOOPINIT, + "Unsupported: Non-blocking assignment to array with " + "compound element type inside loop"); + return true; + } + m_vscpAux(vscp).needsCommitQueue = true; + if (lhsComponents.selLsbp) m_vscpAux(vscp).hasPartialUpdate = true; + } // Record this AstAssignDly for deferred processing m_deferred.emplace_back(); Deferred& record = m_deferred.back(); @@ -378,30 +436,51 @@ class DelayedVisitor final : public VNVisitor { record.lhsComponents = std::move(lhsComponents); record.rhsp = rhsp; record.activep = m_activep; - record.insertionPointp = insertionPointp; + record.insertionPointp = m_insertionPointp; record.suffix = std::move(baseName); record.consecutive = consecutive; // Note consecutive assignment for optimization // If we inserted new statements, the original AstAssignDly ('nodep') // can be deleted. Otherwise, it must be kept as a placeholder. - return insertionPointp != nodep; + return m_insertionPointp != nodep; } } + // Create a temporary variable in the scope of 'vscp',with the given 'name', and with 'dtypep' + // type, with the bits selected by 'sLsbp'/'sWidthp' set to 'insertp', other bits set to zero. + // Returns a read reference to the temporary variable. + AstVarRef* createWidened(FileLine* flp, AstVarScope* vscp, AstNodeDType* dtypep, + AstNodeExpr* sLsbp, AstNodeExpr* sWidthp, const std::string& name, + AstNodeExpr* insertp) { + // Create temporary variable. + AstVarScope* const tp = createNewVarScope(vscp, name, dtypep); + // Zero it + AstConst* const zerop = new AstConst{flp, AstConst::DTyped{}, dtypep}; + zerop->num().setAllBits0(); + insert(new AstAssign{flp, new AstVarRef{flp, tp, VAccess::WRITE}, zerop}); + // Set the selected bits to 'insertp' + AstSel* const selp = new AstSel{flp, new AstVarRef{flp, tp, VAccess::WRITE}, + sLsbp->cloneTreePure(true), sWidthp->cloneTreePure(true)}; + insert(new AstAssign{flp, selp, insertp}); + // This is the expression to get the value of the temporary + return new AstVarRef{flp, tp, VAccess::READ}; + }; + // Convert the given deferred assignment for the given AstVarScope void convertDeferred(Deferred& record) { // Unpack all the parts LhsComponents lhsComponents = std::move(record.lhsComponents); AstActive*& oActivep = record.activep; // Original AstActive the AstAssignDly was under - AstNode* insertionPointp = record.insertionPointp; - const auto insert = [&insertionPointp](AstNode* stmtp) { - insertionPointp->addNextHere(stmtp); - insertionPointp = stmtp; - }; + + // Insertion point + UASSERT_OBJ(!m_insertionPointp, record.insertionPointp, "Insertion point should be null"); + VL_RESTORER(m_insertionPointp); + m_insertionPointp = record.insertionPointp; + // If the original AstAssignDly was kept as a placeholder, we will need to delete it - AstAssignDly* const dlyp = VN_CAST(insertionPointp, AssignDly); + AstAssignDly* const dlyp = VN_CAST(m_insertionPointp, AssignDly); - FileLine* const flp = insertionPointp->fileline(); + FileLine* const flp = m_insertionPointp->fileline(); AstVarScope* const vscp = lhsComponents.refp->varScopep(); // Get/Create 'Post' ordered block to commit the delayed value @@ -421,39 +500,161 @@ class DelayedVisitor final : public VNVisitor { // Ensure it contains the current sensitivities checkActiveSense(lhsComponents.refp, activep, oActivep); - // Create the flag denoting an update is pending - if (record.consecutive) { - // Simplistic optimization. If the previous assignment was immediately before this - // assignment, we can reuse the existing flag. This is good for code like: - // arrayA[0] <= something; - // arrayB[1] <= something; - ++m_statSharedSet; - } else { - // Create new flag - m_setVscp = createNewVarScope(vscp, "__VdlySet" + record.suffix, 1); - // Set it here - insert(new AstAssign{flp, new AstVarRef{flp, m_setVscp, VAccess::WRITE}, - new AstConst{flp, AstConst::BitTrue{}}}); - // Add the 'Pre' ordered reset for the flag - activep->addStmtsp(new AstAssignPre{flp, new AstVarRef{flp, m_setVscp, VAccess::WRITE}, - new AstConst{flp, 0}}); - }; + if (m_vscpAux(vscp).needsCommitQueue) { + if (FileLine* const badFlp = m_vscpAux(vscp).nonDeferredFlp) { + m_insertionPointp->v3warn( + BLKLOOPINIT, + "Unsupported: Non-blocking assignment to array in both " + "loop and suspendable process/fork\n" + << badFlp->warnOther() + << "... Location of non-blocking assignment in suspendable process/fork\n" + << badFlp->warnContextSecondary() << m_insertionPointp->warnOther() + << "... Location of non-blocking assignment inside loop\n"); + return; + } + + // Need special handling for variables that require partial updates + const bool partial = m_vscpAux(vscp).hasPartialUpdate; + + // If partial updates are required, construct the mask and the widened value + AstNodeExpr* valuep = record.rhsp; + AstNodeExpr* maskp = nullptr; + if (partial) { + // Type of array element + AstNodeDType* const eDTypep = [&]() -> AstNodeDType* { + AstNodeDType* dtypep = vscp->dtypep(); + while (AstUnpackArrayDType* const ap = VN_CAST(dtypep, UnpackArrayDType)) { + dtypep = ap->subDTypep(); + } + return dtypep; + }(); + + if (AstNodeExpr* const sLsbp = lhsComponents.selLsbp) { + // This is a partial assignment. Need to create a mask and widen the value to + // element size. + AstConst* const sWidthp = lhsComponents.selWidthp; + + // Create mask value + maskp = [&]() -> AstNodeExpr* { + // Constant mask we can compute here + if (AstConst* const cLsbp = VN_CAST(sLsbp, Const)) { + AstConst* const cp = new AstConst{flp, AstConst::DTyped{}, eDTypep}; + cp->num().setAllBits0(); + const int lsb = cLsbp->toSInt(); + const int msb = lsb + sWidthp->toSInt() - 1; + for (int bit = lsb; bit <= msb; ++bit) cp->num().setBit(bit, '1'); + return cp; + } + + // A non-constant mask we must compute at run-time. + AstConst* const onesp + = new AstConst{flp, AstConst::WidthedValue{}, sWidthp->toSInt(), 0}; + onesp->num().setAllBits1(); + return createWidened(flp, vscp, eDTypep, sLsbp, sWidthp, // + "__VdlyMask" + record.suffix, onesp); + }(); + + // Adjust value to element size + valuep = [&]() -> AstNodeExpr* { + // Constant value with constant select we can compute here + if (AstConst* const cValuep = VN_CAST(valuep, Const)) { + if (AstConst* const cLsbp = VN_CAST(sLsbp, Const)) { // + AstConst* const cp + = new AstConst{flp, AstConst::DTyped{}, eDTypep}; + cp->num().setAllBits0(); + cp->num().opSelInto(cValuep->num(), cLsbp->toSInt(), + sWidthp->toSInt()); + return cp; + } + } + + // A non-constant value we must adjust. + return createWidened(flp, vscp, eDTypep, sLsbp, sWidthp, // + "__VdlyElem" + record.suffix, valuep); + }(); + + // Finished with the sel operands here + VL_DO_DANGLING(lhsComponents.selLsbp->deleteTree(), lhsComponents.selLsbp); + VL_DO_DANGLING(lhsComponents.selWidthp->deleteTree(), lhsComponents.selWidthp); + } else { + // If this assignment is not partial, set mask to ones and we are done + AstConst* const ones = new AstConst{flp, AstConst::DTyped{}, eDTypep}; + ones->num().setAllBits1(); + maskp = ones; + } + } + + // Create/get the commit queue + if (!vscp->user2p()) { + FileLine* const vflp = vscp->fileline(); + + // Statistics + if (partial) { + ++m_commitQueuesPartial; + } else { + ++m_commitQueuesWhole; + } + + // Create the commit queue variable + auto* const cqDTypep = new AstNBACommitQueueDType{vflp, vscp->dtypep(), partial}; + v3Global.rootp()->typeTablep()->addTypesp(cqDTypep); + const std::string name = "__VdlyCommitQueue" + record.suffix; + AstVarScope* const newCqp = createNewVarScope(vscp, name, cqDTypep); + newCqp->varp()->noReset(true); + vscp->user2p(newCqp); + + // Commit it in the 'Post' block + AstCMethodHard* const callp = new AstCMethodHard{ + vflp, new AstVarRef{vflp, newCqp, VAccess::READWRITE}, "commit"}; + callp->dtypeSetVoid(); + callp->addPinsp(lhsComponents.refp->cloneTreePure(false)); + postp->addStmtsp(callp->makeStmt()); + } + AstVarScope* const cqp = VN_AS(vscp->user2p(), VarScope); - // Create/Get 'if (__VdlySet) { ... commit .. }' - AstIf* ifp = nullptr; - if (postp->user2p() == m_setVscp) { - // Optimize as above. If sharing VdlySet *ON SAME VARIABLE*, we can share the 'if' - ifp = VN_AS(postp->user1p(), If); + // Enqueue the update at the original location + AstCMethodHard* const callp + = new AstCMethodHard{flp, new AstVarRef{flp, cqp, VAccess::READWRITE}, "enqueue"}; + callp->dtypeSetVoid(); + callp->addPinsp(valuep); + if (maskp) callp->addPinsp(maskp); + for (AstNodeExpr* const indexp : lhsComponents.arrIdxps) callp->addPinsp(indexp); + insert(callp->makeStmt()); } else { - ifp = new AstIf{flp, new AstVarRef{flp, m_setVscp, VAccess::READ}}; - postp->addStmtsp(ifp); - postp->user1p(ifp); // Remember the associated 'AstIf' - postp->user2p(m_setVscp); // Remember the VdlySet variable used as the condition. - } + // Create the flag denoting an update is pending + if (record.consecutive) { + // Simplistic optimization. If the previous assignment was immediately before this + // assignment, we can reuse the existing flag. This is good for code like: + // arrayA[0] <= something; + // arrayB[1] <= something; + ++m_statSharedSet; + } else { + // Create new flag + m_setVscp = createNewVarScope(vscp, "__VdlySet" + record.suffix, 1); + // Set it here + insert(new AstAssign{flp, new AstVarRef{flp, m_setVscp, VAccess::WRITE}, + new AstConst{flp, AstConst::BitTrue{}}}); + // Add the 'Pre' ordered reset for the flag + activep->addStmtsp(new AstAssignPre{ + flp, new AstVarRef{flp, m_setVscp, VAccess::WRITE}, new AstConst{flp, 0}}); + }; - // Finally assign the delayed value to the reconstructed LHS - AstNodeExpr* const newLhsp = reconstructLhs(lhsComponents, flp); - ifp->addThensp(new AstAssign{flp, newLhsp, record.rhsp}); + // Create/Get 'if (__VdlySet) { ... commit .. }' + AstIf* ifp = nullptr; + if (postp->user2p() == m_setVscp) { + // Optimize as above. If sharing VdlySet *ON SAME VARIABLE*, we can share the 'if' + ifp = VN_AS(postp->user1p(), If); + } else { + ifp = new AstIf{flp, new AstVarRef{flp, m_setVscp, VAccess::READ}}; + postp->addStmtsp(ifp); + postp->user1p(ifp); // Remember the associated 'AstIf' + postp->user2p(m_setVscp); // Remember the VdlySet variable used as the condition. + } + + // Finally assign the delayed value to the reconstructed LHS + AstNodeExpr* const newLhsp = reconstructLhs(lhsComponents, flp); + ifp->addThensp(new AstAssign{flp, newLhsp, record.rhsp}); + } // If the original AstAssignDelay was kept as a placeholder, we can nuke it now. if (dlyp) pushDeletep(dlyp->unlinkFrBack()); @@ -582,10 +783,6 @@ class DelayedVisitor final : public VNVisitor { || (VN_IS(nodep->lhsp(), Sel) && VN_IS(VN_AS(nodep->lhsp(), Sel)->fromp(), ArraySel)); if (isArray) { - if (m_inLoop) { - nodep->v3warn(BLKLOOPINIT, "Unsupported: Delayed assignment to array inside for " - "loops (non-delayed is ok - see docs)"); - } if (const AstBasicDType* const basicp = nodep->lhsp()->dtypep()->basicp()) { // TODO: this message is not covered by tests if (basicp->isEvent()) nodep->v3warn(E_UNSUPPORTED, "Unsupported: event arrays"); @@ -684,6 +881,10 @@ class DelayedVisitor final : public VNVisitor { explicit DelayedVisitor(AstNetlist* nodep) { iterate(nodep); } ~DelayedVisitor() override { V3Stats::addStat("Optimizations, Delayed shared-sets", m_statSharedSet); + V3Stats::addStat("Dynamic NBA, variables needing commit queue with partial updates", + m_commitQueuesPartial); + V3Stats::addStat("Dynamic NBA, variables needing commit queue without partial updates", + m_commitQueuesWhole); } }; diff --git a/src/V3EmitCImp.cpp b/src/V3EmitCImp.cpp index 9fef152b4b2..3bfebebb5f3 100644 --- a/src/V3EmitCImp.cpp +++ b/src/V3EmitCImp.cpp @@ -389,6 +389,7 @@ class EmitCImp final : EmitCFunc { } else if (varp->isParam()) { } else if (varp->isStatic() && varp->isConst()) { } else if (varp->basicp() && varp->basicp()->isTriggerVec()) { + } else if (VN_IS(varp->dtypep(), NBACommitQueueDType)) { } else { int vects = 0; AstNodeDType* elementp = varp->dtypeSkipRefp(); diff --git a/src/V3Localize.cpp b/src/V3Localize.cpp index 0b59c88884e..64692fbd478 100644 --- a/src/V3Localize.cpp +++ b/src/V3Localize.cpp @@ -63,6 +63,8 @@ class LocalizeVisitor final : public VNVisitor { // METHODS bool isOptimizable(AstVarScope* nodep) { + // Don't want to malloc/free the backing store all the time + if (VN_IS(nodep->dtypep(), NBACommitQueueDType)) return false; return ((!nodep->user1() // Not marked as not optimizable, or ... // .. a block temp used in a single CFunc || (nodep->varp()->varType() == VVarType::BLOCKTEMP diff --git a/test_regress/t/t_nba_commit_queue.pl b/test_regress/t/t_nba_commit_queue.pl new file mode 100755 index 00000000000..c7b151c9ce4 --- /dev/null +++ b/test_regress/t/t_nba_commit_queue.pl @@ -0,0 +1,27 @@ +#!/usr/bin/env perl +if (!$::Driver) { use FindBin; exec("$FindBin::Bin/bootstrap.pl", @ARGV, $0); die; } +# DESCRIPTION: Verilator: Verilog Test driver/expect definition +# +# Copyright 2024 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_all => 1); + +compile( + verilator_flags2 => ["-unroll-count 1", "--stats"], + ); + +execute( + check_finished => 1, + ); + +file_grep($Self->{stats}, qr/Dynamic NBA, variables needing commit queue without partial updates\s+(\d+)/i, + 6); +file_grep($Self->{stats}, qr/Dynamic NBA, variables needing commit queue with partial updates\s+(\d+)/i, + 3); + +ok(1); +1; diff --git a/test_regress/t/t_nba_commit_queue.v b/test_regress/t/t_nba_commit_queue.v new file mode 100644 index 00000000000..42848bd6519 --- /dev/null +++ b/test_regress/t/t_nba_commit_queue.v @@ -0,0 +1,296 @@ +// DESCRIPTION: Verilator: Verilog Test module +// +// This file ONLY is placed under the Creative Commons Public Domain, for +// any use, without warranty, 2024 by Wilson Snyder. +// SPDX-License-Identifier: CC0-1.0 + +`define checkh(got ,exp) do \ + if ((got) !== (exp)) begin \ + $write("%%Error: %s:%0d: cyc=%0d result='h%x expected='h%x\n", `__FILE__,`__LINE__, cyc, (got), (exp)); \ + $stop; \ + end \ + while(0) + +`define checkr(got ,exp) do \ + if ((got) !== (exp)) begin \ + $write("%%Error: %s:%0d: cyc=%0d result=%f expected=%f\n", `__FILE__,`__LINE__, cyc, (got), (exp)); \ + $stop; \ + end \ + while(0) + +`define checks(got ,exp) do \ + if ((got) !== (exp)) begin \ + $write("%%Error: %s:%0d: cyc=%0d result=\"%s\" expected=\"%s\"\n", `__FILE__,`__LINE__, cyc, (got), (exp)); \ + $stop; \ + end \ + while(0) + + + +module t(clk); + input clk; + + logic [31:0] cyc = 0; + always @(posedge clk) begin + cyc <= cyc + 1; + if (cyc == 99) begin + $write("*-* All Finished *-*\n"); + $finish; + end + end + + reg [63:0] crc = 64'h5aef0c8d_d70a4497; + always @ (posedge clk) crc <= {crc[62:0], crc[63] ^ crc[2] ^ crc[0]}; + +`define at_posedge_clk_on_cycle(n) always @(posedge clk) if (cyc == n) + + // Case 1: narrow packed variable, whole element updates only - 1D + logic [31:0] array1 [128]; + `at_posedge_clk_on_cycle(0) begin + for (int i = 0 ; i < 128; ++i) array1[i] = 0; + for (int i = 0 ; i < 128; ++i) `checkh(array1[i], 0); + end + `at_posedge_clk_on_cycle(1) begin + for (int i = 0 ; i < 128; ++i) `checkh(array1[i], 0); + for (int i = 0 ; i < 128; ++i) array1[i] <= i; + for (int i = 0 ; i < 128; ++i) `checkh(array1[i], 0); + end + `at_posedge_clk_on_cycle(2) begin + for (int i = 0 ; i < 128; ++i) `checkh(array1[i], i); + for (int i = 0 ; i < 128; ++i) array1[i] <= ~i; + for (int i = 0 ; i < 128; ++i) `checkh(array1[i], i); + end + `at_posedge_clk_on_cycle(3) begin + for (int i = 0 ; i < 128; ++i) `checkh(array1[i], ~i); + for (int i = 0 ; i < 128; ++i) array1[i] <= -1; + for (int i = 0 ; i < 128; ++i) `checkh(array1[i], ~i); + end + `at_posedge_clk_on_cycle(4) begin + for (int i = 0 ; i < 128; ++i) `checkh(array1[i], -1); + end + + // Case 2: wide packed variable, whole element updates only - 1D + logic [127:0] array2 [128]; + `at_posedge_clk_on_cycle(0) begin + for (int i = 0 ; i < 128; ++i) array2[i] = 0; + for (int i = 0 ; i < 128; ++i) `checkh(array2[i], 0); + end + `at_posedge_clk_on_cycle(1) begin + for (int i = 0 ; i < 128; ++i) `checkh(array2[i], 0); + for (int i = 0 ; i < 128; ++i) array2[i] <= {4{i}}; + for (int i = 0 ; i < 128; ++i) `checkh(array2[i], 0); + end + `at_posedge_clk_on_cycle(2) begin + for (int i = 0 ; i < 128; ++i) `checkh(array2[i], {4{i}}); + for (int i = 0 ; i < 128; ++i) array2[i] <= {4{~i}}; + for (int i = 0 ; i < 128; ++i) `checkh(array2[i], {4{i}}); + end + `at_posedge_clk_on_cycle(3) begin + for (int i = 0 ; i < 128; ++i) `checkh(array2[i], {4{~i}}); + for (int i = 0 ; i < 128; ++i) array2[i] <= '1; + for (int i = 0 ; i < 128; ++i) `checkh(array2[i], {4{~i}}); + end + `at_posedge_clk_on_cycle(4) begin + for (int i = 0 ; i < 128; ++i) `checkh(array2[i], ~128'b0); + end + + + // Case 3: wide packed variable, whole element updates only - 2D + logic [127:0] array3 [128][512]; + `at_posedge_clk_on_cycle(0) begin + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 512; ++j) array3[i][j] = 0; + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 512; ++j) `checkh(array3[i][j], 0); + end + `at_posedge_clk_on_cycle(1) begin + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 512; ++j) `checkh(array3[i][j], 0); + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 512; ++j) array3[i][j] <= {4{i}}; + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 512; ++j) `checkh(array3[i][j], 0); + end + `at_posedge_clk_on_cycle(2) begin + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 512; ++j) `checkh(array3[i][j], {4{i}}); + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 512; ++j) array3[i][j] <= {4{~i}}; + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 512; ++j) `checkh(array3[i][j], {4{i}}); + end + `at_posedge_clk_on_cycle(3) begin + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 512; ++j) `checkh(array3[i][j], {4{~i}}); + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 512; ++j) array3[i][j] <= '1; + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 512; ++j) `checkh(array3[i][j], {4{~i}}); + end + `at_posedge_clk_on_cycle(4) begin + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 512; ++j) `checkh(array3[i][j], ~128'b0); + end + + // Case 4: real + real array4 [128]; + `at_posedge_clk_on_cycle(0) begin + for (int i = 0 ; i < 128; ++i) array4[i] = 1e-5; + for (int i = 0 ; i < 128; ++i) `checkr(array4[i], 1e-5); + end + `at_posedge_clk_on_cycle(1) begin + for (int i = 0 ; i < 128; ++i) `checkr(array4[i], 1e-5); + for (int i = 0 ; i < 128; ++i) array4[i] <= 3.14*real'(i); + for (int i = 0 ; i < 128; ++i) `checkr(array4[i], 1e-5); + end + `at_posedge_clk_on_cycle(2) begin + for (int i = 0 ; i < 128; ++i) `checkr(array4[i], 3.14*real'(i)); + for (int i = 0 ; i < 128; ++i) array4[i] <= 2.78*real'(i); + for (int i = 0 ; i < 128; ++i) `checkr(array4[i], 3.14*real'(i)); + end + `at_posedge_clk_on_cycle(3) begin + for (int i = 0 ; i < 128; ++i) `checkr(array4[i], 2.78*real'(i)); + for (int i = 0 ; i < 128; ++i) array4[i] <= 1e50; + for (int i = 0 ; i < 128; ++i) `checkr(array4[i], 2.78*real'(i)); + end + `at_posedge_clk_on_cycle(4) begin + for (int i = 0 ; i < 128; ++i) `checkr(array4[i], 1e50); + end + + // Case 5: narrow packed variable, partial element updates - 1D + logic [31:0] array5 [128]; + `at_posedge_clk_on_cycle(0) begin + for (int i = 0 ; i < 128; ++i) array5[i] = -1; + for (int i = 0 ; i < 128; ++i) `checkh(array5[i], -1); + end + `at_posedge_clk_on_cycle(1) begin + for (int i = 0 ; i < 128; ++i) `checkh(array5[i], -1); + for (int i = 0 ; i < 128; ++i) array5[i][0] <= 1'b0; + for (int i = 0 ; i < 128; ++i) array5[i][1] <= 1'b0; + for (int i = 0 ; i < 128; ++i) array5[i][2] <= 1'b0; + for (int i = 0 ; i < 128; ++i) array5[i][1] <= 1'b1; + for (int i = 0 ; i < 128; ++i) `checkh(array5[i], -1); + end + `at_posedge_clk_on_cycle(2) begin + for (int i = 0 ; i < 128; ++i) `checkh(array5[i], 32'hffff_fffa); + for (int i = 0 ; i < 128; ++i) array5[i][18:16] <= i[3:1]; + for (int i = 0 ; i < 128; ++i) array5[i][19:17] <= ~i[2:0]; + for (int i = 0 ; i < 128; ++i) `checkh(array5[i], 32'hffff_fffa); + end + `at_posedge_clk_on_cycle(3) begin + for (int i = 0 ; i < 128; ++i) `checkh(array5[i], {12'hfff, ~i[2:0], i[1], 16'hfffa}); + for (int i = 0 ; i < 128; ++i) array5[i] <= -1; + for (int i = 0 ; i < 128; ++i) `checkh(array5[i], {12'hfff, ~i[2:0], i[1], 16'hfffa}); + end + `at_posedge_clk_on_cycle(4) begin + for (int i = 0 ; i < 128; ++i) `checkh(array5[i], -1); + end + + // Case 6: wide packed variable, partial element updates - 1D + logic [99:0] array6 [128]; + `at_posedge_clk_on_cycle(0) begin + for (int i = 0 ; i < 128; ++i) array6[i] = -1; + for (int i = 0 ; i < 128; ++i) `checkh(array6[i], -1); + end + `at_posedge_clk_on_cycle(1) begin + for (int i = 0 ; i < 128; ++i) `checkh(array6[i], -1); + for (int i = 0 ; i < 128; ++i) array6[i][80] <= 1'b0; + for (int i = 0 ; i < 128; ++i) array6[i][81] <= 1'b0; + for (int i = 0 ; i < 128; ++i) array6[i][82] <= 1'b0; + for (int i = 0 ; i < 128; ++i) array6[i][81] <= 1'b1; + for (int i = 0 ; i < 128; ++i) `checkh(array6[i], -1); + end + `at_posedge_clk_on_cycle(2) begin + for (int i = 0 ; i < 128; ++i) `checkh(array6[i], 100'hf_fffa_ffff_ffff_ffff_ffff_ffff); + for (int i = 0 ; i < 128; ++i) array6[i][86:84] <= ~i[3:1]; + for (int i = 0 ; i < 128; ++i) array6[i][87:85] <= i[2:0]; + for (int i = 0 ; i < 128; ++i) `checkh(array6[i], 100'hf_fffa_ffff_ffff_ffff_ffff_ffff); + end + `at_posedge_clk_on_cycle(3) begin + for (int i = 0 ; i < 128; ++i) `checkh(array6[i], {12'hfff, i[2:0], ~i[1], 84'ha_ffff_ffff_ffff_ffff_ffff}); + for (int i = 0 ; i < 128; ++i) array6[i] <= -1; + for (int i = 0 ; i < 128; ++i) `checkh(array6[i], {12'hfff, i[2:0], ~i[1], 84'ha_ffff_ffff_ffff_ffff_ffff}); + end + `at_posedge_clk_on_cycle(4) begin + for (int i = 0 ; i < 128; ++i) `checkh(array6[i], -1); + end + + // Case 7: variable partial updates + logic [99:0] array7_nba [128][256]; + logic [99:0] array7_ref [128][256]; + always @(posedge clk) begin + if (cyc == 0) begin + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 256; ++j) array7_nba[i][j] = 100'b0; + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 256; ++j) array7_ref[i][j] = ~100'b0; + end else begin + for (int i = 0 ; i < 128; ++i) for (int j = 0 ; j < 256; ++j) `checkh(array7_nba[i][j], ~array7_ref[i][j]); + for (int i = 0 ; i < 128; ++i) begin + for (int j = 0 ; j < 256; ++j) begin + array7_nba[i[6:0] ^ crc[30+:7]][j[7:0] ^ crc[10+:8]][7'((crc % 10) * 5) +: 5] <= ~crc[4+:5]; + array7_ref[i[6:0] ^ crc[30+:7]][j[7:0] ^ crc[10+:8]][7'((crc % 10) * 5) +: 5] = crc[4+:5]; + end + end + end + end + + // Case 8: Mixed dynamic/non-dynamic + longint array8 [4]; + `at_posedge_clk_on_cycle(0) begin + array8[0] <= 0; + array8[1] <= 0; + array8[2] <= 0; + array8[3] <= 0; + end + `at_posedge_clk_on_cycle(1) begin + `checkh(array8[0], 0); + `checkh(array8[1], 0); + `checkh(array8[2], 0); + `checkh(array8[3], 0); + array8[1] <= 42; + array8[3] <= 63; + for (int i = 1 ; i < 3 ; ++i) array8[i] <= 2*i + 7; + array8[1] <= 74; + end + `at_posedge_clk_on_cycle(3) begin + `checkh(array8[0], 0); + `checkh(array8[1], 74); + `checkh(array8[2], 11); + `checkh(array8[3], 63); + end + + // Case 9: string + string array9 [10]; + `at_posedge_clk_on_cycle(0) begin + for (int i = 0 ; i < 10; ++i) array9[i] = "squid"; + for (int i = 0 ; i < 10; ++i) `checks(array9[i], "squid"); + end + `at_posedge_clk_on_cycle(1) begin + for (int i = 0 ; i < 10; ++i) `checks(array9[i], "squid"); + for (int i = 0 ; i < 10; ++i) array9[i] <= "octopus"; + for (int i = 0 ; i < 10; ++i) `checks(array9[i], "squid"); + end + `at_posedge_clk_on_cycle(2) begin + for (int i = 0 ; i < 10; ++i) `checks(array9[i], "octopus"); + for (int i = 1 ; i < 9; ++i) begin + string tmp; + $sformat(tmp, "%0d-legged-cephalopod", i); + array9[i] <= tmp; + end + for (int i = 0 ; i < 10; ++i) `checks(array9[i], "octopus"); + end + `at_posedge_clk_on_cycle(3) begin + `checks(array9[0], "octopus"); + `checks(array9[1], "1-legged-cephalopod"); + `checks(array9[2], "2-legged-cephalopod"); + `checks(array9[3], "3-legged-cephalopod"); + `checks(array9[4], "4-legged-cephalopod"); + `checks(array9[5], "5-legged-cephalopod"); + `checks(array9[6], "6-legged-cephalopod"); + `checks(array9[7], "7-legged-cephalopod"); + `checks(array9[8], "8-legged-cephalopod"); + `checks(array9[9], "octopus"); + for (int i = 0 ; i < 10; ++i) array9[i] <= "cuttlefish"; + `checks(array9[0], "octopus"); + `checks(array9[1], "1-legged-cephalopod"); + `checks(array9[2], "2-legged-cephalopod"); + `checks(array9[3], "3-legged-cephalopod"); + `checks(array9[4], "4-legged-cephalopod"); + `checks(array9[5], "5-legged-cephalopod"); + `checks(array9[6], "6-legged-cephalopod"); + `checks(array9[7], "7-legged-cephalopod"); + `checks(array9[8], "8-legged-cephalopod"); + `checks(array9[9], "octopus"); + end + `at_posedge_clk_on_cycle(4) begin + for (int i = 0 ; i < 10; ++i) `checks(array9[i], "cuttlefish"); + end + +endmodule diff --git a/test_regress/t/t_order_blkloopinit_bad.out b/test_regress/t/t_order_blkloopinit_bad.out index 56fd5c3ecf3..d34918ab93e 100644 --- a/test_regress/t/t_order_blkloopinit_bad.out +++ b/test_regress/t/t_order_blkloopinit_bad.out @@ -1,5 +1,18 @@ -%Error-BLKLOOPINIT: t/t_order_blkloopinit_bad.v:21:19: Unsupported: Delayed assignment to array inside for loops (non-delayed is ok - see docs) - 21 | array[i] <= 0; - | ^~ +%Error-BLKLOOPINIT: t/t_order_blkloopinit_bad.v:26:20: Unsupported: Non-blocking assignment to array inside loop in suspendable process or fork + : ... note: In instance 't' + 26 | array1[i] <= 0; + | ^~ ... For error description see https://verilator.org/warn/BLKLOOPINIT?v=latest +%Error-BLKLOOPINIT: t/t_order_blkloopinit_bad.v:36:20: Unsupported: Non-blocking assignment to array with compound element type inside loop + : ... note: In instance 't' + 36 | array2[i] <= null; + | ^~ +%Error-BLKLOOPINIT: t/t_order_blkloopinit_bad.v:45:20: Unsupported: Non-blocking assignment to array in both loop and suspendable process/fork + : ... note: In instance 't' + t/t_order_blkloopinit_bad.v:50:20: ... Location of non-blocking assignment in suspendable process/fork + 50 | #1 array3[0] <= 0; + | ^~ + t/t_order_blkloopinit_bad.v:45:20: ... Location of non-blocking assignment inside loop + 45 | array3[i] <= 0; + | ^~ %Error: Exiting due to diff --git a/test_regress/t/t_order_blkloopinit_bad.pl b/test_regress/t/t_order_blkloopinit_bad.pl index a5846c69938..e3fd94eb828 100755 --- a/test_regress/t/t_order_blkloopinit_bad.pl +++ b/test_regress/t/t_order_blkloopinit_bad.pl @@ -11,6 +11,7 @@ scenarios(vlt => 1); lint( + verilator_flags2 => ["--timing"], fails => 1, expect_filename => $Self->{golden_filename}, ); diff --git a/test_regress/t/t_order_blkloopinit_bad.v b/test_regress/t/t_order_blkloopinit_bad.v index d2cef2b314d..db0581a5add 100644 --- a/test_regress/t/t_order_blkloopinit_bad.v +++ b/test_regress/t/t_order_blkloopinit_bad.v @@ -4,6 +4,8 @@ // any use, without warranty, 2020 by Wilson Snyder. // SPDX-License-Identifier: CC0-1.0 +// verilator lint_off MULTIDRIVEN + module t (/*AUTOARG*/ // Outputs o, @@ -14,13 +16,38 @@ module t (/*AUTOARG*/ output int o; localparam SIZE = 65536; - int array [SIZE]; + // Unsupported case 1: Array NBA inside suspendable + int array1 [SIZE]; + always @ (posedge clk) begin + #1; + o <= array1[1]; + for (int i=0; i