2021-06-05 16:40:56 +00:00
|
|
|
// -*- mode: C++; c-file-style: "cc-mode" -*-
|
|
|
|
//*************************************************************************
|
|
|
|
// DESCRIPTION: Verilator: Ast node inline functions
|
|
|
|
//
|
|
|
|
// Code available from: https://verilator.org
|
|
|
|
//
|
|
|
|
//*************************************************************************
|
|
|
|
//
|
2022-01-01 13:26:40 +00:00
|
|
|
// Copyright 2003-2022 by Wilson Snyder. This program is free software; you
|
2021-06-05 16:40:56 +00:00
|
|
|
// 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
|
|
|
|
//
|
|
|
|
//*************************************************************************
|
|
|
|
|
|
|
|
#ifndef VERILATOR_V3ASTINLINES_H_
|
|
|
|
#define VERILATOR_V3ASTINLINES_H_
|
|
|
|
|
2022-09-15 12:10:39 +00:00
|
|
|
#ifndef VERILATOR_V3AST_H_
|
2021-06-05 16:40:56 +00:00
|
|
|
#error "Use V3Ast.h as the include"
|
2022-09-15 12:10:39 +00:00
|
|
|
#include "V3Ast.h" // This helps code analysis tools pick up symbols in V3Ast.h and relaed
|
2021-06-05 16:40:56 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
//######################################################################
|
2022-08-29 13:26:00 +00:00
|
|
|
// Inline METHODS
|
|
|
|
|
2022-09-16 12:17:38 +00:00
|
|
|
int AstNode::width() const { return dtypep() ? dtypep()->width() : 0; }
|
|
|
|
int AstNode::widthMin() const { return dtypep() ? dtypep()->widthMin() : 0; }
|
|
|
|
bool AstNode::width1() const { // V3Const uses to know it can optimize
|
2021-06-05 16:40:56 +00:00
|
|
|
return dtypep() && dtypep()->width() == 1;
|
|
|
|
}
|
2022-09-16 12:17:38 +00:00
|
|
|
int AstNode::widthInstrs() const {
|
2021-06-05 16:40:56 +00:00
|
|
|
return (!dtypep() ? 1 : (dtypep()->isWide() ? dtypep()->widthWords() : 1));
|
|
|
|
}
|
2022-10-18 21:07:09 +00:00
|
|
|
bool AstNode::isDouble() const VL_MT_SAFE {
|
2021-10-22 12:56:48 +00:00
|
|
|
return dtypep() && VN_IS(dtypep(), BasicDType) && VN_AS(dtypep(), BasicDType)->isDouble();
|
2021-06-05 16:40:56 +00:00
|
|
|
}
|
2022-10-18 21:07:09 +00:00
|
|
|
bool AstNode::isString() const VL_MT_SAFE {
|
2021-06-05 16:40:56 +00:00
|
|
|
return dtypep() && dtypep()->basicp() && dtypep()->basicp()->isString();
|
|
|
|
}
|
2022-09-16 12:17:38 +00:00
|
|
|
bool AstNode::isSigned() const { return dtypep() && dtypep()->isSigned(); }
|
2021-06-05 16:40:56 +00:00
|
|
|
|
2022-09-16 12:17:38 +00:00
|
|
|
bool AstNode::isZero() const {
|
2021-10-22 14:15:42 +00:00
|
|
|
return (VN_IS(this, Const) && VN_AS(this, Const)->num().isEqZero());
|
2021-06-05 16:40:56 +00:00
|
|
|
}
|
2022-09-16 12:17:38 +00:00
|
|
|
bool AstNode::isNeqZero() const {
|
2021-10-22 14:15:42 +00:00
|
|
|
return (VN_IS(this, Const) && VN_AS(this, Const)->num().isNeqZero());
|
2021-06-05 16:40:56 +00:00
|
|
|
}
|
2022-09-16 12:17:38 +00:00
|
|
|
bool AstNode::isOne() const { return (VN_IS(this, Const) && VN_AS(this, Const)->num().isEqOne()); }
|
|
|
|
bool AstNode::isAllOnes() const {
|
2021-10-22 14:15:42 +00:00
|
|
|
return (VN_IS(this, Const) && VN_AS(this, Const)->isEqAllOnes());
|
2021-06-05 16:40:56 +00:00
|
|
|
}
|
2022-09-16 12:17:38 +00:00
|
|
|
bool AstNode::isAllOnesV() const {
|
2021-10-22 14:15:42 +00:00
|
|
|
return (VN_IS(this, Const) && VN_AS(this, Const)->isEqAllOnesV());
|
2021-06-05 16:40:56 +00:00
|
|
|
}
|
2022-09-16 12:17:38 +00:00
|
|
|
bool AstNode::sameTree(const AstNode* node2p) const {
|
2021-06-05 16:40:56 +00:00
|
|
|
return sameTreeIter(this, node2p, true, false);
|
|
|
|
}
|
2022-09-16 12:17:38 +00:00
|
|
|
bool AstNode::sameGateTree(const AstNode* node2p) const {
|
2021-06-05 16:40:56 +00:00
|
|
|
return sameTreeIter(this, node2p, true, true);
|
|
|
|
}
|
|
|
|
|
2022-10-18 21:07:09 +00:00
|
|
|
int AstNodeArrayDType::left() const VL_MT_SAFE { return rangep()->leftConst(); }
|
|
|
|
int AstNodeArrayDType::right() const VL_MT_SAFE { return rangep()->rightConst(); }
|
|
|
|
int AstNodeArrayDType::hi() const VL_MT_SAFE { return rangep()->hiConst(); }
|
|
|
|
int AstNodeArrayDType::lo() const VL_MT_SAFE { return rangep()->loConst(); }
|
|
|
|
int AstNodeArrayDType::elementsConst() const VL_MT_SAFE { return rangep()->elementsConst(); }
|
|
|
|
VNumRange AstNodeArrayDType::declRange() const VL_MT_SAFE { return VNumRange{left(), right()}; }
|
2021-06-05 16:40:56 +00:00
|
|
|
|
2022-09-15 12:10:39 +00:00
|
|
|
AstRange::AstRange(FileLine* fl, int left, int right)
|
|
|
|
: ASTGEN_SUPER_Range(fl) {
|
2022-09-15 18:43:56 +00:00
|
|
|
leftp(new AstConst{fl, static_cast<uint32_t>(left)});
|
|
|
|
rightp(new AstConst{fl, static_cast<uint32_t>(right)});
|
2022-09-15 12:10:39 +00:00
|
|
|
}
|
|
|
|
AstRange::AstRange(FileLine* fl, const VNumRange& range)
|
|
|
|
: ASTGEN_SUPER_Range(fl) {
|
2022-09-15 18:43:56 +00:00
|
|
|
leftp(new AstConst{fl, static_cast<uint32_t>(range.left())});
|
|
|
|
rightp(new AstConst{fl, static_cast<uint32_t>(range.right())});
|
2022-09-15 12:10:39 +00:00
|
|
|
}
|
|
|
|
int AstRange::leftConst() const {
|
|
|
|
AstConst* const constp = VN_CAST(leftp(), Const);
|
|
|
|
return (constp ? constp->toSInt() : 0);
|
|
|
|
}
|
|
|
|
int AstRange::rightConst() const {
|
|
|
|
AstConst* const constp = VN_CAST(rightp(), Const);
|
|
|
|
return (constp ? constp->toSInt() : 0);
|
|
|
|
}
|
|
|
|
|
2022-10-18 21:07:09 +00:00
|
|
|
int AstQueueDType::boundConst() const VL_MT_SAFE {
|
2022-09-15 12:10:39 +00:00
|
|
|
AstConst* const constp = VN_CAST(boundp(), Const);
|
|
|
|
return (constp ? constp->toSInt() : 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
AstPin::AstPin(FileLine* fl, int pinNum, AstVarRef* varname, AstNode* exprp)
|
|
|
|
: ASTGEN_SUPER_Pin(fl)
|
|
|
|
, m_pinNum{pinNum}
|
|
|
|
, m_name{varname->name()} {
|
2022-09-15 18:43:56 +00:00
|
|
|
this->exprp(exprp);
|
2022-09-15 12:10:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
AstPackArrayDType::AstPackArrayDType(FileLine* fl, VFlagChildDType, AstNodeDType* dtp,
|
|
|
|
AstRange* rangep)
|
|
|
|
: ASTGEN_SUPER_PackArrayDType(fl) {
|
|
|
|
childDTypep(dtp); // Only for parser
|
|
|
|
refDTypep(nullptr);
|
2022-09-15 18:43:56 +00:00
|
|
|
this->rangep(rangep);
|
2022-09-15 12:10:39 +00:00
|
|
|
dtypep(nullptr); // V3Width will resolve
|
|
|
|
const int width = subDTypep()->width() * rangep->elementsConst();
|
|
|
|
widthForce(width, width);
|
|
|
|
}
|
|
|
|
AstPackArrayDType::AstPackArrayDType(FileLine* fl, AstNodeDType* dtp, AstRange* rangep)
|
|
|
|
: ASTGEN_SUPER_PackArrayDType(fl) {
|
|
|
|
refDTypep(dtp);
|
2022-09-15 18:43:56 +00:00
|
|
|
this->rangep(rangep);
|
2022-09-15 12:10:39 +00:00
|
|
|
dtypep(this);
|
|
|
|
const int width = subDTypep()->width() * rangep->elementsConst();
|
|
|
|
widthForce(width, width);
|
|
|
|
}
|
|
|
|
|
|
|
|
int AstBasicDType::hi() const { return (rangep() ? rangep()->hiConst() : m.m_nrange.hi()); }
|
|
|
|
int AstBasicDType::lo() const { return (rangep() ? rangep()->loConst() : m.m_nrange.lo()); }
|
|
|
|
int AstBasicDType::elements() const {
|
|
|
|
return (rangep() ? rangep()->elementsConst() : m.m_nrange.elements());
|
|
|
|
}
|
|
|
|
bool AstBasicDType::littleEndian() const {
|
|
|
|
return (rangep() ? rangep()->littleEndian() : m.m_nrange.littleEndian());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AstActive::hasClocked() const { return m_sensesp->hasClocked(); }
|
2022-09-15 12:17:00 +00:00
|
|
|
bool AstActive::hasCombo() const { return m_sensesp->hasCombo(); }
|
2022-09-15 12:10:39 +00:00
|
|
|
|
|
|
|
AstElabDisplay::AstElabDisplay(FileLine* fl, VDisplayType dispType, AstNode* exprsp)
|
|
|
|
: ASTGEN_SUPER_ElabDisplay(fl) {
|
2022-09-15 18:43:56 +00:00
|
|
|
addFmtp(new AstSFormatF{fl, AstSFormatF::NoFormat(), exprsp});
|
2022-09-15 12:10:39 +00:00
|
|
|
m_displayType = dispType;
|
|
|
|
}
|
|
|
|
|
|
|
|
AstCStmt::AstCStmt(FileLine* fl, const string& textStmt)
|
|
|
|
: ASTGEN_SUPER_CStmt(fl) {
|
2022-09-15 18:43:56 +00:00
|
|
|
addExprsp(new AstText{fl, textStmt, true});
|
2022-09-15 12:10:39 +00:00
|
|
|
}
|
|
|
|
|
2022-10-12 09:19:21 +00:00
|
|
|
AstCExpr::AstCExpr(FileLine* fl, const string& textStmt, int setwidth, bool cleanOut)
|
|
|
|
: ASTGEN_SUPER_CExpr(fl)
|
2022-09-15 12:10:39 +00:00
|
|
|
, m_cleanOut{cleanOut}
|
|
|
|
, m_pure{true} {
|
2022-09-15 18:43:56 +00:00
|
|
|
addExprsp(new AstText{fl, textStmt, true});
|
2022-09-15 12:10:39 +00:00
|
|
|
if (setwidth) dtypeSetLogicSized(setwidth, VSigning::UNSIGNED);
|
|
|
|
}
|
|
|
|
|
|
|
|
AstVarRef::AstVarRef(FileLine* fl, AstVar* varp, const VAccess& access)
|
|
|
|
: ASTGEN_SUPER_VarRef(fl, varp->name(), varp, access) {}
|
|
|
|
// This form only allowed post-link (see above)
|
|
|
|
AstVarRef::AstVarRef(FileLine* fl, AstVarScope* varscp, const VAccess& access)
|
|
|
|
: ASTGEN_SUPER_VarRef(fl, varscp->varp()->name(), varscp->varp(), access) {
|
|
|
|
varScopep(varscp);
|
|
|
|
}
|
|
|
|
bool AstVarRef::same(const AstVarRef* samep) const {
|
|
|
|
if (varScopep()) {
|
|
|
|
return (varScopep() == samep->varScopep() && access() == samep->access());
|
|
|
|
} else {
|
|
|
|
return (selfPointer() == samep->selfPointer() && varp()->name() == samep->varp()->name()
|
|
|
|
&& access() == samep->access());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bool AstVarRef::sameNoLvalue(AstVarRef* samep) const {
|
|
|
|
if (varScopep()) {
|
|
|
|
return (varScopep() == samep->varScopep());
|
|
|
|
} else {
|
|
|
|
return (selfPointer() == samep->selfPointer()
|
|
|
|
&& (!selfPointer().empty() || !samep->selfPointer().empty())
|
|
|
|
&& varp()->name() == samep->varp()->name());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
AstVarXRef::AstVarXRef(FileLine* fl, AstVar* varp, const string& dotted, const VAccess& access)
|
|
|
|
: ASTGEN_SUPER_VarXRef(fl, varp->name(), varp, access)
|
|
|
|
, m_dotted{dotted} {
|
|
|
|
dtypeFrom(varp);
|
|
|
|
}
|
|
|
|
|
2022-10-12 09:19:21 +00:00
|
|
|
AstStmtExpr* AstNodeExpr::makeStmt() { return new AstStmtExpr{fileline(), this}; }
|
|
|
|
|
2021-06-05 16:40:56 +00:00
|
|
|
#endif // Guard
|