Internals: Add VL_DO_CLEAR delete protections. No functional change intended.

This commit is contained in:
Wilson Snyder 2020-01-18 10:29:49 -05:00
parent 2f4954ca68
commit 09199f79a6
23 changed files with 57 additions and 48 deletions

View File

@ -160,9 +160,13 @@
# define VL_DANGLING(var) do { (var) = NULL; } while(0) # define VL_DANGLING(var) do { (var) = NULL; } while(0)
#endif #endif
///< Perform an e.g. delete, then set variable to NULL to indicate must not use later ///< Perform an e.g. delete, then set variable to NULL to indicate must not use later.
///< Unlike VL_DO_CLEAR the setting of the variable is only for debug reasons.
#define VL_DO_DANGLING(stmt, var) do { do { stmt; } while(0); VL_DANGLING(var); } while(0) #define VL_DO_DANGLING(stmt, var) do { do { stmt; } while(0); VL_DANGLING(var); } while(0)
///< Perform an e.g. delete, then set variable to NULL as a requirement
#define VL_DO_CLEAR(stmt, stmt2) do { do { stmt; } while(0); do { stmt2; } while(0); } while(0)
//========================================================================= //=========================================================================
// C++-2011 // C++-2011

View File

@ -761,7 +761,7 @@ public:
} }
} }
virtual ~CdcVisitor() { virtual ~CdcVisitor() {
if (m_ofp) { delete m_ofp; m_ofp = NULL; } if (m_ofp) VL_DO_CLEAR(delete m_ofp, m_ofp = NULL);
} }
}; };

View File

@ -54,8 +54,8 @@ private:
: m_comment(comment), m_varRefp(vp), m_chgRefp(cp) {} : m_comment(comment), m_varRefp(vp), m_chgRefp(cp) {}
~ToggleEnt() {} ~ToggleEnt() {}
void cleanup() { void cleanup() {
m_varRefp->deleteTree(); m_varRefp = NULL; VL_DO_CLEAR(m_varRefp->deleteTree(), m_varRefp = NULL);
m_chgRefp->deleteTree(); m_chgRefp = NULL; VL_DO_CLEAR(m_chgRefp->deleteTree(), m_chgRefp = NULL);
} }
}; };

View File

@ -2813,7 +2813,7 @@ void EmitCImp::emitImp(AstNodeModule* modp) {
void EmitCImp::maybeSplit(AstNodeModule* modp) { void EmitCImp::maybeSplit(AstNodeModule* modp) {
if (splitNeeded()) { if (splitNeeded()) {
// Close old file // Close old file
delete m_ofp; m_ofp = NULL; VL_DO_CLEAR(delete m_ofp, m_ofp = NULL);
// Open a new file // Open a new file
m_ofp = newOutCFile(modp, !m_fast, true/*source*/, splitFilenumInc()); m_ofp = newOutCFile(modp, !m_fast, true/*source*/, splitFilenumInc());
emitImp(modp); emitImp(modp);
@ -2834,7 +2834,7 @@ void EmitCImp::main(AstNodeModule* modp, bool slow, bool fast) {
if (m_fast) { if (m_fast) {
m_ofp = newOutCFile(modp, !m_fast, false/*source*/); m_ofp = newOutCFile(modp, !m_fast, false/*source*/);
emitInt(modp); emitInt(modp);
delete m_ofp; m_ofp = NULL; VL_DO_CLEAR(delete m_ofp, m_ofp = NULL);
} }
m_ofp = newOutCFile(modp, !m_fast, true/*source*/); m_ofp = newOutCFile(modp, !m_fast, true/*source*/);
@ -2865,7 +2865,7 @@ void EmitCImp::main(AstNodeModule* modp, bool slow, bool fast) {
} }
} }
} }
delete m_ofp; m_ofp = NULL; VL_DO_CLEAR(delete m_ofp, m_ofp = NULL);
} }
//###################################################################### //######################################################################
@ -3192,7 +3192,7 @@ class EmitCTrace : EmitCStmts {
if (splitNeeded()) { if (splitNeeded()) {
// Close old file // Close old file
delete m_ofp; m_ofp = NULL; VL_DO_CLEAR(delete m_ofp, m_ofp = NULL);
// Open a new file // Open a new file
newOutCFile(splitFilenumInc()); newOutCFile(splitFilenumInc());
} }
@ -3273,7 +3273,7 @@ public:
iterate(v3Global.rootp()); iterate(v3Global.rootp());
delete m_ofp; m_ofp = NULL; VL_DO_CLEAR(delete m_ofp, m_ofp = NULL);
} }
}; };

