Support class new.

This commit is contained in:
Wilson Snyder 2020-04-12 18:57:12 -04:00
parent d4b6e2b2b5
commit dba88bae3c
17 changed files with 132 additions and 71 deletions

View File

@ -3567,8 +3567,8 @@ path.
Verilator class support is very limited and in active development. Verilator class support is very limited and in active development.
Verilator supports members, and methods. Verilator doe not support initial Verilator supports members, and methods. Verilator doe not support initial
values on class members, class static members, class new methods, class values on class members, class static members, class extend, or class
extend, or class parameters. parameters.
=head2 Dotted cross-hierarchy references =head2 Dotted cross-hierarchy references

View File

@ -4066,24 +4066,22 @@ public:
} }
}; };
class AstNew : public AstNodeMath { class AstNew : public AstNodeFTaskRef {
// New as constructor // New as constructor
// Don't need the class we are extracting from, as the "fromp()"'s datatype can get us to it // Don't need the class we are extracting from, as the "fromp()"'s datatype can get us to it
// Parents: math|stmt // Parents: math|stmt
// Children: varref|arraysel, math // Children: varref|arraysel, math
public: public:
AstNew(FileLine* fl, AstNode* argsp) AstNew(FileLine* fl, AstNode* pinsp)
: ASTGEN_SUPER(fl) { : ASTGEN_SUPER(fl, false, "new", pinsp) {}
addNOp2p(argsp);
}
ASTNODE_NODE_FUNCS(New) ASTNODE_NODE_FUNCS(New)
virtual V3Hash sameHash() const { return V3Hash(); } virtual V3Hash sameHash() const { return V3Hash(); }
virtual string emitVerilog() { return "new"; } virtual string emitVerilog() { return "new"; }
virtual string emitC() { V3ERROR_NA_RETURN(""); } virtual string emitC() { V3ERROR_NA_RETURN(""); }
virtual bool cleanOut() const { return true; } virtual bool cleanOut() const { return true; }
virtual bool same(const AstNode* samep) const { return true; } virtual bool same(const AstNode* samep) const { return true; }
virtual bool hasDType() const { return true; }
virtual int instrCount() const { return widthInstrs(); } virtual int instrCount() const { return widthInstrs(); }
AstNode* argsp() const { return op2p(); }
}; };
class AstNewCopy : public AstNodeMath { class AstNewCopy : public AstNodeMath {
@ -7040,6 +7038,23 @@ public:
void fromp(AstNode* nodep) { setOp1p(nodep); } void fromp(AstNode* nodep) { setOp1p(nodep); }
}; };
class AstCNew : public AstNodeCCall {
// C++ new() call
// Parents: Anything above an expression
// Children: Args to the function
public:
AstCNew(FileLine* fl, AstCFunc* funcp, AstNode* argsp = NULL)
: ASTGEN_SUPER(fl, funcp, argsp) {
statement(false);
}
// Replacement form for V3Combine
// Note this removes old attachments from the oldp
AstCNew(AstCCall* oldp, AstCFunc* funcp)
: ASTGEN_SUPER(oldp, funcp) {}
virtual bool hasDType() const { return true; }
ASTNODE_NODE_FUNCS(CNew)
};
class AstCReturn : public AstNodeStmt { class AstCReturn : public AstNodeStmt {
// C++ return from a function // C++ return from a function
// Parents: CFUNC/statement // Parents: CFUNC/statement

View File

@ -149,8 +149,13 @@ void V3CCtors::cctorsAll() {
// Process each module in turn // Process each module in turn
AstCFunc* varResetFuncp; AstCFunc* varResetFuncp;
{ {
V3CCtorsVisitor var_reset (modp, "_ctor_var_reset"); V3CCtorsVisitor var_reset(
modp, "_ctor_var_reset",
(VN_IS(modp, Class) ? EmitCBaseVisitor::symClassVar() : ""),
(VN_IS(modp, Class) ? "vlSymsp" : ""),
(VN_IS(modp, Class) ? "if (false && vlSymsp) {} // Prevent unused\n" : ""));
varResetFuncp = var_reset.builtFuncp(); varResetFuncp = var_reset.builtFuncp();
for (AstNode* np = modp->stmtsp(); np; np = np->nextp()) { for (AstNode* np = modp->stmtsp(); np; np = np->nextp()) {
if (AstVar* varp = VN_CAST(np, Var)) { if (AstVar* varp = VN_CAST(np, Var)) {
if (!varp->isIfaceParent() && !varp->isIfaceRef() if (!varp->isIfaceParent() && !varp->isIfaceRef()
@ -164,7 +169,8 @@ void V3CCtors::cctorsAll() {
if (v3Global.opt.coverage()) { if (v3Global.opt.coverage()) {
V3CCtorsVisitor configure_coverage( V3CCtorsVisitor configure_coverage(
modp, "_configure_coverage", modp, "_configure_coverage",
EmitCBaseVisitor::symClassVar() + ", bool first", "vlSymsp, first", EmitCBaseVisitor::symClassVar() + ", bool first",
"vlSymsp, first",
"if (false && vlSymsp && first) {} // Prevent unused\n"); "if (false && vlSymsp && first) {} // Prevent unused\n");
for (AstNode* np = modp->stmtsp(); np; np = np->nextp()) { for (AstNode* np = modp->stmtsp(); np; np = np->nextp()) {
if (AstCoverDecl* coverp = VN_CAST(np, CoverDecl)) { if (AstCoverDecl* coverp = VN_CAST(np, CoverDecl)) {
@ -175,14 +181,6 @@ void V3CCtors::cctorsAll() {
} }
} }
} }
if (VN_IS(modp, Class)) {
AstCFunc* funcp = new AstCFunc(modp->fileline(), "new", NULL, "");
funcp->isConstructor(true);
funcp->isStatic(false);
funcp->slow(false);
modp->addStmtp(funcp);
funcp->addStmtsp(new AstCCall(varResetFuncp->fileline(), varResetFuncp));
}
if (VN_IS(modp, Class)) { if (VN_IS(modp, Class)) {
AstCFunc* funcp = new AstCFunc(modp->fileline(), "~", NULL, ""); AstCFunc* funcp = new AstCFunc(modp->fileline(), "~", NULL, "");
funcp->isDestructor(true); funcp->isDestructor(true);

View File

@ -223,6 +223,10 @@ private:
virtual void visit(AstScopeName* nodep) VL_OVERRIDE { virtual void visit(AstScopeName* nodep) VL_OVERRIDE {
setClean(nodep, true); setClean(nodep, true);
} }
virtual void visit(AstCNew* nodep) VL_OVERRIDE {
iterateChildren(nodep);
setClean(nodep, true);
}
virtual void visit(AstSel* nodep) VL_OVERRIDE { virtual void visit(AstSel* nodep) VL_OVERRIDE {
operandTriop(nodep); operandTriop(nodep);
setClean(nodep, nodep->cleanOut()); setClean(nodep, nodep->cleanOut());

View File

@ -850,8 +850,10 @@ public:
puts(cvtToStr(nodep->fileline()->lineno())); puts(cvtToStr(nodep->fileline()->lineno()));
puts(")"); puts(")");
} }
virtual void visit(AstNew* nodep) VL_OVERRIDE { virtual void visit(AstCNew* nodep) VL_OVERRIDE {
puts("std::make_shared<" + prefixNameProtect(nodep->dtypep()) + ">("); puts("std::make_shared<" + prefixNameProtect(nodep->dtypep()) + ">(");
puts("vlSymsp"); // TODO make this part of argsp, and eliminate when unnecessary
if (nodep->argsp()) puts(", ");
iterateAndNextNull(nodep->argsp()); iterateAndNextNull(nodep->argsp());
puts(")"); puts(")");
} }

View File

@ -67,7 +67,7 @@ class EmitCInlines : EmitCBaseVisitor {
v3Global.needHeavy(true); v3Global.needHeavy(true);
iterateChildren(nodep); iterateChildren(nodep);
} }
virtual void visit(AstNew* nodep) VL_OVERRIDE { virtual void visit(AstCNew* nodep) VL_OVERRIDE {
if (v3Global.opt.savable()) v3error("Unsupported: --savable with dynamic new"); if (v3Global.opt.savable()) v3error("Unsupported: --savable with dynamic new");
iterateChildren(nodep); iterateChildren(nodep);
} }

View File

@ -643,10 +643,9 @@ public:
VSymEnt* foundp = NULL; VSymEnt* foundp = NULL;
while (!foundp) { while (!foundp) {
foundp = lookupSymp->findIdFallback(prefix + dotname); // Might be NULL foundp = lookupSymp->findIdFallback(prefix + dotname); // Might be NULL
if (prefix == "") { if (prefix.empty()) break;
break; string nextPrefix = removeLastInlineScope(prefix);
} if (prefix == nextPrefix) break;
prefix = removeLastInlineScope(prefix);
} }
if (!foundp) baddot = dotname; if (!foundp) baddot = dotname;
return foundp; return foundp;
@ -686,7 +685,8 @@ class LinkDotFindVisitor : public AstNVisitor {
bool m_inRecursion; // Inside a recursive module bool m_inRecursion; // Inside a recursive module
int m_paramNum; // Parameter number, for position based connection int m_paramNum; // Parameter number, for position based connection
int m_beginNum; // Begin block number, 0=none seen int m_beginNum; // Begin block number, 0=none seen
int m_modBeginNum; // Begin block number in module, 0=none seen bool m_explicitNew; // Hit a "new" function
int m_modBeginNum; // Begin block number in module, 0=none seen
// METHODS // METHODS
int debug() { return LinkDotState::debug(); } int debug() { return LinkDotState::debug(); }
@ -723,6 +723,13 @@ class LinkDotFindVisitor : public AstNVisitor {
} }
return NULL; return NULL;
} }
void makeImplicitNew(AstClass* nodep) {
AstFunc* newp = new AstFunc(nodep->fileline(), "new", NULL, NULL);
newp->isConstructor(true);
nodep->addMembersp(newp);
UINFO(8, "Made implicit new for " << nodep->name() << ": " << nodep << endl);
m_statep->insertBlock(m_curSymp, newp->name(), newp, m_packagep);
}
// VISITs // VISITs
virtual void visit(AstNetlist* nodep) VL_OVERRIDE { virtual void visit(AstNetlist* nodep) VL_OVERRIDE {
@ -842,10 +849,13 @@ class LinkDotFindVisitor : public AstNVisitor {
m_paramNum = 0; m_paramNum = 0;
m_beginNum = 0; m_beginNum = 0;
m_modBeginNum = 0; m_modBeginNum = 0;
m_explicitNew = false;
// m_modSymp/m_curSymp for non-packages set by AstCell above this module // m_modSymp/m_curSymp for non-packages set by AstCell above this module
// Iterate // Iterate
iterateChildren(nodep); iterateChildren(nodep);
nodep->user4(true); nodep->user4(true);
// Implicit new needed?
if (!m_explicitNew && m_statep->forPrimary()) makeImplicitNew(nodep);
} }
m_scope = oldscope; m_scope = oldscope;
m_modSymp = oldModSymp; m_modSymp = oldModSymp;
@ -971,6 +981,7 @@ class LinkDotFindVisitor : public AstNVisitor {
// NodeTask: Remember its name for later resolution // NodeTask: Remember its name for later resolution
UINFO(5," "<<nodep<<endl); UINFO(5," "<<nodep<<endl);
UASSERT_OBJ(m_curSymp && m_modSymp, nodep, "Function/Task not under module?"); UASSERT_OBJ(m_curSymp && m_modSymp, nodep, "Function/Task not under module?");
if (nodep->name() == "new") m_explicitNew = true;
// Remember the existing symbol table scope // Remember the existing symbol table scope
VSymEnt* oldCurSymp = m_curSymp; VSymEnt* oldCurSymp = m_curSymp;
{ {
@ -1232,6 +1243,7 @@ public:
m_inRecursion = false; m_inRecursion = false;
m_paramNum = 0; m_paramNum = 0;
m_beginNum = 0; m_beginNum = 0;
m_explicitNew = false;
m_modBeginNum = 0; m_modBeginNum = 0;
// //
iterate(rootp); iterate(rootp);
@ -2423,6 +2435,8 @@ private:
<<nodep->prettyName() <<nodep->prettyName()
<<"'"<<" as a "<<foundp->nodep()->typeName() <<"'"<<" as a "<<foundp->nodep()->typeName()
<<" but expected a task/function"); <<" but expected a task/function");
} else if (VN_IS(nodep, New) && m_statep->forPrearray()) {
// Resolved in V3Width
} else if (nodep->dotted() == "") { } else if (nodep->dotted() == "") {
string suggest = m_statep->suggestSymFallback( string suggest = m_statep->suggestSymFallback(
dotSymp, nodep->name(), LinkNodeMatcherFTask()); dotSymp, nodep->name(), LinkNodeMatcherFTask());

View File

@ -126,6 +126,7 @@ private:
if (nodep->valuep()) clearOptimizable(nodep, "HasInitValue"); if (nodep->valuep()) clearOptimizable(nodep, "HasInitValue");
if (!VarFlags(nodep).m_stdFuncAsn) clearStdOptimizable(nodep, "NoStdAssign"); if (!VarFlags(nodep).m_stdFuncAsn) clearStdOptimizable(nodep, "NoStdAssign");
VarFlags flags (nodep); VarFlags flags (nodep);
if ((nodep->isMovableToBlock() // Blocktemp if ((nodep->isMovableToBlock() // Blocktemp
|| !flags.m_notStd) // Or used only in block || !flags.m_notStd) // Or used only in block
&& !flags.m_notOpt // Optimizable && !flags.m_notOpt // Optimizable

View File

@ -56,6 +56,7 @@ private:
string newname = string("__PVT__")+nodep->name(); string newname = string("__PVT__")+nodep->name();
nodep->name(newname); nodep->name(newname);
nodep->editCountInc(); nodep->editCountInc();
} else if (VN_IS(nodep, CFunc) && VN_CAST(nodep, CFunc)->isConstructor()) {
} else { } else {
string rsvd = m_words.isKeyword(nodep->name()); string rsvd = m_words.isKeyword(nodep->name());
if (rsvd != "") { if (rsvd != "") {

View File

@ -501,6 +501,7 @@ class SplitUnpackedVarVisitor : public AstNVisitor, public SplitVarImpl {
{ {
m_contextp = nodep; m_contextp = nodep;
AstNodeFTask* ftaskp = nodep->taskp(); AstNodeFTask* ftaskp = nodep->taskp();
UASSERT_OBJ(ftaskp, nodep, "Unlinked");
// Iterate arguments of a function/task. // Iterate arguments of a function/task.
for (AstNode *argp = nodep->pinsp(), *paramp = ftaskp->stmtsp(); argp; for (AstNode *argp = nodep->pinsp(), *paramp = ftaskp->stmtsp(); argp;
argp = argp->nextp(), paramp = paramp ? paramp->nextp() : NULL) { argp = argp->nextp(), paramp = paramp ? paramp->nextp() : NULL) {

View File

@ -186,7 +186,7 @@ private:
m_assignwp = NULL; m_assignwp = NULL;
} }
virtual void visit(AstNodeFTaskRef* nodep) VL_OVERRIDE { virtual void visit(AstNodeFTaskRef* nodep) VL_OVERRIDE {
// Includes handling AstMethodCall // Includes handling AstMethodCall, AstNew
if (m_assignwp) { if (m_assignwp) {
// 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?
@ -204,6 +204,7 @@ private:
m_curVxp = getFTaskVertex(nodep); m_curVxp = getFTaskVertex(nodep);
if (nodep->dpiImport()) m_curVxp->noInline(true); if (nodep->dpiImport()) m_curVxp->noInline(true);
if (nodep->classMethod()) m_curVxp->noInline(true); // Until V3Task supports it if (nodep->classMethod()) m_curVxp->noInline(true); // Until V3Task supports it
if (nodep->isConstructor()) m_curVxp->noInline(true);
iterateChildren(nodep); iterateChildren(nodep);
m_curVxp = lastVxp; m_curVxp = lastVxp;
} }
@ -474,7 +475,7 @@ private:
} }
AstNode* createNonInlinedFTask(AstNodeFTaskRef* refp, const string& namePrefix, AstNode* createNonInlinedFTask(AstNodeFTaskRef* refp, const string& namePrefix,
AstVarScope* outvscp) { AstVarScope* outvscp, AstCNew*& cnewpr) {
// outvscp is the variable for functions only, if NULL, it's a task // outvscp is the variable for functions only, if NULL, it's a task
UASSERT_OBJ(refp->taskp(), refp, "Unlinked?"); UASSERT_OBJ(refp->taskp(), refp, "Unlinked?");
AstCFunc* cfuncp = m_statep->ftaskCFuncp(refp->taskp()); AstCFunc* cfuncp = m_statep->ftaskCFuncp(refp->taskp());
@ -483,12 +484,19 @@ private:
AstNode* beginp = new AstComment(refp->fileline(), AstNode* beginp = new AstComment(refp->fileline(),
string("Function: ")+refp->name(), true); string("Function: ")+refp->name(), true);
AstNodeCCall* ccallp; AstNodeCCall* ccallp;
if (AstMethodCall* mrefp = VN_CAST(refp, MethodCall)) { if (AstNew* mrefp = VN_CAST(refp, New)) {
AstCNew* cnewp = new AstCNew(refp->fileline(), cfuncp);
cnewp->dtypep(refp->dtypep());
ccallp = cnewp;
// Parent AstNew will replace with this CNew
cnewpr = cnewp;
} else if (AstMethodCall* mrefp = VN_CAST(refp, MethodCall)) {
ccallp = new AstCMethodCall(refp->fileline(), mrefp->fromp()->unlinkFrBack(), cfuncp); ccallp = new AstCMethodCall(refp->fileline(), mrefp->fromp()->unlinkFrBack(), cfuncp);
beginp->addNext(ccallp);
} else { } else {
ccallp = new AstCCall(refp->fileline(), cfuncp); ccallp = new AstCCall(refp->fileline(), cfuncp);
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());
@ -988,8 +996,10 @@ private:
// so make it unique now. // so make it unique now.
string suffix; // So, make them unique string suffix; // So, make them unique
if (!nodep->taskPublic()) suffix = "_"+m_scopep->nameDotless(); if (!nodep->taskPublic()) suffix = "_"+m_scopep->nameDotless();
string name = ((nodep->name() == "new") ? "new"
: prefix + nodep->name() + suffix);
AstCFunc* cfuncp = new AstCFunc(nodep->fileline(), AstCFunc* cfuncp = new AstCFunc(nodep->fileline(),
prefix + nodep->name() + suffix, name,
m_scopep, m_scopep,
((nodep->taskPublic() && rtnvarp) ((nodep->taskPublic() && rtnvarp)
? rtnvarp->cPubArgType(true, true) ? rtnvarp->cPubArgType(true, true)
@ -1003,6 +1013,7 @@ private:
cfuncp->dpiImportWrapper(nodep->dpiImport()); cfuncp->dpiImportWrapper(nodep->dpiImport());
cfuncp->isStatic(!(nodep->dpiImport() || nodep->taskPublic() || nodep->classMethod())); cfuncp->isStatic(!(nodep->dpiImport() || nodep->taskPublic() || nodep->classMethod()));
cfuncp->pure(nodep->pure()); cfuncp->pure(nodep->pure());
cfuncp->isConstructor(nodep->name() == "new");
//cfuncp->dpiImport // Not set in the wrapper - the called function has it set //cfuncp->dpiImport // Not set in the wrapper - the called function has it set
if (cfuncp->dpiExport()) cfuncp->cname(nodep->cname()); if (cfuncp->dpiExport()) cfuncp->cname(nodep->cname());
@ -1017,6 +1028,10 @@ private:
} else { } else {
// Need symbol table // Need symbol table
cfuncp->argTypes(EmitCBaseVisitor::symClassVar()); cfuncp->argTypes(EmitCBaseVisitor::symClassVar());
if (cfuncp->name() == "new") {
cfuncp->addInitsp(
new AstCStmt(nodep->fileline(), "_ctor_var_reset(vlSymsp);\n"));
}
} }
} }
if (nodep->dpiContext()) { if (nodep->dpiContext()) {
@ -1157,6 +1172,7 @@ private:
m_scopep = NULL; m_scopep = NULL;
} }
virtual void visit(AstNodeFTaskRef* nodep) VL_OVERRIDE { virtual void visit(AstNodeFTaskRef* nodep) VL_OVERRIDE {
// Includes handling AstMethodCall, AstNew
UASSERT_OBJ(nodep->taskp(), nodep, "Unlinked?"); UASSERT_OBJ(nodep->taskp(), nodep, "Unlinked?");
iterateIntoFTask(nodep->taskp()); // First, do hierarchical funcs iterateIntoFTask(nodep->taskp()); // First, do hierarchical funcs
UINFO(4," FTask REF "<<nodep<<endl); UINFO(4," FTask REF "<<nodep<<endl);
@ -1173,15 +1189,21 @@ private:
} }
// Create cloned statements // Create cloned statements
AstNode* beginp; AstNode* beginp;
AstCNew* cnewp = NULL;
if (m_statep->ftaskNoInline(nodep->taskp())) { if (m_statep->ftaskNoInline(nodep->taskp())) {
// This may share VarScope's with a public task, if any. Yuk. // This may share VarScope's with a public task, if any. Yuk.
beginp = createNonInlinedFTask(nodep, namePrefix, outvscp); beginp = createNonInlinedFTask(nodep, namePrefix, outvscp, cnewp /*ref*/);
} else { } else {
beginp = createInlinedFTask(nodep, namePrefix, outvscp); beginp = createInlinedFTask(nodep, namePrefix, outvscp);
} }
// Replace the ref // Replace the ref
AstNode* visitp = NULL; AstNode* visitp = NULL;
if (!nodep->isStatement()) { if (VN_IS(nodep, New)) {
UASSERT_OBJ(!nodep->isStatement(), nodep, "new is non-stmt");
UASSERT_OBJ(cnewp, nodep, "didn't create cnew for new");
nodep->replaceWith(cnewp);
visitp = insertBeforeStmt(nodep, beginp);
} else if (!nodep->isStatement()) {
UASSERT_OBJ(nodep->taskp()->isFunction(), nodep, "func reference to non-function"); UASSERT_OBJ(nodep->taskp()->isFunction(), nodep, "func reference to non-function");
AstVarRef* outrefp = new AstVarRef(nodep->fileline(), outvscp, false); AstVarRef* outrefp = new AstVarRef(nodep->fileline(), outvscp, false);
nodep->replaceWith(outrefp); nodep->replaceWith(outrefp);

View File

@ -1895,7 +1895,7 @@ private:
} }
virtual void visit(AstMethodCall* nodep) VL_OVERRIDE { virtual void visit(AstMethodCall* nodep) VL_OVERRIDE {
UINFO(5," METHODSEL "<<nodep<<endl); UINFO(5, " METHODCALL " << nodep << endl);
if (nodep->didWidth()) return; if (nodep->didWidth()) return;
if (debug()>=9) nodep->dumpTree("-mts-in: "); if (debug()>=9) nodep->dumpTree("-mts-in: ");
// Should check types the method requires, but at present we don't do much // Should check types the method requires, but at present we don't do much
@ -2431,12 +2431,20 @@ private:
return; return;
} }
nodep->dtypep(refp); nodep->dtypep(refp);
if (nodep->argsp()) {
nodep->v3error("Unsupported: new with arguments"); AstClass* classp = refp->classp();
pushDeletep(nodep->argsp()->unlinkFrBackWithNext()); UASSERT_OBJ(classp, nodep, "Unlinked");
//TODO code similar to AstNodeFTaskRef if (AstNodeFTask* ftaskp = VN_CAST(classp->findMember("new"), Func)) {
//userIterateChildren(nodep, WidthVP(SELF, BOTH).p()); nodep->taskp(ftaskp);
nodep->packagep(classp);
} else {
// Either made explicitly or V3LinkDot made implicitly
classp->v3fatalSrc("Can't find class's new");
} }
userIterate(nodep->taskp(), NULL);
userIterateChildren(nodep, NULL);
processFTaskRefArgs(nodep);
} }
virtual void visit(AstNewCopy* nodep) VL_OVERRIDE { virtual void visit(AstNewCopy* nodep) VL_OVERRIDE {
if (nodep->didWidthAndSet()) return; if (nodep->didWidthAndSet()) return;
@ -3406,7 +3414,6 @@ private:
userIterateChildren(nodep, NULL); userIterateChildren(nodep, NULL);
if (nodep->isConstructor()) { if (nodep->isConstructor()) {
// Pretend it's void so less special casing needed when look at dtypes // Pretend it's void so less special casing needed when look at dtypes
nodep->v3error("Unsupported: new constructor");
nodep->dtypeSetVoid(); nodep->dtypeSetVoid();
} else if (nodep->fvarp()) { } else if (nodep->fvarp()) {
m_funcp = VN_CAST(nodep, Func); m_funcp = VN_CAST(nodep, Func);

View File

@ -1,13 +0,0 @@
%Error: t/t_class_new.v:9:13: Unsupported: new constructor
: ... In instance t
9 | function new();
| ^~~
%Error: t/t_class_new.v:16:13: Unsupported: new constructor
: ... In instance t
16 | function new(int i);
| ^~~
%Error: t/t_class_new.v:27:12: Unsupported: new with arguments
: ... In instance t
27 | c2 = new(2);
| ^~~
%Error: Exiting due to

View File

@ -11,13 +11,11 @@ if (!$::Driver) { use FindBin; exec("$FindBin::Bin/bootstrap.pl", @ARGV, $0); di
scenarios(simulator => 1); scenarios(simulator => 1);
compile( compile(
fails => $Self->{vlt_all},
expect_filename => $Self->{golden_filename},
); );
#execute( execute(
# check_finished => 1, check_finished => 1,
# ); );
ok(1); ok(1);
1; 1;

View File

@ -16,16 +16,22 @@ class ClsArg;
function new(int i); function new(int i);
imembera = i + 1; imembera = i + 1;
endfunction endfunction
function int geta;
return imembera;
endfunction
endclass endclass
module t (/*AUTOARG*/); module t (/*AUTOARG*/);
initial begin initial begin
ClsNoArg c1; ClsNoArg c1;
ClsArg c2; ClsArg c2;
c1 = new; c1 = new;
if (c1.imembera != 5) $stop; if (c1.imembera != 5) $stop;
c2 = new(2); c2 = new(2);
if (c2.imembera != 3) $stop; if (c2.imembera != 3) $stop;
if (c2.geta() != 3) $stop;
$write("*-* All Finished *-*\n"); $write("*-* All Finished *-*\n");
$finish; $finish;

View File

@ -1,13 +1,12 @@
%Error: t/t_class_new_bad.v:10:13: Unsupported: new constructor %Error: t/t_class_new_bad.v:31:16: Too many arguments in function call to FUNC 'new'
: ... In instance t 31 | c1 = new(3);
10 | function new(); | ^
| ^~~ %Error: t/t_class_new_bad.v:32:16: Too many arguments in function call to FUNC 'new'
%Error: t/t_class_new_bad.v:17:13: Unsupported: new constructor 32 | c2 = new(3);
: ... In instance t | ^
17 | function new(int i); %Error: t/t_class_new_bad.v:33:12: Missing argument on non-defaulted argument 'i' in function call to FUNC 'new'
| ^~~ 33 | c3 = new();
%Error: t/t_class_new_bad.v:26:12: Unsupported: new with arguments | ^~~
: ... In instance t %Error: Internal Error: t/t_class_new_bad.v:33:12: ../V3Broken.cpp:#: Width != WidthMin
26 | c1 = new(3); 33 | c3 = new();
| ^~~ | ^~~
%Error: Exiting due to

View File

@ -12,6 +12,10 @@ class ClsNoArg;
endfunction endfunction
endclass endclass
class ClsNoNew;
int imembera;
endclass
class ClsArg; class ClsArg;
int imembera; int imembera;
function new(int i); function new(int i);
@ -22,9 +26,11 @@ endclass
module t (/*AUTOARG*/); module t (/*AUTOARG*/);
initial begin initial begin
ClsNoArg c1; ClsNoArg c1;
ClsArg c2; ClsNoNew c2;
ClsArg c3;
c1 = new(3); // Bad, called with arg c1 = new(3); // Bad, called with arg
c2 = new(); // Bad, called without arg c2 = new(3); // Bad, called with arg
c3 = new(); // Bad, called without arg
$stop; $stop;
end end
endmodule endmodule