mirror of
https://github.com/verilator/verilator.git
synced 2025-04-05 12:12:39 +00:00
Internals: Refactor AstNode::checkTreeIter for better stack size (#4420)
This commit is contained in:
parent
6c80457262
commit
0f66262fa1
@ -55,12 +55,16 @@ bool VNUser5InUse::s_userBusy = false;
|
||||
int AstNodeDType::s_uniqueNum = 0;
|
||||
|
||||
//######################################################################
|
||||
// V3AstType
|
||||
// VNType
|
||||
|
||||
const VNTypeInfo VNType::typeInfoTable[] = {
|
||||
#include "V3Ast__gen_type_info.h" // From ./astgen
|
||||
};
|
||||
|
||||
std::ostream& operator<<(std::ostream& os, VNType rhs);
|
||||
|
||||
//######################################################################
|
||||
// Creators
|
||||
// AstNode
|
||||
|
||||
AstNode::AstNode(VNType t, FileLine* fl)
|
||||
: m_type{t}
|
||||
@ -1098,9 +1102,57 @@ bool AstNode::sameTreeIter(const AstNode* node1p, const AstNode* node2p, bool ig
|
||||
void AstNode::checkTreeIter(const AstNode* prevBackp) const VL_MT_STABLE {
|
||||
// private: Check a tree and children
|
||||
UASSERT_OBJ(prevBackp == this->backp(), this, "Back node inconsistent");
|
||||
switch (this->type()) {
|
||||
#include "V3Ast__gen_op_checks.h"
|
||||
default: VL_UNREACHABLE; // LCOV_EXCL_LINE
|
||||
const VNTypeInfo& typeInfo = *type().typeInfo();
|
||||
for (int i = 1; i <= 4; i++) {
|
||||
AstNode* nodep = nullptr;
|
||||
switch (i) {
|
||||
case 1: nodep = op1p(); break;
|
||||
case 2: nodep = op2p(); break;
|
||||
case 3: nodep = op3p(); break;
|
||||
case 4: nodep = op4p(); break;
|
||||
default: this->v3fatalSrc("Bad case"); break;
|
||||
}
|
||||
const char* opName = typeInfo.m_opNamep[i - 1];
|
||||
switch (typeInfo.m_opType[i - 1]) {
|
||||
case VNTypeInfo::OP_UNUSED:
|
||||
UASSERT_OBJ(!nodep, this, typeInfo.m_namep << " must not use " << opName << "()");
|
||||
break;
|
||||
case VNTypeInfo::OP_USED:
|
||||
UASSERT_OBJ(nodep, this,
|
||||
typeInfo.m_namep << " must have non nullptr " << opName << "()");
|
||||
UASSERT_OBJ(!nodep->nextp(), this,
|
||||
typeInfo.m_namep << "::" << opName
|
||||
<< "() cannot have a non nullptr nextp()");
|
||||
nodep->checkTreeIter(this);
|
||||
break;
|
||||
case VNTypeInfo::OP_LIST:
|
||||
if (const AstNode* const headp = nodep) {
|
||||
const AstNode* backp = this;
|
||||
const AstNode* tailp = headp;
|
||||
const AstNode* opp = headp;
|
||||
do {
|
||||
opp->checkTreeIter(backp);
|
||||
UASSERT_OBJ(opp == headp || !opp->nextp() || !opp->m_headtailp, opp,
|
||||
"Headtailp should be null in middle of lists");
|
||||
backp = tailp = opp;
|
||||
opp = opp->nextp();
|
||||
} while (opp);
|
||||
UASSERT_OBJ(headp->m_headtailp == tailp, headp,
|
||||
"Tail in headtailp is inconsistent");
|
||||
UASSERT_OBJ(tailp->m_headtailp == headp, tailp,
|
||||
"Head in headtailp is inconsistent");
|
||||
}
|
||||
break;
|
||||
case VNTypeInfo::OP_OPTIONAL:
|
||||
if (nodep) {
|
||||
UASSERT_OBJ(!nodep->nextp(), this,
|
||||
typeInfo.m_namep << "::" << opName
|
||||
<< "() cannot have a non-nullptr nextp()");
|
||||
nodep->checkTreeIter(this);
|
||||
}
|
||||
break;
|
||||
default: this->v3fatalSrc("Bad case"); break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
14
src/V3Ast.h
14
src/V3Ast.h
@ -98,7 +98,20 @@ using MTaskIdSet = std::set<int>; // Set of mtaskIds for Var sorting
|
||||
|
||||
//######################################################################
|
||||
|
||||
struct VNTypeInfo {
|
||||
const char* m_namep;
|
||||
enum uint8_t {
|
||||
OP_UNUSED,
|
||||
OP_USED,
|
||||
OP_LIST,
|
||||
OP_OPTIONAL,
|
||||
} m_opType[4];
|
||||
const char* m_opNamep[4];
|
||||
};
|
||||
|
||||
class VNType final {
|
||||
static const VNTypeInfo typeInfoTable[];
|
||||
|
||||
public:
|
||||
#include "V3Ast__gen_type_enum.h" // From ./astgen
|
||||
// Above include has:
|
||||
@ -111,6 +124,7 @@ public:
|
||||
constexpr VNType(en _e) VL_MT_SAFE : m_e{_e} {}
|
||||
explicit VNType(int _e)
|
||||
: m_e(static_cast<en>(_e)) {} // Need () or GCC 4.8 false warning
|
||||
const VNTypeInfo* typeInfo() const VL_MT_SAFE { return &typeInfoTable[m_e]; }
|
||||
constexpr operator en() const VL_MT_SAFE { return m_e; }
|
||||
};
|
||||
constexpr bool operator==(const VNType& lhs, const VNType& rhs) VL_PURE {
|
||||
|
122
src/astgen
122
src/astgen
@ -246,7 +246,7 @@ class Cpt:
|
||||
with open_file(self.out_filename) as fho:
|
||||
togen = self.out_lines
|
||||
for line in togen:
|
||||
if type(line) is str:
|
||||
if isinstance(line, str):
|
||||
self.out_lines = [line]
|
||||
else:
|
||||
self.out_lines = []
|
||||
@ -908,6 +908,39 @@ def write_type_tests(prefix, nodeList):
|
||||
################################################################################
|
||||
|
||||
|
||||
def write_ast_type_info(filename):
|
||||
with open_file(filename) as fh:
|
||||
for node in sorted(filter(lambda _: _.isLeaf, AstNodeList),
|
||||
key=lambda _: _.typeId):
|
||||
opTypeList = []
|
||||
opNameList = []
|
||||
for n in range(1, 5):
|
||||
op = node.getOp(n)
|
||||
if not op:
|
||||
opTypeList.append('OP_UNUSED')
|
||||
opNameList.append('op{0}p'.format(n))
|
||||
else:
|
||||
name, monad, _ = op
|
||||
if not monad:
|
||||
opTypeList.append('OP_USED')
|
||||
elif monad == "Optional":
|
||||
opTypeList.append('OP_OPTIONAL')
|
||||
elif monad == "List":
|
||||
opTypeList.append('OP_LIST')
|
||||
opNameList.append(name)
|
||||
# opTypeStr = ', '.join(opTypeList)
|
||||
opTypeStr = ', '.join(
|
||||
['VNTypeInfo::{0}'.format(s) for s in opTypeList])
|
||||
opNameStr = ', '.join(['"{0}"'.format(s) for s in opNameList])
|
||||
fh.write(
|
||||
' {{ "Ast{name}", {{{opTypeStr}}}, {{{opNameStr}}} }},\n'.
|
||||
format(
|
||||
name=node.name,
|
||||
opTypeStr=opTypeStr,
|
||||
opNameStr=opNameStr,
|
||||
))
|
||||
|
||||
|
||||
def write_ast_macros(filename):
|
||||
with open_file(filename) as fh:
|
||||
|
||||
@ -998,91 +1031,6 @@ def write_ast_yystype(filename):
|
||||
node.name[1:]))
|
||||
|
||||
|
||||
def write_ast_op_checks(filename):
|
||||
with open_file(filename) as fh:
|
||||
|
||||
indent = ""
|
||||
|
||||
def emitBlock(pattern, **fmt):
|
||||
fh.write(
|
||||
textwrap.indent(textwrap.dedent(pattern),
|
||||
indent).format(**fmt))
|
||||
|
||||
for node in AstNodeList:
|
||||
if not node.isLeaf:
|
||||
continue
|
||||
|
||||
emitBlock('''\
|
||||
case VNType::at{nodeName}: {{
|
||||
const Ast{nodeName}* const currp = static_cast<const Ast{nodeName}*>(this);
|
||||
''',
|
||||
nodeName=node.name)
|
||||
indent = " "
|
||||
for n in range(1, 5):
|
||||
op = node.getOp(n)
|
||||
emitBlock("// Checking op{n}p\n", n=n)
|
||||
if op:
|
||||
name, monad, kind = op
|
||||
if not monad:
|
||||
emitBlock('''\
|
||||
UASSERT_OBJ(currp->{opName}(), currp, "Ast{nodeName} must have non nullptr {opName}()");
|
||||
UASSERT_OBJ(!currp->{opName}()->nextp(), currp, "Ast{nodeName}::{opName}() cannot have a non nullptr nextp()");
|
||||
currp->{opName}()->checkTreeIter(currp);
|
||||
''',
|
||||
n=n,
|
||||
nodeName=node.name,
|
||||
opName=name)
|
||||
elif monad == "Optional":
|
||||
emitBlock('''\
|
||||
if (Ast{kind}* const opp = currp->{opName}()) {{
|
||||
UASSERT_OBJ(!currp->{opName}()->nextp(), currp, "Ast{nodeName}::{opName}() cannot have a non nullptr nextp()");
|
||||
opp->checkTreeIter(currp);
|
||||
}}
|
||||
''',
|
||||
n=n,
|
||||
nodeName=node.name,
|
||||
opName=name,
|
||||
kind=kind)
|
||||
elif monad == "List":
|
||||
emitBlock('''\
|
||||
if (const Ast{kind}* const headp = currp->{opName}()) {{
|
||||
const AstNode* backp = currp;
|
||||
const Ast{kind}* tailp = headp;
|
||||
const Ast{kind}* opp = headp;
|
||||
do {{
|
||||
opp->checkTreeIter(backp);
|
||||
UASSERT_OBJ(opp == headp || !opp->nextp() || !opp->m_headtailp, opp, "Headtailp should be null in middle of lists");
|
||||
backp = tailp = opp;
|
||||
opp = {next};
|
||||
}} while (opp);
|
||||
// cppcheck-suppress knownConditionTrueFalse
|
||||
if (false && headp && tailp) {{}} // Prevent unused
|
||||
UASSERT_OBJ(headp->m_headtailp == tailp, headp, "Tail in headtailp is inconsistent");
|
||||
UASSERT_OBJ(tailp->m_headtailp == headp, tailp, "Head in headtailp is inconsistent");
|
||||
}}
|
||||
''',
|
||||
n=n,
|
||||
nodeName=node.name,
|
||||
opName=name,
|
||||
kind=kind,
|
||||
next="VN_AS(opp->nextp(), {kind})".format(
|
||||
kind=kind)
|
||||
if kind != "Node" else "opp->nextp()")
|
||||
else:
|
||||
sys.exit("Unknown operand type")
|
||||
else:
|
||||
emitBlock('''\
|
||||
UASSERT_OBJ(!currp->op{n}p(), currp, "Ast{nodeName} does not use op{n}p()");
|
||||
''',
|
||||
n=n,
|
||||
nodeName=node.name)
|
||||
indent = ""
|
||||
emitBlock('''\
|
||||
break;
|
||||
}}
|
||||
''')
|
||||
|
||||
|
||||
################################################################################
|
||||
# DFG code genaration
|
||||
################################################################################
|
||||
@ -1346,9 +1294,9 @@ if Args.classes:
|
||||
write_visitor_defns("Ast", AstNodeList, "VNVisitorConst")
|
||||
write_type_enum("Ast", AstNodeList)
|
||||
write_type_tests("Ast", AstNodeList)
|
||||
write_ast_type_info("V3Ast__gen_type_info.h")
|
||||
write_ast_macros("V3Ast__gen_macros.h")
|
||||
write_ast_yystype("V3Ast__gen_yystype.h")
|
||||
write_ast_op_checks("V3Ast__gen_op_checks.h")
|
||||
# Write Dfg code
|
||||
write_forward_class_decls("Dfg", DfgVertexList)
|
||||
write_visitor_decls("Dfg", DfgVertexList)
|
||||
|
Loading…
Reference in New Issue
Block a user