View File

@ -480,8 +480,7 @@ void EmitCSyms::closeSplit() {
if (!m_ofp || m_ofp == m_ofpBase) return; if (!m_ofp || m_ofp == m_ofpBase) return;
puts("}\n"); puts("}\n");
delete m_ofp; VL_DO_CLEAR(delete m_ofp, m_ofp = NULL);
m_ofp = NULL;
} }
void EmitCSyms::checkSplit(bool usesVfinal) { void EmitCSyms::checkSplit(bool usesVfinal) {

View File

@ -594,7 +594,7 @@ protected:
// Just dispatch to the implementation // Just dispatch to the implementation
VInFilter::VInFilter(const string& command) { m_impp = new VInFilterImp(command); } VInFilter::VInFilter(const string& command) { m_impp = new VInFilterImp(command); }
VInFilter::~VInFilter() { if (m_impp) delete m_impp; m_impp = NULL; } VInFilter::~VInFilter() { if (m_impp) VL_DO_CLEAR(delete m_impp, m_impp = NULL); }
bool VInFilter::readWholefile(const string& filename, VInFilter::StrList& outl) { bool VInFilter::readWholefile(const string& filename, VInFilter::StrList& outl) {
if (!m_impp) v3fatalSrc("readWholefile on invalid filter"); if (!m_impp) v3fatalSrc("readWholefile on invalid filter");

View File

@ -150,9 +150,7 @@ public:
V3GraphEdge* deletep = NULL; V3GraphEdge* deletep = NULL;
for (V3GraphEdge* edgep = vxp->outBeginp(); for (V3GraphEdge* edgep = vxp->outBeginp();
edgep; edgep = edgep->outNextp()) { edgep; edgep = edgep->outNextp()) {
if (deletep) { if (deletep) VL_DO_CLEAR(deletep->unlinkDelete(), deletep = NULL);
deletep->unlinkDelete(); deletep = NULL;
}
// It should be safe to modify the graph, despite using // It should be safe to modify the graph, despite using
// the GraphPathChecker, as none of the modifications will // the GraphPathChecker, as none of the modifications will
// change what can be reached from what, nor should they // change what can be reached from what, nor should they

View File

@ -595,7 +595,7 @@ public:
add_complement_edges(); add_complement_edges();
if (debug()>=6) m_graphp->dumpDotFilePrefixed("comp_preswap"); if (debug()>=6) m_graphp->dumpDotFilePrefixed("comp_preswap");
m_tempNewerReject->unlinkDelete(graphp()); m_tempNewerReject = NULL; VL_DO_CLEAR(m_tempNewerReject->unlinkDelete(graphp()), m_tempNewerReject = NULL);
if (debug()>=6) m_graphp->dumpDotFilePrefixed("comp_out"); if (debug()>=6) m_graphp->dumpDotFilePrefixed("comp_out");
} }
~DfaGraphComplement() {} ~DfaGraphComplement() {}

View File

@ -453,11 +453,11 @@ public:
{ {
m_lifep = new LifeBlock(NULL, m_statep); m_lifep = new LifeBlock(NULL, m_statep);
iterate(nodep); iterate(nodep);
if (m_lifep) { delete m_lifep; m_lifep = NULL; } if (m_lifep) VL_DO_CLEAR(delete m_lifep, m_lifep = NULL);
} }
} }
virtual ~LifeVisitor() { virtual ~LifeVisitor() {
if (m_lifep) { delete m_lifep; m_lifep = NULL; } if (m_lifep) VL_DO_CLEAR(delete m_lifep, m_lifep = NULL);
} }
VL_UNCOPYABLE(LifeVisitor); VL_UNCOPYABLE(LifeVisitor);
}; };

