forked from github/verilator
Merge branch 'master' into develop-v5
This commit is contained in:
commit
fd6275a62b
2
Changes
2
Changes
@ -26,6 +26,8 @@ Verilator 4.227 devel
|
|||||||
|
|
||||||
**Minor:**
|
**Minor:**
|
||||||
|
|
||||||
|
Fix crash in gate optimization of circular logic (#3543). [Bill Flynn]
|
||||||
|
|
||||||
|
|
||||||
Verilator 4.226 2022-08-31
|
Verilator 4.226 2022-08-31
|
||||||
==========================
|
==========================
|
||||||
|
@ -79,7 +79,7 @@ ifeq ($(VL_NOOPT),1)
|
|||||||
CPPFLAGS += -O0
|
CPPFLAGS += -O0
|
||||||
else ifeq ($(VL_DEBUG),)
|
else ifeq ($(VL_DEBUG),)
|
||||||
# Optimize
|
# Optimize
|
||||||
CPPFLAGS += -O2
|
CPPFLAGS += -O3
|
||||||
else
|
else
|
||||||
# Debug
|
# Debug
|
||||||
CPPFLAGS += @CFG_CXXFLAGS_DEBUG@ -DVL_DEBUG -D_GLIBCXX_DEBUG
|
CPPFLAGS += @CFG_CXXFLAGS_DEBUG@ -DVL_DEBUG -D_GLIBCXX_DEBUG
|
||||||
|
@ -234,7 +234,8 @@ inline void AstNode::debugTreeChange(const AstNode* nodep, const char* prefix, i
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
AstNode* AstNode::addNext(AstNode* nodep, AstNode* newp) {
|
template <>
|
||||||
|
AstNode* AstNode::addNext<AstNode, AstNode>(AstNode* nodep, AstNode* newp) {
|
||||||
// Add to m_nextp, returns this
|
// Add to m_nextp, returns this
|
||||||
UDEBUGONLY(UASSERT_OBJ(newp, nodep, "Null item passed to addNext"););
|
UDEBUGONLY(UASSERT_OBJ(newp, nodep, "Null item passed to addNext"););
|
||||||
debugTreeChange(nodep, "-addNextThs: ", __LINE__, false);
|
debugTreeChange(nodep, "-addNextThs: ", __LINE__, false);
|
||||||
@ -272,7 +273,8 @@ AstNode* AstNode::addNext(AstNode* nodep, AstNode* newp) {
|
|||||||
return nodep;
|
return nodep;
|
||||||
}
|
}
|
||||||
|
|
||||||
AstNode* AstNode::addNextNull(AstNode* nodep, AstNode* newp) {
|
template <>
|
||||||
|
AstNode* AstNode::addNextNull<AstNode, AstNode>(AstNode* nodep, AstNode* newp) {
|
||||||
if (!newp) return nodep;
|
if (!newp) return nodep;
|
||||||
return addNext(nodep, newp);
|
return addNext(nodep, newp);
|
||||||
}
|
}
|
||||||
|
35
src/V3Ast.h
35
src/V3Ast.h
@ -1813,16 +1813,27 @@ public:
|
|||||||
|
|
||||||
// METHODS - Tree modifications
|
// METHODS - Tree modifications
|
||||||
// Returns nodep, adds newp to end of nodep's list
|
// Returns nodep, adds newp to end of nodep's list
|
||||||
static AstNode* addNext(AstNode* nodep, AstNode* newp);
|
template <typename T_NodeResult, typename T_NodeNext>
|
||||||
// Returns nodep, adds newp (maybe nullptr) to end of nodep's list
|
static T_NodeResult* addNext(T_NodeResult* nodep, T_NodeNext* newp) {
|
||||||
static AstNode* addNextNull(AstNode* nodep, AstNode* newp);
|
static_assert(std::is_base_of<AstNode, T_NodeResult>::value,
|
||||||
inline AstNode* addNext(AstNode* newp) { return addNext(this, newp); }
|
"'T_NodeResult' must be a subtype of AstNode");
|
||||||
inline AstNode* addNextNull(AstNode* newp) { return addNextNull(this, newp); }
|
static_assert(std::is_base_of<T_NodeResult, T_NodeNext>::value,
|
||||||
void addNextHere(AstNode* newp); // Insert newp at this->nextp
|
"'T_NodeNext' must be a subtype of 'T_NodeResult'");
|
||||||
void addPrev(AstNode* newp) {
|
return static_cast<T_NodeResult*>(addNext<AstNode, AstNode>(nodep, newp));
|
||||||
replaceWith(newp);
|
|
||||||
newp->addNext(this);
|
|
||||||
}
|
}
|
||||||
|
// Returns nodep, adds newp (maybe nullptr) to end of nodep's list
|
||||||
|
template <typename T_NodeResult, typename T_NodeNext>
|
||||||
|
static T_NodeResult* addNextNull(T_NodeResult* nodep, T_NodeNext* newp) {
|
||||||
|
static_assert(std::is_base_of<AstNode, T_NodeResult>::value,
|
||||||
|
"'T_NodeResult' must be a subtype of AstNode");
|
||||||
|
static_assert(std::is_base_of<T_NodeResult, T_NodeNext>::value,
|
||||||
|
"'T_NodeNext' must be a subtype of 'T_NodeResult'");
|
||||||
|
return static_cast<T_NodeResult*>(addNextNull<AstNode, AstNode>(nodep, newp));
|
||||||
|
}
|
||||||
|
inline AstNode* addNext(AstNode* newp);
|
||||||
|
inline AstNode* addNextNull(AstNode* newp);
|
||||||
|
void addNextHere(AstNode* newp); // Insert newp at this->nextp
|
||||||
|
inline void addPrev(AstNode* newp);
|
||||||
void addHereThisAsNext(AstNode* newp); // Adds at old place of this, this becomes next
|
void addHereThisAsNext(AstNode* newp); // Adds at old place of this, this becomes next
|
||||||
void replaceWith(AstNode* newp); // Replace current node in tree with new node
|
void replaceWith(AstNode* newp); // Replace current node in tree with new node
|
||||||
AstNode* unlinkFrBack(VNRelinker* linkerp
|
AstNode* unlinkFrBack(VNRelinker* linkerp
|
||||||
@ -2112,6 +2123,12 @@ public:
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Forward declarations of specializations defined in V3Ast.cpp
|
||||||
|
template <>
|
||||||
|
AstNode* AstNode::addNext<AstNode, AstNode>(AstNode* nodep, AstNode* newp);
|
||||||
|
template <>
|
||||||
|
AstNode* AstNode::addNextNull<AstNode, AstNode>(AstNode* nodep, AstNode* newp);
|
||||||
|
|
||||||
// Specialisations of privateTypeTest
|
// Specialisations of privateTypeTest
|
||||||
#include "V3Ast__gen_impl.h" // From ./astgen
|
#include "V3Ast__gen_impl.h" // From ./astgen
|
||||||
|
|
||||||
|
@ -23,7 +23,14 @@
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
//######################################################################
|
//######################################################################
|
||||||
// Inline ACCESSORS
|
// Inline METHODS
|
||||||
|
|
||||||
|
inline AstNode* AstNode::addNext(AstNode* newp) { return addNext(this, newp); }
|
||||||
|
inline AstNode* AstNode::addNextNull(AstNode* newp) { return addNextNull(this, newp); }
|
||||||
|
inline void AstNode::addPrev(AstNode* newp) {
|
||||||
|
replaceWith(newp);
|
||||||
|
newp->addNext(this);
|
||||||
|
}
|
||||||
|
|
||||||
inline int AstNode::width() const { return dtypep() ? dtypep()->width() : 0; }
|
inline int AstNode::width() const { return dtypep() ? dtypep()->width() : 0; }
|
||||||
inline int AstNode::widthMin() const { return dtypep() ? dtypep()->widthMin() : 0; }
|
inline int AstNode::widthMin() const { return dtypep() ? dtypep()->widthMin() : 0; }
|
||||||
|
@ -2564,9 +2564,6 @@ public:
|
|||||||
AstVar* varp() const { return m_varp; } // [After Link] Pointer to variable
|
AstVar* varp() const { return m_varp; } // [After Link] Pointer to variable
|
||||||
AstScope* scopep() const { return m_scopep; } // Pointer to scope it's under
|
AstScope* scopep() const { return m_scopep; } // Pointer to scope it's under
|
||||||
void scopep(AstScope* nodep) { m_scopep = nodep; }
|
void scopep(AstScope* nodep) { m_scopep = nodep; }
|
||||||
// op1 = Calculation of value of variable, nullptr=complicated
|
|
||||||
AstNode* valuep() const { return op1p(); }
|
|
||||||
void valuep(AstNode* valuep) { addOp1p(valuep); }
|
|
||||||
bool isTrace() const { return m_trace; }
|
bool isTrace() const { return m_trace; }
|
||||||
void trace(bool flag) { m_trace = flag; }
|
void trace(bool flag) { m_trace = flag; }
|
||||||
};
|
};
|
||||||
|
@ -2079,7 +2079,7 @@ private:
|
|||||||
AstSel* const sel2p = new AstSel(conp->fileline(), rhs2p, lsb2, msb2 - lsb2 + 1);
|
AstSel* const sel2p = new AstSel(conp->fileline(), rhs2p, lsb2, msb2 - lsb2 + 1);
|
||||||
// Make new assigns of same flavor as old one
|
// Make new assigns of same flavor as old one
|
||||||
//*** Not cloneTree; just one node.
|
//*** Not cloneTree; just one node.
|
||||||
AstNode* newp = nullptr;
|
AstNodeAssign* newp = nullptr;
|
||||||
if (!need_temp) {
|
if (!need_temp) {
|
||||||
AstNodeAssign* const asn1ap = VN_AS(nodep->cloneType(lc1p, sel1p), NodeAssign);
|
AstNodeAssign* const asn1ap = VN_AS(nodep->cloneType(lc1p, sel1p), NodeAssign);
|
||||||
AstNodeAssign* const asn2ap = VN_AS(nodep->cloneType(lc2p, sel2p), NodeAssign);
|
AstNodeAssign* const asn2ap = VN_AS(nodep->cloneType(lc2p, sel2p), NodeAssign);
|
||||||
|
@ -418,7 +418,6 @@ private:
|
|||||||
void consumedMark();
|
void consumedMark();
|
||||||
void consumedMarkRecurse(GateEitherVertex* vertexp);
|
void consumedMarkRecurse(GateEitherVertex* vertexp);
|
||||||
void consumedMove();
|
void consumedMove();
|
||||||
void replaceAssigns();
|
|
||||||
void dedupe();
|
void dedupe();
|
||||||
void mergeAssigns();
|
void mergeAssigns();
|
||||||
void decomposeClkVectors();
|
void decomposeClkVectors();
|
||||||
@ -455,7 +454,6 @@ private:
|
|||||||
m_graph.dumpDotFilePrefixed("gate_opt");
|
m_graph.dumpDotFilePrefixed("gate_opt");
|
||||||
// Rewrite assignments
|
// Rewrite assignments
|
||||||
consumedMove();
|
consumedMove();
|
||||||
replaceAssigns();
|
|
||||||
}
|
}
|
||||||
virtual void visit(AstNodeModule* nodep) override {
|
virtual void visit(AstNodeModule* nodep) override {
|
||||||
VL_RESTORER(m_modp);
|
VL_RESTORER(m_modp);
|
||||||
@ -665,14 +663,7 @@ void GateVisitor::optimizeSignals(bool allowMultiIn) {
|
|||||||
while (V3GraphEdge* const edgep = vvertexp->inBeginp()) {
|
while (V3GraphEdge* const edgep = vvertexp->inBeginp()) {
|
||||||
VL_DO_DANGLING(edgep->unlinkDelete(), edgep);
|
VL_DO_DANGLING(edgep->unlinkDelete(), edgep);
|
||||||
}
|
}
|
||||||
// Clone tree so we remember it for tracing, and keep the pointer
|
// Mark the vertex so we don't mark it as being unconsumed in the next step
|
||||||
// to the "ALWAYS" part of the tree as part of this statement
|
|
||||||
// That way if a later signal optimization that
|
|
||||||
// retained a pointer to the always can
|
|
||||||
// optimize it further
|
|
||||||
VL_DO_DANGLING(vvertexp->varScp()->valuep(logicp->unlinkFrBack()), logicp);
|
|
||||||
// Mark the vertex so we don't mark it as being
|
|
||||||
// unconsumed in the next step
|
|
||||||
vvertexp->user(true);
|
vvertexp->user(true);
|
||||||
logicVertexp->user(true);
|
logicVertexp->user(true);
|
||||||
}
|
}
|
||||||
@ -704,45 +695,6 @@ bool GateVisitor::elimLogicOkOutputs(GateLogicVertex* consumeVertexp,
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void GateVisitor::replaceAssigns() {
|
|
||||||
for (V3GraphVertex* itp = m_graph.verticesBeginp(); itp; itp = itp->verticesNextp()) {
|
|
||||||
if (const GateVarVertex* const vvertexp = dynamic_cast<GateVarVertex*>(itp)) {
|
|
||||||
// Take the Comments/assigns that were moved to the VarScope and change them to a
|
|
||||||
// simple value assignment
|
|
||||||
const AstVarScope* const vscp = vvertexp->varScp();
|
|
||||||
if (vscp->valuep() && !VN_IS(vscp->valuep(), NodeMath)) {
|
|
||||||
// if (debug() > 9) vscp->dumpTree(cout, "-vscPre: ");
|
|
||||||
while (AstNode* delp = VN_CAST(vscp->valuep(), Comment)) {
|
|
||||||
VL_DO_DANGLING(delp->unlinkFrBack()->deleteTree(), delp);
|
|
||||||
}
|
|
||||||
if (AstInitial* const delp = VN_CAST(vscp->valuep(), Initial)) {
|
|
||||||
AstNode* const bodyp = delp->bodysp();
|
|
||||||
bodyp->unlinkFrBackWithNext();
|
|
||||||
delp->replaceWith(bodyp);
|
|
||||||
VL_DO_DANGLING(delp->deleteTree(), delp);
|
|
||||||
}
|
|
||||||
if (AstAlways* const delp = VN_CAST(vscp->valuep(), Always)) {
|
|
||||||
AstNode* const bodyp = delp->bodysp();
|
|
||||||
bodyp->unlinkFrBackWithNext();
|
|
||||||
delp->replaceWith(bodyp);
|
|
||||||
VL_DO_DANGLING(delp->deleteTree(), delp);
|
|
||||||
}
|
|
||||||
if (AstNodeAssign* const delp = VN_CAST(vscp->valuep(), NodeAssign)) {
|
|
||||||
AstNode* const rhsp = delp->rhsp();
|
|
||||||
rhsp->unlinkFrBack();
|
|
||||||
delp->replaceWith(rhsp);
|
|
||||||
VL_DO_DANGLING(delp->deleteTree(), delp);
|
|
||||||
}
|
|
||||||
// if (debug() > 9) {vscp->dumpTree(cout, "-vscDone: "); cout<<endl;}
|
|
||||||
if (!VN_IS(vscp->valuep(), NodeMath) || vscp->valuep()->nextp()) {
|
|
||||||
vscp->dumpTree(std::cerr, "vscStrange: ");
|
|
||||||
vscp->v3fatalSrc("Value of varscope not mathematical");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//----------------------------------------------------------------------
|
//----------------------------------------------------------------------
|
||||||
|
|
||||||
void GateVisitor::consumedMark() {
|
void GateVisitor::consumedMark() {
|
||||||
@ -1044,11 +996,18 @@ static void eliminate(AstNode* logicp,
|
|||||||
const std::unordered_map<AstVarScope*, AstNode*>& substitutions,
|
const std::unordered_map<AstVarScope*, AstNode*>& substitutions,
|
||||||
GateDedupeVarVisitor* varVisp) {
|
GateDedupeVarVisitor* varVisp) {
|
||||||
|
|
||||||
const std::function<void(AstNodeVarRef*)> visit
|
// Recursion filter holding already replaced variables
|
||||||
= [&substitutions, &visit, varVisp](AstNodeVarRef* nodep) -> void {
|
std::unordered_set<const AstVarScope*> replaced(substitutions.size() * 2);
|
||||||
|
|
||||||
|
const std::function<void(AstNodeVarRef*)> visit = [&, varVisp](AstNodeVarRef* nodep) -> void {
|
||||||
// See if this variable has a substitution
|
// See if this variable has a substitution
|
||||||
const auto& it = substitutions.find(nodep->varScopep());
|
AstVarScope* const vscp = nodep->varScopep();
|
||||||
|
const auto& it = substitutions.find(vscp);
|
||||||
if (it == substitutions.end()) return;
|
if (it == substitutions.end()) return;
|
||||||
|
|
||||||
|
// Do not substitute circular logic
|
||||||
|
if (!replaced.insert(vscp).second) return;
|
||||||
|
|
||||||
AstNode* const substp = it->second;
|
AstNode* const substp = it->second;
|
||||||
|
|
||||||
// Substitute in the new tree
|
// Substitute in the new tree
|
||||||
@ -1075,6 +1034,9 @@ static void eliminate(AstNode* logicp,
|
|||||||
VL_DO_DANGLING(nodep->deleteTree(), nodep);
|
VL_DO_DANGLING(nodep->deleteTree(), nodep);
|
||||||
// Recursively substitute the new tree
|
// Recursively substitute the new tree
|
||||||
newp->foreach<AstNodeVarRef>(visit);
|
newp->foreach<AstNodeVarRef>(visit);
|
||||||
|
|
||||||
|
// Remove from recursion filter
|
||||||
|
replaced.erase(vscp);
|
||||||
};
|
};
|
||||||
|
|
||||||
logicp->foreach<AstNodeVarRef>(visit);
|
logicp->foreach<AstNodeVarRef>(visit);
|
||||||
@ -1143,11 +1105,6 @@ private:
|
|||||||
while (V3GraphEdge* const inedgep = vvertexp->inBeginp()) {
|
while (V3GraphEdge* const inedgep = vvertexp->inBeginp()) {
|
||||||
VL_DO_DANGLING(inedgep->unlinkDelete(), inedgep);
|
VL_DO_DANGLING(inedgep->unlinkDelete(), inedgep);
|
||||||
}
|
}
|
||||||
// replaceAssigns() does the deleteTree on lvertexNodep in a later step
|
|
||||||
AstNode* lvertexNodep = lvertexp->nodep();
|
|
||||||
lvertexNodep->unlinkFrBack();
|
|
||||||
vvertexp->varScp()->valuep(lvertexNodep);
|
|
||||||
lvertexNodep = nullptr;
|
|
||||||
vvertexp->user(true);
|
vvertexp->user(true);
|
||||||
lvertexp->user(true);
|
lvertexp->user(true);
|
||||||
}
|
}
|
||||||
@ -1538,16 +1495,5 @@ void GateVisitor::decomposeClkVectors() {
|
|||||||
void V3Gate::gateAll(AstNetlist* nodep) {
|
void V3Gate::gateAll(AstNetlist* nodep) {
|
||||||
UINFO(2, __FUNCTION__ << ": " << endl);
|
UINFO(2, __FUNCTION__ << ": " << endl);
|
||||||
{ const GateVisitor visitor{nodep}; } // Destruct before checking
|
{ const GateVisitor visitor{nodep}; } // Destruct before checking
|
||||||
|
|
||||||
nodep->foreach<AstVarScope>([](AstVarScope* nodep) {
|
|
||||||
if (AstNodeAssign* const assp = VN_CAST(nodep->valuep(), NodeAssign)) {
|
|
||||||
UINFO(5, " Removeassign " << assp << endl);
|
|
||||||
AstNode* const valuep = assp->rhsp();
|
|
||||||
valuep->unlinkFrBack();
|
|
||||||
assp->replaceWith(valuep);
|
|
||||||
VL_DO_DANGLING(assp->deleteTree(), assp);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
V3Global::dumpCheckGlobalTree("gate", 0, v3Global.opt.dumpTreeLevel(__FILE__) >= 3);
|
V3Global::dumpCheckGlobalTree("gate", 0, v3Global.opt.dumpTreeLevel(__FILE__) >= 3);
|
||||||
}
|
}
|
||||||
|
@ -67,10 +67,10 @@ void V3ParseImp::parserClear() {
|
|||||||
//======================================================================
|
//======================================================================
|
||||||
// V3ParseGrammar functions requiring bison state
|
// V3ParseGrammar functions requiring bison state
|
||||||
|
|
||||||
AstNode* V3ParseGrammar::argWrapList(AstNode* nodep) {
|
AstArg* V3ParseGrammar::argWrapList(AstNode* nodep) {
|
||||||
// Convert list of expressions to list of arguments
|
// Convert list of expressions to list of arguments
|
||||||
if (!nodep) return nullptr;
|
if (!nodep) return nullptr;
|
||||||
AstNode* outp = nullptr;
|
AstArg* outp = nullptr;
|
||||||
AstBegin* const tempp = new AstBegin(nodep->fileline(), "[EditWrapper]", nodep);
|
AstBegin* const tempp = new AstBegin(nodep->fileline(), "[EditWrapper]", nodep);
|
||||||
while (nodep) {
|
while (nodep) {
|
||||||
AstNode* const nextp = nodep->nextp();
|
AstNode* const nextp = nodep->nextp();
|
||||||
|
@ -372,7 +372,7 @@ LogicByScope fixCuts(AstNetlist* netlistp, const std::vector<VarVertex*>& cutVer
|
|||||||
for (AstVarScope* const vscp : lvtx2Cuts[lvtxp]) {
|
for (AstVarScope* const vscp : lvtx2Cuts[lvtxp]) {
|
||||||
AstVarRef* const refp = new AstVarRef{flp, vscp, VAccess::READ};
|
AstVarRef* const refp = new AstVarRef{flp, vscp, VAccess::READ};
|
||||||
AstSenItem* const nextp = new AstSenItem{flp, VEdgeType::ET_HYBRID, refp};
|
AstSenItem* const nextp = new AstSenItem{flp, VEdgeType::ET_HYBRID, refp};
|
||||||
senItemsp = VN_AS(AstNode::addNext(senItemsp, nextp), SenItem);
|
senItemsp = AstNode::addNext(senItemsp, nextp);
|
||||||
}
|
}
|
||||||
AstSenTree* const senTree = new AstSenTree{flp, senItemsp};
|
AstSenTree* const senTree = new AstSenTree{flp, senItemsp};
|
||||||
// Add logic to result with new sensitivity
|
// Add logic to result with new sensitivity
|
||||||
|
@ -139,12 +139,13 @@ class SliceVisitor final : public VNVisitor {
|
|||||||
// Left and right could have different msb/lsbs/endianness, but #elements is common
|
// Left and right could have different msb/lsbs/endianness, but #elements is common
|
||||||
// and all variables are realigned to start at zero
|
// and all variables are realigned to start at zero
|
||||||
// Assign of a little endian'ed slice to a big endian one must reverse the elements
|
// Assign of a little endian'ed slice to a big endian one must reverse the elements
|
||||||
AstNode* newlistp = nullptr;
|
AstNodeAssign* newlistp = nullptr;
|
||||||
const int elements = arrayp->rangep()->elementsConst();
|
const int elements = arrayp->rangep()->elementsConst();
|
||||||
for (int offset = 0; offset < elements; ++offset) {
|
for (int offset = 0; offset < elements; ++offset) {
|
||||||
AstNode* const newp = nodep->cloneType // AstNodeAssign
|
AstNodeAssign* const newp
|
||||||
(cloneAndSel(nodep->lhsp(), elements, offset),
|
= VN_AS(nodep->cloneType(cloneAndSel(nodep->lhsp(), elements, offset),
|
||||||
cloneAndSel(nodep->rhsp(), elements, offset));
|
cloneAndSel(nodep->rhsp(), elements, offset)),
|
||||||
|
NodeAssign);
|
||||||
if (debug() >= 9) newp->dumpTree(cout, "-new ");
|
if (debug() >= 9) newp->dumpTree(cout, "-new ");
|
||||||
newlistp = AstNode::addNextNull(newlistp, newp);
|
newlistp = AstNode::addNextNull(newlistp, newp);
|
||||||
}
|
}
|
||||||
|
@ -280,11 +280,7 @@ private:
|
|||||||
addIgnore(ignoreReasonp);
|
addIgnore(ignoreReasonp);
|
||||||
} else {
|
} else {
|
||||||
++m_statSigs;
|
++m_statSigs;
|
||||||
if (AstNode* const valuep = m_traVscp->valuep()) {
|
|
||||||
m_traValuep = valuep->cloneTree(false);
|
|
||||||
} else {
|
|
||||||
m_traValuep = new AstVarRef{m_traVscp->fileline(), m_traVscp, VAccess::READ};
|
m_traValuep = new AstVarRef{m_traVscp->fileline(), m_traVscp, VAccess::READ};
|
||||||
}
|
|
||||||
// Recurse into data type of the signal. The visit methods will add AstTraceDecls.
|
// Recurse into data type of the signal. The visit methods will add AstTraceDecls.
|
||||||
iterate(m_traVscp->varp()->dtypep()->skipRefToEnump());
|
iterate(m_traVscp->varp()->dtypep()->skipRefToEnump());
|
||||||
// Cleanup
|
// Cleanup
|
||||||
|
@ -30,6 +30,7 @@
|
|||||||
|
|
||||||
#include "V3Ast.h"
|
#include "V3Ast.h"
|
||||||
#include "V3Global.h"
|
#include "V3Global.h"
|
||||||
|
#include "V3Stats.h"
|
||||||
#include "V3String.h"
|
#include "V3String.h"
|
||||||
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
@ -475,4 +476,5 @@ public:
|
|||||||
void V3Undriven::undrivenAll(AstNetlist* nodep) {
|
void V3Undriven::undrivenAll(AstNetlist* nodep) {
|
||||||
UINFO(2, __FUNCTION__ << ": " << endl);
|
UINFO(2, __FUNCTION__ << ": " << endl);
|
||||||
{ UndrivenVisitor{nodep}; }
|
{ UndrivenVisitor{nodep}; }
|
||||||
|
if (v3Global.opt.stats()) V3Stats::statsStage("undriven");
|
||||||
}
|
}
|
||||||
|
@ -102,7 +102,7 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
// METHODS
|
// METHODS
|
||||||
AstNode* argWrapList(AstNode* nodep);
|
AstArg* argWrapList(AstNode* nodep);
|
||||||
bool allTracingOn(FileLine* fl) {
|
bool allTracingOn(FileLine* fl) {
|
||||||
return v3Global.opt.trace() && m_tracingParse && fl->tracingOn();
|
return v3Global.opt.trace() && m_tracingParse && fl->tracingOn();
|
||||||
}
|
}
|
||||||
@ -5100,7 +5100,7 @@ idArrayedForeach<nodep>: // IEEE: id + select (under foreach expression)
|
|||||||
| idArrayed '[' expr ',' loop_variables ']'
|
| idArrayed '[' expr ',' loop_variables ']'
|
||||||
{ $3 = AstNode::addNextNull($3, $5); $$ = new AstSelLoopVars($2, $1, $3); }
|
{ $3 = AstNode::addNextNull($3, $5); $$ = new AstSelLoopVars($2, $1, $3); }
|
||||||
| idArrayed '[' ',' loop_variables ']'
|
| idArrayed '[' ',' loop_variables ']'
|
||||||
{ $4 = AstNode::addNextNull(new AstEmpty{$3}, $4); $$ = new AstSelLoopVars($2, $1, $4); }
|
{ $4 = AstNode::addNextNull(static_cast<AstNode*>(new AstEmpty{$3}), $4); $$ = new AstSelLoopVars($2, $1, $4); }
|
||||||
;
|
;
|
||||||
|
|
||||||
// VarRef without any dots or vectorizaion
|
// VarRef without any dots or vectorizaion
|
||||||
|
18
test_regress/t/t_gate_loop.pl
Executable file
18
test_regress/t/t_gate_loop.pl
Executable file
@ -0,0 +1,18 @@
|
|||||||
|
#!/usr/bin/env perl
|
||||||
|
if (!$::Driver) { use FindBin; exec("$FindBin::Bin/bootstrap.pl", @ARGV, $0); die; }
|
||||||
|
# DESCRIPTION: Verilator: Verilog Test driver/expect definition
|
||||||
|
#
|
||||||
|
# Copyright 2022 by Geza Lore. 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(simulator => 1);
|
||||||
|
|
||||||
|
compile(
|
||||||
|
verilator_flags2 => ["-Wno-UNOPTFLAT"]
|
||||||
|
);
|
||||||
|
|
||||||
|
ok(1);
|
||||||
|
1;
|
14
test_regress/t/t_gate_loop.v
Normal file
14
test_regress/t/t_gate_loop.v
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
// DESCRIPTION: Verilator: Verilog Test module
|
||||||
|
//
|
||||||
|
// This file ONLY is placed under the Creative Commons Public Domain, for
|
||||||
|
// any use, without warranty, 2022 by Geza Lore.
|
||||||
|
// SPDX-License-Identifier: CC0-1.0
|
||||||
|
|
||||||
|
module t;
|
||||||
|
wire a;
|
||||||
|
wire b;
|
||||||
|
wire c;
|
||||||
|
assign a = b;
|
||||||
|
assign b = c;
|
||||||
|
assign c = a;
|
||||||
|
endmodule
|
Loading…
Reference in New Issue
Block a user