Internals: astgen: Detect bad node types after edits.

Also add checks for nodes that can be multiple types with syntax
`AstNode<AstNodeExpr|AstNodeDType>`
This commit is contained in:
Wilson Snyder 2024-09-30 22:25:28 -04:00
parent 0547108e3f
commit 03012da11c
4 changed files with 56 additions and 19 deletions

View File

@ -1121,7 +1121,7 @@ public:
bool isCompound() const override { return true; } bool isCompound() const override { return true; }
}; };
class AstRefDType final : public AstNodeDType { class AstRefDType final : public AstNodeDType {
// @astgen op1 := typeofp : Optional[AstNode] // @astgen op1 := typeofp : Optional[AstNode<AstNodeExpr|AstNodeDType>]
// @astgen op2 := classOrPackageOpp : Optional[AstNodeExpr] // @astgen op2 := classOrPackageOpp : Optional[AstNodeExpr]
// @astgen op3 := paramsp : List[AstPin] // @astgen op3 := paramsp : List[AstPin]
// //

View File

@ -589,7 +589,7 @@ public:
}; };
class AstAttrOf final : public AstNodeExpr { class AstAttrOf final : public AstNodeExpr {
// Return a value of a attribute, for example a LSB or array LSB of a signal // Return a value of a attribute, for example a LSB or array LSB of a signal
// @astgen op1 := fromp : Optional[AstNode] // Expr or DType // @astgen op1 := fromp : Optional[AstNode<AstNodeExpr|AstNodeDType>]
// @astgen op2 := dimp : Optional[AstNodeExpr] // @astgen op2 := dimp : Optional[AstNodeExpr]
VAttrType m_attrType; // What sort of extraction VAttrType m_attrType; // What sort of extraction
public: public:

View File

@ -1407,7 +1407,7 @@ private:
}; };
class AstPin final : public AstNode { class AstPin final : public AstNode {
// A port or parameter assignment on an instantiation // A port or parameter assignment on an instantiation
// @astgen op1 := exprp : Optional[AstNode] // NodeExpr or NodeDType (nullptr if unconnected) // @astgen op1 := exprp : Optional[AstNode<AstNodeExpr|AstNodeDType>] // nullptr=unconnected
// //
// @astgen ptr := m_modVarp : Optional[AstVar] // Input/output connects to on submodule // @astgen ptr := m_modVarp : Optional[AstVar] // Input/output connects to on submodule
// @astgen ptr := m_modPTypep : Optional[AstParamTypeDType] // Param type connects to on sub // @astgen ptr := m_modPTypep : Optional[AstParamTypeDType] // Param type connects to on sub
@ -1781,7 +1781,8 @@ class AstVar final : public AstNode {
// @astgen op2 := delayp : Optional[AstDelay] // Net delay // @astgen op2 := delayp : Optional[AstDelay] // Net delay
// Initial value that never changes (static const), or constructor argument for // Initial value that never changes (static const), or constructor argument for
// MTASKSTATE variables // MTASKSTATE variables
// @astgen op3 := valuep : Optional[AstNode] // May be a DType for type parameter defaults // @astgen op3 := valuep : Optional[AstNode<AstNodeExpr|AstNodeDType>]
// Value is a DType for type parameter defaults
// @astgen op4 := attrsp : List[AstNode] // Attributes during early parse // @astgen op4 := attrsp : List[AstNode] // Attributes during early parse
// @astgen ptr := m_sensIfacep : Optional[AstIface] // Interface type to which reads from this // @astgen ptr := m_sensIfacep : Optional[AstIface] // Interface type to which reads from this
// var are sensitive // var are sensitive
@ -2559,7 +2560,7 @@ public:
class AstBracketRange final : public AstNodeRange { class AstBracketRange final : public AstNodeRange {
// Parser only concept "[lhsp]", an AstUnknownRange, QueueRange or Range, // Parser only concept "[lhsp]", an AstUnknownRange, QueueRange or Range,
// unknown until lhsp type is determined // unknown until lhsp type is determined
// @astgen op1 := elementsp : AstNode // Expr or DType // @astgen op1 := elementsp : AstNode<AstNodeExpr|AstNodeDType>
public: public:
AstBracketRange(FileLine* fl, AstNode* elementsp) AstBracketRange(FileLine* fl, AstNode* elementsp)
: ASTGEN_SUPER_BracketRange(fl) { : ASTGEN_SUPER_BracketRange(fl) {

View File

@ -65,9 +65,9 @@ class Node:
assert not self.isCompleted assert not self.isCompleted
self._subClasses.append(subClass) self._subClasses.append(subClass)
def addOp(self, n, name, monad, kind): def addOp(self, n, name, monad, kind, legals):
assert 1 <= n <= 4 assert 1 <= n <= 4
self._ops[n] = (name, monad, kind) self._ops[n] = (name, monad, kind, legals)
self._arity = max(self._arity, n) self._arity = max(self._arity, n)
def getOp(self, n): def getOp(self, n):
@ -79,9 +79,9 @@ class Node:
return self.superClass.getOp(n) return self.superClass.getOp(n)
return None return None
def addPtr(self, name, monad, kind): def addPtr(self, name, monad, kind, legals):
name = re.sub(r'^m_', '', name) name = re.sub(r'^m_', '', name)
self._ptrs.append({'name': name, 'monad': monad, 'kind': kind}) self._ptrs.append({'name': name, 'monad': monad, 'kind': kind, 'legals': legals})
# Computes derived properties over entire class hierarchy. # Computes derived properties over entire class hierarchy.
# No more changes to the hierarchy are allowed once this was called # No more changes to the hierarchy are allowed once this was called
@ -519,7 +519,8 @@ def partitionAndStrip(string, separator):
def parseOpType(string): def parseOpType(string):
match = re.match(r'^(\w+)\[(\w+)\]$', string) # Return [Optional/List, AstNodeKind, LegalAstKinds]
match = re.match(r'^(\w+)\[(.*?)\]$', string)
if match: if match:
monad, kind = match.groups() monad, kind = match.groups()
if monad not in ("Optional", "List"): if monad not in ("Optional", "List"):
@ -527,9 +528,16 @@ def parseOpType(string):
kind = parseOpType(kind) kind = parseOpType(kind)
if not kind or kind[0]: if not kind or kind[0]:
return None return None
return monad, kind[1] return monad, kind[1], kind[2]
if re.match(r'^Ast(\w+)$', string): match = re.match(r'^Ast(\w+)<(.*?)>$', string)
return "", string[3:] if match:
kind = match.group(1)
legals = match.group(2)
legals = re.sub(r'\bAst', '', legals)
return "", kind, legals
match = re.match(r'^Ast(\w+)$', string)
if match:
return "", match.group(1), match.group(1)
return None return None
@ -889,7 +897,7 @@ def write_ast_type_info(filename):
opTypeList.append('OP_UNUSED') opTypeList.append('OP_UNUSED')
opNameList.append('op{0}p'.format(n)) opNameList.append('op{0}p'.format(n))
else: else:
name, monad, _ = op name, monad, _, _ = op
if not monad: if not monad:
opTypeList.append('OP_USED') opTypeList.append('OP_USED')
elif monad == "Optional": elif monad == "Optional":
@ -930,6 +938,34 @@ def write_ast_impl(filename):
emitBlock(" BROKEN_RTN(!m_{name});\n" + emitBlock(" BROKEN_RTN(!m_{name});\n" +
" BROKEN_RTN(!m_{name}->brokeExists());\n", " BROKEN_RTN(!m_{name}->brokeExists());\n",
name=ptr['name']) name=ptr['name'])
if ptr['legals'] != '':
emitBlock(" BROKEN_RTN(m_{name} && !(", name=ptr['name'])
eor = ""
for legal in ptr['legals'].split('|'):
# We use privateTypeTest, as VN_IS would assert that we know
# the type is correct, but we want to check regardless,
# to find errors after raw node edits/replacements
emitBlock("{eor}privateTypeTest<Ast{legal}>(m_{name})",
eor=eor,
name=ptr['name'],
legal=legal)
eor = " || "
emitBlock("));\n")
for i in range(1, 5):
op = node.getOp(i)
if op is None:
continue
name, _, _, legals = op
if legals != '':
emitBlock(" BROKEN_RTN({name}() && !(", name=name)
eor = ""
for legal in legals.split('|'):
emitBlock("{eor}privateTypeTest<Ast{legal}>({name}())",
eor=eor,
name=name,
legal=legal)
eor = " || "
emitBlock("));\n")
# Node's broken rules can be specialized by declaring broken() # Node's broken rules can be specialized by declaring broken()
emitBlock(" return Ast{t}::broken();\n", t=node.name) emitBlock(" return Ast{t}::broken();\n", t=node.name)
emitBlock("}}\n", t=node.name) emitBlock("}}\n", t=node.name)
@ -959,7 +995,7 @@ def write_ast_impl(filename):
op = node.getOp(i) op = node.getOp(i)
if op is None: if op is None:
continue continue
name, _, _ = op name, _, _, _ = op
emitBlock(" dumpNodeListJson(str, {name}(), \"{name}\", indent);\n", name=name) emitBlock(" dumpNodeListJson(str, {name}(), \"{name}\", indent);\n", name=name)
emitBlock("}}\n") emitBlock("}}\n")
@ -1025,7 +1061,7 @@ def write_ast_macros(filename):
op = node.getOp(n) op = node.getOp(n)
if not op: if not op:
continue continue
name, monad, kind = op name, monad, kind, _ = op
retrieve = ("VN_DBG_AS(op{n}p(), {kind})" retrieve = ("VN_DBG_AS(op{n}p(), {kind})"
if kind != "Node" else "op{n}p()").format(n=n, kind=kind) if kind != "Node" else "op{n}p()").format(n=n, kind=kind)
superOp = node.superClass.getOp(n) superOp = node.superClass.getOp(n)
@ -1113,7 +1149,7 @@ def write_dfg_macros(filename):
s=node.superClass.name) s=node.superClass.name)
for n in range(1, node.arity + 1): for n in range(1, node.arity + 1):
name, _, _ = node.getOp(n) name, _, _, _ = node.getOp(n)
emitBlock('''\ emitBlock('''\
DfgVertex* {name}() const {{ return source<{n}>(); }} DfgVertex* {name}() const {{ return source<{n}>(); }}
void {name}(DfgVertex* vtxp) {{ relinkSource<{n}>(vtxp); }} void {name}(DfgVertex* vtxp) {{ relinkSource<{n}>(vtxp); }}
@ -1300,9 +1336,9 @@ for node in AstNodeList:
for n in range(1, node.arity + 1): for n in range(1, node.arity + 1):
op = node.getOp(n) op = node.getOp(n)
if op is not None: if op is not None:
name, monad, kind = op name, monad, kind, legals = op
assert monad == "", "Cannot represent AstNode as DfgVertex" assert monad == "", "Cannot represent AstNode as DfgVertex"
vertex.addOp(n, name, "", "") vertex.addOp(n, name, "", "", "")
# Compute derived properties over the whole DfgVertex hierarchy # Compute derived properties over the whole DfgVertex hierarchy
DfgVertices["Vertex"].complete() DfgVertices["Vertex"].complete()