View File

@ -1553,7 +1553,7 @@ V3Options::V3Options() {
} }
V3Options::~V3Options() { V3Options::~V3Options() {
delete m_impp; m_impp = NULL; VL_DO_CLEAR(delete m_impp, m_impp = NULL);
} }
void V3Options::setDebugMode(int level) { void V3Options::setDebugMode(int level) {

View File

@ -57,11 +57,11 @@ extern void yyerrorf(const char* format, ...);
V3ParseImp::~V3ParseImp() { V3ParseImp::~V3ParseImp() {
for (std::deque<string*>::iterator it = m_stringps.begin(); it != m_stringps.end(); ++it) { for (std::deque<string*>::iterator it = m_stringps.begin(); it != m_stringps.end(); ++it) {
delete (*it); VL_DO_DANGLING(delete *it, *it);
} }
m_stringps.clear(); m_stringps.clear();
for (std::deque<V3Number*>::iterator it = m_numberps.begin(); it != m_numberps.end(); ++it) { for (std::deque<V3Number*>::iterator it = m_numberps.begin(); it != m_numberps.end(); ++it) {
delete (*it); VL_DO_DANGLING(delete *it, *it);
} }
m_numberps.clear(); m_numberps.clear();
lexDestroy(); lexDestroy();
@ -297,7 +297,7 @@ V3Parse::V3Parse(AstNetlist* rootp, VInFilter* filterp, V3ParseSym* symp) {
m_impp = new V3ParseImp(rootp, filterp, symp); m_impp = new V3ParseImp(rootp, filterp, symp);
} }
V3Parse::~V3Parse() { V3Parse::~V3Parse() {
delete m_impp; m_impp = NULL; VL_DO_CLEAR(delete m_impp, m_impp = NULL);
} }
void V3Parse::parseFile(FileLine* fileline, const string& modname, bool inLibrary, void V3Parse::parseFile(FileLine* fileline, const string& modname, bool inLibrary,
const string& errmsg) { const string& errmsg) {

View File

@ -82,5 +82,5 @@ void V3ParseImp::lexNew() {
} }
void V3ParseImp::lexDestroy() { void V3ParseImp::lexDestroy() {
if (m_lexerp) { delete m_lexerp; m_lexerp = NULL; } if (m_lexerp) VL_DO_CLEAR(delete m_lexerp, m_lexerp = NULL);
} }

View File

@ -1411,7 +1411,7 @@ private:
// Remove and free the connecting edge. Must do this before // Remove and free the connecting edge. Must do this before
// propagating CP's below. // propagating CP's below.
m_sb.removeElem(mergeCanp); m_sb.removeElem(mergeCanp);
mergeEdgep->unlinkDelete(); mergeEdgep=NULL; VL_DO_CLEAR(mergeEdgep->unlinkDelete(), mergeEdgep=NULL);
} }
// This also updates cost and stepCost on recipientp // This also updates cost and stepCost on recipientp
@ -1467,7 +1467,7 @@ private:
partMergeEdgesFrom(m_mtasksp, recipientp, donorp, &m_sb); partMergeEdgesFrom(m_mtasksp, recipientp, donorp, &m_sb);
// Delete the donorp mtask from the graph // Delete the donorp mtask from the graph
donorp->unlinkDelete(m_mtasksp); donorp = NULL; VL_DO_CLEAR(donorp->unlinkDelete(m_mtasksp), donorp = NULL);
m_mergesSinceRescore++; m_mergesSinceRescore++;

View File

@ -189,7 +189,7 @@ class V3PreLex {
} }
~V3PreLex() { ~V3PreLex() {
while (!m_streampStack.empty()) { delete m_streampStack.top(); m_streampStack.pop(); } while (!m_streampStack.empty()) { delete m_streampStack.top(); m_streampStack.pop(); }
yy_delete_buffer(m_bufferState); m_bufferState = NULL; VL_DO_CLEAR(yy_delete_buffer(m_bufferState), m_bufferState = NULL);
} }
// Called by V3PreLex.l from lexer // Called by V3PreLex.l from lexer

View File

@ -285,7 +285,7 @@ public:
m_lexp->debug(debug()>=5 ? debug() : 0); // See also V3PreProc::debug() method m_lexp->debug(debug()>=5 ? debug() : 0); // See also V3PreProc::debug() method
} }
~V3PreProcImp() { ~V3PreProcImp() {
if (m_lexp) { delete m_lexp; m_lexp = NULL; } if (m_lexp) VL_DO_CLEAR(delete m_lexp, m_lexp = NULL);
} }
}; };

