From 3c144ada530419e6ee29675819a6b4c70f0fd968 Mon Sep 17 00:00:00 2001 From: Geza Lore Date: Sun, 29 Oct 2023 01:12:27 +0100 Subject: [PATCH] Delete AstNode user5 (#4638) This saves about 5% memory. V3AstUserAllocator is appropriate for most use cases, performance is marginally up as we are mostly D-cache bound on large designs. --- docs/internals.rst | 6 ++-- src/V3Ast.cpp | 3 -- src/V3Ast.h | 27 ---------------- src/V3AstNodes.cpp | 1 - src/V3AstUserAllocator.h | 19 ++++------- src/V3Delayed.cpp | 60 +++++++++++++++++++---------------- src/V3Gate.cpp | 64 +++++++++++++++++++++++-------------- src/V3Inline.cpp | 11 +++---- src/V3InstrCount.cpp | 26 +++++++-------- src/V3InstrCount.h | 2 +- src/V3LinkDot.cpp | 24 +++++++------- src/V3MergeCond.cpp | 21 ++++++------- src/V3Param.cpp | 58 +++++++++++++++++----------------- src/V3Partition.cpp | 4 +-- src/V3Simulate.h | 68 ++++++++++++++++++++++------------------ src/V3Timing.cpp | 6 ++-- src/V3Tristate.cpp | 59 +++++++++++++++++++--------------- 17 files changed, 226 insertions(+), 233 deletions(-) diff --git a/docs/internals.rst b/docs/internals.rst index fba09bd72..30a27bb5e 100644 --- a/docs/internals.rst +++ b/docs/internals.rst @@ -1222,9 +1222,9 @@ There are three ways data is passed between visitor functions. 2. User attributes. Each ``AstNode`` (**Note.** The AST node, not the visitor) has five user attributes, which may be accessed as an - integer using the ``user1()`` through ``user5()`` methods, or as a + integer using the ``user1()`` through ``user4()`` methods, or as a pointer (of type ``AstNUser``) using the ``user1p()`` through - ``user5p()`` methods (a common technique lifted from graph traversal + ``user4p()`` methods (a common technique lifted from graph traversal packages). A visitor first clears the one it wants to use by calling @@ -1653,7 +1653,7 @@ Source file and line file is ``a``, the 26th is ``z``, the 27th is ``aa``, and so on. User pointers - Shows the value of the node's user1p...user5p, if non-NULL. + Shows the value of the node's user1p...user4p, if non-NULL. Data type Many nodes have an explicit data type. "@dt=0x..." indicates the diff --git a/src/V3Ast.cpp b/src/V3Ast.cpp index f7441655c..973814442 100644 --- a/src/V3Ast.cpp +++ b/src/V3Ast.cpp @@ -41,13 +41,11 @@ uint32_t VNUser1InUse::s_userCntGbl = 0; // Hot cache line, leave adjacent uint32_t VNUser2InUse::s_userCntGbl = 0; // Hot cache line, leave adjacent uint32_t VNUser3InUse::s_userCntGbl = 0; // Hot cache line, leave adjacent uint32_t VNUser4InUse::s_userCntGbl = 0; // Hot cache line, leave adjacent -uint32_t VNUser5InUse::s_userCntGbl = 0; // Hot cache line, leave adjacent bool VNUser1InUse::s_userBusy = false; bool VNUser2InUse::s_userBusy = false; bool VNUser3InUse::s_userBusy = false; bool VNUser4InUse::s_userBusy = false; -bool VNUser5InUse::s_userBusy = false; int AstNodeDType::s_uniqueNum = 0; @@ -1235,7 +1233,6 @@ void AstNode::dumpPtrs(std::ostream& os) const { if (user2p()) os << " user2p=" << cvtToHex(user2p()); if (user3p()) os << " user3p=" << cvtToHex(user3p()); if (user4p()) os << " user4p=" << cvtToHex(user4p()); - if (user5p()) os << " user5p=" << cvtToHex(user5p()); if (m_iterpp) { os << " iterpp=" << cvtToHex(m_iterpp); // This may cause address sanitizer failures as iterpp can be stale diff --git a/src/V3Ast.h b/src/V3Ast.h index 3adf9580b..d2bb7068f 100644 --- a/src/V3Ast.h +++ b/src/V3Ast.h @@ -1611,17 +1611,6 @@ public: static void clear() { clearcnt(4, s_userCntGbl/*ref*/, s_userBusy/*ref*/); } static void check() { checkcnt(4, s_userCntGbl/*ref*/, s_userBusy/*ref*/); } }; -class VNUser5InUse final : VNUserInUseBase { -protected: - friend class AstNode; - static uint32_t s_userCntGbl; // Count of which usage of userp() this is - static bool s_userBusy; // Count is in use -public: - VNUser5InUse() { allocate(5, s_userCntGbl/*ref*/, s_userBusy/*ref*/); } - ~VNUser5InUse() { free (5, s_userCntGbl/*ref*/, s_userBusy/*ref*/); } - static void clear() { clearcnt(5, s_userCntGbl/*ref*/, s_userBusy/*ref*/); } - static void check() { checkcnt(5, s_userCntGbl/*ref*/, s_userBusy/*ref*/); } -}; // clang-format on //###################################################################### @@ -1825,8 +1814,6 @@ class AstNode VL_NOT_FINAL { uint32_t m_user3Cnt = 0; // Mark of when userp was set uint32_t m_user4Cnt = 0; // Mark of when userp was set VNUser m_user4u{0}; // Contains any information the user iteration routine wants - VNUser m_user5u{0}; // Contains any information the user iteration routine wants - uint32_t m_user5Cnt = 0; // Mark of when userp was set // METHODS void op1p(AstNode* nodep) { @@ -2074,20 +2061,6 @@ public: int user4Inc(int val = 1) { int v = user4(); user4(v + val); return v; } int user4SetOnce() { int v = user4(); if (!v) user4(1); return v; } // Better for cache than user4Inc() static void user4ClearTree() { VNUser4InUse::clear(); } // Clear userp()'s across the entire tree - - VNUser user5u() const VL_MT_STABLE { - // Slows things down measurably, so disabled by default - //UASSERT_STATIC(VNUser5InUse::s_userBusy, "user5p used without AstUserInUse"); - return ((m_user5Cnt == VNUser5InUse::s_userCntGbl) ? m_user5u : VNUser{0}); - } - AstNode* user5p() const VL_MT_STABLE { return user5u().toNodep(); } - void user5u(const VNUser& user) { m_user5u = user; m_user5Cnt = VNUser5InUse::s_userCntGbl; } - void user5p(void* userp) { user5u(VNUser{userp}); } - void user5(int val) { user5u(VNUser{val}); } - int user5() const { return user5u().toInt(); } - int user5Inc(int val = 1) { int v = user5(); user5(v + val); return v; } - int user5SetOnce() { int v = user5(); if (!v) user5(1); return v; } // Better for cache than user5Inc() - static void user5ClearTree() { VNUser5InUse::clear(); } // Clear userp()'s across the entire tree // clang-format on #ifdef VL_DEBUG diff --git a/src/V3AstNodes.cpp b/src/V3AstNodes.cpp index 37b16324e..d9cef86c2 100644 --- a/src/V3AstNodes.cpp +++ b/src/V3AstNodes.cpp @@ -1379,7 +1379,6 @@ void AstNode::dump(std::ostream& str) const { if (user2p()) str << " u2=" << nodeAddr(user2p()); if (user3p()) str << " u3=" << nodeAddr(user3p()); if (user4p()) str << " u4=" << nodeAddr(user4p()); - if (user5p()) str << " u5=" << nodeAddr(user5p()); if (hasDType()) { // Final @ so less likely to by accident read it as a nodep if (dtypep() == this) { diff --git a/src/V3AstUserAllocator.h b/src/V3AstUserAllocator.h index 521b0b1f6..a30917a71 100644 --- a/src/V3AstUserAllocator.h +++ b/src/V3AstUserAllocator.h @@ -29,7 +29,7 @@ template class AstUserAllocatorBase VL_NOT_FINAL { - static_assert(1 <= T_UserN && T_UserN <= 5, "Wrong user pointer number"); + static_assert(1 <= T_UserN && T_UserN <= 4, "Wrong user pointer number"); static_assert(std::is_base_of::value, "T_Node must be an AstNode type"); private: @@ -45,11 +45,8 @@ private: } else if VL_CONSTEXPR_CXX17 (T_UserN == 3) { const VNUser user = nodep->user3u(); return user.to(); - } else if VL_CONSTEXPR_CXX17 (T_UserN == 4) { - const VNUser user = nodep->user4u(); - return user.to(); } else { - const VNUser user = nodep->user5u(); + const VNUser user = nodep->user4u(); return user.to(); } } @@ -61,10 +58,8 @@ private: nodep->user2u(VNUser{userp}); } else if VL_CONSTEXPR_CXX17 (T_UserN == 3) { nodep->user3u(VNUser{userp}); - } else if VL_CONSTEXPR_CXX17 (T_UserN == 4) { - nodep->user4u(VNUser{userp}); } else { - nodep->user5u(VNUser{userp}); + nodep->user4u(VNUser{userp}); } } @@ -76,10 +71,8 @@ protected: VNUser2InUse::check(); } else if VL_CONSTEXPR_CXX17 (T_UserN == 3) { VNUser3InUse::check(); - } else if VL_CONSTEXPR_CXX17 (T_UserN == 4) { - VNUser4InUse::check(); } else { - VNUser5InUse::check(); + VNUser4InUse::check(); } } @@ -107,6 +100,8 @@ public: // Get a pointer to the user data if exists, otherwise nullptr T_Data* tryGet(const T_Node* nodep) { return getUserp(nodep); } + + void clear() { m_allocated.clear(); } }; // User pointer allocator classes. T_Node is the type of node the allocator should be applied to @@ -120,7 +115,5 @@ template class AstUser3Allocator final : public AstUserAllocatorBase {}; template class AstUser4Allocator final : public AstUserAllocatorBase {}; -template -class AstUser5Allocator final : public AstUserAllocatorBase {}; #endif // Guard diff --git a/src/V3Delayed.cpp b/src/V3Delayed.cpp index 0b4450376..da769536e 100644 --- a/src/V3Delayed.cpp +++ b/src/V3Delayed.cpp @@ -52,6 +52,7 @@ #include "V3Delayed.h" +#include "V3AstUserAllocator.h" #include "V3Stats.h" #include @@ -65,27 +66,33 @@ class DelayedVisitor final : public VNVisitor { private: // NODE STATE // Cleared each module: - // AstVarScope::user1p() -> AstVarScope*. Points to temp var created. - // AstVarScope::user2p() -> AstActive*. Points to activity block of signal - // (valid when AstVarScope::user1p is valid) - // AstVarScope::user3p() -> AstAlwaysPost*. Post block for this variable used for - // AssignDlys in suspendable processes - // AstVarScope::user4p() -> AstAlwaysPost*. Post block for this variable - // AstVarScope::user5p() -> AstVarRef*. Last blocking or non-blocking reference + // AstVarScope::user1p() -> aux // AstVar::user2() -> bool. Set true if already made warning + // AstVarRef::user1() -> bool. Set true if was blocking reference // AstVarRef::user2() -> bool. Set true if already processed - // AstVarRef::user5() -> bool. Set true if was blocking reference + // AstAlwaysPost::user1() -> AstIf*. Last IF (__Vdlyvset__) created under this AlwaysPost // AstAlwaysPost::user2() -> ActActive*. Points to activity block of signal // (valid when AstAlwaysPost::user4p is valid) - // AstAlwaysPost::user4() -> AstIf*. Last IF (__Vdlyvset__) created under this AlwaysPost // Cleared each scope/active: // AstAssignDly::user3() -> AstVarScope*. __Vdlyvset__ created for this assign // AstAlwaysPost::user3() -> AstVarScope*. __Vdlyvset__ last referenced in IF const VNUser1InUse m_inuser1; const VNUser2InUse m_inuser2; const VNUser3InUse m_inuser3; - const VNUser4InUse m_inuser4; - const VNUser5InUse m_inuser5; + + struct AuxAstVarScope final { + // Points to temp (shadow) variable created. + AstVarScope* delayVscp = nullptr; + // Points to activity block of signal (valid when AstVarScope::user1p is valid) + AstActive* activep = nullptr; + // Post block for this variable used for AssignDlys in suspendable processes + AstAlwaysPost* suspFinalp = nullptr; + // Post block for this variable + AstAlwaysPost* finalp = nullptr; + // Last blocking or non-blocking reference + AstNodeVarRef* lastRefp = nullptr; + }; + AstUser1Allocator m_vscpAux; // STATE - across all visitors std::unordered_map m_scopeVecMap; // Next var number for each scope @@ -116,14 +123,14 @@ private: // Ignore if warning is disabled on this reference (used by V3Force). if (nodep->fileline()->warnIsOff(V3ErrorCode::BLKANDNBLK)) return; if (m_ignoreBlkAndNBlk) return; - if (blocking) nodep->user5(true); + if (blocking) nodep->user1(true); AstVarScope* const vscp = nodep->varScopep(); // UINFO(4, " MVU " << blocking << " " << nodep << endl); - const AstNode* const lastrefp = vscp->user5p(); + const AstNode* const lastrefp = m_vscpAux(vscp).lastRefp; if (!lastrefp) { - vscp->user5p(nodep); + m_vscpAux(vscp).lastRefp = nodep; } else { - const bool last_was_blocking = lastrefp->user5(); + const bool last_was_blocking = lastrefp->user1(); if (last_was_blocking != blocking) { const AstNode* nonblockingp = blocking ? nodep : lastrefp; if (const AstNode* np = containingAssignment(nonblockingp)) nonblockingp = np; @@ -351,7 +358,7 @@ private: UINFO(9, " & " << varrefp << endl); AstAlwaysPost* finalp = nullptr; if (m_inSuspendableOrFork) { - finalp = VN_AS(varrefp->varScopep()->user3p(), AlwaysPost); + finalp = m_vscpAux(varrefp->varScopep()).suspFinalp; if (!finalp) { FileLine* const flp = nodep->fileline(); finalp = new AstAlwaysPost{flp, nullptr, nullptr}; @@ -359,13 +366,13 @@ private: if (!m_procp->user3p()) { AstActive* const newactp = createActive(varrefp); m_procp->user3p(newactp); - varrefp->varScopep()->user3p(finalp); + m_vscpAux(varrefp->varScopep()).suspFinalp = finalp; } AstActive* const actp = VN_AS(m_procp->user3p(), Active); actp->addStmtsp(finalp); } } else { - finalp = VN_AS(varrefp->varScopep()->user4p(), AlwaysPost); + finalp = m_vscpAux(varrefp->varScopep()).finalp; if (finalp) { AstActive* const oldactivep = VN_AS(finalp->user2p(), Active); checkActivePost(varrefp, oldactivep); @@ -375,7 +382,7 @@ private: UINFO(9, " Created " << finalp << endl); AstActive* const newactp = createActive(varrefp); newactp->addStmtsp(finalp); - varrefp->varScopep()->user4p(finalp); + m_vscpAux(varrefp->varScopep()).finalp = finalp; finalp->user2p(newactp); if (setinitp) newactp->addStmtsp(setinitp); } @@ -384,7 +391,7 @@ private: if (finalp->user3p() == setvscp) { // Optimize as above; if sharing Vdlyvset *ON SAME VARIABLE*, // we can share the IF statement too - postLogicp = VN_AS(finalp->user4p(), If); + postLogicp = VN_AS(finalp->user1p(), If); UASSERT_OBJ(postLogicp, nodep, "Delayed assignment misoptimized; prev var found w/o associated IF"); } else { @@ -393,7 +400,7 @@ private: UINFO(9, " Created " << postLogicp << endl); finalp->addStmtsp(postLogicp); finalp->user3p(setvscp); // Remember IF's vset variable - finalp->user4p(postLogicp); // and the associated IF, as we may be able to reuse it + finalp->user1p(postLogicp); // and the associated IF, as we may be able to reuse it } postLogicp->addThensp(new AstAssign{nodep->fileline(), selectsp, valreadp}); if (m_inSuspendableOrFork) { @@ -565,12 +572,11 @@ private: } AstVarScope* const oldvscp = nodep->varScopep(); UASSERT_OBJ(oldvscp, nodep, "Var didn't get varscoped in V3Scope.cpp"); - AstVarScope* dlyvscp = VN_AS(oldvscp->user1p(), VarScope); + AstVarScope* dlyvscp = m_vscpAux(oldvscp).delayVscp; if (dlyvscp) { // Multiple use of delayed variable - AstActive* const oldactivep = VN_AS(dlyvscp->user2p(), Active); + AstActive* const oldactivep = m_vscpAux(dlyvscp).activep; checkActivePost(nodep, oldactivep); - } - if (!dlyvscp) { // First use of this delayed variable + } else { // First use of this delayed variable const string newvarname = string{"__Vdly__"} + nodep->varp()->shortName(); dlyvscp = createVarSc(oldvscp, newvarname, 0, nullptr); AstNodeAssign* const prep = new AstAssignPre{ @@ -582,10 +588,10 @@ private: new AstVarRef{nodep->fileline(), oldvscp, VAccess::WRITE}, new AstVarRef{nodep->fileline(), dlyvscp, VAccess::READ}}; postp->lhsp()->user2(true); // Don't detect this assignment - oldvscp->user1p(dlyvscp); // So we can find it later + m_vscpAux(oldvscp).delayVscp = dlyvscp; // So we can find it later // Make new ACTIVE with identical sensitivity tree AstActive* const newactp = createActive(nodep); - dlyvscp->user2p(newactp); + m_vscpAux(dlyvscp).activep = newactp; newactp->addStmtsp(prep); // Add to FRONT of statements newactp->addStmtsp(postp); } diff --git a/src/V3Gate.cpp b/src/V3Gate.cpp index a641b1e9f..df38ef6df 100644 --- a/src/V3Gate.cpp +++ b/src/V3Gate.cpp @@ -766,15 +766,22 @@ class GateDedupeHash final : public V3DupFinderUserSame { private: // NODE STATE // Ast*::user2p -> parent AstNodeAssign* for this rhsp - // Ast*::user3p -> AstActive* of assign, for isSame() in test for duplicate - // Set to nullptr if this assign's tree was later replaced - // Ast*::user5p -> AstNode* of assign if condition, for isSame() in test for duplicate - // Set to nullptr if this assign's tree was later replaced + // AstNodeExpr::user3p -> see AuxAstNodeExpr via m_aux // VNUser1InUse m_inuser1; (Allocated for use in GateVisitor) // VNUser2InUse m_inuser2; (Allocated for use in GateVisitor) const VNUser3InUse m_inuser3; // VNUser4InUse m_inuser4; (Allocated for use in V3Hasher via V3DupFinder) - const VNUser5InUse m_inuser5; + + struct AuxAstNodeExpr final { + // AstActive* of assign, for isSame() in test for duplicate. Set to nullptr if this + // assign's tree was later replaced + AstActive* activep = nullptr; + // AstNodeExpr* of assign if condition, for isSame() in test for duplicate. Set to nullptr + // if this assign's tree was later replaced + AstNodeExpr* condp = nullptr; + }; + + AstUser3Allocator m_auxNodeExpr; V3DupFinder m_dupFinder; // Duplicate finder for rhs of assigns std::unordered_set m_nodeDeleteds; // Any node in this hash was deleted @@ -814,41 +821,49 @@ public: // or could be a rhs variable in a tree we're not replacing (or not yet anyways) void hashReplace(AstNode* oldp, AstNode* newp) { UINFO(9, "replacing " << (void*)oldp << " with " << (void*)newp << endl); - // We could update the user3p and user5p but the resulting node + // We could update the activep and condp but the resulting node // still has incorrect hash. We really need to remove all hash on // the whole hash entry tree involving the replaced node and // rehash. That's complicated and this is rare, so just remove it // from consideration. m_nodeDeleteds.insert(oldp); } - bool isReplaced(AstNode* nodep) { + bool isReplaced(AstNodeExpr* nodep) { // Assignment may have been hashReplaced, if so consider non-match (effectively removed) - UASSERT_OBJ(!VN_IS(nodep, NodeAssign), nodep, "Dedup attempt on non-assign"); - AstNode* const extra1p = nodep->user3p(); - AstNode* const extra2p = nodep->user5p(); - return ((extra1p && m_nodeDeleteds.find(extra1p) != m_nodeDeleteds.end()) - || (extra2p && m_nodeDeleteds.find(extra2p) != m_nodeDeleteds.end())); + const auto& aux = m_auxNodeExpr(nodep); + AstActive* const activep = aux.activep; + AstNodeExpr* const condp = aux.condp; + return (activep && m_nodeDeleteds.count(activep)) + || (condp && m_nodeDeleteds.count(condp)); } // Callback from V3DupFinder::findDuplicate bool isSame(AstNode* node1p, AstNode* node2p) override { + AstNodeExpr* const expr1p = VN_AS(node1p, NodeExpr); + AstNodeExpr* const expr2p = VN_AS(node2p, NodeExpr); + // Assignment may have been hashReplaced, if so consider non-match (effectively removed) - if (isReplaced(node1p) || isReplaced(node2p)) { + if (isReplaced(expr1p) || isReplaced(expr2p)) { // UINFO(9, "isSame hit on replaced "<<(void*)node1p<<" "<<(void*)node2p<user3p(), node2p->user3p()) && same(node1p->user5p(), node2p->user5p()) + + const auto& aux1 = m_auxNodeExpr(expr1p); + const auto& aux2 = m_auxNodeExpr(expr2p); + + return same(aux1.activep, aux2.activep) && same(aux1.condp, aux2.condp) && node1p->user2p()->type() == node2p->user2p()->type(); } - const AstNodeAssign* hashAndFindDupe(AstNodeAssign* assignp, AstNode* extra1p, - AstNode* extra2p) { + const AstNodeAssign* hashAndFindDupe(AstNodeAssign* assignp, AstActive* extra1p, + AstNodeExpr* extra2p) { // Legal for extra1p/2p to be nullptr, we'll compare with other assigns with extras also // nullptr - AstNode* const rhsp = assignp->rhsp(); + AstNodeExpr* const rhsp = assignp->rhsp(); rhsp->user2p(assignp); - rhsp->user3p(extra1p); - rhsp->user5p(extra2p); + auto& aux = m_auxNodeExpr(rhsp); + aux.activep = extra1p; + aux.condp = extra2p; const auto inserted = m_dupFinder.insert(rhsp); const auto dupit = m_dupFinder.findDuplicate(rhsp, this); @@ -865,13 +880,14 @@ public: void check() { for (const auto& itr : m_dupFinder) { - AstNode* const nodep = itr.second; - const AstNode* const activep = nodep->user3p(); - const AstNode* const condVarp = nodep->user5p(); + AstNodeExpr* const nodep = VN_AS(itr.second, NodeExpr); + const auto& aux = m_auxNodeExpr(nodep); + const AstActive* const activep = aux.activep; + const AstNodeExpr* const condVarp = aux.condp; if (!isReplaced(nodep)) { // This class won't break if activep isn't an active, or // ifVar isn't a var, but this is checking the caller's construction. - UASSERT_OBJ(!activep || (!VN_DELETED(activep) && VN_IS(activep, Active)), nodep, + UASSERT_OBJ(!activep || (!VN_DELETED(activep)), nodep, "V3DupFinder check failed, lost active pointer"); UASSERT_OBJ(!condVarp || !VN_DELETED(condVarp), nodep, "V3DupFinder check failed, lost if pointer"); @@ -899,7 +915,7 @@ private: // STATE GateDedupeHash m_ghash; // Hash used to find dupes of rhs of assign AstNodeAssign* m_assignp = nullptr; // Assign found for dedupe - AstNode* m_ifCondp = nullptr; // IF condition that assign is under + AstNodeExpr* m_ifCondp = nullptr; // IF condition that assign is under bool m_always = false; // Assign is under an always bool m_dedupable = true; // Determined the assign to be dedupable diff --git a/src/V3Inline.cpp b/src/V3Inline.cpp index dbc3e7bb9..ec3b52149 100644 --- a/src/V3Inline.cpp +++ b/src/V3Inline.cpp @@ -343,8 +343,8 @@ private: newdp->cellName(newcellp->name()); // Tag the old ifacerefp to ensure it leaves no stale // reference to the inlined cell. - newdp->user5(false); - ifacerefp->user5(true); + newdp->user1(false); + ifacerefp->user1(true); } } } @@ -454,8 +454,8 @@ class InlineVisitor final : public VNVisitor { private: // NODE STATE // Cleared entire netlist - // AstIfaceRefDType::user5p() // Whether the cell pointed to by this - // // AstIfaceRefDType has been inlined + // AstIfaceRefDType::user1() // Whether the cell pointed to by this + // // AstIfaceRefDType has been inlined // Input: // AstNodeModule::user1p() // ModuleState instance (via m_moduleState) // Cleared each cell @@ -464,7 +464,6 @@ private: // AstVar::user3() // bool Don't alias the user2, keep it as signal // AstCell::user4 // AstCell* of the created clone const VNUser4InUse m_inuser4; - const VNUser5InUse m_inuser5; ModuleStateUser1Allocator& m_moduleState; @@ -586,7 +585,7 @@ private: m_modp = nullptr; } void visit(AstIfaceRefDType* nodep) override { - if (nodep->user5()) { + if (nodep->user1()) { // The cell has been removed so let's make sure we don't leave a reference to it // This dtype may still be in use by the AstAssignVarScope created earlier // but that'll get cleared up later diff --git a/src/V3InstrCount.cpp b/src/V3InstrCount.cpp index b7dfff1e1..1881d2fff 100644 --- a/src/V3InstrCount.cpp +++ b/src/V3InstrCount.cpp @@ -32,9 +32,9 @@ VL_DEFINE_DEBUG_FUNCTIONS; class InstrCountVisitor final : public VNVisitorConst { private: // NODE STATE - // AstNode::user4() -> int. Path cost + 1, 0 means don't dump - // AstNode::user5() -> bool. Processed if assertNoDups - const VNUser4InUse m_inuser4; + // AstNode::user1() -> bool. Processed if assertNoDups + // AstNode::user2() -> int. Path cost + 1, 0 means don't dump + const VNUser2InUse m_inuser2; // MEMBERS uint32_t m_instrCount = 0; // Running count of instructions @@ -99,10 +99,10 @@ private: // (which at the V3Order stage represent verilog tasks, not to // the CFuncs that V3Order will generate.) So don't check for // collisions in CFuncs. - UASSERT_OBJ(!nodep->user5p(), nodep, + UASSERT_OBJ(!nodep->user1p(), nodep, "Node originally inserted below logic vertex " - << static_cast(nodep->user5p())); - nodep->user5p(const_cast(reinterpret_cast(m_startNodep))); + << static_cast(nodep->user1p())); + nodep->user1p(const_cast(reinterpret_cast(m_startNodep))); } // Save the count, and add it back in during ~VisitBase This allows @@ -118,7 +118,7 @@ private: if (!m_ignoreRemaining) m_instrCount += savedCount; } void markCost(AstNode* nodep) { - if (m_osp) nodep->user4(m_instrCount + 1); // Else don't mark to avoid writeback + if (m_osp) nodep->user2(m_instrCount + 1); // Else don't mark to avoid writeback } // VISITORS @@ -185,10 +185,10 @@ private: reset(); if (ifCount >= elseCount) { m_instrCount = savedCount + ifCount; - if (nodep->elsesp()) nodep->elsesp()->user4(0); // Don't dump it + if (nodep->elsesp()) nodep->elsesp()->user2(0); // Don't dump it } else { m_instrCount = savedCount + elseCount; - if (nodep->thensp()) nodep->thensp()->user4(0); // Don't dump it + if (nodep->thensp()) nodep->thensp()->user2(0); // Don't dump it } } void visit(AstNodeCond* nodep) override { @@ -212,10 +212,10 @@ private: reset(); if (ifCount < elseCount) { m_instrCount = savedCount + elseCount; - if (nodep->thenp()) nodep->thenp()->user4(0); // Don't dump it + if (nodep->thenp()) nodep->thenp()->user2(0); // Don't dump it } else { m_instrCount = savedCount + ifCount; - if (nodep->elsep()) nodep->elsep()->user4(0); // Don't dump it + if (nodep->elsep()) nodep->elsep()->user2(0); // Don't dump it } } void visit(AstCAwait* nodep) override { @@ -289,7 +289,7 @@ private: class InstrCountDumpVisitor final : public VNVisitorConst { private: // NODE STATE - // AstNode::user4() -> int. Path cost, 0 means don't dump + // AstNode::user2() -> int. Path cost, 0 means don't dump // MEMBERS std::ostream* const m_osp; // Dump file @@ -310,7 +310,7 @@ private: string indent() const { return string(m_depth, ':') + " "; } void visit(AstNode* nodep) override { ++m_depth; - if (unsigned costPlus1 = nodep->user4()) { + if (unsigned costPlus1 = nodep->user2()) { *m_osp << " " << indent() << "cost " << std::setw(6) << std::left << (costPlus1 - 1) << " " << nodep << '\n'; iterateChildrenConst(nodep); diff --git a/src/V3InstrCount.h b/src/V3InstrCount.h index fc4978439..d50da1708 100644 --- a/src/V3InstrCount.h +++ b/src/V3InstrCount.h @@ -36,7 +36,7 @@ public: // If nodep is an AstActive, returns 0. // If nodep contains nested AstActives, raises an error. // - // If assertNoDups is true, marks user5 on each AstNode scanned. Then + // If assertNoDups is true, marks user1 on each AstNode scanned. Then // if we see the same node twice (across more than one call to count, // potentially) raises an error. // Optional osp is stream to dump critical path to. diff --git a/src/V3LinkDot.cpp b/src/V3LinkDot.cpp index 3e53ce387..0652060fd 100644 --- a/src/V3LinkDot.cpp +++ b/src/V3LinkDot.cpp @@ -2004,10 +2004,7 @@ private: // *::user2p() -> See LinkDotState // *::user3() // bool. Processed // *::user4() -> See LinkDotState - // Cleared on Cell - // AstVar::user5() // bool. True if pin used in this cell const VNUser3InUse m_inuser3; - const VNUser5InUse m_inuser5; // TYPES enum DotPosition : uint8_t { @@ -2034,6 +2031,7 @@ private: // They are added to the set only in linkDotPrimary. bool m_insideClassExtParam = false; // Inside a class from m_extendsParam bool m_explicitSuperNew = false; // Hit a "super.new" call inside a "new" function + std::map m_usedPins; // Pin used in this cell, map to duplicate struct DotStates { DotPosition m_dotPos; // Scope part of dotted resolution @@ -2162,15 +2160,15 @@ private: UASSERT_OBJ(ifaceTopVarp, nodep, "Can't find interface var ref: " << findName); return ifaceTopVarp; } - void markAndCheckPinDup(AstNode* nodep, AstNode* refp, const char* whatp) { - if (refp->user5p() && refp->user5p() != nodep) { + void markAndCheckPinDup(AstPin* nodep, AstNode* refp, const char* whatp) { + const auto pair = m_usedPins.emplace(refp, nodep); + if (!pair.second) { + AstNode* const origp = pair.first->second; nodep->v3error("Duplicate " << whatp << " connection: " << nodep->prettyNameQ() << '\n' << nodep->warnContextPrimary() << '\n' - << refp->user5p()->warnOther() - << "... Location of original " << whatp << " connection\n" - << refp->user5p()->warnContextSecondary()); - } else { - refp->user5p(nodep); + << origp->warnOther() << "... Location of original " + << whatp << " connection\n" + << origp->warnContextSecondary()); } } VSymEnt* getCreateClockingEventSymEnt(AstClocking* clockingp) { @@ -2314,7 +2312,7 @@ private: void visit(AstCell* nodep) override { // Cell: Recurse inside or cleanup not founds checkNoDot(nodep); - AstNode::user5ClearTree(); + m_usedPins.clear(); UASSERT_OBJ(nodep->modp(), nodep, "Cell has unlinked module"); // V3LinkCell should have errored out VL_RESTORER(m_cellp); @@ -2343,7 +2341,7 @@ private: void visit(AstClassRefDType* nodep) override { // Cell: Recurse inside or cleanup not founds checkNoDot(nodep); - AstNode::user5ClearTree(); + m_usedPins.clear(); UASSERT_OBJ(nodep->classp(), nodep, "ClassRef has unlinked class"); UASSERT_OBJ(m_statep->forPrimary() || !nodep->paramsp(), nodep, "class reference parameter not removed by V3Param"); @@ -2905,7 +2903,7 @@ private: void visit(AstClassOrPackageRef* nodep) override { // Class: Recurse inside or cleanup not founds // checkNoDot not appropriate, can be under a dot - AstNode::user5ClearTree(); + m_usedPins.clear(); UASSERT_OBJ(m_statep->forPrimary() || VN_IS(nodep->classOrPackageNodep(), ParamTypeDType) || nodep->classOrPackagep(), nodep, "ClassRef has unlinked class"); diff --git a/src/V3MergeCond.cpp b/src/V3MergeCond.cpp index b9e08ea22..e5b078daa 100644 --- a/src/V3MergeCond.cpp +++ b/src/V3MergeCond.cpp @@ -159,11 +159,10 @@ class CodeMotionAnalysisVisitor final : public VNVisitorConst { // NODE STATE // AstNodeStmt::user3 -> StmtProperties (accessed via m_stmtProperties, managed externally, // see MergeCondVisitor::process) + // AstNodeExpr::user3 -> AstNodeStmt*: Set on a condition node, points to the last + // conditional with that condition so far encountered in the same + // AstNode list // AstNode::user4 -> Used by V3Hasher - // AstNode::user5 -> AstNode*: Set on a condition node, points to the last conditional - // with that condition so far encountered in the same AstNode list - - VNUser5InUse m_user5InUse; StmtPropertiesAllocator& m_stmtProperties; @@ -212,14 +211,14 @@ class CodeMotionAnalysisVisitor final : public VNVisitorConst { // First time seeing this condition in the current list dupFinder.insert(condp); // Remember last statement with this condition (which is this statement) - condp->user5p(nodep); + condp->user3p(nodep); } else { // Seen a conditional with the same condition earlier in the current list - AstNode* const firstp = dit->second; + AstNodeExpr* const firstp = VN_AS(dit->second, NodeExpr); // Add to properties for easy retrieval during optimization - m_propsp->m_prevWithSameCondp = static_cast(firstp->user5p()); + m_propsp->m_prevWithSameCondp = static_cast(firstp->user3p()); // Remember last statement with this condition (which is this statement) - firstp->user5p(nodep); + firstp->user3p(nodep); } } } @@ -430,13 +429,13 @@ private: // NODE STATE // AstVar::user1 -> bool: Set for variables referenced by m_mgCondp // (Only below MergeCondVisitor::process). - // AstNode::user2 -> bool: Marking node as included in merge because cheap to - // duplicate + // AstNode::user2 -> bool: Marking node as included in merge because cheap to duplicate // (Only below MergeCondVisitor::process). // AstNodeStmt::user3 -> StmtProperties // (Only below MergeCondVisitor::process). + // AstNodeExpr::user3 -> See CodeMotionAnalysisVisitor + // (Only below MergeCondVisitor::process). // AstNode::user4 -> See CodeMotionAnalysisVisitor/CodeMotionOptimizeVisitor - // AstNode::user5 -> See CodeMotionAnalysisVisitor // STATE VDouble0 m_statMerges; // Statistic tracking diff --git a/src/V3Param.cpp b/src/V3Param.cpp index 53c01c77c..ffe79fa1f 100644 --- a/src/V3Param.cpp +++ b/src/V3Param.cpp @@ -218,21 +218,21 @@ public: class ParamProcessor final { // NODE STATE - Local - // AstVar::user4() // int Global parameter number (for naming new module) + // AstVar::user3() // int Global parameter number (for naming new module) // // (0=not processed, 1=iterated, but no number, // // 65+ parameter numbered) // NODE STATE - Shared with ParamVisitor - // AstClass::user4p() // AstClass* Unchanged copy of the parameterized class node. + // AstClass::user3p() // AstClass* Unchanged copy of the parameterized class node. // The class node may be modified according to parameter // values and an unchanged copy is needed to instantiate // classes with different parameters. - // AstNodeModule::user5() // bool True if processed - // AstGenFor::user5() // bool True if processed - // AstVar::user5() // bool True if constant propagated - // AstCell::user5p() // string* Generate portion of hierarchical name - const VNUser4InUse m_inuser4; - const VNUser5InUse m_inuser5; - // User1/2/3 used by constant function simulations + // AstNodeModule::user2() // bool True if processed + // AstGenFor::user2() // bool True if processed + // AstVar::user2() // bool True if constant propagated + // AstCell::user2p() // string* Generate portion of hierarchical name + const VNUser2InUse m_inuser2; + const VNUser3InUse m_inuser3; + // User1 used by constant function simulations // TYPES // Note may have duplicate entries @@ -284,20 +284,20 @@ class ParamProcessor final { char ch = varp->name()[0]; ch = std::toupper(ch); if (ch < 'A' || ch > 'Z') ch = 'Z'; - varp->user4(usedLetter[static_cast(ch)] * 256 + ch); + varp->user3(usedLetter[static_cast(ch)] * 256 + ch); usedLetter[static_cast(ch)]++; } } else if (AstParamTypeDType* const typep = VN_CAST(stmtp, ParamTypeDType)) { const char ch = 'T'; - typep->user4(usedLetter[static_cast(ch)] * 256 + ch); + typep->user3(usedLetter[static_cast(ch)] * 256 + ch); usedLetter[static_cast(ch)]++; } } } static string paramSmallName(AstNodeModule* modp, AstNode* varp) { - if (varp->user4() <= 1) makeSmallNames(modp); - int index = varp->user4() / 256; - const char ch = varp->user4() & 255; + if (varp->user3() <= 1) makeSmallNames(modp); + int index = varp->user3() / 256; + const char ch = varp->user3() & 255; string st = cvtToStr(ch); while (index) { st += cvtToStr(char((index % 25) + 'A')); @@ -571,8 +571,8 @@ class ParamProcessor final { // Note all module internal variables will be re-linked to the new modules by clone // However links outside the module (like on the upper cells) will not. AstNodeModule* newmodp; - if (srcModp->user4p()) { - newmodp = VN_CAST(srcModp->user4p()->cloneTree(false), NodeModule); + if (srcModp->user3p()) { + newmodp = VN_CAST(srcModp->user3p()->cloneTree(false), NodeModule); } else { newmodp = srcModp->cloneTree(false); } @@ -583,7 +583,7 @@ class ParamProcessor final { } newmodp->name(newname); - newmodp->user5(false); // We need to re-recurse this module once changed + newmodp->user2(false); // We need to re-recurse this module once changed newmodp->recursive(false); newmodp->recursiveClone(false); // Only the first generation of clone holds this property @@ -614,7 +614,7 @@ class ParamProcessor final { // Grab all I/O so we can remap our pins later // Note we allow multiple users of a parameterized model, // thus we need to stash this info. - collectPins(clonemapp, newmodp, srcModp->user4p()); + collectPins(clonemapp, newmodp, srcModp->user3p()); // Relink parameter vars to the new module relinkPins(clonemapp, paramsp); // Fix any interface references @@ -853,14 +853,14 @@ class ParamProcessor final { if (!any_overrides) { UINFO(8, "Cell parameters all match original values, skipping expansion.\n"); - // If it's the first use of the default instance, create a copy and store it in user4p. - // user4p will also be used to check if the default instance is used. - if (!srcModpr->user4p() && VN_IS(srcModpr, Class)) { + // If it's the first use of the default instance, create a copy and store it in user3p. + // user3p will also be used to check if the default instance is used. + if (!srcModpr->user3p() && VN_IS(srcModpr, Class)) { AstClass* classCopyp = VN_AS(srcModpr, Class)->cloneTree(false); // It is a temporary copy of the original class node, stored in order to create // another instances. It is needed only during class instantiation. m_deleter.pushDeletep(classCopyp); - srcModpr->user4p(classCopyp); + srcModpr->user3p(classCopyp); storeOriginalParams(classCopyp); } } else if (AstNodeModule* const paramedModp @@ -1002,9 +1002,9 @@ class ParamVisitor final : public VNVisitor { AstNodeModule* const modp = itm->second; workQueue.erase(itm); - // Process once; note user5 will be cleared on specialization, so we will do the + // Process once; note user2 will be cleared on specialization, so we will do the // specialized module if needed - if (!modp->user5SetOnce()) { + if (!modp->user2SetOnce()) { // TODO: this really should be an assert, but classes and hier_blocks are // special... @@ -1040,9 +1040,9 @@ class ParamVisitor final : public VNVisitor { // Update path string someInstanceName(modp->someInstanceName()); - if (const string* const genHierNamep = cellp->user5u().to()) { + if (const string* const genHierNamep = cellp->user2u().to()) { someInstanceName += *genHierNamep; - cellp->user5p(nullptr); + cellp->user2p(nullptr); VL_DO_DANGLING(delete genHierNamep, genHierNamep); } @@ -1077,7 +1077,7 @@ class ParamVisitor final : public VNVisitor { void visitCellOrClassRef(AstNode* nodep, bool isIface) { // Must do ifaces first, so push to list and do in proper order string* const genHierNamep = new std::string{m_generateHierName}; - nodep->user5p(genHierNamep); + nodep->user2p(genHierNamep); // Visit parameters in the instantiation. iterateChildren(nodep); m_cellps.emplace(!isIface, nodep); @@ -1133,7 +1133,7 @@ class ParamVisitor final : public VNVisitor { // Make sure all parameters are constantified void visit(AstVar* nodep) override { - if (nodep->user5SetOnce()) return; // Process once + if (nodep->user2SetOnce()) return; // Process once iterateChildren(nodep); if (nodep->isParam()) { if (!nodep->valuep() && !VN_IS(m_modp, Class)) { @@ -1428,7 +1428,7 @@ public: for (AstNodeModule* const modp : modps) netlistp->addModulesp(modp); for (AstClass* const classp : m_paramClasses) { - if (!classp->user4p()) { + if (!classp->user3p()) { // The default value isn't referenced, so it can be removed VL_DO_DANGLING(pushDeletep(classp->unlinkFrBack()), classp); } else { diff --git a/src/V3Partition.cpp b/src/V3Partition.cpp index 26403446d..a1a502a06 100644 --- a/src/V3Partition.cpp +++ b/src/V3Partition.cpp @@ -2662,9 +2662,9 @@ uint32_t V3Partition::setupMTaskDeps(V3Graph* mtasksp) { // coarsening algorithm assumes that the graph is connected. m_entryMTaskp = new LogicMTask{mtasksp, nullptr}; - // The V3InstrCount within LogicMTask will set user5 on each AST + // The V3InstrCount within LogicMTask will set user1 on each AST // node, to assert that we never count any node twice. - const VNUser5InUse user5inUse; + const VNUser1InUse user1inUse; // Create the LogicMTasks for each MTaskMoveVertex for (V3GraphVertex *vtxp = m_fineDepsGraphp->verticesBeginp(), *nextp; vtxp; vtxp = nextp) { diff --git a/src/V3Simulate.h b/src/V3Simulate.h index bc36bf0d7..5343af244 100644 --- a/src/V3Simulate.h +++ b/src/V3Simulate.h @@ -36,6 +36,7 @@ #include "verilatedos.h" #include "V3Ast.h" +#include "V3AstUserAllocator.h" #include "V3Error.h" #include "V3Task.h" #include "V3Width.h" @@ -76,19 +77,25 @@ private: // NODE STATE // Cleared on each always/assignw const VNUser1InUse m_inuser1; - const VNUser2InUse m_inuser2; - const VNUser3InUse m_inuser3; - // Checking: - // AstVar(Scope)::user1() -> VarUsage. Set true to indicate tracking as lvalue/rvalue - // Simulating: - // AstConst::user2() -> bool. This AstConst (allocated by this class) is in use - // AstVar(Scope)::user3() -> AstConst*. Input value of variable or node - // (and output for non-delayed assignments) - // AstVar(Scope)::user2() -> AstCont*. Output value of variable (delayed assignments) + // AstVar/AstVarScope::user1p() -> See AuxAstVar via m_varAux + // AstConst::user1() -> bool. This AstConst (allocated by this class) is in use enum VarUsage : uint8_t { VU_NONE = 0, VU_LV = 1, VU_RV = 2, VU_LVDLY = 4 }; + struct AuxVariable final { + // Checking: + // Set true to indicate tracking as lvalue/rvalue + uint8_t usage = VU_NONE; + // Simulating: + // Output value of variable (delayed assignments) + AstNodeExpr* outValuep = nullptr; + // Input value of variable or node (and output for non-delayed assignments) + AstNodeExpr* valuep = nullptr; + }; + + AstUser1Allocator m_varAux; + // STATE // Major mode bool m_checkOnly; ///< Checking only (no simulation) mode @@ -223,14 +230,14 @@ private: bool allocNewConst = true; if (!freeList.empty()) { constp = freeList.front(); - if (!constp->user2()) { + if (!constp->user1()) { // Front of free list is free, reuse it (otherwise allocate new node) allocNewConst = false; // No need to allocate // Mark the AstConst node as used, and move it to the back of the free list. This // ensures that when all AstConst instances within the list are used, then the // front of the list will be marked as used, in which case the enclosing 'if' will // fail and we fall back to allocation. - constp->user2(1); + constp->user1(1); freeList.pop_front(); freeList.push_back(constp); // configure const @@ -241,7 +248,7 @@ private: // Need to allocate new constant constp = new AstConst{nodep->fileline(), AstConst::DTyped{}, nodep->dtypep()}; // Mark as in use, add to free list for later reuse - constp->user2(1); + constp->user1(1); freeList.push_back(constp); } return constp; @@ -273,7 +280,7 @@ private: } AstConst* newConst(AstNode* nodep) { // Set a constant value for this node - if (!VN_IS(nodep->user3p(), Const)) { + if (!VN_IS(m_varAux(nodep).valuep, Const)) { AstConst* const constp = allocConst(nodep); setValue(nodep, constp); return constp; @@ -283,7 +290,7 @@ private: } AstConst* newOutConst(AstNode* nodep) { // Set a var-output constant value for this node - if (!VN_IS(nodep->user2p(), Const)) { + if (!VN_IS(m_varAux(nodep).outValuep, Const)) { AstConst* const constp = allocConst(nodep); setOutValue(nodep, constp); return constp; @@ -293,10 +300,10 @@ private: } public: - AstNodeExpr* fetchValueNull(AstNode* nodep) { return VN_AS(nodep->user3p(), NodeExpr); } + AstNodeExpr* fetchValueNull(AstNode* nodep) { return m_varAux(nodep).valuep; } private: - AstNodeExpr* fetchOutValueNull(AstNode* nodep) { return VN_AS(nodep->user2p(), NodeExpr); } + AstNodeExpr* fetchOutValueNull(AstNode* nodep) { return m_varAux(nodep).outValuep; } AstConst* fetchConstNull(AstNode* nodep) { return VN_CAST(fetchValueNull(nodep), Const); } AstConst* fetchOutConstNull(AstNode* nodep) { return VN_CAST(fetchOutValueNull(nodep), Const); @@ -332,15 +339,15 @@ public: } private: - void setValue(AstNode* nodep, const AstNodeExpr* valuep) { + void setValue(AstNode* nodep, AstNodeExpr* valuep) { UASSERT_OBJ(valuep, nodep, "Simulate setting null value"); UINFO(9, " set val " << valuep->name() << " on " << nodep << endl); - nodep->user3p((void*)valuep); + m_varAux(nodep).valuep = valuep; } - void setOutValue(AstNode* nodep, const AstNodeExpr* valuep) { + void setOutValue(AstNode* nodep, AstNodeExpr* valuep) { UASSERT_OBJ(valuep, nodep, "Simulate setting null value"); UINFO(9, " set oval " << valuep->name() << " on " << nodep << endl); - nodep->user2p((void*)valuep); + m_varAux(nodep).outValuep = valuep; } void checkNodeInfo(AstNode* nodep, bool ignorePredict = false) { @@ -429,26 +436,26 @@ private: clearOptimizable(nodep, "Array references/not basic"); if (nodep->access().isWriteOrRW()) { if (m_inDlyAssign) { - if (!(vscp->user1() & VU_LVDLY)) { - vscp->user1(vscp->user1() | VU_LVDLY); + if (!(m_varAux(vscp).usage & VU_LVDLY)) { + m_varAux(vscp).usage |= VU_LVDLY; if (m_checkOnly) varRefCb(nodep); } } else { // nondly asn - if (!(vscp->user1() & VU_LV)) { - if (!m_params && (vscp->user1() & VU_RV)) { + if (!(m_varAux(vscp).usage & VU_LV)) { + if (!m_params && (m_varAux(vscp).usage & VU_RV)) { clearOptimizable(nodep, "Var read & write"); } - vscp->user1(vscp->user1() | VU_LV); + m_varAux(vscp).usage |= VarUsage::VU_LV; if (m_checkOnly) varRefCb(nodep); } } } if (nodep->access().isReadOrRW()) { - if (!(vscp->user1() & VU_RV)) { - if (!m_params && (vscp->user1() & VU_LV)) { + if (!(m_varAux(vscp).usage & VU_RV)) { + if (!m_params && (m_varAux(vscp).usage & VU_LV)) { clearOptimizable(nodep, "Var write & read"); } - vscp->user1(vscp->user1() | VU_RV); + m_varAux(vscp).usage |= VU_RV; const bool isConst = (nodep->varp()->isConst() || nodep->varp()->isParam()) && nodep->varp()->valuep(); AstNodeExpr* const valuep @@ -717,7 +724,7 @@ private: AstNodeExpr* const valuep = newTrackedClone(fetchValue(nodep->rhsp())); UINFO(9, " set val[" << index << "] = " << valuep << endl); // Values are in the "real" tree under the InitArray so can eventually extract it, - // Not in the usual setValue (pointed to by user2/3p) + // Not in the usual setValue (via m_varAux) initp->addIndexValuep(index, valuep); if (debug() >= 9) initp->dumpTree("- array: "); assignOutValue(nodep, vscp, initp); @@ -1224,8 +1231,7 @@ public: m_jumpp = nullptr; AstNode::user1ClearTree(); - AstNode::user2ClearTree(); // Also marks all elements in m_constps as free - AstNode::user3ClearTree(); + m_varAux.clear(); } void mainTableCheck(AstNode* nodep) { setMode(true /*scoped*/, true /*checking*/, false /*params*/); diff --git a/src/V3Timing.cpp b/src/V3Timing.cpp index 0d170c90c..6484a7247 100644 --- a/src/V3Timing.cpp +++ b/src/V3Timing.cpp @@ -183,7 +183,7 @@ private: const VNUser1InUse m_user1InUse; const VNUser2InUse m_user2InUse; const VNUser3InUse m_user3InUse; - const VNUser5InUse m_user5InUse; + const VNUser4InUse m_user4InUse; // STATE VMemberMap m_memberMap; // Member names cached for fast lookup @@ -214,8 +214,8 @@ private: classp = VN_CAST(funcp->scopep()->modp(), Class); } } - if (!nodep->user5p()) nodep->user5p(new NeedsProcDepVtx{&m_procGraph, nodep, classp}); - return nodep->user5u().to(); + if (!nodep->user4p()) nodep->user4p(new NeedsProcDepVtx{&m_procGraph, nodep, classp}); + return nodep->user4u().to(); } // Pass timing flag between nodes bool passFlag(const AstNode* from, AstNode* to, NodeFlag flag) { diff --git a/src/V3Tristate.cpp b/src/V3Tristate.cpp index 444a49ebc..4807365af 100644 --- a/src/V3Tristate.cpp +++ b/src/V3Tristate.cpp @@ -119,6 +119,7 @@ #include "V3Tristate.h" +#include "V3AstUserAllocator.h" #include "V3Graph.h" #include "V3Inst.h" #include "V3Stats.h" @@ -176,8 +177,8 @@ public: class TristateGraph final { // NODE STATE - // AstVar::user5p -> TristateVertex* for variable being built - // VNUser5InUse m_inuser5; // In visitor below + // AstVar::user4p -> TristateVertex* for variable being built + // VNUser4InUse m_inuser4; // In visitor below // TYPES public: @@ -197,11 +198,11 @@ private: // METHODS TristateVertex* makeVertex(AstNode* nodep) { - TristateVertex* vertexp = reinterpret_cast(nodep->user5p()); + TristateVertex* vertexp = reinterpret_cast(nodep->user4p()); if (!vertexp) { UINFO(6, " New vertex " << nodep << endl); vertexp = new TristateVertex{&m_graph, nodep}; - nodep->user5p(vertexp); + nodep->user4p(vertexp); } return vertexp; } @@ -279,7 +280,7 @@ public: } } m_graph.clear(); - AstNode::user5ClearTree(); // Wipe all node user5p's that point to vertexes + AstNode::user4ClearTree(); // Wipe all node user4p's that point to vertexes } void graphWalk(AstNodeModule* nodep) { UINFO(9, " Walking " << nodep << endl); @@ -298,7 +299,7 @@ public: if (!nodep) return; // Skip vars, because they may be connected to more than one varref if (!VN_IS(nodep, Var)) { - TristateVertex* const vertexp = reinterpret_cast(nodep->user5p()); + TristateVertex* const vertexp = reinterpret_cast(nodep->user4p()); if (vertexp) vertexp->unlinkDelete(&m_graph); } deleteVerticesFromSubtreeRecurse(nodep->op1p()); @@ -308,15 +309,15 @@ public: } void setTristate(AstNode* nodep) { makeVertex(nodep)->isTristate(true); } bool isTristate(AstNode* nodep) { - const TristateVertex* const vertexp = reinterpret_cast(nodep->user5p()); + const TristateVertex* const vertexp = reinterpret_cast(nodep->user4p()); return vertexp && vertexp->isTristate(); } bool feedsTri(AstNode* nodep) { - const TristateVertex* const vertexp = reinterpret_cast(nodep->user5p()); + const TristateVertex* const vertexp = reinterpret_cast(nodep->user4p()); return vertexp && vertexp->feedsTri(); } void didProcess(AstNode* nodep) { - TristateVertex* const vertexp = reinterpret_cast(nodep->user5p()); + TristateVertex* const vertexp = reinterpret_cast(nodep->user4p()); if (!vertexp) { // Not v3errorSrc as no reason to stop the world nodep->v3error("Unsupported tristate construct (not in propagation graph): " @@ -390,16 +391,21 @@ class TristateVisitor final : public TristateBaseVisitor { // NODE STATE // *::user1p -> pointer to output enable __en expressions // *::user2 -> int - already visited, see U2_ enum - // AstVar::user3p -> AstPull* pullup/pulldown direction (input Var's user3p) - // AstVar::user4p -> AstVar* pointer to output __out var (input Var's user2p) + // AstVar::user3p -> See AuxAstVar via m_varAux // See TristateGraph: - // AstVar::user5p -> TristateVertex* for variable being built - // AstStmt*::user5p -> TristateVertex* for this statement + // AstVar::user4p -> TristateVertex* for variable being built + // AstStmt::user4p -> TristateVertex* for this statement const VNUser1InUse m_inuser1; const VNUser2InUse m_inuser2; const VNUser3InUse m_inuser3; const VNUser4InUse m_inuser4; - const VNUser5InUse m_inuser5; + + struct AuxAstVar final { + AstPull* pullp = nullptr; // pullup/pulldown direction + AstVar* outVarp = nullptr; // output __out var + }; + + AstUser3Allocator m_varAux; // TYPES struct RefStrength { @@ -531,14 +537,14 @@ class TristateVisitor final : public TristateBaseVisitor { } AstVar* getCreateOutVarp(AstVar* invarp) { // Return the master __out for the specified input variable - if (!invarp->user4p()) { + if (!m_varAux(invarp).outVarp) { AstVar* const newp = new AstVar{invarp->fileline(), VVarType::MODULETEMP, invarp->name() + "__out", invarp}; UINFO(9, " newout " << newp << endl); modAddStmtp(invarp, newp); - invarp->user4p(newp); // find outvar given invarp + m_varAux(invarp).outVarp = newp; // find outvar given invarp } - return VN_AS(invarp->user4p(), Var); + return m_varAux(invarp).outVarp; } AstVar* getCreateUnconnVarp(AstNode* fromp, AstNodeDType* dtypep) { AstVar* const newp = new AstVar{fromp->fileline(), VVarType::MODULETEMP, @@ -566,9 +572,9 @@ class TristateVisitor final : public TristateBaseVisitor { } void setPullDirection(AstVar* varp, AstPull* pullp) { - const AstPull* const oldpullp = static_cast(varp->user3p()); + const AstPull* const oldpullp = m_varAux(varp).pullp; if (!oldpullp) { - varp->user3p(pullp); // save off to indicate the pull direction + m_varAux(varp).pullp = pullp; // save off to indicate the pull direction } else { if (oldpullp->direction() != pullp->direction()) { pullp->v3warn(E_UNSUPPORTED, "Unsupported: Conflicting pull directions.\n" @@ -725,8 +731,8 @@ class TristateVisitor final : public TristateBaseVisitor { envarp = getCreateEnVarp(invarp); // direction will be sen in visit(AstPin*) // outvarp->user1p(envarp); - outvarp->user3p(invarp->user3p()); // AstPull* propagation - if (invarp->user3p()) UINFO(9, "propagate pull to " << outvarp << endl); + m_varAux(outvarp).pullp = m_varAux(invarp).pullp; // AstPull* propagation + if (m_varAux(invarp).pullp) UINFO(9, "propagate pull to " << outvarp << endl); } else if (invarp->user1p()) { envarp = VN_AS(invarp->user1p(), Var); // From CASEEQ, foo === 1'bz } @@ -784,7 +790,7 @@ class TristateVisitor final : public TristateBaseVisitor { if (!outvarp) { // This is the final pre-forced resolution of the tristate, so we apply // the pull direction to any undriven pins. - const AstPull* const pullp = static_cast(lhsp->user3p()); + const AstPull* const pullp = m_varAux(lhsp).pullp; bool pull1 = pullp && pullp->direction() == 1; // Else default is down AstNodeExpr* undrivenp; @@ -1435,7 +1441,7 @@ class TristateVisitor final : public TristateBaseVisitor { associateLogic(nodep, varrefp->varp()); } else { // Replace any pullup/pulldowns with assignw logic and set the - // direction of the pull in the user3() data on the var. Given + // direction of the pull in the var aux data. Given // the complexity of merging tristate drivers at any level, the // current limitation of this implementation is that a pullup/down // gets applied to all bits of a bus and a bus cannot have drivers @@ -1448,7 +1454,8 @@ class TristateVisitor final : public TristateBaseVisitor { } if (!m_graphing) { nodep->unlinkFrBack(); - VL_DO_DANGLING(pushDeletep(nodep), nodep); // Node must persist as user3p points to it + // Node must persist as var aux data points to it + VL_DO_DANGLING(pushDeletep(nodep), nodep); } } @@ -1563,7 +1570,7 @@ class TristateVisitor final : public TristateBaseVisitor { // Create new output pin const AstAssignW* outAssignp = nullptr; // If reconnected, the related assignment AstPin* outpinp = nullptr; - AstVar* const outModVarp = static_cast(nodep->modVarp()->user4p()); + AstVar* const outModVarp = m_varAux(nodep->modVarp()).outVarp; if (!outModVarp) { // At top, no need for __out as might be input only. Otherwise resolvable. if (!m_modp->isTop()) { @@ -1638,7 +1645,7 @@ class TristateVisitor final : public TristateBaseVisitor { // Propagate any pullups/pulldowns upwards if necessary if (exprrefp) { - if (AstPull* const pullp = static_cast(nodep->modVarp()->user3p())) { + if (AstPull* const pullp = m_varAux(nodep->modVarp()).pullp) { UINFO(9, "propagate pull on " << exprrefp << endl); setPullDirection(exprrefp->varp(), pullp); }