View File

@ -164,12 +164,12 @@ public:
void deleteUnusedAssign() { void deleteUnusedAssign() {
// If there are unused assignments in this var, kill them // If there are unused assignments in this var, kill them
if (!m_whole.m_use && !m_wordUse && m_whole.m_assignp) { if (!m_whole.m_use && !m_wordUse && m_whole.m_assignp) {
deleteAssign(m_whole.m_assignp); m_whole.m_assignp = NULL; VL_DO_CLEAR(deleteAssign(m_whole.m_assignp), m_whole.m_assignp = NULL);
} }
for (unsigned i=0; i<m_words.size(); i++) { for (unsigned i=0; i<m_words.size(); i++) {
if (!m_whole.m_use && !m_words[i].m_use if (!m_whole.m_use && !m_words[i].m_use
&& m_words[i].m_assignp && !m_words[i].m_complex) { && m_words[i].m_assignp && !m_words[i].m_complex) {
deleteAssign(m_words[i].m_assignp); m_words[i].m_assignp = NULL; VL_DO_CLEAR(deleteAssign(m_words[i].m_assignp), m_words[i].m_assignp = NULL);
} }
} }
} }

View File

@ -195,7 +195,7 @@ private:
// of multiple statements. Perhaps someday make all wassigns into always's? // of multiple statements. Perhaps someday make all wassigns into always's?
UINFO(5," IM_WireRep "<<m_assignwp<<endl); UINFO(5," IM_WireRep "<<m_assignwp<<endl);
m_assignwp->convertToAlways(); m_assignwp->convertToAlways();
pushDeletep(m_assignwp); m_assignwp = NULL; VL_DO_CLEAR(pushDeletep(m_assignwp), m_assignwp = NULL);
} }
// We make multiple edges if a task is called multiple times from another task. // We make multiple edges if a task is called multiple times from another task.
UASSERT_OBJ(nodep->taskp(), nodep, "Unlinked task"); UASSERT_OBJ(nodep->taskp(), nodep, "Unlinked task");
@ -470,7 +470,8 @@ private:
{ {
AstBegin* tempp = new AstBegin(beginp->fileline(), "[EditWrapper]", beginp); AstBegin* tempp = new AstBegin(beginp->fileline(), "[EditWrapper]", beginp);
TaskRelinkVisitor visit (tempp); TaskRelinkVisitor visit (tempp);
tempp->stmtsp()->unlinkFrBackWithNext(); VL_DO_DANGLING(tempp->deleteTree(), tempp); tempp->stmtsp()->unlinkFrBackWithNext();
VL_DO_DANGLING(tempp->deleteTree(), tempp);
} }
// //
if (debug()>=9) { beginp->dumpTreeAndNext(cout, "-iotask: "); } if (debug()>=9) { beginp->dumpTreeAndNext(cout, "-iotask: "); }
@ -488,8 +489,8 @@ private:
string("Function: ")+refp->name(), true); string("Function: ")+refp->name(), true);
AstCCall* ccallp = new AstCCall(refp->fileline(), cfuncp, NULL); AstCCall* ccallp = new AstCCall(refp->fileline(), cfuncp, NULL);
beginp->addNext(ccallp); beginp->addNext(ccallp);
// Convert complicated outputs to temp signals
// Convert complicated outputs to temp signals
V3TaskConnects tconnects = V3Task::taskConnects(refp, refp->taskp()->stmtsp()); V3TaskConnects tconnects = V3Task::taskConnects(refp, refp->taskp()->stmtsp());
for (V3TaskConnects::iterator it=tconnects.begin(); it!=tconnects.end(); ++it) { for (V3TaskConnects::iterator it=tconnects.begin(); it!=tconnects.end(); ++it) {
AstVar* portp = it->first; AstVar* portp = it->first;
@ -520,7 +521,7 @@ private:
// Correct lvalue; we didn't know when we linked // Correct lvalue; we didn't know when we linked
// This is slightly scary; are we sure no decisions were made // This is slightly scary; are we sure no decisions were made
// before here based on this not being a lvalue? // before here based on this not being a lvalue?
// Doesn't seem so; V3Unknown uses it earlier, but works ok. // Seems correct assumption; V3Unknown uses it earlier, but works ok.
V3LinkLValue::linkLValueSet(pinp); V3LinkLValue::linkLValueSet(pinp);
// Even if it's referencing a varref, we still make a temporary // Even if it's referencing a varref, we still make a temporary
@ -1073,7 +1074,8 @@ private:
{ {
AstBegin* tempp = new AstBegin(cfuncp->fileline(), "[EditWrapper]", cfuncp); AstBegin* tempp = new AstBegin(cfuncp->fileline(), "[EditWrapper]", cfuncp);
TaskRelinkVisitor visit (tempp); TaskRelinkVisitor visit (tempp);
tempp->stmtsp()->unlinkFrBackWithNext(); VL_DO_DANGLING(tempp->deleteTree(), tempp); tempp->stmtsp()->unlinkFrBackWithNext();
VL_DO_DANGLING(tempp->deleteTree(), tempp);
} }
// Delete rest of cloned task and return new func // Delete rest of cloned task and return new func
VL_DO_DANGLING(pushDeletep(nodep), nodep); VL_DO_DANGLING(pushDeletep(nodep), nodep);
@ -1410,7 +1412,8 @@ V3TaskConnects V3Task::taskConnects(AstNodeFTaskRef* nodep, AstNode* taskStmtsp)
UINFO(9,"Default pin for "<<portp<<endl); UINFO(9,"Default pin for "<<portp<<endl);
AstArg* newp = new AstArg(nodep->fileline(), portp->name(), newvaluep); AstArg* newp = new AstArg(nodep->fileline(), portp->name(), newvaluep);
if (tconnects[i].second) { // Have a "NULL" pin already defined for it if (tconnects[i].second) { // Have a "NULL" pin already defined for it
tconnects[i].second->unlinkFrBack()->deleteTree(); tconnects[i].second = NULL; VL_DO_CLEAR(tconnects[i].second->unlinkFrBack()->deleteTree(),
tconnects[i].second = NULL);
} }
tconnects[i].second = newp; tconnects[i].second = newp;
reorganize = true; reorganize = true;

View File

@ -227,7 +227,7 @@ private:
iterate(varp->dtypeSkipRefp()); iterate(varp->dtypeSkipRefp());
} }
// Cleanup // Cleanup
if (m_traValuep) { m_traValuep->deleteTree(); m_traValuep = NULL; } if (m_traValuep) VL_DO_CLEAR(m_traValuep->deleteTree(), m_traValuep = NULL);
} }
m_traVscp = NULL; m_traVscp = NULL;
m_traValuep = NULL; m_traValuep = NULL;
@ -272,7 +272,7 @@ private:
i - nodep->lsb()); i - nodep->lsb());
iterate(subtypep); iterate(subtypep);
m_traValuep->deleteTree(); m_traValuep = NULL; VL_DO_CLEAR(m_traValuep->deleteTree(), m_traValuep = NULL);
} }
m_traShowname = oldShowname; m_traShowname = oldShowname;
m_traValuep = oldValuep; m_traValuep = oldValuep;
@ -298,7 +298,7 @@ private:
(i - nodep->lsb())*subtypep->width(), (i - nodep->lsb())*subtypep->width(),
subtypep->width()); subtypep->width());
iterate(subtypep); iterate(subtypep);
m_traValuep->deleteTree(); m_traValuep = NULL; VL_DO_CLEAR(m_traValuep->deleteTree(), m_traValuep = NULL);
} }
m_traShowname = oldShowname; m_traShowname = oldShowname;
m_traValuep = oldValuep; m_traValuep = oldValuep;
@ -329,7 +329,7 @@ private:
m_traValuep->cloneTree(true), m_traValuep->cloneTree(true),
itemp->lsb(), subtypep->width()); itemp->lsb(), subtypep->width());
iterate(subtypep); iterate(subtypep);
m_traValuep->deleteTree(); m_traValuep = NULL; VL_DO_CLEAR(m_traValuep->deleteTree(), m_traValuep = NULL);
} else { // Else union, replicate fields } else { // Else union, replicate fields
iterate(subtypep); iterate(subtypep);
} }

View File

@ -86,7 +86,8 @@ private:
// Wire assigns must become always statements to deal with insertion // Wire assigns must become always statements to deal with insertion
// of multiple statements. Perhaps someday make all wassigns into always's? // of multiple statements. Perhaps someday make all wassigns into always's?
UINFO(5," IM_WireRep "<<m_assignwp<<endl); UINFO(5," IM_WireRep "<<m_assignwp<<endl);
m_assignwp->convertToAlways(); pushDeletep(m_assignwp); m_assignwp = NULL; m_assignwp->convertToAlways();
VL_DO_CLEAR(pushDeletep(m_assignwp), m_assignwp = NULL);
} }
bool needDly = (m_assigndlyp != NULL); bool needDly = (m_assigndlyp != NULL);
if (m_assigndlyp) { if (m_assigndlyp) {
@ -95,7 +96,8 @@ private:
AstNode* newp = new AstAssign(m_assigndlyp->fileline(), AstNode* newp = new AstAssign(m_assigndlyp->fileline(),
m_assigndlyp->lhsp()->unlinkFrBackWithNext(), m_assigndlyp->lhsp()->unlinkFrBackWithNext(),
m_assigndlyp->rhsp()->unlinkFrBackWithNext()); m_assigndlyp->rhsp()->unlinkFrBackWithNext());
m_assigndlyp->replaceWith(newp); pushDeletep(m_assigndlyp); m_assigndlyp = NULL; m_assigndlyp->replaceWith(newp);
VL_DO_CLEAR(pushDeletep(m_assigndlyp), m_assigndlyp = NULL);
} }
AstNode* prep = nodep; AstNode* prep = nodep;

View File

@ -184,7 +184,7 @@ private:
SimulateVisitor simvis; SimulateVisitor simvis;
AstNode* clonep = nodep->cloneTree(true); AstNode* clonep = nodep->cloneTree(true);
simvis.mainCheckTree(clonep); simvis.mainCheckTree(clonep);
pushDeletep(clonep); clonep = NULL; VL_DO_CLEAR(pushDeletep(clonep), clonep = NULL);
return simvis.optimizable(); return simvis.optimizable();
} }
@ -202,7 +202,7 @@ private:
clonep = tempp->stmtsp()->unlinkFrBackWithNext(); clonep = tempp->stmtsp()->unlinkFrBackWithNext();
tempp->deleteTree(); tempp->deleteTree();
tempp = NULL; tempp = NULL;
pushDeletep(m_varValuep); m_varValuep = NULL; VL_DO_CLEAR(pushDeletep(m_varValuep), m_varValuep = NULL);
} }
SimulateVisitor simvis; SimulateVisitor simvis;
simvis.mainParamEmulate(clonep); simvis.mainParamEmulate(clonep);
@ -329,7 +329,7 @@ private:
string nname = m_beginName + "__BRA__" + index + "__KET__"; string nname = m_beginName + "__BRA__" + index + "__KET__";
oneloopp = new AstBegin(oneloopp->fileline(), nname, oneloopp, true); oneloopp = new AstBegin(oneloopp->fileline(), nname, oneloopp, true);
} }
pushDeletep(m_varValuep); m_varValuep = NULL; VL_DO_CLEAR(pushDeletep(m_varValuep), m_varValuep = NULL);
if (newbodysp) newbodysp->addNext(oneloopp); if (newbodysp) newbodysp->addNext(oneloopp);
else newbodysp = oneloopp; else newbodysp = oneloopp;

View File

@ -112,7 +112,7 @@ AstNetlist* V3Global::makeNetlist() {
void V3Global::checkTree() { rootp()->checkTree(); } void V3Global::checkTree() { rootp()->checkTree(); }
void V3Global::clear() { void V3Global::clear() {
if (m_rootp) { m_rootp->deleteTree(); m_rootp = NULL; } if (m_rootp) VL_DO_CLEAR(m_rootp->deleteTree(), m_rootp = NULL);
} }
void V3Global::readFiles() { void V3Global::readFiles() {

View File

@ -110,7 +110,7 @@ public:
VlcTests() {} VlcTests() {}
~VlcTests() { ~VlcTests() {
for (VlcTests::ByName::iterator it=begin(); it!=end(); ++it) { for (VlcTests::ByName::iterator it=begin(); it!=end(); ++it) {
delete *it; *it=NULL; VL_DO_CLEAR(delete *it, *it=NULL);
} }
} }

View File

@ -123,7 +123,7 @@ public:
} }
void setVarDecl(AstVarType type) { m_varDecl = type; } void setVarDecl(AstVarType type) { m_varDecl = type; }
void setDType(AstNodeDType* dtypep) { void setDType(AstNodeDType* dtypep) {
if (m_varDTypep) { m_varDTypep->deleteTree(); m_varDTypep=NULL; } // It was cloned, so this is safe. if (m_varDTypep) VL_DO_CLEAR(m_varDTypep->deleteTree(), m_varDTypep=NULL); // It was cloned, so this is safe.
m_varDTypep = dtypep; m_varDTypep = dtypep;
} }
AstPackage* unitPackage(FileLine* fl) { AstPackage* unitPackage(FileLine* fl) {
@ -2362,7 +2362,10 @@ etcInst<nodep>: // IEEE: module_instantiation + gate_instantiation + udp_insta
instDecl<nodep>: instDecl<nodep>:
id parameter_value_assignmentE {INSTPREP($<fl>1,*$1,$2);} instnameList ';' id parameter_value_assignmentE {INSTPREP($<fl>1,*$1,$2);} instnameList ';'
{ $$ = $4; GRAMMARP->m_impliedDecl=false; { $$ = $4; GRAMMARP->m_impliedDecl=false;
if (GRAMMARP->m_instParamp) { GRAMMARP->m_instParamp->deleteTree(); GRAMMARP->m_instParamp = NULL; } } if (GRAMMARP->m_instParamp) {
VL_DO_CLEAR(GRAMMARP->m_instParamp->deleteTree(),
GRAMMARP->m_instParamp = NULL);
} }
// // IEEE: interface_identifier' .' modport_identifier list_of_interface_identifiers // // IEEE: interface_identifier' .' modport_identifier list_of_interface_identifiers
| id/*interface*/ '.' id/*modport*/ | id/*interface*/ '.' id/*modport*/
{ VARRESET_NONLIST(AstVarType::IFACEREF); { VARRESET_NONLIST(AstVarType::IFACEREF);