mirror of
https://github.com/verilator/verilator.git
synced 2025-01-10 08:37:35 +00:00
3575 lines
157 KiB
C++
3575 lines
157 KiB
C++
// -*- mode: C++; c-file-style: "cc-mode" -*-
|
||
//*************************************************************************
|
||
// DESCRIPTION: Verilator: Expression width calculations
|
||
//
|
||
// Code available from: http://www.veripool.org/verilator
|
||
//
|
||
//*************************************************************************
|
||
//
|
||
// Copyright 2003-2016 by Wilson Snyder. This program is free software; you 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.
|
||
//
|
||
// Verilator is distributed in the hope that it will be useful,
|
||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
// GNU General Public License for more details.
|
||
//
|
||
//*************************************************************************
|
||
// V3Width's Transformations:
|
||
// Top down traversal:
|
||
// Determine width of sub-expressions
|
||
// width() = # bits upper expression wants, 0 for anything-goes
|
||
// widthUnsized() = # bits for unsized constant, or 0 if it's sized
|
||
// widthMin() = Alternative acceptable width for linting, or width() if sized
|
||
// Determine this subop's width, can be either:
|
||
// Fixed width X
|
||
// Unsized, min width X ('d5 is unsized, min 3 bits.)
|
||
// Pass up:
|
||
// width() = # bits this expression generates
|
||
// widthSized() = true if all constants sized, else false
|
||
// Compute size of this expression
|
||
// Lint warn about mismatches
|
||
// If expr size != subop fixed, bad
|
||
// If expr size < subop unsized minimum, bad
|
||
// If expr size != subop, edit netlist
|
||
// For == and similar ops, if multibit underneath, add a REDOR
|
||
// If subop larger, add a EXTRACT
|
||
// If subop smaller, add a EXTEND
|
||
// Pass size to sub-expressions if required (+/-* etc)
|
||
// FINAL = true.
|
||
// Subexpressions lint and extend as needed
|
||
//
|
||
//*************************************************************************
|
||
// Signedness depends on:
|
||
// Decimal numbers are signed
|
||
// Based numbers are unsigned unless 's' prefix
|
||
// Comparison results are unsigned
|
||
// Bit&Part selects are unsigned, even if whole
|
||
// Concatenates are unsigned
|
||
// Ignore signedness of self-determined:
|
||
// shift rhs, ** rhs, x?: lhs, concat and replicate members
|
||
// Else, if any operand unsigned, output unsigned
|
||
//
|
||
// Real number rules:
|
||
// Real numbers are real (duh)
|
||
// Reals convert to integers by rounding
|
||
// Reals init to 0.0
|
||
// Logicals convert compared to zero
|
||
// If any operand is real, result is real
|
||
//*************************************************************************
|
||
// V3Width is the only visitor that uses vup. We could switch to using userp,
|
||
// though note some iterators operate on next() and so would need to pass the
|
||
// same value on each nextp().
|
||
//*************************************************************************
|
||
// See notes in internal.txt about misuse of iterateAndNext and use of
|
||
// acceptSubtreeReturnEdits.
|
||
//*************************************************************************
|
||
|
||
#include "config_build.h"
|
||
#include "verilatedos.h"
|
||
#include <cstdio>
|
||
#include <cstdarg>
|
||
#include <unistd.h>
|
||
|
||
#include "V3Global.h"
|
||
#include "V3Width.h"
|
||
#include "V3Number.h"
|
||
#include "V3Const.h"
|
||
#include "V3String.h"
|
||
#include "V3Task.h"
|
||
|
||
// More code; this file was getting too large; see actions there
|
||
#define _V3WIDTH_CPP_
|
||
#include "V3WidthCommit.h"
|
||
|
||
//######################################################################
|
||
|
||
enum Stage { PRELIM=1,FINAL=2,BOTH=3 }; // Numbers are a bitmask <0>=prelim, <1>=final
|
||
ostream& operator<<(ostream& str, const Stage& rhs) { return str<<("-PFB"[(int)rhs]); }
|
||
|
||
enum Determ {
|
||
SELF, // Self-determined
|
||
CONTEXT, // Context-determined
|
||
ASSIGN // Assignment-like where sign comes from RHS only
|
||
};
|
||
ostream& operator<<(ostream& str, const Determ& rhs) {
|
||
static const char* s_det[] = {"SELF","CNTX","ASSN"};
|
||
return str<<s_det[rhs];
|
||
}
|
||
|
||
//######################################################################
|
||
// Width state, as a visitor of each AstNode
|
||
|
||
class WidthVP : public AstNUser {
|
||
// Parameters to pass down hierarchy with visit functions.
|
||
AstNodeDType* m_dtypep; // Parent's data type to resolve to
|
||
Stage m_stage; // If true, report errors
|
||
public:
|
||
WidthVP(AstNodeDType* dtypep, Stage stage)
|
||
: m_dtypep(dtypep), m_stage(stage) {
|
||
// Prelim doesn't look at assignments, so shouldn't need a dtype, however AstPattern uses them
|
||
}
|
||
WidthVP(Determ determ, Stage stage)
|
||
: m_dtypep(NULL), m_stage(stage) {
|
||
if (determ != SELF && stage != PRELIM) v3fatalSrc("Context-determined width request only allowed as prelim step");
|
||
}
|
||
bool selfDtm() const { return m_dtypep==NULL; }
|
||
AstNodeDType* dtypep() const {
|
||
// Detect where overrideDType is probably the intended call
|
||
if (!m_dtypep) v3fatalSrc("Width dtype request on self-determined or preliminary VUP");
|
||
return m_dtypep;
|
||
}
|
||
AstNodeDType* dtypeNullp() const { return m_dtypep; }
|
||
AstNodeDType* dtypeOverridep(AstNodeDType* defaultp) const {
|
||
if (m_stage == PRELIM) v3fatalSrc("Parent dtype should be a final-stage action");
|
||
return m_dtypep ? m_dtypep : defaultp;
|
||
}
|
||
int width() const {
|
||
if (!m_dtypep) v3fatalSrc("Width request on self-determined or preliminary VUP");
|
||
return m_dtypep->width();
|
||
}
|
||
int widthMin() const {
|
||
if (!m_dtypep) v3fatalSrc("Width request on self-determined or preliminary VUP");
|
||
return m_dtypep->widthMin();
|
||
}
|
||
bool prelim() const { return m_stage & PRELIM; }
|
||
bool final() const { return m_stage & FINAL; }
|
||
void dump(ostream& str) const {
|
||
if (!this) {
|
||
str<<" VUP(NULL)";
|
||
} else if (!m_dtypep) {
|
||
str<<" VUP(s="<<m_stage<<",self)";
|
||
} else {
|
||
str<<" VUP(s="<<m_stage<<",dt="<<(void*)dtypep()<<")";
|
||
}
|
||
}
|
||
};
|
||
ostream& operator<<(ostream& str, const WidthVP* vup) {
|
||
vup->dump(str);
|
||
return str;
|
||
}
|
||
|
||
//######################################################################
|
||
|
||
class WidthVisitor : public AstNVisitor {
|
||
private:
|
||
// TYPES
|
||
typedef map<pair<AstNodeDType*,AstAttrType>, AstVar*> TableMap;
|
||
typedef map<int,AstPatMember*> PatVecMap;
|
||
|
||
// STATE
|
||
bool m_paramsOnly; // Computing parameter value; limit operation
|
||
AstRange* m_cellRangep; // Range for arrayed instantiations, NULL for normal instantiations
|
||
AstFunc* m_funcp; // Current function
|
||
AstInitial* m_initialp; // Current initial block
|
||
AstAttrOf* m_attrp; // Current attribute
|
||
bool m_doGenerate; // Do errors later inside generate statement
|
||
int m_dtTables; // Number of created data type tables
|
||
TableMap m_tableMap; // Created tables so can remove duplicates
|
||
|
||
// ENUMS
|
||
enum ExtendRule {
|
||
EXTEND_EXP, // Extend if expect sign and node signed, e.g. node=y in ADD(x,y), "x + y"
|
||
EXTEND_ZERO, // Extend with zeros. e.g. node=y in EQ(x,y), "x == y"
|
||
EXTEND_LHS, // Extend with sign if node signed. e.g. node=y in ASSIGN(y,x), "x = y"
|
||
EXTEND_OFF // No extension
|
||
};
|
||
|
||
// METHODS
|
||
static int debug() { return V3Width::debug(); }
|
||
|
||
// VISITORS
|
||
// Naming: width_O{outputtype}_L{lhstype}_R{rhstype}_W{widthing}_S{signing}
|
||
// Where type:
|
||
// _O1=boolean (width 1 unsigned)
|
||
// _Ou=unsigned
|
||
// _Os=signed
|
||
// _Ous=unsigned or signed
|
||
// _Or=real
|
||
// _Ox=anything
|
||
|
||
// Widths: 1 bit out, lhs 1 bit; Real: converts via compare with 0
|
||
virtual void visit(AstLogNot* nodep, AstNUser* vup) { visit_log_not(nodep,vup); }
|
||
// Widths: 1 bit out, lhs 1 bit, rhs 1 bit; Real: converts via compare with 0
|
||
virtual void visit(AstLogAnd* nodep, AstNUser* vup) { visit_log_and_or(nodep,vup); }
|
||
virtual void visit(AstLogOr* nodep, AstNUser* vup) { visit_log_and_or(nodep,vup); }
|
||
virtual void visit(AstLogIf* nodep, AstNUser* vup) { visit_log_and_or(nodep,vup); } // Conversion from real not in IEEE, but a fallout
|
||
virtual void visit(AstLogIff* nodep, AstNUser* vup) { visit_log_and_or(nodep,vup); } // Conversion from real not in IEEE, but a fallout
|
||
|
||
// Widths: 1 bit out, Any width lhs
|
||
virtual void visit(AstRedAnd* nodep, AstNUser* vup) { visit_red_and_or(nodep,vup); }
|
||
virtual void visit(AstRedOr* nodep, AstNUser* vup) { visit_red_and_or(nodep,vup); }
|
||
virtual void visit(AstRedXnor* nodep, AstNUser* vup){ visit_red_and_or(nodep,vup); }
|
||
virtual void visit(AstRedXor* nodep,AstNUser* vup) { visit_red_and_or(nodep,vup); }
|
||
virtual void visit(AstOneHot* nodep,AstNUser* vup) { visit_red_and_or(nodep,vup); }
|
||
virtual void visit(AstOneHot0* nodep,AstNUser* vup) { visit_red_and_or(nodep,vup); }
|
||
virtual void visit(AstIsUnknown* nodep,AstNUser* vup) { visit_red_unknown(nodep,vup); } // Allow real
|
||
|
||
// These have different node types, as they operate differently
|
||
// Must add to case statement below,
|
||
// Widths: 1 bit out, lhs width == rhs width. real if lhs|rhs real
|
||
virtual void visit(AstEq* nodep, AstNUser* vup) { visit_cmp_eq_gt(nodep,vup,true); }
|
||
virtual void visit(AstNeq* nodep, AstNUser* vup) { visit_cmp_eq_gt(nodep,vup,true); }
|
||
virtual void visit(AstGt* nodep, AstNUser* vup) { visit_cmp_eq_gt(nodep,vup,true); }
|
||
virtual void visit(AstGte* nodep, AstNUser* vup) { visit_cmp_eq_gt(nodep,vup,true); }
|
||
virtual void visit(AstLt* nodep, AstNUser* vup) { visit_cmp_eq_gt(nodep,vup,true); }
|
||
virtual void visit(AstLte* nodep, AstNUser* vup) { visit_cmp_eq_gt(nodep,vup,true); }
|
||
virtual void visit(AstGtS* nodep, AstNUser* vup) { visit_cmp_eq_gt(nodep,vup,true); }
|
||
virtual void visit(AstGteS* nodep, AstNUser* vup) { visit_cmp_eq_gt(nodep,vup,true); }
|
||
virtual void visit(AstLtS* nodep, AstNUser* vup) { visit_cmp_eq_gt(nodep,vup,true); }
|
||
virtual void visit(AstLteS* nodep, AstNUser* vup) { visit_cmp_eq_gt(nodep,vup,true); }
|
||
virtual void visit(AstEqCase* nodep, AstNUser* vup) { visit_cmp_eq_gt(nodep,vup,true); }
|
||
virtual void visit(AstNeqCase* nodep, AstNUser* vup) { visit_cmp_eq_gt(nodep,vup,true); }
|
||
// ... These comparisons don't allow reals
|
||
virtual void visit(AstEqWild* nodep, AstNUser* vup) { visit_cmp_eq_gt(nodep,vup,false); }
|
||
virtual void visit(AstNeqWild* nodep, AstNUser* vup) { visit_cmp_eq_gt(nodep,vup,false); }
|
||
// ... Real compares
|
||
virtual void visit(AstEqD* nodep, AstNUser* vup) { visit_cmp_real(nodep,vup); }
|
||
virtual void visit(AstNeqD* nodep, AstNUser* vup) { visit_cmp_real(nodep,vup); }
|
||
virtual void visit(AstLtD* nodep, AstNUser* vup) { visit_cmp_real(nodep,vup); }
|
||
virtual void visit(AstLteD* nodep, AstNUser* vup) { visit_cmp_real(nodep,vup); }
|
||
virtual void visit(AstGtD* nodep, AstNUser* vup) { visit_cmp_real(nodep,vup); }
|
||
virtual void visit(AstGteD* nodep, AstNUser* vup) { visit_cmp_real(nodep,vup); }
|
||
// ... String compares
|
||
virtual void visit(AstEqN* nodep, AstNUser* vup) { visit_cmp_string(nodep,vup); }
|
||
virtual void visit(AstNeqN* nodep, AstNUser* vup) { visit_cmp_string(nodep,vup); }
|
||
virtual void visit(AstLtN* nodep, AstNUser* vup) { visit_cmp_string(nodep,vup); }
|
||
virtual void visit(AstLteN* nodep, AstNUser* vup) { visit_cmp_string(nodep,vup); }
|
||
virtual void visit(AstGtN* nodep, AstNUser* vup) { visit_cmp_string(nodep,vup); }
|
||
virtual void visit(AstGteN* nodep, AstNUser* vup) { visit_cmp_string(nodep,vup); }
|
||
|
||
// Widths: out width = lhs width = rhs width
|
||
// Signed: Output signed iff LHS & RHS signed.
|
||
// Real: Not allowed
|
||
virtual void visit(AstAnd* nodep, AstNUser* vup) { visit_boolmath_and_or(nodep,vup); }
|
||
virtual void visit(AstOr* nodep, AstNUser* vup) { visit_boolmath_and_or(nodep,vup); }
|
||
virtual void visit(AstXnor* nodep, AstNUser* vup) { visit_boolmath_and_or(nodep,vup); }
|
||
virtual void visit(AstXor* nodep, AstNUser* vup) { visit_boolmath_and_or(nodep,vup); }
|
||
virtual void visit(AstBufIf1* nodep, AstNUser* vup) { visit_boolmath_and_or(nodep,vup); } // Signed behavior changing in 3.814
|
||
// Width: Max(Lhs,Rhs) sort of.
|
||
// Real: If either side real
|
||
// Signed: If both sides real
|
||
virtual void visit(AstAdd* nodep, AstNUser* vup) { visit_add_sub_replace(nodep,vup,true); }
|
||
virtual void visit(AstSub* nodep, AstNUser* vup) { visit_add_sub_replace(nodep,vup,true); }
|
||
virtual void visit(AstDiv* nodep, AstNUser* vup) { visit_add_sub_replace(nodep,vup,true); }
|
||
virtual void visit(AstMul* nodep, AstNUser* vup) { visit_add_sub_replace(nodep,vup,true); }
|
||
// These can't promote to real
|
||
virtual void visit(AstModDiv* nodep, AstNUser* vup) { visit_add_sub_replace(nodep,vup,false); }
|
||
virtual void visit(AstModDivS* nodep, AstNUser* vup) { visit_add_sub_replace(nodep,vup,false); }
|
||
virtual void visit(AstMulS* nodep, AstNUser* vup) { visit_add_sub_replace(nodep,vup,false); }
|
||
virtual void visit(AstDivS* nodep, AstNUser* vup) { visit_add_sub_replace(nodep,vup,false); }
|
||
// Widths: out width = lhs width, but upper matters
|
||
// Signed: Output signed iff LHS signed; unary operator
|
||
// Unary promote to real
|
||
virtual void visit(AstNegate* nodep, AstNUser* vup) { visit_negate_not(nodep,vup,true); }
|
||
// Unary never real
|
||
virtual void visit(AstNot* nodep, AstNUser* vup) { visit_negate_not(nodep,vup,false); }
|
||
|
||
// Real: inputs and output real
|
||
virtual void visit(AstAddD* nodep, AstNUser* vup) { visit_real_add_sub(nodep,vup); }
|
||
virtual void visit(AstSubD* nodep, AstNUser* vup) { visit_real_add_sub(nodep,vup); }
|
||
virtual void visit(AstDivD* nodep, AstNUser* vup) { visit_real_add_sub(nodep,vup); }
|
||
virtual void visit(AstMulD* nodep, AstNUser* vup) { visit_real_add_sub(nodep,vup); }
|
||
virtual void visit(AstPowD* nodep, AstNUser* vup) { visit_real_add_sub(nodep,vup); }
|
||
// Real: Output real
|
||
virtual void visit(AstNegateD* nodep, AstNUser* vup) { visit_real_neg_ceil(nodep,vup); }
|
||
virtual void visit(AstCeilD* nodep, AstNUser* vup) { visit_real_neg_ceil(nodep,vup); }
|
||
virtual void visit(AstExpD* nodep, AstNUser* vup) { visit_real_neg_ceil(nodep,vup); }
|
||
virtual void visit(AstFloorD* nodep, AstNUser* vup) { visit_real_neg_ceil(nodep,vup); }
|
||
virtual void visit(AstLogD* nodep, AstNUser* vup) { visit_real_neg_ceil(nodep,vup); }
|
||
virtual void visit(AstLog10D* nodep, AstNUser* vup) { visit_real_neg_ceil(nodep,vup); }
|
||
virtual void visit(AstSqrtD* nodep, AstNUser* vup) { visit_real_neg_ceil(nodep,vup); }
|
||
|
||
// Widths: out signed/unsigned width = lhs width, input un|signed
|
||
virtual void visit(AstSigned* nodep, AstNUser* vup) { visit_signed_unsigned(nodep,vup,AstNumeric::SIGNED); }
|
||
virtual void visit(AstUnsigned* nodep, AstNUser* vup) { visit_signed_unsigned(nodep,vup,AstNumeric::UNSIGNED); }
|
||
|
||
// Widths: Output width from lhs, rhs<33 bits
|
||
// Signed: If lhs signed
|
||
virtual void visit(AstShiftL* nodep, AstNUser* vup) { visit_shift(nodep,vup); }
|
||
virtual void visit(AstShiftR* nodep, AstNUser* vup) { visit_shift(nodep,vup); }
|
||
// ShiftRS converts to ShiftR, but not vice-versa
|
||
virtual void visit(AstShiftRS* nodep, AstNUser* vup) { visit_shift(nodep,vup); }
|
||
|
||
//========
|
||
// Widths: Output real, input integer signed
|
||
virtual void visit(AstBitsToRealD* nodep, AstNUser* vup) { visit_Or_Lu64(nodep,vup); }
|
||
virtual void visit(AstIToRD* nodep, AstNUser* vup) { visit_Or_Ls32(nodep,vup); }
|
||
|
||
// Widths: Output integer signed, input real
|
||
virtual void visit(AstRToIS* nodep, AstNUser* vup) { visit_Os32_Lr(nodep,vup); }
|
||
virtual void visit(AstRToIRoundS* nodep, AstNUser* vup) { visit_Os32_Lr(nodep,vup); }
|
||
|
||
// Widths: Output integer unsigned, input real
|
||
virtual void visit(AstRealToBits* nodep, AstNUser* vup) { visit_Ou64_Lr(nodep,vup); }
|
||
|
||
// Widths: Constant, terminal
|
||
virtual void visit(AstTime* nodep, AstNUser*) { nodep->dtypeSetUInt64(); }
|
||
virtual void visit(AstTimeD* nodep, AstNUser*) { nodep->dtypeSetDouble(); }
|
||
virtual void visit(AstTestPlusArgs* nodep, AstNUser*) { nodep->dtypeSetSigned32(); }
|
||
virtual void visit(AstScopeName* nodep, AstNUser*) { nodep->dtypeSetUInt64(); } // A pointer, but not that it matters
|
||
|
||
// Special cases. So many....
|
||
virtual void visit(AstNodeCond* nodep, AstNUser* vup) {
|
||
// op=cond?expr1:expr2
|
||
// Signed: Output signed iff RHS & THS signed (presumed, not in IEEE)
|
||
// See IEEE-2012 11.4.11 and Table 11-21.
|
||
// LHS is self-determined
|
||
// Width: max(RHS,THS)
|
||
// Real: Output real if either expression is real, non-real argument gets converted
|
||
if (vup->c()->prelim()) { // First stage evaluation
|
||
// Just once, do the conditional, expect one bit out.
|
||
iterateCheckBool(nodep,"Conditional Test",nodep->condp(),BOTH);
|
||
// Determine sub expression widths only relying on what's in the subops
|
||
// CONTEXT determined, but need data type for pattern assignments
|
||
nodep->expr1p()->iterateAndNext(*this,WidthVP(vup->c()->dtypeNullp(),PRELIM).p());
|
||
nodep->expr2p()->iterateAndNext(*this,WidthVP(vup->c()->dtypeNullp(),PRELIM).p());
|
||
// Calculate width of this expression.
|
||
// First call (prelim()) vup->c()->width() is probably zero, so we'll return
|
||
// the size of this subexpression only.
|
||
// Second call (final()) vup->c()->width() is probably the expression size, so
|
||
// the expression includes the size of the output too.
|
||
if (nodep->expr1p()->isDouble() || nodep->expr2p()->isDouble()) {
|
||
nodep->dtypeSetDouble();
|
||
} else {
|
||
int width = max(nodep->expr1p()->width(), nodep->expr2p()->width());
|
||
int mwidth = max(nodep->expr1p()->widthMin(), nodep->expr2p()->widthMin());
|
||
bool issigned = nodep->expr1p()->isSigned() && nodep->expr2p()->isSigned();
|
||
nodep->dtypeSetLogicSized(width,mwidth,AstNumeric::fromBool(issigned));
|
||
}
|
||
}
|
||
if (vup->c()->final()) {
|
||
AstNodeDType* expDTypep = vup->c()->dtypeOverridep(nodep->dtypep());
|
||
AstNodeDType* subDTypep = expDTypep;
|
||
nodep->dtypeFrom(expDTypep);
|
||
// Error report and change sizes for suboperands of this node.
|
||
iterateCheck(nodep,"Conditional True", nodep->expr1p(),CONTEXT,FINAL,subDTypep,EXTEND_EXP);
|
||
iterateCheck(nodep,"Conditional False",nodep->expr2p(),CONTEXT,FINAL,subDTypep,EXTEND_EXP);
|
||
}
|
||
}
|
||
virtual void visit(AstConcat* nodep, AstNUser* vup) {
|
||
// Real: Not allowed (assumed)
|
||
// Signed: unsigned output, input either (assumed)
|
||
// IEEE-2012 Table 11-21, and 11.8.1:
|
||
// LHS, RHS is self-determined
|
||
// signed: Unsigned (11.8.1)
|
||
// width: LHS + RHS
|
||
if (vup->c()->prelim()) {
|
||
iterateCheckSizedSelf(nodep,"LHS",nodep->lhsp(),SELF,BOTH);
|
||
iterateCheckSizedSelf(nodep,"RHS",nodep->rhsp(),SELF,BOTH);
|
||
nodep->dtypeSetLogicSized(nodep->lhsp()->width() + nodep->rhsp()->width(),
|
||
nodep->lhsp()->widthMin() + nodep->rhsp()->widthMin(),
|
||
AstNumeric::UNSIGNED);
|
||
// Cleanup zero width Verilog2001 {x,{0{foo}}} now,
|
||
// otherwise having width(0) will cause later assertions to fire
|
||
if (AstReplicate* repp=nodep->lhsp()->castReplicate()) {
|
||
if (repp->width()==0) { // Keep rhs
|
||
nodep->replaceWith(nodep->rhsp()->unlinkFrBack());
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
return;
|
||
}
|
||
}
|
||
if (AstReplicate* repp=nodep->rhsp()->castReplicate()) {
|
||
if (repp->width()==0) { // Keep lhs
|
||
nodep->replaceWith(nodep->lhsp()->unlinkFrBack());
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
return;
|
||
}
|
||
}
|
||
if (nodep->lhsp()->isString()
|
||
|| nodep->rhsp()->isString()) {
|
||
AstNode* newp = new AstConcatN (nodep->fileline(),nodep->lhsp()->unlinkFrBack(),
|
||
nodep->rhsp()->unlinkFrBack());
|
||
nodep->replaceWith(newp);
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
return;
|
||
}
|
||
}
|
||
if (vup->c()->final()) {
|
||
if (!nodep->dtypep()->widthSized()) {
|
||
// See also error in V3Number
|
||
nodep->v3warn(WIDTHCONCAT,"Unsized numbers/parameters not allowed in concatenations.");
|
||
}
|
||
}
|
||
}
|
||
virtual void visit(AstConcatN* nodep, AstNUser* vup) {
|
||
// String concatenate.
|
||
// Already did AstConcat simplifications
|
||
if (vup->c()->prelim()) {
|
||
iterateCheckString(nodep,"LHS",nodep->lhsp(),BOTH);
|
||
iterateCheckString(nodep,"RHS",nodep->rhsp(),BOTH);
|
||
nodep->dtypeSetString();
|
||
}
|
||
if (vup->c()->final()) {
|
||
if (!nodep->dtypep()->widthSized()) {
|
||
// See also error in V3Number
|
||
nodep->v3warn(WIDTHCONCAT,"Unsized numbers/parameters not allowed in concatenations.");
|
||
}
|
||
}
|
||
}
|
||
virtual void visit(AstReplicate* nodep, AstNUser* vup) {
|
||
// IEEE-2012 Table 11-21:
|
||
// LHS, RHS is self-determined
|
||
// width: value(LHS) * width(RHS)
|
||
if (vup->c()->prelim()) {
|
||
iterateCheckSizedSelf(nodep,"LHS",nodep->lhsp(),SELF,BOTH);
|
||
iterateCheckSizedSelf(nodep,"RHS",nodep->rhsp(),SELF,BOTH);
|
||
V3Const::constifyParamsEdit(nodep->rhsp()); // rhsp may change
|
||
AstConst* constp = nodep->rhsp()->castConst();
|
||
if (!constp) { nodep->v3error("Replication value isn't a constant."); return; }
|
||
uint32_t times = constp->toUInt();
|
||
if (times==0 && !nodep->backp()->castConcat()) { // Concat Visitor will clean it up.
|
||
nodep->v3error("Replication value of 0 is only legal under a concatenation."); times=1;
|
||
}
|
||
if (nodep->lhsp()->isString()) {
|
||
AstNode* newp = new AstReplicateN(nodep->fileline(),nodep->lhsp()->unlinkFrBack(),
|
||
nodep->rhsp()->unlinkFrBack());
|
||
nodep->replaceWith(newp);
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
return;
|
||
} else {
|
||
nodep->dtypeSetLogicSized((nodep->lhsp()->width() * times),
|
||
(nodep->lhsp()->widthMin() * times),
|
||
AstNumeric::UNSIGNED);
|
||
}
|
||
}
|
||
if (vup->c()->final()) {
|
||
if (!nodep->dtypep()->widthSized()) {
|
||
// See also error in V3Number
|
||
nodep->v3warn(WIDTHCONCAT,"Unsized numbers/parameters not allowed in replications.");
|
||
}
|
||
}
|
||
}
|
||
virtual void visit(AstReplicateN* nodep, AstNUser* vup) {
|
||
// Replicate with string
|
||
if (vup->c()->prelim()) {
|
||
iterateCheckString(nodep,"LHS",nodep->lhsp(),BOTH);
|
||
iterateCheckSizedSelf(nodep,"RHS",nodep->rhsp(),SELF,BOTH);
|
||
V3Const::constifyParamsEdit(nodep->rhsp()); // rhsp may change
|
||
AstConst* constp = nodep->rhsp()->castConst();
|
||
if (!constp) { nodep->v3error("Replication value isn't a constant."); return; }
|
||
uint32_t times = constp->toUInt();
|
||
if (times==0 && !nodep->backp()->castConcat()) { // Concat Visitor will clean it up.
|
||
nodep->v3error("Replication value of 0 is only legal under a concatenation.");
|
||
}
|
||
nodep->dtypeSetString();
|
||
}
|
||
if (vup->c()->final()) {
|
||
if (!nodep->dtypep()->widthSized()) {
|
||
// See also error in V3Number
|
||
nodep->v3warn(WIDTHCONCAT,"Unsized numbers/parameters not allowed in replications.");
|
||
}
|
||
}
|
||
}
|
||
virtual void visit(AstNodeStream* nodep, AstNUser* vup) {
|
||
if (vup->c()->prelim()) {
|
||
iterateCheckSizedSelf(nodep,"LHS",nodep->lhsp(),SELF,BOTH);
|
||
iterateCheckSizedSelf(nodep,"RHS",nodep->rhsp(),SELF,BOTH);
|
||
V3Const::constifyParamsEdit(nodep->rhsp()); // rhsp may change
|
||
AstConst* constp = nodep->rhsp()->castConst();
|
||
AstBasicDType* basicp = nodep->rhsp()->castBasicDType();
|
||
if (!constp && !basicp) { nodep->v3error("Slice size isn't a constant or basic data type."); return; }
|
||
if (basicp) { // Convert data type to a constant size
|
||
AstConst* newp = new AstConst(basicp->fileline(), basicp->width());
|
||
nodep->rhsp()->replaceWith(newp);
|
||
pushDeletep(basicp);
|
||
} else {
|
||
uint32_t sliceSize = constp->toUInt();
|
||
if (!sliceSize) { nodep->v3error("Slice size cannot be zero."); return; }
|
||
}
|
||
nodep->dtypeSetLogicSized((nodep->lhsp()->width()),
|
||
(nodep->lhsp()->widthMin()),
|
||
AstNumeric::UNSIGNED);
|
||
}
|
||
if (vup->c()->final()) {
|
||
if (!nodep->dtypep()->widthSized()) {
|
||
// See also error in V3Number
|
||
nodep->v3warn(WIDTHCONCAT,"Unsized numbers/parameters not allowed in streams.");
|
||
}
|
||
}
|
||
}
|
||
virtual void visit(AstRange* nodep, AstNUser* vup) {
|
||
// Real: Not allowed
|
||
// Signed: unsigned output, input either
|
||
// Convert all range values to constants
|
||
UINFO(6,"RANGE "<<nodep<<endl);
|
||
V3Const::constifyParamsEdit(nodep->msbp()); // May relink pointed to node
|
||
V3Const::constifyParamsEdit(nodep->lsbp()); // May relink pointed to node
|
||
checkConstantOrReplace(nodep->msbp(), "MSB of bit range isn't a constant");
|
||
checkConstantOrReplace(nodep->lsbp(), "LSB of bit range isn't a constant");
|
||
int msb = nodep->msbConst();
|
||
int lsb = nodep->lsbConst();
|
||
if (msb<lsb) {
|
||
// Little endian bits are legal, just remember to swap
|
||
// Warning is in V3Width to avoid false warnings when in "off" generate if's
|
||
nodep->littleEndian(!nodep->littleEndian());
|
||
// Internally we'll always have msb() be the greater number
|
||
// We only need to correct when doing [] AstSel extraction,
|
||
// and when tracing the vector.
|
||
nodep->msbp()->swapWith(nodep->lsbp());
|
||
}
|
||
if (vup->c()->prelim()) {
|
||
// Don't need to iterate because V3Const already constified
|
||
int width = nodep->elementsConst();
|
||
if (width > (1<<28)) nodep->v3error("Width of bit range is huge; vector of over 1billion bits: 0x"<<hex<<width);
|
||
// Note width() not set on range; use elementsConst()
|
||
if (nodep->littleEndian() && !nodep->backp()->castUnpackArrayDType()) {
|
||
nodep->v3warn(LITENDIAN,"Little bit endian vector: MSB < LSB of bit range: "<<nodep->lsbConst()<<":"<<nodep->msbConst());
|
||
}
|
||
}
|
||
}
|
||
|
||
virtual void visit(AstSel* nodep, AstNUser* vup) {
|
||
// Signed: always unsigned; Real: Not allowed
|
||
// LSB is self-determined (IEEE 2012 11.5.1)
|
||
// We also use SELs to shorten a signed constant etc, in this case they are signed.
|
||
if (nodep->didWidth()) return;
|
||
if (vup->c()->prelim()) {
|
||
if (debug()>=9) nodep->dumpTree(cout,"-selWidth: ");
|
||
nodep->fromp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p());
|
||
nodep->lsbp()->iterateAndNext(*this,WidthVP(SELF,PRELIM).p());
|
||
checkCvtUS(nodep->fromp());
|
||
iterateCheckSizedSelf(nodep,"Select Width",nodep->widthp(),SELF,BOTH);
|
||
iterateCheckSizedSelf(nodep,"Select LHS",nodep->lhsp(),SELF,BOTH);
|
||
V3Const::constifyParamsEdit(nodep->widthp()); // widthp may change
|
||
AstConst* widthConstp = nodep->widthp()->castConst();
|
||
if (!widthConstp) {
|
||
nodep->v3error("Width of bit extract isn't a constant");
|
||
nodep->dtypeSetLogicBool(); return;
|
||
}
|
||
int width = nodep->widthConst();
|
||
if (!nodep->dtypep()) nodep->v3fatalSrc("dtype wasn't set") // by V3WidthSel
|
||
if (nodep->lsbp()->castConst()
|
||
&& nodep->msbConst() < nodep->lsbConst()) {
|
||
nodep->v3error("Unsupported: MSB < LSB of bit extract: "
|
||
<<nodep->msbConst()<<"<"<<nodep->lsbConst());
|
||
width = (nodep->lsbConst() - nodep->msbConst() + 1);
|
||
nodep->dtypeSetLogicSized(width,width,AstNumeric::UNSIGNED);
|
||
nodep->widthp()->replaceWith(new AstConst(nodep->widthp()->fileline(),
|
||
width));
|
||
nodep->lsbp()->replaceWith(new AstConst(nodep->lsbp()->fileline(), 0));
|
||
}
|
||
// We're extracting, so just make sure the expression is at least wide enough.
|
||
if (nodep->fromp()->width() < width) {
|
||
nodep->v3error("Extracting "<<width
|
||
<<" bits from only "<<nodep->fromp()->width()<<" bit number");
|
||
// Extend it.
|
||
AstNodeDType* subDTypep = nodep->findLogicDType(width,width,nodep->fromp()->dtypep()->numeric());
|
||
widthCheckSized(nodep,"errorless...",nodep->fromp(),subDTypep,EXTEND_EXP,false/*noerror*/);
|
||
}
|
||
// Check bit indexes.
|
||
// What is the MSB? We want the true MSB, not one starting at 0,
|
||
// because a 4 bit index is required to look at a one-bit variable[15:15] and 5 bits for [15:-2]
|
||
int frommsb = nodep->fromp()->width() - 1;
|
||
int fromlsb = 0;
|
||
int elw = nodep->declElWidth(); // Must adjust to tell user bit ranges
|
||
if (nodep->declRange().ranged()) {
|
||
frommsb = nodep->declRange().hiMaxSelect()*elw + (elw-1); // Corrected for negative lsb
|
||
fromlsb = nodep->declRange().lo()*elw;
|
||
} else {
|
||
//nodep->v3fatalSrc("Should have been declRanged in V3WidthSel");
|
||
}
|
||
int selwidth = V3Number::log2b(frommsb+1-1)+1; // Width to address a bit
|
||
AstNodeDType* selwidthDTypep = nodep->findLogicDType(selwidth,selwidth,nodep->lsbp()->dtypep()->numeric());
|
||
nodep->fromp()->iterateAndNext(*this,WidthVP(SELF,FINAL).p());
|
||
nodep->lsbp()->iterateAndNext(*this,WidthVP(SELF,FINAL).p());
|
||
if (widthBad(nodep->lsbp(),selwidthDTypep)
|
||
&& nodep->lsbp()->width()!=32) {
|
||
if (!nodep->fileline()->warnIsOff(V3ErrorCode::WIDTH)) {
|
||
nodep->v3warn(WIDTH,"Bit extraction of var["<<(frommsb/elw)<<":"<<(fromlsb/elw)<<"] requires "
|
||
<<(selwidth/elw)<<" bit index, not "
|
||
<<(nodep->lsbp()->width()/elw)
|
||
<<(nodep->lsbp()->width()!=nodep->lsbp()->widthMin()
|
||
?" or "+cvtToStr(nodep->lsbp()->widthMin()/elw):"")
|
||
<<" bits.");
|
||
UINFO(1," Related node: "<<nodep<<endl);
|
||
}
|
||
}
|
||
if (nodep->lsbp()->castConst() && nodep->msbConst() > frommsb) {
|
||
// See also warning in V3Const
|
||
// We need to check here, because the widthCheckSized may silently
|
||
// add another SEL which will lose the out-of-range check
|
||
//
|
||
// We don't want to trigger an error here if we are just
|
||
// evaluating type sizes for a generate block condition. We
|
||
// should only trigger the error if the out-of-range access is
|
||
// actually generated.
|
||
if (m_doGenerate) {
|
||
UINFO(5, "Selection index out of range inside generate."<<endl);
|
||
} else {
|
||
nodep->v3warn(SELRANGE,"Selection index out of range: "
|
||
<<nodep->msbConst()<<":"<<nodep->lsbConst()
|
||
<<" outside "<<frommsb<<":"<<fromlsb);
|
||
UINFO(1," Related node: "<<nodep<<endl);
|
||
}
|
||
}
|
||
// iterate FINAL is two blocks above
|
||
//
|
||
// If we have a width problem with GENERATE etc, this will reduce
|
||
// it down and mask it, so we have no chance of finding a real
|
||
// error in the future. So don't do this for them.
|
||
if (!m_doGenerate) {
|
||
// lsbp() must be self-determined, however for performance we want the select to be
|
||
// truncated to fit within the maximum select range, e.g. turn Xs outside of the select
|
||
// into something fast which pulls from within the array.
|
||
widthCheckSized(nodep,"Extract Range",nodep->lsbp(),selwidthDTypep,EXTEND_EXP,false/*NOWARN*/);
|
||
}
|
||
}
|
||
}
|
||
|
||
virtual void visit(AstArraySel* nodep, AstNUser* vup) {
|
||
// Signed/Real: Output signed iff LHS signed/real; binary operator
|
||
// Note by contrast, bit extract selects are unsigned
|
||
// LSB is self-determined (IEEE 2012 11.5.1)
|
||
if (vup->c()->prelim()) {
|
||
iterateCheckSizedSelf(nodep,"Bit select",nodep->bitp(),SELF,BOTH);
|
||
nodep->fromp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
//
|
||
int frommsb;
|
||
int fromlsb;
|
||
AstNodeDType* fromDtp = nodep->fromp()->dtypep()->skipRefp();
|
||
if (AstUnpackArrayDType* adtypep = fromDtp->castUnpackArrayDType()) {
|
||
frommsb = adtypep->msb();
|
||
fromlsb = adtypep->lsb();
|
||
if (fromlsb>frommsb) {int t=frommsb; frommsb=fromlsb; fromlsb=t; }
|
||
// However, if the lsb<0 we may go negative, so need more bits!
|
||
if (fromlsb < 0) frommsb += -fromlsb;
|
||
nodep->dtypeFrom(adtypep->subDTypep()); // Need to strip off array reference
|
||
}
|
||
else {
|
||
// Note PackArrayDType doesn't use an ArraySel but a normal Sel.
|
||
UINFO(1," Related dtype: "<<fromDtp<<endl);
|
||
nodep->v3fatalSrc("Array reference exceeds dimension of array");
|
||
frommsb = fromlsb = 0;
|
||
}
|
||
int selwidth = V3Number::log2b(frommsb+1-1)+1; // Width to address a bit
|
||
AstNodeDType* selwidthDTypep = nodep->findLogicDType(selwidth,selwidth,nodep->bitp()->dtypep()->numeric());
|
||
if (widthBad(nodep->bitp(),selwidthDTypep)
|
||
&& nodep->bitp()->width()!=32) {
|
||
nodep->v3warn(WIDTH,"Bit extraction of array["<<frommsb<<":"<<fromlsb<<"] requires "
|
||
<<selwidth<<" bit index, not "
|
||
<<nodep->bitp()->width()
|
||
<<(nodep->bitp()->width()!=nodep->bitp()->widthMin()
|
||
?" or "+cvtToStr(nodep->bitp()->widthMin()):"")
|
||
<<" bits.");
|
||
if (!nodep->fileline()->warnIsOff(V3ErrorCode::WIDTH)) {
|
||
UINFO(1," Related node: "<<nodep<<endl);
|
||
UINFO(1," Related dtype: "<<nodep->dtypep()<<endl);
|
||
}
|
||
}
|
||
if (!m_doGenerate) {
|
||
// Must check bounds before adding a select that truncates the bound
|
||
// Note we've already subtracted off LSB
|
||
if (nodep->bitp()->castConst() && (nodep->bitp()->castConst()->toSInt() > (frommsb-fromlsb)
|
||
|| nodep->bitp()->castConst()->toSInt() < 0)) {
|
||
nodep->v3warn(SELRANGE,"Selection index out of range: "
|
||
<<(nodep->bitp()->castConst()->toSInt()+fromlsb)
|
||
<<" outside "<<frommsb<<":"<<fromlsb);
|
||
UINFO(1," Related node: "<<nodep<<endl);
|
||
}
|
||
widthCheckSized(nodep,"Extract Range",nodep->bitp(),selwidthDTypep,EXTEND_EXP,false/*NOWARN*/);
|
||
}
|
||
}
|
||
}
|
||
|
||
virtual void visit(AstSelBit* nodep, AstNUser* vup) {
|
||
// Just a quick check as after V3Param these nodes instead are AstSel's
|
||
nodep->fromp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p()); //FINAL in AstSel
|
||
nodep->rhsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p()); //FINAL in AstSel
|
||
nodep->thsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p()); //FINAL in AstSel
|
||
nodep->attrp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
AstNode* selp = V3Width::widthSelNoIterEdit(nodep); if (selp!=nodep) { nodep=NULL; selp->iterate(*this,vup); return; }
|
||
nodep->v3fatalSrc("AstSelBit should disappear after widthSel");
|
||
}
|
||
virtual void visit(AstSelExtract* nodep, AstNUser* vup) {
|
||
// Just a quick check as after V3Param these nodes instead are AstSel's
|
||
nodep->fromp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p()); //FINAL in AstSel
|
||
nodep->rhsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p()); //FINAL in AstSel
|
||
nodep->thsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p()); //FINAL in AstSel
|
||
nodep->attrp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
AstNode* selp = V3Width::widthSelNoIterEdit(nodep); if (selp!=nodep) { nodep=NULL; selp->iterate(*this,vup); return; }
|
||
nodep->v3fatalSrc("AstSelExtract should disappear after widthSel");
|
||
}
|
||
virtual void visit(AstSelPlus* nodep, AstNUser* vup) {
|
||
nodep->fromp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p()); //FINAL in AstSel
|
||
nodep->rhsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p()); //FINAL in AstSel
|
||
nodep->thsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p()); //FINAL in AstSel
|
||
nodep->attrp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
AstNode* selp = V3Width::widthSelNoIterEdit(nodep); if (selp!=nodep) { nodep=NULL; selp->iterate(*this,vup); return; }
|
||
nodep->v3fatalSrc("AstSelPlus should disappear after widthSel");
|
||
}
|
||
virtual void visit(AstSelMinus* nodep, AstNUser* vup) {
|
||
nodep->fromp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p()); //FINAL in AstSel
|
||
nodep->rhsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p()); //FINAL in AstSel
|
||
nodep->thsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p()); //FINAL in AstSel
|
||
nodep->attrp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
AstNode* selp = V3Width::widthSelNoIterEdit(nodep); if (selp!=nodep) { nodep=NULL; selp->iterate(*this,vup); return; }
|
||
nodep->v3fatalSrc("AstSelMinus should disappear after widthSel");
|
||
}
|
||
|
||
virtual void visit(AstExtend* nodep, AstNUser* vup) {
|
||
// Only created by this process, so we know width from here down is correct.
|
||
}
|
||
virtual void visit(AstExtendS* nodep, AstNUser* vup) {
|
||
// Only created by this process, so we know width from here down is correct.
|
||
}
|
||
virtual void visit(AstConst* nodep, AstNUser* vup) {
|
||
// The node got setup with the signed/real state of the node.
|
||
// However a later operation may have changed the node->signed w/o changing
|
||
// the number's sign. So we don't: nodep->dtypeChgSigned(nodep->num().isSigned());
|
||
if (vup && vup->c()->prelim()) {
|
||
if (nodep->num().isString()) {
|
||
nodep->dtypeSetString();
|
||
} else if (nodep->num().sized()) {
|
||
nodep->dtypeChgWidth(nodep->num().width(), nodep->num().width());
|
||
} else {
|
||
nodep->dtypeChgWidth(nodep->num().width(), nodep->num().widthMin());
|
||
}
|
||
}
|
||
// We don't size the constant until we commit the widths, as need parameters
|
||
// to remain unsized, and numbers to remain unsized to avoid backp() warnings
|
||
}
|
||
virtual void visit(AstRand* nodep, AstNUser* vup) {
|
||
if (vup->c()->prelim()) {
|
||
nodep->dtypeSetSigned32(); // Says the spec
|
||
}
|
||
}
|
||
virtual void visit(AstUCFunc* nodep, AstNUser* vup) {
|
||
// Give it the size the user wants.
|
||
if (vup && vup->c()->prelim()) {
|
||
nodep->dtypeSetLogicSized(32,1,AstNumeric::UNSIGNED); // We don't care
|
||
// All arguments seek their natural sizes
|
||
nodep->iterateChildren(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
if (vup->c()->final()) {
|
||
AstNodeDType* expDTypep = vup->c()->dtypeOverridep(nodep->dtypep());
|
||
nodep->dtypeFrom(expDTypep); // Assume user knows the rules; go with the flow
|
||
if (nodep->width()>64) nodep->v3error("Unsupported: $c can't generate wider than 64 bits");
|
||
}
|
||
}
|
||
virtual void visit(AstCLog2* nodep, AstNUser* vup) {
|
||
if (vup->c()->prelim()) {
|
||
iterateCheckSizedSelf(nodep,"LHS",nodep->lhsp(),SELF,BOTH);
|
||
nodep->dtypeSetSigned32();
|
||
}
|
||
}
|
||
virtual void visit(AstPow* nodep, AstNUser* vup) {
|
||
// Pow is special, output sign only depends on LHS sign, but function result depends on both signs
|
||
// RHS is self-determined (IEEE)
|
||
// Real if either side is real (as with AstAdd)
|
||
if (vup->c()->prelim()) {
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p());
|
||
nodep->rhsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p());
|
||
if (nodep->lhsp()->isDouble() || nodep->rhsp()->isDouble()) {
|
||
spliceCvtD(nodep->lhsp());
|
||
spliceCvtD(nodep->rhsp());
|
||
replaceWithDVersion(nodep); VL_DANGLING(nodep);
|
||
return;
|
||
}
|
||
|
||
checkCvtUS(nodep->lhsp());
|
||
iterateCheckSizedSelf(nodep,"RHS",nodep->rhsp(),SELF,BOTH);
|
||
nodep->dtypeFrom(nodep->lhsp());
|
||
}
|
||
|
||
if (vup->c()->final()) {
|
||
AstNodeDType* expDTypep = vup->c()->dtypeOverridep(nodep->dtypep());
|
||
nodep->dtypeFrom(expDTypep);
|
||
// rhs already finalized in iterate_shift_prelim
|
||
iterateCheck(nodep,"LHS",nodep->lhsp(),SELF,FINAL,nodep->dtypep(),EXTEND_EXP);
|
||
if (nodep->width()>64) nodep->v3error("Unsupported: Large >64bit ** power operator not implemented.");
|
||
AstNode* newp = NULL; // No change
|
||
if (nodep->lhsp()->isSigned() && nodep->rhsp()->isSigned()) {
|
||
newp = new AstPowSS (nodep->fileline(), nodep->lhsp()->unlinkFrBack(),
|
||
nodep->rhsp()->unlinkFrBack());
|
||
} else if (nodep->lhsp()->isSigned() && !nodep->rhsp()->isSigned()) {
|
||
newp = new AstPowSU (nodep->fileline(), nodep->lhsp()->unlinkFrBack(),
|
||
nodep->rhsp()->unlinkFrBack());
|
||
} else if (!nodep->lhsp()->isSigned() && nodep->rhsp()->isSigned()) {
|
||
newp = new AstPowUS (nodep->fileline(), nodep->lhsp()->unlinkFrBack(),
|
||
nodep->rhsp()->unlinkFrBack());
|
||
}
|
||
if (newp) {
|
||
newp->dtypeFrom(nodep);
|
||
UINFO(9,"powOld "<<nodep<<endl);
|
||
UINFO(9,"powNew "<<newp<<endl);
|
||
nodep->replaceWith(newp); VL_DANGLING(nodep);
|
||
}
|
||
}
|
||
}
|
||
virtual void visit(AstPowSU* nodep, AstNUser* vup) {
|
||
// POWSU/SS/US only created here, dtype already determined, so nothing to do in this function
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
nodep->rhsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
virtual void visit(AstPowSS* nodep, AstNUser* vup) {
|
||
// POWSU/SS/US only created here, dtype already determined, so nothing to do in this function
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
nodep->rhsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
virtual void visit(AstPowUS* nodep, AstNUser* vup) {
|
||
// POWSU/SS/US only created here, dtype already determined, so nothing to do in this function
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
nodep->rhsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
virtual void visit(AstCountOnes* nodep, AstNUser* vup) {
|
||
if (vup->c()->prelim()) {
|
||
iterateCheckSizedSelf(nodep,"LHS",nodep->lhsp(),SELF,BOTH);
|
||
// If it's a 32 bit number, we need a 6 bit number as we need to return '32'.
|
||
int selwidth = V3Number::log2b(nodep->lhsp()->width())+1;
|
||
nodep->dtypeSetLogicSized(selwidth,selwidth,AstNumeric::UNSIGNED); // Spec doesn't indicate if an integer
|
||
}
|
||
}
|
||
virtual void visit(AstCvtPackString* nodep, AstNUser* vup) {
|
||
// Opaque returns, so arbitrary
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
// Type set in constructor
|
||
}
|
||
virtual void visit(AstAttrOf* nodep, AstNUser*) {
|
||
AstAttrOf* oldAttr = m_attrp;
|
||
m_attrp = nodep;
|
||
nodep->fromp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
// Don't iterate children, don't want to lose VarRef.
|
||
switch (nodep->attrType()) {
|
||
case AstAttrType::VAR_BASE:
|
||
case AstAttrType::MEMBER_BASE:
|
||
case AstAttrType::ENUM_BASE:
|
||
// Soon to be handled in V3LinkWidth SEL generation, under attrp() and newSubLsbOf
|
||
break;
|
||
case AstAttrType::DIM_DIMENSIONS:
|
||
case AstAttrType::DIM_UNPK_DIMENSIONS: {
|
||
if (!nodep->fromp() || !nodep->fromp()->dtypep()) nodep->v3fatalSrc("Unsized expression");
|
||
pair<uint32_t,uint32_t> dim = nodep->fromp()->dtypep()->dimensions(true);
|
||
int val = (nodep->attrType()==AstAttrType::DIM_UNPK_DIMENSIONS
|
||
? dim.second : (dim.first+dim.second));
|
||
nodep->replaceWith(new AstConst(nodep->fileline(), AstConst::Signed32(), val)); nodep->deleteTree(); VL_DANGLING(nodep);
|
||
break;
|
||
}
|
||
case AstAttrType::DIM_BITS:
|
||
case AstAttrType::DIM_HIGH:
|
||
case AstAttrType::DIM_INCREMENT:
|
||
case AstAttrType::DIM_LEFT:
|
||
case AstAttrType::DIM_LOW:
|
||
case AstAttrType::DIM_RIGHT:
|
||
case AstAttrType::DIM_SIZE: {
|
||
if (!nodep->fromp() || !nodep->fromp()->dtypep()) nodep->v3fatalSrc("Unsized expression");
|
||
pair<uint32_t,uint32_t> dim = nodep->fromp()->dtypep()->skipRefp()->dimensions(true);
|
||
uint32_t msbdim = dim.first+dim.second;
|
||
if (!nodep->dimp() || nodep->dimp()->castConst() || msbdim<1) {
|
||
int dim = !nodep->dimp() ? 1 : nodep->dimp()->castConst()->toSInt();
|
||
AstConst* newp = dimensionValue(nodep->fromp()->dtypep(), nodep->attrType(), dim);
|
||
nodep->replaceWith(newp); nodep->deleteTree(); VL_DANGLING(nodep);
|
||
}
|
||
else { // Need a runtime lookup table. Yuk.
|
||
if (!nodep->fromp() || !nodep->fromp()->dtypep()) nodep->v3fatalSrc("Unsized expression");
|
||
AstVar* varp = dimensionVarp(nodep->fromp()->dtypep(), nodep->attrType(), msbdim);
|
||
AstNode* dimp = nodep->dimp()->unlinkFrBack();
|
||
AstVarRef* varrefp = new AstVarRef(nodep->fileline(), varp, false);
|
||
varrefp->packagep(v3Global.rootp()->dollarUnitPkgAddp());
|
||
AstNode* newp = new AstArraySel(nodep->fileline(), varrefp, dimp);
|
||
nodep->replaceWith(newp); nodep->deleteTree(); VL_DANGLING(nodep);
|
||
}
|
||
break;
|
||
}
|
||
default: {
|
||
// Everything else resolved earlier
|
||
nodep->dtypeSetLogicSized(32,1,AstNumeric::UNSIGNED); // Approximation, unsized 32
|
||
UINFO(1,"Missing ATTR type case node: "<<nodep<<endl);
|
||
nodep->v3fatalSrc("Missing ATTR type case");
|
||
break;
|
||
}
|
||
}
|
||
m_attrp = oldAttr;
|
||
}
|
||
virtual void visit(AstText* nodep, AstNUser* vup) {
|
||
// Only used in CStmts which don't care....
|
||
}
|
||
|
||
// DTYPES
|
||
virtual void visit(AstNodeArrayDType* nodep, AstNUser*) {
|
||
if (nodep->didWidthAndSet()) return; // This node is a dtype & not both PRELIMed+FINALed
|
||
if (nodep->childDTypep()) nodep->refDTypep(moveChildDTypeEdit(nodep));
|
||
// Iterate into subDTypep() to resolve that type and update pointer.
|
||
nodep->refDTypep(iterateEditDTypep(nodep, nodep->subDTypep()));
|
||
// Cleanup array size
|
||
nodep->rangep()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
nodep->dtypep(nodep); // The array itself, not subDtype
|
||
if (nodep->castUnpackArrayDType()) {
|
||
// Historically array elements have width of the ref type not the full array
|
||
nodep->widthFromSub(nodep->subDTypep());
|
||
} else {
|
||
int width = nodep->subDTypep()->width() * nodep->rangep()->elementsConst();
|
||
nodep->widthForce(width,width);
|
||
}
|
||
UINFO(4,"dtWidthed "<<nodep<<endl);
|
||
}
|
||
virtual void visit(AstBasicDType* nodep, AstNUser*) {
|
||
if (nodep->didWidthAndSet()) return; // This node is a dtype & not both PRELIMed+FINALed
|
||
if (nodep->generic()) return; // Already perfect
|
||
if (nodep->rangep()) {
|
||
nodep->rangep()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
// Because this DType has a unique child range, we know it's not pointed at by
|
||
// other nodes unless they are referencing this type. Furthermore the width()
|
||
// calculation would return identical values. Therefore we can directly replace the width
|
||
nodep->widthForce(nodep->rangep()->elementsConst(), nodep->rangep()->elementsConst());
|
||
}
|
||
else if (nodep->implicit()) {
|
||
// Parameters may notice implicitness and change to different dtype
|
||
nodep->widthForce(1,1);
|
||
}
|
||
// else width in node is correct; it was set based on keyword().width()
|
||
// at construction time. Ditto signed, so "unsigned byte" etc works right.
|
||
nodep->cvtRangeConst();
|
||
// TODO: If BasicDType now looks like a generic type, we can convert to a real generic dtype
|
||
// Instead for now doing this in V3WidthCommit
|
||
UINFO(4,"dtWidthed "<<nodep<<endl);
|
||
}
|
||
virtual void visit(AstConstDType* nodep, AstNUser*) {
|
||
if (nodep->didWidthAndSet()) return; // This node is a dtype & not both PRELIMed+FINALed
|
||
// Move any childDTypep to instead be in global AstTypeTable.
|
||
// This way if this node gets deleted and some other dtype points to it
|
||
// it won't become a dead pointer. This doesn't change the object pointed to.
|
||
if (nodep->childDTypep()) nodep->refDTypep(moveChildDTypeEdit(nodep));
|
||
// Iterate into subDTypep() to resolve that type and update pointer.
|
||
nodep->refDTypep(iterateEditDTypep(nodep, nodep->subDTypep()));
|
||
nodep->iterateChildren(*this);
|
||
nodep->dtypep(nodep); // Should already be set, but be clear it's not the subDType
|
||
nodep->widthFromSub(nodep->subDTypep());
|
||
UINFO(4,"dtWidthed "<<nodep<<endl);
|
||
}
|
||
virtual void visit(AstRefDType* nodep, AstNUser*) {
|
||
if (nodep->didWidthAndSet()) return; // This node is a dtype & not both PRELIMed+FINALed
|
||
nodep->iterateChildren(*this);
|
||
if (nodep->subDTypep()) nodep->refDTypep(iterateEditDTypep(nodep, nodep->subDTypep()));
|
||
nodep->dtypeFrom(nodep->dtypeSkipRefp());
|
||
nodep->widthFromSub(nodep->subDTypep());
|
||
UINFO(4,"dtWidthed "<<nodep<<endl);
|
||
}
|
||
virtual void visit(AstTypedef* nodep, AstNUser*) {
|
||
if (nodep->didWidthAndSet()) return; // This node is a dtype & not both PRELIMed+FINALed
|
||
if (nodep->childDTypep()) nodep->dtypep(moveChildDTypeEdit(nodep));
|
||
nodep->iterateChildren(*this);
|
||
nodep->dtypep(iterateEditDTypep(nodep, nodep->subDTypep()));
|
||
}
|
||
virtual void visit(AstCastParse* nodep, AstNUser* vup) {
|
||
// nodep->dtp could be data type, or a primary_constant
|
||
// Don't iterate lhsp, will deal with that once convert the type
|
||
V3Const::constifyParamsEdit(nodep->dtp()); // itemp may change
|
||
if (AstConst* constp = nodep->dtp()->castConst()) {
|
||
constp->unlinkFrBack();
|
||
AstNode* newp = new AstCastSize(nodep->fileline(), nodep->lhsp()->unlinkFrBack(), constp);
|
||
nodep->replaceWith(newp);
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
newp->accept(*this,vup);
|
||
} else {
|
||
nodep->v3error("Unsupported: Cast to "<<nodep->dtp()->prettyTypeName());
|
||
nodep->replaceWith(nodep->lhsp()->unlinkFrBack());
|
||
}
|
||
}
|
||
virtual void visit(AstCast* nodep, AstNUser* vup) {
|
||
if (nodep->childDTypep()) nodep->dtypep(moveChildDTypeEdit(nodep));
|
||
nodep->dtypep(iterateEditDTypep(nodep, nodep->dtypep()));
|
||
//if (debug()) nodep->dumpTree(cout," CastPre: ");
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
// When more general casts are supported, the cast elimination will be done later.
|
||
// For now, replace it ASAP, so widthing can propagate easily
|
||
// The cast may change signing, but we don't know the sign yet. Make it so.
|
||
// Note we don't sign lhsp() that would make the algorithm O(n^2) if lots of casting.
|
||
AstBasicDType* basicp = nodep->dtypep()->basicp();
|
||
if (!basicp) nodep->v3fatalSrc("Unimplemented: Casting non-simple data type");
|
||
// When implement more complicated types need to convert childDTypep to dtypep() not as a child
|
||
if (!basicp->isDouble() && !nodep->lhsp()->isDouble()) {
|
||
// Note widthCheckSized might modify nodep->lhsp()
|
||
AstNodeDType* subDTypep = nodep->findLogicDType(nodep->width(),nodep->width(),
|
||
nodep->lhsp()->dtypep()->numeric());
|
||
widthCheckSized(nodep,"Cast",nodep->lhsp(),subDTypep,EXTEND_EXP,false);
|
||
}
|
||
AstNode* newp = nodep->lhsp()->unlinkFrBack();
|
||
if (basicp->isDouble() && !newp->isDouble()) {
|
||
newp = new AstIToRD(nodep->fileline(), newp);
|
||
} else if (!basicp->isDouble() && newp->isDouble()) {
|
||
if (basicp->isSigned()) {
|
||
newp = new AstRToIRoundS(nodep->fileline(), newp);
|
||
} else {
|
||
newp = new AstUnsigned(nodep->fileline(),
|
||
new AstRToIS(nodep->fileline(), newp));
|
||
}
|
||
} else if (basicp->isSigned() && !newp->isSigned()) {
|
||
newp = new AstSigned(nodep->fileline(), newp);
|
||
} else if (!basicp->isSigned() && newp->isSigned()) {
|
||
newp = new AstUnsigned(nodep->fileline(), newp);
|
||
} else {
|
||
//newp = newp; // Can just remove cast
|
||
}
|
||
nodep->replaceWith(newp);
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
//if (debug()) newp->dumpTree(cout," CastOut: ");
|
||
}
|
||
virtual void visit(AstCastSize* nodep, AstNUser* vup) {
|
||
// IEEE: Signedness of result is same as self-determined signedness
|
||
// However, the result is same as BITSEL, so we do not sign extend the LHS
|
||
if (!nodep->rhsp()->castConst()) nodep->v3fatalSrc("Unsupported: Non-const cast of size");
|
||
//if (debug()) nodep->dumpTree(cout," CastSizePre: ");
|
||
if (vup->c()->prelim()) {
|
||
int width = nodep->rhsp()->castConst()->toSInt();
|
||
if (width < 1) { nodep->v3error("Size-changing cast to zero or negative size"); width=1; }
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(SELF,PRELIM).p());
|
||
AstBasicDType* underDtp = nodep->lhsp()->dtypep()->castBasicDType();
|
||
if (!underDtp) {
|
||
underDtp = nodep->lhsp()->dtypep()->basicp();
|
||
}
|
||
if (!underDtp) {
|
||
nodep->v3error("Unsupported: Size-changing cast on non-basic data type");
|
||
underDtp = nodep->findLogicBoolDType()->castBasicDType();
|
||
}
|
||
// A cast propagates its size to the lower expression and is included in the maximum
|
||
// width, so 23'(1'b1 + 1'b1) uses 23-bit math, but 1'(2'h2 * 2'h1) uses two-bit math.
|
||
// However the output width is exactly that requested.
|
||
// So two steps, first do the calculation's width (max of the two widths)
|
||
{
|
||
int calcWidth = max(width, underDtp->width());
|
||
AstNodeDType* calcDtp = (underDtp->keyword().isFourstate()
|
||
? nodep->findLogicDType(calcWidth, calcWidth, underDtp->numeric())
|
||
: nodep->findBitDType(calcWidth, calcWidth, underDtp->numeric()));
|
||
nodep->dtypep(calcDtp);
|
||
// We ignore warnings as that is sort of the point of a cast
|
||
iterateCheck(nodep,"Cast expr",nodep->lhsp(),CONTEXT,FINAL,calcDtp,EXTEND_EXP,false);
|
||
}
|
||
if (debug()) nodep->dumpTree(cout," CastSizeClc: ");
|
||
// Next step, make the proper output width
|
||
{
|
||
AstNodeDType* outDtp = (underDtp->keyword().isFourstate()
|
||
? nodep->findLogicDType(width, width, underDtp->numeric())
|
||
: nodep->findBitDType(width, width, underDtp->numeric()));
|
||
nodep->dtypep(outDtp);
|
||
// We ignore warnings as that is sort of the point of a cast
|
||
widthCheckSized(nodep,"Cast expr",nodep->lhsp(),outDtp,EXTEND_EXP,false);
|
||
}
|
||
}
|
||
if (vup->c()->final()) {
|
||
// CastSize not needed once sizes determined
|
||
AstNode* underp = nodep->lhsp()->unlinkFrBack();
|
||
nodep->replaceWith(underp);
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
}
|
||
//if (debug()) nodep->dumpTree(cout," CastSizeOut: ");
|
||
}
|
||
virtual void visit(AstVar* nodep, AstNUser* vup) {
|
||
//if (debug()) nodep->dumpTree(cout," InitPre: ");
|
||
// Must have deterministic constant width
|
||
// We can't skip this step when width()!=0, as creating a AstVar
|
||
// with non-constant range gets size 1, not size 0. So use didWidth().
|
||
if (nodep->didWidth()) return;
|
||
if (nodep->doingWidth()) { // Early exit if have circular parameter definition
|
||
if (!nodep->valuep()) nodep->v3fatalSrc("circular, but without value");
|
||
nodep->v3error("Variable's initial value is circular: "<<nodep->prettyName());
|
||
pushDeletep(nodep->valuep()->unlinkFrBack());
|
||
nodep->valuep(new AstConst(nodep->fileline(), AstConst::LogicTrue()));
|
||
nodep->dtypeFrom(nodep->valuep());
|
||
nodep->didWidth(true);
|
||
return;
|
||
}
|
||
nodep->doingWidth(true);
|
||
// Make sure dtype is sized
|
||
if (nodep->childDTypep()) nodep->dtypep(moveChildDTypeEdit(nodep));
|
||
nodep->dtypep(iterateEditDTypep(nodep, nodep->dtypep()));
|
||
if (!nodep->dtypep()) nodep->v3fatalSrc("No dtype determined for var");
|
||
if (nodep->isIO() && !(nodep->dtypeSkipRefp()->castBasicDType()
|
||
|| nodep->dtypeSkipRefp()->castNodeArrayDType()
|
||
|| nodep->dtypeSkipRefp()->castNodeClassDType())) {
|
||
nodep->v3error("Unsupported: Inputs and outputs must be simple data types");
|
||
}
|
||
if (nodep->dtypep()->skipRefToConstp()->castConstDType()) {
|
||
nodep->isConst(true);
|
||
}
|
||
// Parameters if implicit untyped inherit from what they are assigned to
|
||
AstBasicDType* bdtypep = nodep->dtypep()->castBasicDType();
|
||
bool didchk = false;
|
||
bool implicitParam = nodep->isParam() && bdtypep && bdtypep->implicit();
|
||
if (implicitParam) {
|
||
if (nodep->valuep()) {
|
||
nodep->valuep()->iterateAndNext(*this,WidthVP(nodep->dtypep(),PRELIM).p());
|
||
UINFO(9,"implicitParamPRELIMIV "<<nodep->valuep()<<endl);
|
||
// Although nodep will get a different width for parameters just below,
|
||
// we want the init numbers to retain their width/minwidth until parameters are replaced.
|
||
// This prevents width warnings at the location the parameter is substituted in
|
||
if (nodep->valuep()->isDouble()) {
|
||
nodep->dtypeSetDouble(); VL_DANGLING(bdtypep);
|
||
} else {
|
||
int width=0;
|
||
AstBasicDType* valueBdtypep = nodep->valuep()->dtypep()->basicp();
|
||
bool issigned = false;
|
||
if (bdtypep->isNosign()) {
|
||
if (valueBdtypep && valueBdtypep->isSigned()) issigned=true;
|
||
} else issigned = bdtypep->isSigned();
|
||
if (nodep->valuep()->dtypep()->widthSized()) {
|
||
width = nodep->valuep()->width();
|
||
} else {
|
||
if (nodep->valuep()->width()>32) nodep->valuep()->v3warn(WIDTH,"Assigning >32 bit to unranged parameter (defaults to 32 bits)");
|
||
width = 32;
|
||
}
|
||
// Can't just inherit valuep()->dtypep() as mwidth might not equal width
|
||
if (width==1) {
|
||
// one bit parameter is same as "parameter [0] foo", not "parameter logic foo"
|
||
// as you can extract "foo[0]" from a parameter but not a wire
|
||
nodep->dtypeChgWidthSigned(width, nodep->valuep()->widthMin(),
|
||
AstNumeric::fromBool(issigned));
|
||
nodep->dtypep(nodep->findLogicRangeDType
|
||
(VNumRange(0,0,false),
|
||
nodep->valuep()->widthMin(),
|
||
AstNumeric::fromBool(issigned)));
|
||
} else {
|
||
nodep->dtypeChgWidthSigned(width, nodep->valuep()->widthMin(),
|
||
AstNumeric::fromBool(issigned));
|
||
}
|
||
didchk = true;
|
||
}
|
||
iterateCheckAssign(nodep,"Initial value",nodep->valuep(),FINAL,nodep->dtypep());
|
||
UINFO(9,"implicitParamFromIV "<<nodep->valuep()<<endl);
|
||
//UINFO below will print variable nodep
|
||
} else {
|
||
// Or, if nothing assigned, they're integral
|
||
nodep->dtypeSetSigned32(); VL_DANGLING(bdtypep);
|
||
}
|
||
}
|
||
else if (bdtypep && bdtypep->implicit()) { // Implicits get converted to size 1
|
||
nodep->dtypeSetLogicSized(1,1,bdtypep->numeric()); VL_DANGLING(bdtypep);
|
||
}
|
||
if (nodep->valuep() && !didchk) {
|
||
//if (debug()) nodep->dumpTree(cout," final: ");
|
||
// AstPattern requires assignments to pass datatype on PRELIM
|
||
nodep->valuep()->iterateAndNext(*this,WidthVP(nodep->dtypep(),PRELIM).p());
|
||
iterateCheckAssign(nodep,"Initial value",nodep->valuep(),FINAL,nodep->dtypep());
|
||
}
|
||
UINFO(4,"varWidthed "<<nodep<<endl);
|
||
//if (debug()) nodep->dumpTree(cout," InitOut: ");
|
||
nodep->didWidth(true);
|
||
nodep->doingWidth(false);
|
||
}
|
||
virtual void visit(AstNodeVarRef* nodep, AstNUser* vup) {
|
||
if (nodep->didWidth()) return;
|
||
if (!nodep->varp()) nodep->v3fatalSrc("Unlinked varref");
|
||
if (!nodep->varp()->didWidth()) {
|
||
// Var hasn't been widthed, so make it so.
|
||
nodep->varp()->iterate(*this);
|
||
}
|
||
//if (debug()>=9) { nodep->dumpTree(cout," VRin "); nodep->varp()->dumpTree(cout," forvar "); }
|
||
// Note genvar's are also entered as integers
|
||
nodep->dtypeFrom(nodep->varp());
|
||
if (nodep->backp()->castNodeAssign() && nodep->lvalue()) { // On LHS
|
||
if (!nodep->widthMin()) v3fatalSrc("LHS var should be size complete");
|
||
}
|
||
//if (debug()>=9) nodep->dumpTree(cout," VRout ");
|
||
if (nodep->lvalue() && nodep->varp()->isConst()
|
||
&& !m_paramsOnly
|
||
&& !m_initialp) { // Too loose, but need to allow our generated first assignment
|
||
// // Move this to a property of the AstInitial block
|
||
nodep->v3error("Assigning to const variable: "<<nodep->prettyName());
|
||
}
|
||
nodep->didWidth(true);
|
||
}
|
||
|
||
virtual void visit(AstEnumDType* nodep, AstNUser* vup) {
|
||
if (nodep->didWidthAndSet()) return; // This node is a dtype & not both PRELIMed+FINALed
|
||
UINFO(5," ENUMDTYPE "<<nodep<<endl);
|
||
if (nodep->childDTypep()) nodep->refDTypep(moveChildDTypeEdit(nodep));
|
||
nodep->refDTypep(iterateEditDTypep(nodep, nodep->subDTypep()));
|
||
nodep->dtypep(nodep);
|
||
nodep->widthFromSub(nodep->subDTypep());
|
||
// Assign widths
|
||
nodep->itemsp()->iterateAndNext(*this,WidthVP(nodep->dtypep(),BOTH).p());
|
||
// Assign missing values
|
||
V3Number num (nodep->fileline(), nodep->width(), 0);
|
||
V3Number one (nodep->fileline(), nodep->width(), 1);
|
||
map<V3Number,AstEnumItem*> inits;
|
||
for (AstEnumItem* itemp = nodep->itemsp(); itemp; itemp=itemp->nextp()->castEnumItem()) {
|
||
if (itemp->valuep()) {
|
||
if (debug()>=9) { UINFO(0,"EnumInit "<<itemp<<endl); itemp->valuep()->dumpTree(cout,"-EnumInit: "); }
|
||
V3Const::constifyParamsEdit(itemp->valuep()); // itemp may change
|
||
if (!itemp->valuep()->castConst()) {
|
||
itemp->valuep()->v3error("Enum value isn't a constant");
|
||
itemp->valuep()->unlinkFrBack()->deleteTree();
|
||
continue;
|
||
}
|
||
// TODO IEEE says assigning sized number that is not same size as enum is illegal
|
||
}
|
||
if (!itemp->valuep()) {
|
||
if (num.isEqZero() && itemp != nodep->itemsp())
|
||
itemp->v3error("Enum value wrapped around"); // IEEE says illegal
|
||
if (!nodep->dtypep()->basicp()
|
||
&& !nodep->dtypep()->basicp()->keyword().isIntNumeric()) {
|
||
itemp->v3error("Enum names without values only allowed on numeric types");
|
||
// as can't +1 to resolve them.
|
||
}
|
||
itemp->valuep(new AstConst(itemp->fileline(), num));
|
||
}
|
||
num.opAssign(itemp->valuep()->castConst()->num());
|
||
// Look for duplicates
|
||
if (inits.find(num) != inits.end()) { // IEEE says illegal
|
||
itemp->v3error("Overlapping enumeration value: "<<itemp->prettyName()<<endl
|
||
<<inits.find(num)->second->warnMore()
|
||
<<"... Location of original declaration");
|
||
} else {
|
||
inits.insert(make_pair(num,itemp));
|
||
}
|
||
num.opAdd(one, itemp->valuep()->castConst()->num());
|
||
}
|
||
}
|
||
virtual void visit(AstEnumItem* nodep, AstNUser* vup) {
|
||
UINFO(5," ENUMITEM "<<nodep<<endl);
|
||
AstNodeDType* vdtypep = vup->c()->dtypep();
|
||
if (!vdtypep) nodep->v3fatalSrc("ENUMITEM not under ENUM");
|
||
nodep->dtypep(vdtypep);
|
||
if (nodep->valuep()) { // else the value will be assigned sequentially
|
||
// Default type is int, but common to assign narrower values, so minwidth from value
|
||
nodep->valuep()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p());
|
||
int mwidth = nodep->valuep()->widthMin(); // Value determines minwidth
|
||
nodep->dtypeChgWidth(nodep->width(), mwidth);
|
||
iterateCheck(nodep,"Enum value",nodep->valuep(),CONTEXT,FINAL,nodep->dtypep(),EXTEND_EXP);
|
||
}
|
||
}
|
||
virtual void visit(AstEnumItemRef* nodep, AstNUser* vup) {
|
||
if (!nodep->itemp()->didWidth()) {
|
||
// We need to do the whole enum en-mass
|
||
AstNode* enump = nodep->itemp();
|
||
if (!enump) nodep->v3fatalSrc("EnumItemRef not linked");
|
||
for (; enump; enump=enump->backp()) {
|
||
if (enump->castEnumDType()) break;
|
||
}
|
||
if (!enump) nodep->v3fatalSrc("EnumItemRef can't deref back to an Enum");
|
||
enump->iterate(*this,vup); VL_DANGLING(enump); // parent's connection to enump may be relinked
|
||
}
|
||
nodep->dtypeFrom(nodep->itemp());
|
||
}
|
||
virtual void visit(AstInitArray* nodep, AstNUser* vup) {
|
||
// InitArray has type of the array; children are array values
|
||
if (vup->c()->prelim()) { // First stage evaluation
|
||
AstNodeDType* vdtypep = vup->c()->dtypep();
|
||
if (!vdtypep) nodep->v3fatalSrc("InitArray type not assigned by AstPattern/Var visitor");
|
||
nodep->dtypep(vdtypep);
|
||
if (AstNodeArrayDType* arrayp = vdtypep->castNodeArrayDType()) {
|
||
nodep->iterateChildren(*this,WidthVP(arrayp->subDTypep(),BOTH).p());
|
||
} else {
|
||
nodep->v3fatalSrc("InitArray on non-array");
|
||
}
|
||
}
|
||
}
|
||
virtual void visit(AstInside* nodep, AstNUser* vup) {
|
||
nodep->exprp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p());
|
||
for (AstNode* nextip, *itemp = nodep->itemsp(); itemp; itemp=nextip) {
|
||
nextip = itemp->nextp(); // Prelim may cause the node to get replaced
|
||
itemp->iterate(*this,WidthVP(CONTEXT,PRELIM).p()); VL_DANGLING(itemp);
|
||
}
|
||
// Take width as maximum across all items
|
||
int width = nodep->exprp()->width();
|
||
int mwidth = nodep->exprp()->widthMin();
|
||
for (AstNode* itemp = nodep->itemsp(); itemp; itemp=itemp->nextp()) {
|
||
width = max(width,itemp->width());
|
||
mwidth = max(mwidth,itemp->widthMin());
|
||
}
|
||
// Apply width
|
||
AstNodeDType* subDTypep = nodep->findLogicDType(width,mwidth,nodep->exprp()->dtypep()->numeric());
|
||
iterateCheck(nodep,"Inside expression",nodep->exprp(),CONTEXT,FINAL,subDTypep,EXTEND_EXP);
|
||
for (AstNode* itemp = nodep->itemsp(); itemp; itemp=itemp->nextp()) {
|
||
iterateCheck(nodep,"Inside Item",itemp,CONTEXT,FINAL,subDTypep,EXTEND_EXP);
|
||
}
|
||
nodep->dtypeSetLogicBool();
|
||
if (debug()>=9) nodep->dumpTree(cout,"-inside-in: ");
|
||
// Now rip out the inside and replace with simple math
|
||
AstNode* newp = NULL;
|
||
for (AstNode* nextip, *itemp = nodep->itemsp(); itemp; itemp=nextip) {
|
||
nextip = itemp->nextp(); // Will be unlinking
|
||
AstNode* inewp;
|
||
if (AstInsideRange* irangep = itemp->castInsideRange()) {
|
||
// Similar logic in V3Case
|
||
inewp = new AstAnd(itemp->fileline(),
|
||
new AstGte(itemp->fileline(),
|
||
nodep->exprp()->cloneTree(true),
|
||
irangep->lhsp()->unlinkFrBack()),
|
||
new AstLte(itemp->fileline(),
|
||
nodep->exprp()->cloneTree(true),
|
||
irangep->rhsp()->unlinkFrBack()));
|
||
} else {
|
||
inewp = new AstEqWild(itemp->fileline(),
|
||
nodep->exprp()->cloneTree(true),
|
||
itemp->unlinkFrBack());
|
||
}
|
||
if (newp) newp = new AstOr(nodep->fileline(), newp, inewp);
|
||
else newp = inewp;
|
||
}
|
||
if (!newp) newp = new AstConst(nodep->fileline(), AstConst::LogicFalse());
|
||
if (debug()>=9) newp->dumpTree(cout,"-inside-out: ");
|
||
nodep->replaceWith(newp); pushDeletep(nodep); VL_DANGLING(nodep);
|
||
}
|
||
virtual void visit(AstInsideRange* nodep, AstNUser* vup) {
|
||
// Just do each side; AstInside will rip these nodes out later
|
||
nodep->lhsp()->iterateAndNext(*this,vup);
|
||
nodep->rhsp()->iterateAndNext(*this,vup);
|
||
nodep->dtypeFrom(nodep->lhsp());
|
||
}
|
||
|
||
virtual void visit(AstIfaceRefDType* nodep, AstNUser* vup) {
|
||
if (nodep->didWidthAndSet()) return; // This node is a dtype & not both PRELIMed+FINALed
|
||
UINFO(5," IFACEREF "<<nodep<<endl);
|
||
nodep->iterateChildren(*this, vup);
|
||
nodep->dtypep(nodep);
|
||
nodep->widthForce(1, 1); // Not really relevant
|
||
UINFO(4,"dtWidthed "<<nodep<<endl);
|
||
}
|
||
virtual void visit(AstNodeClassDType* nodep, AstNUser* vup) {
|
||
if (nodep->didWidthAndSet()) return; // This node is a dtype & not both PRELIMed+FINALed
|
||
UINFO(5," NODECLASS "<<nodep<<endl);
|
||
//if (debug()>=9) nodep->dumpTree("-class-in--");
|
||
if (!nodep->packed()) {
|
||
nodep->v3warn(UNPACKED, "Unsupported: Unpacked struct/union");
|
||
}
|
||
nodep->iterateChildren(*this); // First size all members
|
||
nodep->repairMemberCache();
|
||
// Determine bit assignments and width
|
||
nodep->dtypep(nodep);
|
||
int lsb = 0;
|
||
int width = 0;
|
||
// MSB is first, so go backwards
|
||
AstMemberDType* itemp;
|
||
for (itemp = nodep->membersp(); itemp && itemp->nextp(); itemp=itemp->nextp()->castMemberDType()) ;
|
||
for (AstMemberDType* backip; itemp; itemp=backip) {
|
||
backip = itemp->backp()->castMemberDType();
|
||
itemp->lsb(lsb);
|
||
if (nodep->castUnionDType()) {
|
||
width = max(width, itemp->width());
|
||
} else {
|
||
lsb += itemp->width();
|
||
width += itemp->width();
|
||
}
|
||
}
|
||
nodep->widthForce(width,width); // Signing stays as-is, as parsed from declaration
|
||
//if (debug()>=9) nodep->dumpTree("-class-out-");
|
||
}
|
||
virtual void visit(AstMemberDType* nodep, AstNUser* vup) {
|
||
if (nodep->didWidthAndSet()) return; // This node is a dtype & not both PRELIMed+FINALed
|
||
if (nodep->childDTypep()) nodep->refDTypep(moveChildDTypeEdit(nodep));
|
||
// Iterate into subDTypep() to resolve that type and update pointer.
|
||
nodep->refDTypep(iterateEditDTypep(nodep, nodep->subDTypep()));
|
||
nodep->dtypep(nodep); // The member itself, not subDtype
|
||
nodep->widthFromSub(nodep->subDTypep());
|
||
}
|
||
virtual void visit(AstMemberSel* nodep, AstNUser* vup) {
|
||
UINFO(5," MEMBERSEL "<<nodep<<endl);
|
||
if (debug()>=9) nodep->dumpTree("-mbs-in: ");
|
||
nodep->iterateChildren(*this,WidthVP(SELF,BOTH).p());
|
||
if (debug()>=9) nodep->dumpTree("-mbs-ic: ");
|
||
// Find the fromp dtype - should be a class
|
||
AstNodeDType* fromDtp = nodep->fromp()->dtypep()->skipRefToEnump();
|
||
UINFO(9," from dt "<<fromDtp<<endl);
|
||
AstMemberDType* memberp = NULL; // NULL=error below
|
||
if (AstNodeClassDType* adtypep = fromDtp->castNodeClassDType()) {
|
||
// No need to width-resolve the class, as it was done when we did the child
|
||
memberp = adtypep->findMember(nodep->name());
|
||
if (!memberp) {
|
||
nodep->v3error("Member '"<<nodep->prettyName()<<"' not found in structure");
|
||
}
|
||
}
|
||
else if (fromDtp->castEnumDType()) {
|
||
// Method call on enum without following parenthesis, e.g. "ENUM.next"
|
||
// Convert this into a method call, and let that visitor figure out what to do next
|
||
AstNode* newp = new AstMethodSel(nodep->fileline(), nodep->fromp()->unlinkFrBack(), nodep->name(), NULL);
|
||
nodep->replaceWith(newp);
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
newp->accept(*this,vup);
|
||
return;
|
||
}
|
||
else {
|
||
nodep->v3error("Member selection of non-struct/union object '"
|
||
<<nodep->fromp()->prettyTypeName()<<"' which is a '"<<nodep->fromp()->dtypep()->prettyTypeName()<<"'");
|
||
}
|
||
if (memberp) {
|
||
if (m_attrp) { // Looking for the base of the attribute
|
||
nodep->dtypep(memberp);
|
||
UINFO(9," MEMBERSEL(attr) -> "<<nodep<<endl);
|
||
UINFO(9," dt-> "<<nodep->dtypep()<<endl);
|
||
} else {
|
||
AstSel* newp = new AstSel(nodep->fileline(), nodep->fromp()->unlinkFrBack(),
|
||
memberp->lsb(), memberp->width());
|
||
newp->dtypep(memberp->subDTypep()->skipRefToEnump()); // Must skip over the member to find the union; as the member may disappear later
|
||
newp->didWidth(true); // Don't replace dtype with basic type
|
||
UINFO(9," MEMBERSEL -> "<<newp<<endl);
|
||
UINFO(9," dt-> "<<newp->dtypep()<<endl);
|
||
nodep->replaceWith(newp);
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
// Should be able to treat it as a normal-ish nodesel - maybe. The lhsp() will be strange until this stage; create the number here?
|
||
}
|
||
}
|
||
if (!memberp) { // Very bogus, but avoids core dump
|
||
nodep->replaceWith(new AstConst(nodep->fileline(), AstConst::LogicFalse()));
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
}
|
||
}
|
||
|
||
virtual void visit(AstMethodSel* nodep, AstNUser* vup) {
|
||
UINFO(5," METHODSEL "<<nodep<<endl);
|
||
if (debug()>=9) nodep->dumpTree("-mts-in: ");
|
||
// Should check types the method requires, but at present we don't do much
|
||
nodep->fromp()->accept(*this,WidthVP(SELF,BOTH).p());
|
||
for (AstArg* argp = nodep->pinsp()->castArg(); argp; argp = argp->nextp()->castArg()) {
|
||
if (argp->exprp()) argp->exprp()->accept(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
// Find the fromp dtype - should be a class
|
||
if (!nodep->fromp() || !nodep->fromp()->dtypep()) nodep->v3fatalSrc("Unsized expression");
|
||
AstNodeDType* fromDtp = nodep->fromp()->dtypep()->skipRefToEnump();
|
||
UINFO(9," from dt "<<fromDtp<<endl);
|
||
if (AstEnumDType* adtypep = fromDtp->castEnumDType()) {
|
||
// Method call on enum without following parenthesis, e.g. "ENUM.next"
|
||
// Convert this into a method call, and let that visitor figure out what to do next
|
||
if (adtypep) {}
|
||
if (nodep->name() == "num"
|
||
|| nodep->name() == "first"
|
||
|| nodep->name() == "last") {
|
||
// Constant value
|
||
AstConst* newp = NULL;
|
||
if (nodep->pinsp()) nodep->v3error("Arguments passed to enum.num method, but it does not take arguments");
|
||
if (nodep->name() == "num") {
|
||
int items = 0;
|
||
for (AstNode* itemp = adtypep->itemsp(); itemp; itemp = itemp->nextp()) ++items;
|
||
newp = new AstConst(nodep->fileline(), AstConst::Signed32(), items);
|
||
} else if (nodep->name() == "first") {
|
||
AstEnumItem* itemp = adtypep->itemsp();
|
||
if (!itemp) newp = new AstConst(nodep->fileline(), AstConst::Signed32(), 0); // Spec doesn't say what to do
|
||
else newp = itemp->valuep()->cloneTree(false)->castConst(); // A const
|
||
} else if (nodep->name() == "last") {
|
||
AstEnumItem* itemp = adtypep->itemsp();
|
||
while (itemp && itemp->nextp()) itemp = itemp->nextp()->castEnumItem();
|
||
if (!itemp) newp = new AstConst(nodep->fileline(), AstConst::Signed32(), 0); // Spec doesn't say what to do
|
||
else newp = itemp->valuep()->cloneTree(false)->castConst(); // A const
|
||
}
|
||
if (!newp) nodep->v3fatalSrc("Enum method (perhaps enum item) not const");
|
||
newp->fileline(nodep->fileline()); // Use method's filename/line number to be clearer; may have warning disables
|
||
nodep->replaceWith(newp);
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
}
|
||
else if (nodep->name() == "name"
|
||
|| nodep->name() == "next"
|
||
|| nodep->name() == "prev") {
|
||
AstAttrType attrType;
|
||
if (nodep->name() == "name") attrType = AstAttrType::ENUM_NAME;
|
||
else if (nodep->name() == "next") attrType = AstAttrType::ENUM_NEXT;
|
||
else if (nodep->name() == "prev") attrType = AstAttrType::ENUM_PREV;
|
||
else nodep->v3fatalSrc("Bad case");
|
||
|
||
if (nodep->pinsp() && nodep->name() == "name") {
|
||
nodep->v3error("Arguments passed to enum.name method, but it does not take arguments");
|
||
} else if (nodep->pinsp() && !(nodep->pinsp()->castArg()->exprp()->castConst()
|
||
&& nodep->pinsp()->castArg()->exprp()->castConst()->toUInt()==1
|
||
&& !nodep->pinsp()->nextp())) {
|
||
nodep->v3error("Unsupported: Arguments passed to enum.next method");
|
||
}
|
||
// Need a runtime lookup table. Yuk.
|
||
// Ideally we would have a fast algorithm when a number is
|
||
// of small width and complete and so can use an array, and
|
||
// a map for when the value is many bits and sparse.
|
||
uint64_t msbdim = 0;
|
||
{
|
||
for (AstEnumItem* itemp = adtypep->itemsp(); itemp; itemp = itemp->nextp()->castEnumItem()) {
|
||
AstConst* vconstp = itemp->valuep()->castConst();
|
||
if (!vconstp) nodep->v3fatalSrc("Enum item without constified value");
|
||
if (vconstp->toUQuad() >= msbdim) msbdim = vconstp->toUQuad();
|
||
}
|
||
if (adtypep->itemsp()->width() > 64 || msbdim >= 1024) {
|
||
nodep->v3error("Unsupported; enum next/prev method on enum with > 10 bits");
|
||
return;
|
||
}
|
||
}
|
||
AstVar* varp = enumVarp(adtypep, attrType, msbdim);
|
||
AstVarRef* varrefp = new AstVarRef(nodep->fileline(), varp, false);
|
||
varrefp->packagep(v3Global.rootp()->dollarUnitPkgAddp());
|
||
AstNode* newp = new AstArraySel(nodep->fileline(), varrefp, nodep->fromp()->unlinkFrBack());
|
||
nodep->replaceWith(newp); nodep->deleteTree(); VL_DANGLING(nodep);
|
||
} else {
|
||
nodep->v3error("Unknown built-in enum method '"<<nodep->fromp()->prettyTypeName()<<"'");
|
||
}
|
||
}
|
||
else {
|
||
nodep->v3error("Unsupported: Member call on non-enum object '"
|
||
<<nodep->fromp()->prettyTypeName()<<"' which is a '"<<nodep->fromp()->dtypep()->prettyTypeName()<<"'");
|
||
}
|
||
}
|
||
|
||
virtual void visit(AstPattern* nodep, AstNUser* vup) {
|
||
if (nodep->didWidthAndSet()) return;
|
||
UINFO(9,"PATTERN "<<nodep<<endl);
|
||
if (nodep->childDTypep()) nodep->dtypep(moveChildDTypeEdit(nodep)); // data_type '{ pattern }
|
||
if (!nodep->dtypep() && vup->c()->dtypeNullp()) { // Get it from parent assignment/pin/etc
|
||
nodep->dtypep(vup->c()->dtypep());
|
||
}
|
||
AstNodeDType* vdtypep = nodep->dtypep();
|
||
if (!vdtypep) nodep->v3error("Unsupported/Illegal: Assignment pattern member not underneath a supported construct: "<<nodep->backp()->prettyTypeName());
|
||
{
|
||
vdtypep = vdtypep->skipRefp();
|
||
nodep->dtypep(vdtypep);
|
||
UINFO(9," adtypep "<<vdtypep<<endl);
|
||
nodep->dtypep(vdtypep);
|
||
// Determine replication count, and replicate initial value as widths need to be individually determined
|
||
for (AstPatMember* patp = nodep->itemsp()->castPatMember(); patp; patp = patp->nextp()->castPatMember()) {
|
||
int times = visitPatMemberRep(patp);
|
||
for (int i=1; i<times; i++) {
|
||
AstNode* newp = patp->cloneTree(false);
|
||
patp->addNextHere(newp);
|
||
// This loop will see the new elements as part of nextp()
|
||
}
|
||
}
|
||
// Convert any PatMember with multiple items to multiple PatMembers
|
||
for (AstPatMember* patp = nodep->itemsp()->castPatMember(); patp; patp = patp->nextp()->castPatMember()) {
|
||
if (patp->lhssp()->nextp()) {
|
||
// Can't just addNext, as would add to end of all members. So detach, add next and reattach
|
||
AstNRelinker relinkHandle;
|
||
patp->unlinkFrBack(&relinkHandle);
|
||
while (AstNode* movep = patp->lhssp()->nextp()) {
|
||
movep->unlinkFrBack(); // Not unlinkFrBackWithNext, just one
|
||
AstPatMember* newp = new AstPatMember(patp->fileline(), movep, patp->keyp()->cloneTree(true), NULL);
|
||
patp->addNext(newp);
|
||
}
|
||
relinkHandle.relink(patp);
|
||
}
|
||
}
|
||
AstPatMember* defaultp = NULL;
|
||
for (AstPatMember* patp = nodep->itemsp()->castPatMember(); patp; patp = patp->nextp()->castPatMember()) {
|
||
if (patp->isDefault()) {
|
||
if (defaultp) nodep->v3error("Multiple '{ default: } clauses");
|
||
defaultp = patp;
|
||
patp->unlinkFrBack();
|
||
}
|
||
}
|
||
while (AstConstDType* classp = vdtypep->castConstDType()) {
|
||
vdtypep = classp->subDTypep()->skipRefp();
|
||
}
|
||
if (AstNodeClassDType* classp = vdtypep->castNodeClassDType()) {
|
||
// Due to "default" and tagged patterns, we need to determine
|
||
// which member each AstPatMember corresponds to before we can
|
||
// determine the dtypep for that PatMember's value, and then
|
||
// width the initial value appropriately.
|
||
typedef map<AstMemberDType*,AstPatMember*> PatMap;
|
||
PatMap patmap;
|
||
{
|
||
AstMemberDType* memp = classp->membersp();
|
||
AstPatMember* patp = nodep->itemsp()->castPatMember();
|
||
for (; memp || patp; ) {
|
||
if (patp) {
|
||
if (patp->keyp()) {
|
||
if (AstText* textp = patp->keyp()->castText()) {
|
||
memp = classp->findMember(textp->text());
|
||
if (!memp) {
|
||
patp->keyp()->v3error("Assignment pattern key '"<<textp->text()<<"' not found as member");
|
||
continue;
|
||
}
|
||
} else {
|
||
patp->keyp()->v3error("Assignment pattern key not supported/understood: "<<patp->keyp()->prettyTypeName());
|
||
}
|
||
}
|
||
}
|
||
if (memp && !patp) {
|
||
// Missing init elements, warn below
|
||
memp=NULL; patp=NULL; break;
|
||
} else if (!memp && patp) { patp->v3error("Assignment pattern contains too many elements");
|
||
memp=NULL; patp=NULL; break;
|
||
} else {
|
||
patmap.insert(make_pair(memp, patp));
|
||
}
|
||
// Next
|
||
if (memp) memp = memp->nextp()->castMemberDType();
|
||
if (patp) patp = patp->nextp()->castPatMember();
|
||
}
|
||
}
|
||
AstNode* newp = NULL;
|
||
for (AstMemberDType* memp = classp->membersp(); memp; memp=memp->nextp()->castMemberDType()) {
|
||
PatMap::iterator it = patmap.find(memp);
|
||
AstPatMember* newpatp = NULL;
|
||
AstPatMember* patp = NULL;
|
||
if (it == patmap.end()) {
|
||
if (defaultp) {
|
||
newpatp = defaultp->cloneTree(false);
|
||
patp = newpatp;
|
||
}
|
||
else {
|
||
if (!classp->castUnionDType()) {
|
||
patp->v3error("Assignment pattern missed initializing elements: "<<memp->prettyTypeName());
|
||
}
|
||
}
|
||
} else {
|
||
patp = it->second;
|
||
}
|
||
if (patp) {
|
||
// Determine initial values
|
||
vdtypep = memp;
|
||
patp->dtypep(memp);
|
||
patp->accept(*this,WidthVP(memp,BOTH).p());
|
||
// Convert to concat for now
|
||
AstNode* valuep = patp->lhssp()->unlinkFrBack();
|
||
if (valuep->castConst()) {
|
||
// Forming a AstConcat will cause problems with unsized (uncommitted sized) constants
|
||
if (AstNode* newp = WidthCommitVisitor::newIfConstCommitSize(valuep->castConst())) {
|
||
pushDeletep(valuep); VL_DANGLING(valuep);
|
||
valuep = newp;
|
||
}
|
||
}
|
||
if (!newp) newp = valuep;
|
||
else {
|
||
AstConcat* concatp = new AstConcat(patp->fileline(), newp, valuep);
|
||
newp = concatp;
|
||
newp->dtypeSetLogicSized(concatp->lhsp()->width()+concatp->rhsp()->width(),
|
||
concatp->lhsp()->width()+concatp->rhsp()->width(),
|
||
nodep->dtypep()->numeric());
|
||
}
|
||
}
|
||
if (newpatp) { pushDeletep(newpatp); VL_DANGLING(newpatp); }
|
||
}
|
||
if (newp) nodep->replaceWith(newp);
|
||
else nodep->v3error("Assignment pattern with no members");
|
||
pushDeletep(nodep); VL_DANGLING(nodep); // Deletes defaultp also, if present
|
||
}
|
||
else if (vdtypep->castNodeArrayDType()) {
|
||
AstNodeArrayDType* arrayp = vdtypep->castNodeArrayDType();
|
||
VNumRange range = arrayp->declRange();
|
||
PatVecMap patmap = patVectorMap(nodep, range);
|
||
UINFO(9,"ent "<<range.hi()<<" to "<<range.lo()<<endl);
|
||
AstNode* newp = NULL;
|
||
for (int ent=range.hi(); ent>=range.lo(); --ent) {
|
||
AstPatMember* newpatp = NULL;
|
||
AstPatMember* patp = NULL;
|
||
PatVecMap::iterator it=patmap.find(ent);
|
||
if (it == patmap.end()) {
|
||
if (defaultp) {
|
||
newpatp = defaultp->cloneTree(false);
|
||
patp = newpatp;
|
||
}
|
||
else {
|
||
nodep->v3error("Assignment pattern missed initializing elements: "<<ent);
|
||
}
|
||
} else {
|
||
patp = it->second;
|
||
patmap.erase(it);
|
||
}
|
||
|
||
if (patp) {
|
||
// Determine initial values
|
||
vdtypep = arrayp->subDTypep();
|
||
// Don't want the RHS an array
|
||
patp->dtypep(vdtypep);
|
||
// Determine values - might be another InitArray
|
||
patp->accept(*this,WidthVP(patp->dtypep(),BOTH).p());
|
||
// Convert to InitArray or constify immediately
|
||
AstNode* valuep = patp->lhssp()->unlinkFrBack();
|
||
if (valuep->castConst()) {
|
||
// Forming a AstConcat will cause problems with unsized (uncommitted sized) constants
|
||
if (AstNode* newp = WidthCommitVisitor::newIfConstCommitSize(valuep->castConst())) {
|
||
pushDeletep(valuep); VL_DANGLING(valuep);
|
||
valuep = newp;
|
||
}
|
||
}
|
||
if (arrayp->castUnpackArrayDType()) {
|
||
if (!newp) {
|
||
newp = new AstInitArray(nodep->fileline(), arrayp, valuep);
|
||
} else {
|
||
// We iterate hi()..lo() as that is what packed needs,
|
||
// but INITARRAY needs lo() first
|
||
newp->castInitArray()->initsp()->addHereThisAsNext(valuep);
|
||
}
|
||
} else { // Packed. Convert to concat for now.
|
||
if (!newp) newp = valuep;
|
||
else {
|
||
AstConcat* concatp = new AstConcat(patp->fileline(), newp, valuep);
|
||
newp = concatp;
|
||
newp->dtypeSetLogicSized(concatp->lhsp()->width()+concatp->rhsp()->width(),
|
||
concatp->lhsp()->width()+concatp->rhsp()->width(),
|
||
nodep->dtypep()->numeric());
|
||
}
|
||
}
|
||
}
|
||
if (newpatp) { pushDeletep(newpatp); VL_DANGLING(newpatp); }
|
||
}
|
||
if (!patmap.empty()) nodep->v3error("Assignment pattern with too many elements");
|
||
if (newp) nodep->replaceWith(newp);
|
||
else nodep->v3error("Assignment pattern with no members");
|
||
//if (debug()>=9) newp->dumpTree("-apat-out: ");
|
||
pushDeletep(nodep); VL_DANGLING(nodep); // Deletes defaultp also, if present
|
||
}
|
||
else if (vdtypep->castBasicDType()
|
||
&& vdtypep->castBasicDType()->isRanged()) {
|
||
AstBasicDType* bdtypep = vdtypep->castBasicDType();
|
||
VNumRange range = bdtypep->declRange();
|
||
PatVecMap patmap = patVectorMap(nodep, range);
|
||
UINFO(9,"ent "<<range.hi()<<" to "<<range.lo()<<endl);
|
||
AstNode* newp = NULL;
|
||
for (int ent=range.hi(); ent>=range.lo(); --ent) {
|
||
AstPatMember* newpatp = NULL;
|
||
AstPatMember* patp = NULL;
|
||
PatVecMap::iterator it=patmap.find(ent);
|
||
if (it == patmap.end()) {
|
||
if (defaultp) {
|
||
newpatp = defaultp->cloneTree(false);
|
||
patp = newpatp;
|
||
}
|
||
else {
|
||
nodep->v3error("Assignment pattern missed initializing elements: "<<ent);
|
||
}
|
||
} else {
|
||
patp = it->second;
|
||
patmap.erase(it);
|
||
}
|
||
if (patp) {
|
||
// Determine initial values
|
||
vdtypep = nodep->findLogicBoolDType();
|
||
// Don't want the RHS an array
|
||
patp->dtypep(vdtypep);
|
||
// Determine values - might be another InitArray
|
||
patp->accept(*this,WidthVP(patp->dtypep(),BOTH).p());
|
||
// Convert to InitArray or constify immediately
|
||
AstNode* valuep = patp->lhssp()->unlinkFrBack();
|
||
if (valuep->castConst()) {
|
||
// Forming a AstConcat will cause problems with unsized (uncommitted sized) constants
|
||
if (AstNode* newp = WidthCommitVisitor::newIfConstCommitSize(valuep->castConst())) {
|
||
pushDeletep(valuep); VL_DANGLING(valuep);
|
||
valuep = newp;
|
||
}
|
||
}
|
||
{ // Packed. Convert to concat for now.
|
||
if (!newp) newp = valuep;
|
||
else {
|
||
AstConcat* concatp = new AstConcat(patp->fileline(), newp, valuep);
|
||
newp = concatp;
|
||
newp->dtypeSetLogicSized(concatp->lhsp()->width()+concatp->rhsp()->width(),
|
||
concatp->lhsp()->width()+concatp->rhsp()->width(),
|
||
nodep->dtypep()->numeric());
|
||
}
|
||
}
|
||
}
|
||
if (newpatp) { pushDeletep(newpatp); VL_DANGLING(newpatp); }
|
||
}
|
||
if (!patmap.empty()) nodep->v3error("Assignment pattern with too many elements");
|
||
if (newp) nodep->replaceWith(newp);
|
||
else nodep->v3error("Assignment pattern with no members");
|
||
//if (debug()>=9) newp->dumpTree("-apat-out: ");
|
||
pushDeletep(nodep); VL_DANGLING(nodep); // Deletes defaultp also, if present
|
||
} else {
|
||
nodep->v3error("Unsupported: Assignment pattern applies against non struct/union: "<<vdtypep->prettyTypeName());
|
||
}
|
||
}
|
||
}
|
||
virtual void visit(AstPatMember* nodep, AstNUser* vup) {
|
||
AstNodeDType* vdtypep = vup->c()->dtypeNullp();
|
||
if (!vdtypep) nodep->v3fatalSrc("Pattern member type not assigned by AstPattern visitor");
|
||
nodep->dtypep(vdtypep);
|
||
UINFO(9," PATMEMBER "<<nodep<<endl);
|
||
if (nodep->lhssp()->nextp()) nodep->v3fatalSrc("PatMember value should be singular w/replicates removed");
|
||
// Need to propagate assignment type downwards, even on prelim
|
||
nodep->iterateChildren(*this,WidthVP(nodep->dtypep(),BOTH).p());
|
||
iterateCheck(nodep,"Pattern value",nodep->lhssp(),CONTEXT,FINAL,vdtypep,EXTEND_LHS);
|
||
}
|
||
int visitPatMemberRep(AstPatMember* nodep) {
|
||
uint32_t times = 1;
|
||
if (nodep->repp()) { // else repp()==NULL shorthand for rep count 1
|
||
iterateCheckSizedSelf(nodep,"LHS",nodep->repp(),SELF,BOTH);
|
||
V3Const::constifyParamsEdit(nodep->repp()); // repp may change
|
||
AstConst* constp = nodep->repp()->castConst();
|
||
if (!constp) { nodep->v3error("Replication value isn't a constant."); times=0; }
|
||
else times = constp->toUInt();
|
||
if (times==0) { nodep->v3error("Pattern replication value of 0 is not legal."); times=1; }
|
||
nodep->repp()->unlinkFrBackWithNext()->deleteTree(); // Done with replicate before cloning
|
||
}
|
||
return times;
|
||
}
|
||
|
||
virtual void visit(AstPslClocked* nodep, AstNUser* vup) {
|
||
if (vup->c()->prelim()) { // First stage evaluation
|
||
iterateCheckBool(nodep,"Property",nodep->propp(),BOTH);
|
||
nodep->sensesp()->iterateAndNext(*this);
|
||
if (nodep->disablep()) {
|
||
iterateCheckBool(nodep,"Disable",nodep->disablep(),BOTH); // it's like an if() condition.
|
||
}
|
||
nodep->dtypeSetLogicBool();
|
||
}
|
||
}
|
||
|
||
//--------------------
|
||
// Top levels
|
||
|
||
virtual void visit(AstNodeCase* nodep, AstNUser* vup) {
|
||
// IEEE-2012 12.5:
|
||
// Width: MAX(expr, all items)
|
||
// Signed: Only if expr, and all items signed
|
||
assertAtStatement(nodep,vup);
|
||
nodep->exprp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p());
|
||
for (AstCaseItem* nextip, *itemp = nodep->itemsp(); itemp; itemp=nextip) {
|
||
nextip = itemp->nextp()->castCaseItem(); // Prelim may cause the node to get replaced
|
||
if (!nodep->castGenCase()) itemp->bodysp()->iterateAndNext(*this);
|
||
for (AstNode* nextcp, *condp = itemp->condsp(); condp; condp=nextcp) {
|
||
nextcp = condp->nextp(); // Prelim may cause the node to get replaced
|
||
condp->iterate(*this,WidthVP(CONTEXT,PRELIM).p()); VL_DANGLING(condp);
|
||
}
|
||
}
|
||
|
||
// Take width as maximum across all items, if any is real whole thing is real
|
||
AstNodeDType* subDTypep = nodep->exprp()->dtypep();
|
||
for (AstCaseItem* itemp = nodep->itemsp(); itemp; itemp=itemp->nextp()->castCaseItem()) {
|
||
for (AstNode* condp = itemp->condsp(); condp; condp=condp->nextp()) {
|
||
if (condp->dtypep() != subDTypep) {
|
||
if (condp->dtypep()->isDouble()) {
|
||
subDTypep = nodep->findDoubleDType();
|
||
} else {
|
||
int width = max(subDTypep->width(),condp->width());
|
||
int mwidth = max(subDTypep->widthMin(),condp->widthMin());
|
||
bool issigned = subDTypep->isSigned() && condp->isSigned();
|
||
subDTypep = nodep->findLogicDType(width,mwidth,AstNumeric::fromBool(issigned));
|
||
}
|
||
}
|
||
}
|
||
}
|
||
// Apply width
|
||
iterateCheck(nodep,"Case expression",nodep->exprp(),CONTEXT,FINAL,subDTypep,EXTEND_LHS);
|
||
for (AstCaseItem* itemp = nodep->itemsp(); itemp; itemp=itemp->nextp()->castCaseItem()) {
|
||
for (AstNode* nextcp, *condp = itemp->condsp(); condp; condp=nextcp) {
|
||
nextcp = condp->nextp(); // Final may cause the node to get replaced
|
||
iterateCheck(nodep,"Case Item",condp,CONTEXT,FINAL,subDTypep,EXTEND_LHS);
|
||
}
|
||
}
|
||
}
|
||
virtual void visit(AstNodeFor* nodep, AstNUser* vup) {
|
||
assertAtStatement(nodep,vup);
|
||
nodep->initsp()->iterateAndNext(*this);
|
||
iterateCheckBool(nodep,"For Test Condition",nodep->condp(),BOTH); // it's like an if() condition.
|
||
if (!nodep->castGenFor()) nodep->bodysp()->iterateAndNext(*this);
|
||
nodep->incsp()->iterateAndNext(*this);
|
||
|
||
}
|
||
virtual void visit(AstRepeat* nodep, AstNUser* vup) {
|
||
assertAtStatement(nodep,vup);
|
||
nodep->countp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
nodep->bodysp()->iterateAndNext(*this);
|
||
}
|
||
virtual void visit(AstWhile* nodep, AstNUser* vup) {
|
||
assertAtStatement(nodep,vup);
|
||
nodep->precondsp()->iterateAndNext(*this);
|
||
iterateCheckBool(nodep,"For Test Condition",nodep->condp(),BOTH); // it's like an if() condition.
|
||
nodep->bodysp()->iterateAndNext(*this);
|
||
nodep->incsp()->iterateAndNext(*this);
|
||
}
|
||
virtual void visit(AstNodeIf* nodep, AstNUser* vup) {
|
||
assertAtStatement(nodep,vup);
|
||
//if (debug()) nodep->dumpTree(cout," IfPre: ");
|
||
if (!nodep->castGenIf()) { // for m_paramsOnly
|
||
nodep->ifsp()->iterateAndNext(*this);
|
||
nodep->elsesp()->iterateAndNext(*this);
|
||
}
|
||
iterateCheckBool(nodep,"If",nodep->condp(),BOTH); // it's like an if() condition.
|
||
//if (debug()) nodep->dumpTree(cout," IfOut: ");
|
||
}
|
||
|
||
virtual void visit(AstNodeAssign* nodep, AstNUser* vup) {
|
||
// IEEE-2012 10.7, 11.8.2, 11.8.3, 11.5: (Careful of 11.8.1 which is
|
||
// only one step; final dtype depends on assign LHS.)
|
||
// Determine RHS type width and signing
|
||
// Propagate type down to *non-self-determined* operators
|
||
// Real propagates only across one operator if one side is real - handled in each visitor.
|
||
// Then LHS sign-extends only if *RHS* is signed
|
||
assertAtStatement(nodep,vup);
|
||
//if (debug()) nodep->dumpTree(cout," AssignPre: ");
|
||
{
|
||
//if (debug()) nodep->dumpTree(cout,"- assin: ");
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
if (!nodep->lhsp()->dtypep()) nodep->v3fatalSrc("How can LHS be untyped?");
|
||
if (!nodep->lhsp()->dtypep()->widthSized()) nodep->v3fatalSrc("How can LHS be unsized?");
|
||
nodep->dtypeFrom(nodep->lhsp());
|
||
//
|
||
// AstPattern needs to know the proposed data type of the lhs, so pass on the prelim
|
||
nodep->rhsp()->iterateAndNext(*this,WidthVP(nodep->dtypep(),PRELIM).p());
|
||
//
|
||
//if (debug()) nodep->dumpTree(cout,"- assign: ");
|
||
AstNodeDType* lhsDTypep = nodep->lhsp()->dtypep(); // Note we use rhsp for context determined
|
||
iterateCheckAssign(nodep,"Assign RHS",nodep->rhsp(),FINAL,lhsDTypep);
|
||
//if (debug()) nodep->dumpTree(cout," AssignOut: ");
|
||
}
|
||
}
|
||
|
||
virtual void visit(AstSFormatF* nodep, AstNUser* vup) {
|
||
// Excludes NodeDisplay, see below
|
||
if (vup && !vup->c()->prelim()) return; // Can be called as statement or function
|
||
// Just let all arguments seek their natural sizes
|
||
nodep->iterateChildren(*this,WidthVP(SELF,BOTH).p());
|
||
//
|
||
UINFO(9," Display in "<<nodep->text()<<endl);
|
||
string newFormat;
|
||
bool inPct = false;
|
||
AstNode* argp = nodep->exprsp();
|
||
string txt = nodep->text();
|
||
string fmt;
|
||
for (string::const_iterator it = txt.begin(); it!=txt.end(); ++it) {
|
||
char ch = *it;
|
||
if (!inPct && ch=='%') {
|
||
inPct = true;
|
||
fmt = ch;
|
||
} else if (inPct && (isdigit(ch) || ch=='.')) {
|
||
fmt += ch;
|
||
} else if (tolower(inPct)) {
|
||
inPct = false;
|
||
bool added = false;
|
||
switch (tolower(ch)) {
|
||
case '%': break; // %% - just output a %
|
||
case 'm': break; // %m - auto insert "name"
|
||
case 'l': break; // %m - auto insert "library"
|
||
case 'd': { // Convert decimal to either 'd' or '#'
|
||
if (argp && argp->isSigned()) { // Convert it
|
||
ch = '~';
|
||
}
|
||
if (argp) argp=argp->nextp();
|
||
break;
|
||
}
|
||
case 'p': { // Packed
|
||
// Very hacky and non-compliant; print strings as strings, otherwise as hex
|
||
if (argp && argp->dtypep()->basicp()->isString()) { // Convert it
|
||
added = true;
|
||
newFormat += "\"%@\"";
|
||
} else {
|
||
added = true;
|
||
newFormat += "'h%0h";
|
||
}
|
||
if (argp) argp=argp->nextp();
|
||
break;
|
||
}
|
||
case 's': { // Convert string to pack string
|
||
if (argp && argp->dtypep()->basicp()->isString()) { // Convert it
|
||
ch = '@';
|
||
}
|
||
if (argp) argp=argp->nextp();
|
||
break;
|
||
}
|
||
default: { // Most operators, just move to next argument
|
||
if (argp) argp=argp->nextp();
|
||
break;
|
||
}
|
||
} // switch
|
||
if (!added) {
|
||
fmt += ch;
|
||
newFormat += fmt;
|
||
}
|
||
} else {
|
||
newFormat += ch;
|
||
}
|
||
}
|
||
nodep->text(newFormat);
|
||
UINFO(9," Display out "<<nodep->text()<<endl);
|
||
}
|
||
virtual void visit(AstDisplay* nodep, AstNUser* vup) {
|
||
assertAtStatement(nodep,vup);
|
||
if (nodep->filep()) {
|
||
iterateCheckFileDesc(nodep,nodep->filep(),BOTH);
|
||
}
|
||
// Just let all arguments seek their natural sizes
|
||
nodep->iterateChildren(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
virtual void visit(AstFOpen* nodep, AstNUser* vup) {
|
||
// Although a system function in IEEE, here a statement which sets the file pointer (MCD)
|
||
assertAtStatement(nodep,vup);
|
||
iterateCheckFileDesc(nodep,nodep->filep(),BOTH);
|
||
nodep->filenamep()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
nodep->modep()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
virtual void visit(AstFClose* nodep, AstNUser* vup) {
|
||
assertAtStatement(nodep,vup);
|
||
iterateCheckFileDesc(nodep,nodep->filep(),BOTH);
|
||
}
|
||
virtual void visit(AstFEof* nodep, AstNUser* vup) {
|
||
if (vup->c()->prelim()) {
|
||
iterateCheckFileDesc(nodep,nodep->filep(),BOTH);
|
||
nodep->dtypeSetLogicSized(32,1,AstNumeric::SIGNED); // Spec says integer return
|
||
}
|
||
}
|
||
virtual void visit(AstFFlush* nodep, AstNUser* vup) {
|
||
assertAtStatement(nodep,vup);
|
||
if (nodep->filep()) {
|
||
iterateCheckFileDesc(nodep,nodep->filep(),BOTH);
|
||
}
|
||
}
|
||
virtual void visit(AstFGetC* nodep, AstNUser* vup) {
|
||
if (vup->c()->prelim()) {
|
||
iterateCheckFileDesc(nodep,nodep->filep(),BOTH);
|
||
nodep->dtypeSetLogicSized(32,8,AstNumeric::SIGNED); // Spec says integer return
|
||
}
|
||
}
|
||
virtual void visit(AstFGetS* nodep, AstNUser* vup) {
|
||
if (vup->c()->prelim()) {
|
||
nodep->dtypeSetSigned32(); // Spec says integer return
|
||
iterateCheckFileDesc(nodep,nodep->filep(),BOTH);
|
||
nodep->strgp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
}
|
||
virtual void visit(AstFScanF* nodep, AstNUser* vup) {
|
||
if (vup->c()->prelim()) {
|
||
nodep->dtypeSetSigned32(); // Spec says integer return
|
||
iterateCheckFileDesc(nodep,nodep->filep(),BOTH);
|
||
nodep->exprsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
}
|
||
virtual void visit(AstSScanF* nodep, AstNUser* vup) {
|
||
if (vup->c()->prelim()) {
|
||
nodep->dtypeSetSigned32(); // Spec says integer return
|
||
nodep->fromp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
nodep->exprsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
}
|
||
virtual void visit(AstSysIgnore* nodep, AstNUser* vup) {
|
||
nodep->exprsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
virtual void visit(AstSystemF* nodep, AstNUser* vup) {
|
||
if (vup->c()->prelim()) {
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
nodep->dtypeSetSigned32(); // Spec says integer return
|
||
}
|
||
}
|
||
virtual void visit(AstSystemT* nodep, AstNUser* vup) {
|
||
assertAtStatement(nodep,vup);
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
virtual void visit(AstReadMem* nodep, AstNUser* vup) {
|
||
assertAtStatement(nodep,vup);
|
||
nodep->filenamep()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
nodep->memp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
if (!nodep->memp()->dtypep()->skipRefp()->castUnpackArrayDType()) {
|
||
nodep->memp()->v3error("Unsupported: $readmem into other than unpacked array");
|
||
}
|
||
nodep->lsbp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
nodep->msbp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
virtual void visit(AstValuePlusArgs* nodep, AstNUser* vup) {
|
||
if (vup->c()->prelim()) {
|
||
nodep->exprsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
nodep->dtypeSetSigned32(); // Spec says integer return
|
||
}
|
||
}
|
||
virtual void visit(AstUCStmt* nodep, AstNUser* vup) {
|
||
// Just let all arguments seek their natural sizes
|
||
assertAtStatement(nodep,vup);
|
||
nodep->iterateChildren(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
virtual void visit(AstPslCover* nodep, AstNUser* vup) {
|
||
assertAtStatement(nodep,vup);
|
||
iterateCheckBool(nodep,"Property",nodep->propp(),BOTH); // it's like an if() condition.
|
||
nodep->stmtsp()->iterateAndNext(*this);
|
||
}
|
||
virtual void visit(AstVAssert* nodep, AstNUser* vup) {
|
||
assertAtStatement(nodep,vup);
|
||
iterateCheckBool(nodep,"Property",nodep->propp(),BOTH); // it's like an if() condition.
|
||
nodep->passsp()->iterateAndNext(*this);
|
||
nodep->failsp()->iterateAndNext(*this);
|
||
}
|
||
virtual void visit(AstPin* nodep, AstNUser*) {
|
||
//if (debug()) nodep->dumpTree(cout,"- PinPre: ");
|
||
// TOP LEVEL NODE
|
||
if (nodep->modVarp() && nodep->modVarp()->isGParam()) {
|
||
// Widthing handled as special init() case
|
||
nodep->iterateChildren(*this,WidthVP(SELF,BOTH).p());
|
||
} else if (!m_paramsOnly) {
|
||
if (!nodep->modVarp()->didWidth()) {
|
||
// Var hasn't been widthed, so make it so.
|
||
nodep->modVarp()->iterate(*this);
|
||
}
|
||
if (!nodep->exprp()) { // No-connect
|
||
return;
|
||
}
|
||
// Very much like like an assignment, but which side is LH/RHS
|
||
// depends on pin being a in/output/inout.
|
||
nodep->exprp()->iterateAndNext(*this,WidthVP(nodep->modVarp()->dtypep(),PRELIM).p());
|
||
AstNodeDType* pinDTypep = nodep->modVarp()->dtypep();
|
||
AstNodeDType* conDTypep = nodep->exprp()->dtypep();
|
||
AstNodeDType* subDTypep = pinDTypep;
|
||
int pinwidth = pinDTypep->width();
|
||
int conwidth = conDTypep->width();
|
||
if (conDTypep == pinDTypep) { // If match, we're golden
|
||
nodep->exprp()->iterateAndNext(*this,WidthVP(subDTypep,FINAL).p());
|
||
}
|
||
else if (m_cellRangep) {
|
||
int numInsts = m_cellRangep->elementsConst();
|
||
if (conwidth == pinwidth) {
|
||
// Arrayed instants: widths match so connect to each instance
|
||
subDTypep = conDTypep; // = same expr dtype
|
||
} else if (conwidth == numInsts*pinwidth) {
|
||
// Arrayed instants: one bit for each of the instants (each assign is 1 pinwidth wide)
|
||
subDTypep = conDTypep; // = same expr dtype (but numInst*pin_dtype)
|
||
} else {
|
||
// Must be a error according to spec
|
||
// (Because we need to know if to connect to one or all instants)
|
||
nodep->v3error(ucfirst(nodep->prettyOperatorName())<<" as part of a module instance array"
|
||
<<" requires "<<pinwidth<<" or "<<pinwidth*numInsts
|
||
<<" bits, but connection's "<<nodep->exprp()->prettyTypeName()
|
||
<<" generates "<<conwidth<<" bits.");
|
||
subDTypep = conDTypep; // = same expr dtype
|
||
}
|
||
nodep->exprp()->iterateAndNext(*this,WidthVP(subDTypep,FINAL).p());
|
||
} else {
|
||
if (nodep->modVarp()->isTristate()) {
|
||
if (pinwidth != conwidth) {
|
||
nodep->v3error("Unsupported: "<<ucfirst(nodep->prettyOperatorName())
|
||
<<" to inout signal requires "<<pinwidth
|
||
<<" bits, but connection's "<<nodep->exprp()->prettyTypeName()
|
||
<<" generates "<<conwidth<<" bits.");
|
||
// otherwise would need some mess to force both sides to proper size
|
||
}
|
||
}
|
||
// TODO Simple dtype checking, should be a more general check
|
||
bool hiArray = nodep->exprp()->dtypep()->skipRefp()->castUnpackArrayDType();
|
||
bool loArray = nodep->modVarp()->dtypep()->skipRefp()->castUnpackArrayDType();
|
||
if (loArray != hiArray && pinwidth != conwidth) {
|
||
nodep->v3error("Illegal "<<nodep->prettyOperatorName()<<","
|
||
<<" mismatch between port which is"<<(hiArray?"":" not")<<" an array,"
|
||
<<" and expression which is"<<(loArray?"":" not")<<" an array.");
|
||
UINFO(1," Related lo: "<<nodep->exprp()->dtypep()->skipRefp()<<endl);
|
||
UINFO(1," Related hi: "<<nodep->modVarp()->dtypep()->skipRefp()<<endl);
|
||
}
|
||
iterateCheckAssign(nodep,"pin connection",nodep->exprp(),FINAL,subDTypep);
|
||
}
|
||
}
|
||
//if (debug()) nodep->dumpTree(cout,"- PinOut: ");
|
||
}
|
||
virtual void visit(AstCell* nodep, AstNUser*) {
|
||
if (!m_paramsOnly) {
|
||
if (nodep->modp()->castNotFoundModule()) {
|
||
// We've resolved parameters and hit a module that we couldn't resolve. It's
|
||
// finally time to report it.
|
||
// Note only here in V3Width as this is first visitor after V3Dead.
|
||
nodep->v3error("Cannot find file containing module: "<<nodep->modName());
|
||
v3Global.opt.filePathLookedMsg(nodep->fileline(), nodep->modName());
|
||
}
|
||
if (nodep->rangep()) {
|
||
m_cellRangep = nodep->rangep();
|
||
nodep->rangep()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
}
|
||
nodep->pinsp()->iterateAndNext(*this);
|
||
}
|
||
nodep->paramsp()->iterateAndNext(*this);
|
||
m_cellRangep = NULL;
|
||
}
|
||
virtual void visit(AstGatePin* nodep, AstNUser* vup) {
|
||
if (vup->c()->prelim()) {
|
||
nodep->rangep()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
nodep->exprp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p());
|
||
nodep->dtypeFrom(nodep->rangep());
|
||
// Very much like like an pin
|
||
AstNodeDType* conDTypep = nodep->exprp()->dtypep();
|
||
int numInsts = nodep->rangep()->elementsConst();
|
||
int pinwidth = numInsts;
|
||
int conwidth = conDTypep->width();
|
||
if (conwidth == 1 && pinwidth > 1) { // Multiple connections
|
||
AstNodeDType* subDTypep = nodep->findLogicDType(1,1, conDTypep->numeric());
|
||
nodep->exprp()->iterateAndNext(*this,WidthVP(subDTypep,FINAL).p());
|
||
AstNode* newp = new AstReplicate(nodep->fileline(),
|
||
nodep->exprp()->unlinkFrBack(),
|
||
numInsts);
|
||
nodep->replaceWith(newp);
|
||
}
|
||
else {
|
||
// Eliminating so pass down all of vup
|
||
nodep->exprp()->iterateAndNext(*this,vup);
|
||
nodep->replaceWith(nodep->exprp()->unlinkFrBack());
|
||
}
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
}
|
||
}
|
||
virtual void visit(AstNodeFTask* nodep, AstNUser* vup) {
|
||
// Grab width from the output variable (if it's a function)
|
||
if (nodep->didWidth()) return;
|
||
UINFO(5," FTASK "<<nodep<<endl);
|
||
if (nodep->doingWidth()) {
|
||
nodep->v3error("Unsupported: Recursive function or task call");
|
||
nodep->dtypeSetLogicBool();
|
||
nodep->didWidth(true);
|
||
return;
|
||
}
|
||
// Function hasn't been widthed, so make it so.
|
||
nodep->doingWidth(true); // Would use user1 etc, but V3Width called from too many places to spend a user
|
||
nodep->iterateChildren(*this);
|
||
if (nodep->fvarp()) {
|
||
m_funcp = nodep->castFunc();
|
||
if (!m_funcp) nodep->v3fatalSrc("FTask with function variable, but isn't a function");
|
||
nodep->dtypeFrom(nodep->fvarp()); // Which will get it from fvarp()->dtypep()
|
||
}
|
||
nodep->didWidth(true);
|
||
nodep->doingWidth(false);
|
||
m_funcp = NULL;
|
||
}
|
||
virtual void visit(AstReturn* nodep, AstNUser* vup) {
|
||
// IEEE: Assignment-like context
|
||
assertAtStatement(nodep,vup);
|
||
if (!m_funcp) {
|
||
if (nodep->lhsp()) { // Return w/o value ok other places
|
||
nodep->v3error("Return with return value isn't underneath a function");
|
||
}
|
||
} else {
|
||
if (nodep->lhsp()) {
|
||
// Function hasn't been widthed, so make it so.
|
||
nodep->dtypeFrom(m_funcp->fvarp());
|
||
// AstPattern requires assignments to pass datatype on PRELIM
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(nodep->dtypep(),PRELIM).p());
|
||
iterateCheckAssign(nodep,"Return value",nodep->lhsp(),FINAL,nodep->dtypep());
|
||
}
|
||
}
|
||
}
|
||
|
||
virtual void visit(AstFuncRef* nodep, AstNUser* vup) {
|
||
visit(nodep->castNodeFTaskRef(), vup);
|
||
nodep->dtypeFrom(nodep->taskp());
|
||
//if (debug()) nodep->dumpTree(cout," FuncOut: ");
|
||
}
|
||
virtual void visit(AstNodeFTaskRef* nodep, AstNUser* vup) {
|
||
// For arguments, is assignment-like context; see IEEE rules in AstNodeAssign
|
||
// Function hasn't been widthed, so make it so.
|
||
UINFO(5, " FTASKREF "<<nodep<<endl);
|
||
if (!nodep->taskp()) nodep->v3fatalSrc("Unlinked");
|
||
if (nodep->didWidth()) return;
|
||
nodep->taskp()->iterate(*this);
|
||
//
|
||
// And do the arguments to the task/function too
|
||
for (int accept_mode=0; accept_mode<3; accept_mode++) { // Avoid duplicate code; just do inner stuff several times
|
||
reloop:
|
||
V3TaskConnects tconnects = V3Task::taskConnects(nodep, nodep->taskp()->stmtsp());
|
||
for (V3TaskConnects::iterator it=tconnects.begin(); it!=tconnects.end(); ++it) {
|
||
AstVar* portp = it->first;
|
||
AstArg* argp = it->second;
|
||
AstNode* pinp = argp->exprp();
|
||
if (pinp!=NULL) { // Else argument error we'll find later
|
||
if (accept_mode==0) {
|
||
// Prelim may cause the node to get replaced; we've lost our
|
||
// pointer, so need to iterate separately later
|
||
if (portp->attrSFormat()
|
||
&& (!pinp->castSFormatF() || pinp->nextp())) { // Not already done
|
||
UINFO(4," sformat via metacomment: "<<nodep<<endl);
|
||
AstNRelinker handle;
|
||
argp->unlinkFrBackWithNext(&handle); // Format + additional args, if any
|
||
AstNode* argsp = NULL;
|
||
while (AstArg* nextargp = argp->nextp()->castArg()) {
|
||
// cppcheck-suppress nullPointer
|
||
argsp = argsp->addNext(nextargp->exprp()->unlinkFrBackWithNext()); // Expression goes to SFormatF
|
||
nextargp->unlinkFrBack()->deleteTree(); // Remove the call's Arg wrapper
|
||
}
|
||
string format;
|
||
if (pinp->castConst()) format = pinp->castConst()->num().toString();
|
||
else pinp->v3error("Format to $display-like function must have constant format string");
|
||
pushDeletep(argp); VL_DANGLING(argp);
|
||
AstSFormatF* newp = new AstSFormatF(nodep->fileline(), format, false, argsp);
|
||
if (!newp->scopeNamep() && newp->formatScopeTracking()) {
|
||
newp->scopeNamep(new AstScopeName(newp->fileline()));
|
||
}
|
||
handle.relink(new AstArg(newp->fileline(), "", newp));
|
||
// Connection list is now incorrect (has extra args in it).
|
||
goto reloop; // so exit early; next loop will correct it
|
||
}
|
||
else if (portp->basicp() && portp->basicp()->keyword()==AstBasicDTypeKwd::STRING
|
||
&& !pinp->castCvtPackString()
|
||
&& !pinp->castSFormatF() // Already generates a string
|
||
&& !(pinp->castVarRef() && pinp->castVarRef()->varp()->basicp()->keyword()==AstBasicDTypeKwd::STRING)) {
|
||
UINFO(4," Add CvtPackString: "<<pinp<<endl);
|
||
AstNRelinker handle;
|
||
pinp->unlinkFrBack(&handle); // No next, that's the next pin
|
||
AstNode* newp = new AstCvtPackString(pinp->fileline(), pinp);
|
||
handle.relink(newp);
|
||
pinp = newp;
|
||
}
|
||
// AstPattern requires assignments to pass datatype on PRELIM
|
||
pinp->accept(*this,WidthVP(portp->dtypep(),PRELIM).p()); VL_DANGLING(pinp);
|
||
} else if (accept_mode==1) {
|
||
// Change data types based on above accept completion
|
||
if (portp->isDouble()) {
|
||
spliceCvtD(pinp); VL_DANGLING(pinp);
|
||
}
|
||
} else if (accept_mode==2) {
|
||
// Do PRELIM again, because above accept may have exited early due to node replacement
|
||
pinp->accept(*this,WidthVP(portp->dtypep(),PRELIM).p());
|
||
if ((portp->isOutput() || portp->isInout())
|
||
&& pinp->width() != portp->width()) {
|
||
pinp->v3error("Unsupported: Function output argument '"<<portp->prettyName()<<"'"
|
||
<<" requires "<<portp->width()
|
||
<<" bits, but connection's "<<pinp->prettyTypeName()
|
||
<<" generates "<<pinp->width()<<" bits.");
|
||
// otherwise would need some mess to force both sides to proper size
|
||
// (get an ASSIGN with EXTEND on the lhs instead of rhs)
|
||
}
|
||
if (!portp->basicp() || portp->basicp()->isOpaque()) {
|
||
pinp->accept(*this,WidthVP(portp->dtypep(),FINAL).p());
|
||
} else {
|
||
iterateCheckAssign(nodep,"Function Argument",pinp,FINAL,portp->dtypep());
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
nodep->didWidth(true);
|
||
}
|
||
virtual void visit(AstInitial* nodep, AstNUser* vup) {
|
||
assertAtStatement(nodep,vup);
|
||
m_initialp = nodep;
|
||
nodep->iterateChildren(*this);
|
||
m_initialp = NULL;
|
||
}
|
||
virtual void visit(AstNetlist* nodep, AstNUser*) {
|
||
// Iterate modules backwards, in bottom-up order. That's faster
|
||
nodep->iterateChildrenBackwards(*this);
|
||
}
|
||
|
||
//--------------------
|
||
// Default
|
||
virtual void visit(AstNodeMath* nodep, AstNUser*) {
|
||
nodep->v3fatalSrc("Visit function missing? Widthed function missing for math node: "<<nodep);
|
||
nodep->iterateChildren(*this);
|
||
}
|
||
virtual void visit(AstNode* nodep, AstNUser* vup) {
|
||
// Default: Just iterate
|
||
if (vup) nodep->v3fatalSrc("Visit function missing? Widthed expectation for this node: "<<nodep);
|
||
nodep->iterateChildren(*this);
|
||
}
|
||
|
||
//----------------------------------------------------------------------
|
||
// WIDTH METHODs -- all iterate
|
||
|
||
void visit_Or_Lu64(AstNodeUniop* nodep, AstNUser* vup) {
|
||
// CALLER: AstBitsToRealD
|
||
// Real: Output real
|
||
// LHS presumed self-determined, then coerced to real
|
||
if (vup->c()->prelim()) { // First stage evaluation
|
||
nodep->dtypeSetDouble();
|
||
AstNodeDType* subDTypep = nodep->findLogicDType(64,64, AstNumeric::UNSIGNED);
|
||
// Self-determined operand
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(SELF,PRELIM).p());
|
||
iterateCheck(nodep,"LHS",nodep->lhsp(),SELF,FINAL,subDTypep,EXTEND_EXP);
|
||
}
|
||
}
|
||
void visit_Or_Ls32(AstNodeUniop* nodep, AstNUser* vup) {
|
||
// CALLER: AstIToRD
|
||
// Real: Output real
|
||
// LHS presumed self-determined, then coerced to real
|
||
if (vup->c()->prelim()) { // First stage evaluation
|
||
nodep->dtypeSetDouble();
|
||
AstNodeDType* subDTypep = nodep->findLogicDType(32,32, AstNumeric::SIGNED);
|
||
// Self-determined operand
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(SELF,PRELIM).p());
|
||
iterateCheck(nodep,"LHS",nodep->lhsp(),SELF,FINAL,subDTypep,EXTEND_EXP);
|
||
}
|
||
}
|
||
void visit_Os32_Lr(AstNodeUniop* nodep, AstNUser* vup) {
|
||
// CALLER: RToI
|
||
// Real: LHS real
|
||
// LHS presumed self-determined, then coerced to real
|
||
if (vup->c()->prelim()) { // First stage evaluation
|
||
iterateCheckReal(nodep,"LHS",nodep->lhsp(),BOTH);
|
||
nodep->dtypeSetSigned32();
|
||
}
|
||
}
|
||
void visit_Ou64_Lr(AstNodeUniop* nodep, AstNUser* vup) {
|
||
// CALLER: RealToBits
|
||
// Real: LHS real
|
||
// LHS presumed self-determined, then coerced to real
|
||
if (vup->c()->prelim()) { // First stage evaluation
|
||
iterateCheckReal(nodep,"LHS",nodep->lhsp(),BOTH);
|
||
nodep->dtypeSetUInt64();
|
||
}
|
||
}
|
||
|
||
void visit_log_not(AstNode* nodep, AstNUser* vup) {
|
||
// CALLER: LogNot
|
||
// Width-check: lhs 1 bit
|
||
// Real: Allowed; implicitly compares with zero
|
||
// We calculate the width of the UNDER expression.
|
||
// We then check its width to see if it's legal, and edit if not
|
||
// We finally set the width of our output
|
||
// IEEE-2012: Table 11-21 and 11.8.1 (same as RedAnd):
|
||
// LHS is self-determined
|
||
// Width: 1 bit out
|
||
// Sign: unsigned out (11.8.1)
|
||
if (nodep->op2p()) nodep->v3fatalSrc("For unary ops only!");
|
||
if (vup->c()->prelim()) {
|
||
iterateCheckBool(nodep,"LHS",nodep->op1p(),BOTH);
|
||
nodep->dtypeSetLogicBool();
|
||
}
|
||
}
|
||
void visit_log_and_or(AstNodeBiop* nodep, AstNUser* vup) {
|
||
// CALLER: LogAnd, LogOr, LogIf, LogIff
|
||
// Widths: 1 bit out, lhs 1 bit, rhs 1 bit
|
||
// IEEE-2012 Table 11-21:
|
||
// LHS is self-determined
|
||
// RHS is self-determined
|
||
if (vup->c()->prelim()) {
|
||
iterateCheckBool(nodep,"LHS",nodep->lhsp(),BOTH);
|
||
iterateCheckBool(nodep,"RHS",nodep->rhsp(),BOTH);
|
||
nodep->dtypeSetLogicBool();
|
||
}
|
||
}
|
||
void visit_red_and_or(AstNodeUniop* nodep, AstNUser* vup) {
|
||
// CALLER: RedAnd, RedOr, ...
|
||
// Signed: Output unsigned, Lhs/Rhs/etc non-real (presumed, not in IEEE)
|
||
// IEEE-2012: Table 11-21 and 11.8.1:
|
||
// LHS is self-determined
|
||
// Width: 1 bit out
|
||
// Sign: unsigned out (11.8.1)
|
||
if (vup->c()->prelim()) {
|
||
iterateCheckSizedSelf(nodep,"LHS",nodep->lhsp(),SELF,BOTH);
|
||
nodep->dtypeSetLogicBool();
|
||
}
|
||
}
|
||
void visit_red_unknown(AstNodeUniop* nodep, AstNUser* vup) {
|
||
// CALLER: IsUnknown
|
||
// Signed: Output unsigned, Lhs/Rhs/etc non-real (presumed, not in IEEE)
|
||
// IEEE-2012: Table 11-21 and 11.8.1:
|
||
// LHS is self-determined
|
||
// Width: 1 bit out
|
||
// Sign: unsigned out (11.8.1)
|
||
if (vup->c()->prelim()) {
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(SELF,BOTH).p());
|
||
nodep->dtypeSetLogicBool();
|
||
}
|
||
}
|
||
|
||
void visit_cmp_eq_gt(AstNodeBiop* nodep, AstNUser* vup, bool realok) {
|
||
// CALLER: AstEq, AstGt, ..., AstLtS
|
||
// Real allowed if and only if real_lhs set
|
||
// See IEEE-2012 11.4.4, and 11.8.1:
|
||
// Widths: 1 bit out, width is max of LHS or RHS
|
||
// Sign: signed compare (not output) if both signed, compare is signed, width mismatches sign extend
|
||
// else, compare is unsigned, **zero-extends**
|
||
// Real: If either real, other side becomes real and real compare
|
||
// TODO: chandle/class handle/iface handle: WildEq/WildNeq same as Eq/Neq
|
||
// TODO: chandle/class handle/iface handle only allowed to self-compare or against null
|
||
// TODO: chandle/class handle/iface handle no relational compares
|
||
if (!nodep->rhsp()) nodep->v3fatalSrc("For binary ops only!");
|
||
if (vup->c()->prelim()) {
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p());
|
||
nodep->rhsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p());
|
||
if (nodep->lhsp()->isDouble() || nodep->rhsp()->isDouble()) {
|
||
if (!realok) nodep->v3error("Real not allowed as operand to in ?== operator");
|
||
if (AstNodeBiop* newp=replaceWithDVersion(nodep)) { VL_DANGLING(nodep);
|
||
nodep = newp; // Process new node instead
|
||
iterateCheckReal(nodep,"LHS",nodep->lhsp(),FINAL);
|
||
iterateCheckReal(nodep,"RHS",nodep->rhsp(),FINAL);
|
||
}
|
||
} else if (nodep->lhsp()->isString() || nodep->rhsp()->isString()) {
|
||
if (AstNodeBiop* newp=replaceWithNVersion(nodep)) { VL_DANGLING(nodep);
|
||
nodep = newp; // Process new node instead
|
||
iterateCheckString(nodep,"LHS",nodep->lhsp(),FINAL);
|
||
iterateCheckString(nodep,"RHS",nodep->rhsp(),FINAL);
|
||
}
|
||
} else {
|
||
bool signedFl = nodep->lhsp()->isSigned() && nodep->rhsp()->isSigned();
|
||
if (AstNodeBiop* newp=replaceWithUOrSVersion(nodep, signedFl)) { VL_DANGLING(nodep);
|
||
nodep = newp; // Process new node instead
|
||
}
|
||
int width = max(nodep->lhsp()->width(), nodep->rhsp()->width());
|
||
int ewidth = max(nodep->lhsp()->widthMin(), nodep->rhsp()->widthMin());
|
||
AstNodeDType* subDTypep = nodep->findLogicDType(width, ewidth,
|
||
AstNumeric::fromBool(signedFl));
|
||
iterateCheck(nodep,"LHS",nodep->lhsp(),CONTEXT,FINAL,subDTypep,signedFl?EXTEND_LHS:EXTEND_ZERO);
|
||
iterateCheck(nodep,"RHS",nodep->rhsp(),CONTEXT,FINAL,subDTypep,signedFl?EXTEND_LHS:EXTEND_ZERO);
|
||
}
|
||
nodep->dtypeSetLogicBool();
|
||
}
|
||
}
|
||
void visit_cmp_real(AstNodeBiop* nodep, AstNUser* vup) {
|
||
// CALLER: EqD, LtD
|
||
// Widths: 1 bit out, lhs width == rhs width
|
||
// Signed compare (not output) if both sides signed
|
||
// Real if and only if real_allow set
|
||
// IEEE, 11.4.4: relational compares (<,>,<=,>=,==,===,!=,!==) use "zero padding" on unsigned
|
||
if (!nodep->rhsp()) nodep->v3fatalSrc("For binary ops only!");
|
||
if (vup->c()->prelim()) {
|
||
// See similar handling in visit_cmp_eq_gt where created
|
||
iterateCheckReal(nodep,"LHS",nodep->lhsp(),BOTH);
|
||
iterateCheckReal(nodep,"RHS",nodep->rhsp(),BOTH);
|
||
nodep->dtypeSetLogicBool();
|
||
}
|
||
}
|
||
void visit_cmp_string(AstNodeBiop* nodep, AstNUser* vup) {
|
||
// CALLER: EqN, LtN
|
||
// Widths: 1 bit out, lhs width == rhs width
|
||
// String compare (not output)
|
||
// Real if and only if real_allow set
|
||
if (!nodep->rhsp()) nodep->v3fatalSrc("For binary ops only!");
|
||
if (vup->c()->prelim()) {
|
||
// See similar handling in visit_cmp_eq_gt where created
|
||
iterateCheckString(nodep,"LHS",nodep->lhsp(),BOTH);
|
||
iterateCheckString(nodep,"RHS",nodep->rhsp(),BOTH);
|
||
nodep->dtypeSetLogicBool();
|
||
}
|
||
}
|
||
|
||
void visit_negate_not(AstNodeUniop* nodep, AstNUser* vup, bool real_ok) {
|
||
// CALLER: (real_ok=false) Not
|
||
// CALLER: (real_ok=true) Negate
|
||
// Signed: From lhs
|
||
// IEEE-2012 Table 11-21:
|
||
// Widths: out width = lhs width
|
||
if (nodep->op2p()) nodep->v3fatalSrc("For unary ops only!");
|
||
if (vup->c()->prelim()) {
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p());
|
||
if (!real_ok) checkCvtUS(nodep->lhsp());
|
||
}
|
||
if (real_ok && nodep->lhsp()->isDouble()) {
|
||
spliceCvtD(nodep->lhsp());
|
||
if (AstNodeUniop* newp=replaceWithDVersion(nodep)) { VL_DANGLING(nodep);
|
||
nodep = newp; // Process new node instead
|
||
iterateCheckReal(nodep,"LHS",nodep->lhsp(),BOTH);
|
||
nodep->dtypeSetDouble();
|
||
return;
|
||
}
|
||
} else {
|
||
// Note there aren't yet uniops that need version changes
|
||
// So no need to call replaceWithUOrSVersion(nodep, nodep->isSigned())
|
||
}
|
||
if (vup->c()->prelim()) {
|
||
nodep->dtypeFrom(nodep->lhsp());
|
||
}
|
||
if (vup->c()->final()) {
|
||
AstNodeDType* expDTypep = vup->c()->dtypeOverridep(nodep->dtypep());
|
||
nodep->dtypep(expDTypep); // Propagate expression type to negation
|
||
AstNodeDType* subDTypep = expDTypep;
|
||
iterateCheck(nodep,"LHS",nodep->lhsp(),CONTEXT,FINAL,subDTypep,EXTEND_EXP);
|
||
}
|
||
}
|
||
|
||
void visit_signed_unsigned(AstNodeUniop* nodep, AstNUser* vup, AstNumeric rs_out) {
|
||
// CALLER: Signed, Unsigned
|
||
// Width: lhs is self determined width
|
||
// See IEEE-2012 6.24.1:
|
||
// Width: Returns packed array, of size $bits(expression).
|
||
// Sign: Output sign is as specified by operation
|
||
// TODO: Type: Two-state if input is two-state, else four-state
|
||
if (nodep->op2p()) nodep->v3fatalSrc("For unary ops only!");
|
||
if (vup->c()->prelim()) {
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(SELF,PRELIM).p());
|
||
checkCvtUS(nodep->lhsp());
|
||
int width = nodep->lhsp()->width();
|
||
AstNodeDType* expDTypep = nodep->findLogicDType(width,width,rs_out);
|
||
nodep->dtypep(expDTypep);
|
||
AstNodeDType* subDTypep = expDTypep;
|
||
// The child's width is self determined
|
||
iterateCheck(nodep,"LHS",nodep->lhsp(),SELF,FINAL,subDTypep,EXTEND_EXP);
|
||
}
|
||
}
|
||
|
||
void visit_shift(AstNodeBiop* nodep, AstNUser* vup) {
|
||
// CALLER: ShiftL, ShiftR, ShiftRS
|
||
// Widths: Output width from lhs, rhs<33 bits
|
||
// Signed: Output signed iff LHS signed; unary operator
|
||
// See IEEE 2012 11.4.10:
|
||
// RHS is self-determined. RHS is always treated as unsigned, has no effect on result.
|
||
iterate_shift_prelim(nodep,vup);
|
||
nodep->dtypeChgSigned(nodep->lhsp()->isSigned());
|
||
AstNodeBiop* newp = iterate_shift_final(nodep,vup); VL_DANGLING(nodep);
|
||
if (newp) {} // Ununused
|
||
}
|
||
void iterate_shift_prelim(AstNodeBiop* nodep, AstNUser* vup) {
|
||
// Shifts
|
||
// See IEEE-2012 11.4.10 and Table 11-21.
|
||
// RHS is self-determined. RHS is always treated as unsigned, has no effect on result.
|
||
if (vup->c()->prelim()) {
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(SELF,PRELIM).p());
|
||
checkCvtUS(nodep->lhsp());
|
||
iterateCheckSizedSelf(nodep,"RHS",nodep->rhsp(),SELF,BOTH);
|
||
nodep->dtypeFrom(nodep->lhsp());
|
||
}
|
||
}
|
||
AstNodeBiop* iterate_shift_final(AstNodeBiop* nodep, AstNUser* vup) {
|
||
// Nodep maybe edited
|
||
if (vup->c()->final()) {
|
||
AstNodeDType* expDTypep = vup->c()->dtypeOverridep(nodep->dtypep());
|
||
AstNodeDType* subDTypep = expDTypep;
|
||
nodep->dtypeFrom(expDTypep);
|
||
// ShiftRS converts to ShiftR, but not vice-versa
|
||
if (nodep->castShiftRS()) {
|
||
if (AstNodeBiop* newp=replaceWithUOrSVersion(nodep, nodep->isSigned())) { VL_DANGLING(nodep);
|
||
nodep = newp; // Process new node instead
|
||
}
|
||
}
|
||
bool warnOn = true;
|
||
// No warning if "X = 1'b1<<N"; assume user is doing what they want
|
||
if (nodep->lhsp()->isOne() && nodep->backp()->castNodeAssign()) warnOn = false;
|
||
iterateCheck(nodep,"LHS",nodep->lhsp(),CONTEXT,FINAL,subDTypep,EXTEND_EXP,warnOn);
|
||
if (nodep->rhsp()->width()>32) {
|
||
AstConst* shiftp = nodep->rhsp()->castConst();
|
||
if (shiftp && shiftp->num().mostSetBitP1() <= 32) {
|
||
// If (number)<<96'h1, then make it into (number)<<32'h1
|
||
V3Number num (shiftp->fileline(), 32, 0); num.opAssign(shiftp->num());
|
||
AstNode* shiftp = nodep->rhsp();
|
||
nodep->rhsp()->replaceWith(new AstConst(shiftp->fileline(), num));
|
||
shiftp->deleteTree(); VL_DANGLING(shiftp);
|
||
} else {
|
||
nodep->rhsp()->v3error("Unsupported: Shifting of by over 32-bit number isn't supported."
|
||
<<" (This isn't a shift of 32 bits, but a shift of 2^32, or 4 billion!)\n");
|
||
}
|
||
}
|
||
}
|
||
return nodep; // May edit
|
||
}
|
||
|
||
void visit_boolmath_and_or(AstNodeBiop* nodep, AstNUser* vup) {
|
||
// CALLER: And, Or, Xor, ...
|
||
// Lint widths: out width = lhs width = rhs width
|
||
// Signed: if lhs & rhs signed
|
||
// IEEE-2012 Table 11-21:
|
||
// Width: max(LHS, RHS)
|
||
if (!nodep->rhsp()) nodep->v3fatalSrc("For binary ops only!");
|
||
// If errors are off, we need to follow the spec; thus we really need to do the max()
|
||
// because the rhs could be larger, and we need to have proper editing to get the widths
|
||
// to be the same for our operations.
|
||
if (vup->c()->prelim()) { // First stage evaluation
|
||
// Determine expression widths only relying on what's in the subops
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p());
|
||
nodep->rhsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p());
|
||
checkCvtUS(nodep->lhsp());
|
||
checkCvtUS(nodep->rhsp());
|
||
int width = max(nodep->lhsp()->width(), nodep->rhsp()->width());
|
||
int mwidth = max(nodep->lhsp()->widthMin(), nodep->rhsp()->widthMin());
|
||
bool expSigned = (nodep->lhsp()->isSigned() && nodep->rhsp()->isSigned());
|
||
nodep->dtypeChgWidthSigned(width,mwidth,AstNumeric::fromBool(expSigned));
|
||
}
|
||
if (vup->c()->final()) {
|
||
AstNodeDType* expDTypep = vup->c()->dtypeOverridep(nodep->dtypep());
|
||
AstNodeDType* subDTypep = expDTypep;
|
||
nodep->dtypeFrom(expDTypep);
|
||
// Error report and change sizes for suboperands of this node.
|
||
iterateCheck(nodep,"LHS",nodep->lhsp(),CONTEXT,FINAL,subDTypep,EXTEND_EXP);
|
||
iterateCheck(nodep,"RHS",nodep->rhsp(),CONTEXT,FINAL,subDTypep,EXTEND_EXP);
|
||
}
|
||
}
|
||
|
||
void visit_add_sub_replace(AstNodeBiop* nodep, AstNUser* vup, bool real_ok) {
|
||
// CALLER: (real_ok=false) AddS, SubS, ...
|
||
// CALLER: (real_ok=true) Add, Sub, ...
|
||
// Widths: out width = lhs width = rhs width
|
||
// Signed: Replace operator with signed operator, or signed to unsigned
|
||
// Real: Replace operator with real operator
|
||
// IEEE-2012 Table 11-21:
|
||
// Width: max(LHS, RHS)
|
||
// If errors are off, we need to follow the spec; thus we really need to do the max()
|
||
// because the rhs could be larger, and we need to have proper editing to get the widths
|
||
// to be the same for our operations.
|
||
//
|
||
//if (debug()>=9) { UINFO(0,"-rus "<<vup->c()<<endl); nodep->dumpTree(cout,"-rusin-"); }
|
||
if (vup->c()->prelim()) { // First stage evaluation
|
||
// Determine expression widths only relying on what's in the subops
|
||
nodep->lhsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p());
|
||
nodep->rhsp()->iterateAndNext(*this,WidthVP(CONTEXT,PRELIM).p());
|
||
if (!real_ok) {
|
||
checkCvtUS(nodep->lhsp());
|
||
checkCvtUS(nodep->rhsp());
|
||
}
|
||
if (nodep->lhsp()->isDouble() || nodep->rhsp()->isDouble()) {
|
||
spliceCvtD(nodep->lhsp());
|
||
spliceCvtD(nodep->rhsp());
|
||
if (AstNodeBiop* newp=replaceWithDVersion(nodep)) { VL_DANGLING(nodep);
|
||
nodep = newp; // Process new node instead
|
||
}
|
||
nodep->dtypeSetDouble();
|
||
iterateCheckReal(nodep,"LHS",nodep->lhsp(),FINAL);
|
||
iterateCheckReal(nodep,"RHS",nodep->rhsp(),FINAL);
|
||
return;
|
||
} else {
|
||
int width = max(nodep->lhsp()->width(), nodep->rhsp()->width());
|
||
int mwidth = max(nodep->lhsp()->widthMin(), nodep->rhsp()->widthMin());
|
||
bool expSigned = (nodep->lhsp()->isSigned() && nodep->rhsp()->isSigned());
|
||
nodep->dtypeChgWidthSigned(width,mwidth,AstNumeric::fromBool(expSigned));
|
||
}
|
||
}
|
||
if (vup->c()->final()) {
|
||
// Parent's data type was computed using the max(upper, nodep->dtype)
|
||
AstNodeDType* expDTypep = vup->c()->dtypeOverridep(nodep->dtypep());
|
||
AstNodeDType* subDTypep = expDTypep;
|
||
nodep->dtypeFrom(expDTypep);
|
||
if (AstNodeBiop* newp=replaceWithUOrSVersion(nodep, expDTypep->isSigned())) { VL_DANGLING(nodep);
|
||
nodep = newp; // Process new node instead
|
||
}
|
||
// Some warning suppressions
|
||
bool lhsWarn=true; bool rhsWarn = true;
|
||
if (nodep->castAdd() || nodep->castSub()) {
|
||
if (subDTypep->widthMin() == (nodep->lhsp()->widthMin()+1)) lhsWarn=false; // Warn if user wants extra bit from carry
|
||
if (subDTypep->widthMin() == (nodep->rhsp()->widthMin()+1)) rhsWarn=false; // Warn if user wants extra bit from carry
|
||
} else if (nodep->castMul() || nodep->castMulS()) {
|
||
if (subDTypep->widthMin() >= (nodep->lhsp()->widthMin())) lhsWarn=false;
|
||
if (subDTypep->widthMin() >= (nodep->rhsp()->widthMin())) rhsWarn=false;
|
||
}
|
||
// Final call, so make sure children check their sizes
|
||
// Error report and change sizes for suboperands of this node.
|
||
iterateCheck(nodep,"LHS",nodep->lhsp(),CONTEXT,FINAL,subDTypep,EXTEND_EXP,lhsWarn);
|
||
iterateCheck(nodep,"RHS",nodep->rhsp(),CONTEXT,FINAL,subDTypep,EXTEND_EXP,rhsWarn);
|
||
}
|
||
//if (debug()>=9) nodep->dumpTree(cout,"-rusou-");
|
||
}
|
||
void visit_real_add_sub(AstNodeBiop* nodep, AstNUser* vup) {
|
||
// CALLER: AddD, MulD, ...
|
||
if (vup->c()->prelim()) { // First stage evaluation
|
||
// Note similar steps in visit_add_sub_replace promotion to double
|
||
iterateCheckReal(nodep,"LHS",nodep->lhsp(),BOTH);
|
||
iterateCheckReal(nodep,"RHS",nodep->rhsp(),BOTH);
|
||
nodep->dtypeSetDouble();
|
||
}
|
||
}
|
||
void visit_real_neg_ceil(AstNodeUniop* nodep, AstNUser* vup) {
|
||
// CALLER: Negate, Ceil, Log, ...
|
||
if (vup->c()->prelim()) { // First stage evaluation
|
||
// See alsl visit_negate_not conversion
|
||
iterateCheckReal(nodep,"LHS",nodep->lhsp(),BOTH);
|
||
nodep->dtypeSetDouble();
|
||
}
|
||
}
|
||
|
||
//----------------------------------------------------------------------
|
||
// LOWER LEVEL WIDTH METHODS (none iterate)
|
||
|
||
bool widthBad (AstNode* nodep, AstNodeDType* expDTypep) {
|
||
int expWidth = expDTypep->width();
|
||
int expWidthMin = expDTypep->widthMin();
|
||
if (!nodep->dtypep()) nodep->v3fatalSrc("Under node "<<nodep->prettyTypeName()<<" has no dtype?? Missing Visitor func?");
|
||
if (nodep->width()==0) nodep->v3fatalSrc("Under node "<<nodep->prettyTypeName()<<" has no expected width?? Missing Visitor func?");
|
||
if (expWidth==0) nodep->v3fatalSrc("Node "<<nodep->prettyTypeName()<<" has no expected width?? Missing Visitor func?");
|
||
if (expWidthMin==0) expWidthMin = expWidth;
|
||
if (nodep->dtypep()->widthSized() && nodep->width() != expWidthMin) return true;
|
||
if (!nodep->dtypep()->widthSized() && nodep->widthMin() > expWidthMin) return true;
|
||
return false;
|
||
}
|
||
|
||
void fixWidthExtend (AstNode* nodep, AstNodeDType* expDTypep, ExtendRule extendRule) {
|
||
// Fix the width mismatch by extending or truncating bits
|
||
// *ONLY* call this from checkWidth()
|
||
// Truncation is rarer, but can occur: parameter [3:0] FOO = 64'h12312;
|
||
// A(CONSTwide)+B becomes A(CONSTwidened)+B
|
||
// A(somewide)+B becomes A(TRUNC(somewide,width))+B
|
||
// or A(EXTRACT(somewide,width,0))+B
|
||
// Sign extension depends on the type of the *present*
|
||
// node, while the output dtype is the *expected* sign.
|
||
// It is reasonable to have sign extension with unsigned output,
|
||
// for example $unsigned(a)+$signed(b), the SIGNED(B) will be unsigned dtype out
|
||
UINFO(4," widthExtend_(r="<<extendRule<<") old: "<<nodep<<endl);
|
||
if (extendRule == EXTEND_OFF) return;
|
||
AstConst* constp = nodep->castConst();
|
||
int expWidth = expDTypep->width();
|
||
if (constp && !nodep->isSigned()) {
|
||
// Save later constant propagation work, just right-size it.
|
||
V3Number num (nodep->fileline(), expWidth);
|
||
num.opAssign(constp->num());
|
||
num.isSigned(expDTypep->isSigned());
|
||
AstNode* newp = new AstConst(nodep->fileline(), num);
|
||
constp->replaceWith(newp);
|
||
pushDeletep(constp); VL_DANGLING(constp); VL_DANGLING(nodep);
|
||
nodep=newp;
|
||
} else if (expWidth<nodep->width()) {
|
||
// Trunc - Extract
|
||
AstNRelinker linker;
|
||
nodep->unlinkFrBack(&linker);
|
||
AstNode* newp = new AstSel(nodep->fileline(), nodep, 0, expWidth);
|
||
newp->didWidth(true); // Don't replace dtype with unsigned
|
||
linker.relink(newp);
|
||
nodep=newp;
|
||
} else {
|
||
// Extend
|
||
AstNRelinker linker;
|
||
nodep->unlinkFrBack(&linker);
|
||
bool doSigned = false;
|
||
switch (extendRule) {
|
||
case EXTEND_ZERO: doSigned = false; break;
|
||
case EXTEND_EXP: doSigned = nodep->isSigned() && expDTypep->isSigned(); break;
|
||
case EXTEND_LHS: doSigned = nodep->isSigned(); break;
|
||
default: nodep->v3fatalSrc("bad case");
|
||
}
|
||
AstNode* newp = (doSigned
|
||
? (new AstExtendS(nodep->fileline(), nodep))->castNode()
|
||
: (new AstExtend (nodep->fileline(), nodep))->castNode());
|
||
linker.relink(newp);
|
||
nodep=newp;
|
||
}
|
||
if (expDTypep->isDouble() && !nodep->isDouble()) {
|
||
// For AstVar init() among others
|
||
// TODO do all to-real and to-integer conversions in this function rather than in callers
|
||
AstNode* newp = spliceCvtD(nodep);
|
||
nodep = newp;
|
||
}
|
||
nodep->dtypeFrom(expDTypep);
|
||
UINFO(4," _new: "<<nodep<<endl);
|
||
}
|
||
|
||
void fixWidthReduce (AstNode* nodep) {
|
||
// Fix the width mismatch by adding a reduction OR operator
|
||
// IF (A(CONSTwide)) becomes IF (A(CONSTreduced))
|
||
// IF (A(somewide)) becomes IF (A(REDOR(somewide)))
|
||
// Attempt to fix it quietly
|
||
int expWidth = 1;
|
||
int expSigned = false;
|
||
UINFO(4," widthReduce_old: "<<nodep<<endl);
|
||
AstConst* constp = nodep->castConst();
|
||
if (constp) {
|
||
V3Number num (nodep->fileline(), expWidth);
|
||
num.opRedOr(constp->num());
|
||
num.isSigned(expSigned);
|
||
AstNode* newp = new AstConst(nodep->fileline(), num);
|
||
constp->replaceWith(newp);
|
||
constp->deleteTree(); VL_DANGLING(constp); VL_DANGLING(nodep);
|
||
nodep=newp;
|
||
} else {
|
||
AstNRelinker linker;
|
||
nodep->unlinkFrBack(&linker);
|
||
AstNode* newp = new AstRedOr(nodep->fileline(), nodep);
|
||
linker.relink(newp);
|
||
nodep=newp;
|
||
}
|
||
nodep->dtypeChgWidthSigned(expWidth,expWidth,AstNumeric::fromBool(expSigned));
|
||
UINFO(4," _new: "<<nodep<<endl);
|
||
}
|
||
|
||
bool fixAutoExtend (AstNode*& nodepr, int expWidth) {
|
||
// For SystemVerilog '0,'1,'x,'z, autoextend and don't warn
|
||
if (AstConst* constp = nodepr->castConst()) {
|
||
if (constp->num().autoExtend() && !constp->num().sized() && constp->width()==1) {
|
||
// Make it the proper size. Careful of proper extension of 0's/1's
|
||
V3Number num (constp->fileline(), expWidth);
|
||
num.opRepl(constp->num(), expWidth); // {width{'1}}
|
||
AstNode* newp = new AstConst(constp->fileline(), num);
|
||
// Spec says always unsigned with proper width
|
||
if (debug()>4) constp->dumpTree(cout," fixAutoExtend_old: ");
|
||
if (debug()>4) newp->dumpTree(cout," _new: ");
|
||
constp->replaceWith(newp);
|
||
constp->deleteTree(); VL_DANGLING(constp);
|
||
// Tell caller the new constp, and that we changed it.
|
||
nodepr = newp;
|
||
return true;
|
||
}
|
||
}
|
||
return false; // No change
|
||
}
|
||
|
||
void iterateCheckFileDesc (AstNode* nodep, AstNode* underp, Stage stage) {
|
||
if (stage != BOTH) nodep->v3fatalSrc("Bad call");
|
||
// underp may change as a result of replacement
|
||
underp = underp->acceptSubtreeReturnEdits(*this,WidthVP(SELF,PRELIM).p());
|
||
AstNodeDType* expDTypep = underp->findUInt32DType();
|
||
underp = iterateCheck(nodep,"file_descriptor",underp,SELF,FINAL,expDTypep,EXTEND_EXP);
|
||
if (underp) {} // cppcheck
|
||
}
|
||
void iterateCheckReal (AstNode* nodep, const char* side, AstNode* underp, Stage stage) {
|
||
// Coerce child to real if not already. Child is self-determined
|
||
// e.g. nodep=ADDD, underp=ADD in ADDD(ADD(a,b), real-CONST)
|
||
// Don't need separate PRELIM and FINAL(double) calls;
|
||
// as if resolves to double, the BOTH correctly resolved double,
|
||
// otherwise self-determined was correct
|
||
// underp may change as a result of replacement
|
||
if (stage & PRELIM) {
|
||
underp = underp->acceptSubtreeReturnEdits(*this,WidthVP(SELF,PRELIM).p());
|
||
}
|
||
if (stage & FINAL) {
|
||
AstNodeDType* expDTypep = nodep->findDoubleDType();
|
||
underp = iterateCheck(nodep,side,underp,SELF,FINAL,expDTypep,EXTEND_EXP);
|
||
}
|
||
if (underp) {} // cppcheck
|
||
}
|
||
void iterateCheckString (AstNode* nodep, const char* side, AstNode* underp, Stage stage) {
|
||
if (stage & PRELIM) {
|
||
underp = underp->acceptSubtreeReturnEdits(*this,WidthVP(SELF,PRELIM).p());
|
||
}
|
||
if (stage & FINAL) {
|
||
AstNodeDType* expDTypep = nodep->findStringDType();
|
||
underp = iterateCheck(nodep,side,underp,SELF,FINAL,expDTypep,EXTEND_EXP);
|
||
}
|
||
if (underp) {} // cppcheck
|
||
}
|
||
void iterateCheckSizedSelf (AstNode* nodep, const char* side, AstNode* underp,
|
||
Determ determ, Stage stage) {
|
||
// Coerce child to any sized-number data type; child is self-determined i.e. isolated from expected type.
|
||
// e.g. nodep=CONCAT, underp=lhs in CONCAT(lhs,rhs)
|
||
if (determ != SELF) nodep->v3fatalSrc("Bad call");
|
||
if (stage != FINAL && stage != BOTH) nodep->v3fatalSrc("Bad call");
|
||
// underp may change as a result of replacement
|
||
if (stage & PRELIM) underp = underp->acceptSubtreeReturnEdits(*this,WidthVP(SELF,PRELIM).p());
|
||
underp = checkCvtUS(underp);
|
||
AstNodeDType* expDTypep = underp->dtypep();
|
||
underp = iterateCheck(nodep,side,underp,SELF,FINAL,expDTypep,EXTEND_EXP);
|
||
if (underp) {} // cppcheck
|
||
}
|
||
void iterateCheckAssign(AstNode* nodep, const char* side,
|
||
AstNode* rhsp, Stage stage, AstNodeDType* lhsDTypep) {
|
||
// Check using assignment-like context rules
|
||
//if (debug()) nodep->dumpTree(cout,"-checkass: ");
|
||
if (stage != FINAL) nodep->v3fatalSrc("Bad width call");
|
||
// We iterate and size the RHS based on the result of RHS evaluation
|
||
bool lhsStream = (nodep->castNodeAssign()
|
||
&& nodep->castNodeAssign()->lhsp()->castNodeStream());
|
||
rhsp = iterateCheck(nodep,side,rhsp,ASSIGN,FINAL,lhsDTypep,lhsStream?EXTEND_OFF:EXTEND_LHS);
|
||
//if (debug()) nodep->dumpTree(cout,"-checkout: ");
|
||
if (rhsp) {} // cppcheck
|
||
}
|
||
|
||
void iterateCheckBool (AstNode* nodep, const char* side, AstNode* underp, Stage stage) {
|
||
if (stage != BOTH) nodep->v3fatalSrc("Bad call"); // Booleans always self-determined so do BOTH at once
|
||
// Underp is used in a self-determined but boolean context, reduce a multibit number to one bit
|
||
// stage is always BOTH so not passed as argument
|
||
// underp may change as a result of replacement
|
||
if (!underp) underp->v3fatalSrc("Node has no type");
|
||
underp = underp->acceptSubtreeReturnEdits(*this,WidthVP(SELF,BOTH).p());
|
||
if (!underp || !underp->dtypep()) underp->v3fatalSrc("Node has no type"); // Perhaps forgot to do a prelim visit on it?
|
||
//
|
||
// For DOUBLE under a logical op, add implied test against zero, never a warning
|
||
if (underp && underp->isDouble()) {
|
||
UINFO(6," spliceCvtCmpD0: "<<underp<<endl);
|
||
AstNRelinker linker;
|
||
underp->unlinkFrBack(&linker);
|
||
AstNode* newp = new AstNeqD(nodep->fileline(), underp,
|
||
new AstConst(nodep->fileline(), AstConst::RealDouble(), 0.0));
|
||
linker.relink(newp);
|
||
} else if (!underp->dtypep()->basicp()) {
|
||
nodep->v3error("Logical Operator "<<nodep->prettyTypeName()
|
||
<<" expects a non-complex data type on the "<<side<<".");
|
||
underp->replaceWith(new AstConst(nodep->fileline(), AstConst::LogicFalse()));
|
||
pushDeletep(underp); VL_DANGLING(underp);
|
||
} else {
|
||
bool bad = widthBad(underp,nodep->findLogicBoolDType());
|
||
if (bad) {
|
||
bool warnOn = true; // Not used
|
||
if (warnOn) {
|
||
if (debug()>4) nodep->backp()->dumpTree(cout," back: ");
|
||
nodep->v3warn(WIDTH,"Logical Operator "<<nodep->prettyTypeName()
|
||
<<" expects 1 bit on the "<<side<<", but "<<side<<"'s "
|
||
<<underp->prettyTypeName()<<" generates "<<underp->width()
|
||
<<(underp->width()!=underp->widthMin()
|
||
?" or "+cvtToStr(underp->widthMin()):"")
|
||
<<" bits.");
|
||
}
|
||
fixWidthReduce(underp); VL_DANGLING(underp);//Changed
|
||
}
|
||
}
|
||
}
|
||
|
||
AstNode* iterateCheck (AstNode* nodep, const char* side, AstNode* underp,
|
||
Determ determ, Stage stage, AstNodeDType* expDTypep,
|
||
ExtendRule extendRule,
|
||
bool warnOn=true) {
|
||
// Perform data type check on underp, which is underneath nodep used for error reporting
|
||
// Returns the new underp
|
||
// Conversion to/from doubles and integers are before iterating.
|
||
if (stage != FINAL) nodep->v3fatalSrc("Bad state to iterateCheck");
|
||
if (!underp || !underp->dtypep()) underp->v3fatalSrc("Node has no type"); // Perhaps forgot to do a prelim visit on it?
|
||
if (expDTypep == underp->dtypep()) { // Perfect
|
||
underp = underp->acceptSubtreeReturnEdits(*this,WidthVP(SELF,FINAL).p());
|
||
} else if (expDTypep->isDouble() && underp->isDouble()) { // Also good
|
||
underp = underp->acceptSubtreeReturnEdits(*this,WidthVP(SELF,FINAL).p());
|
||
} else if (expDTypep->isDouble() && !underp->isDouble()) {
|
||
underp = spliceCvtD(underp);
|
||
underp = underp->acceptSubtreeReturnEdits(*this,WidthVP(SELF,FINAL).p());
|
||
} else if (!expDTypep->isDouble() && underp->isDouble()) {
|
||
underp = spliceCvtS(underp, true); // Round RHS
|
||
underp = underp->acceptSubtreeReturnEdits(*this,WidthVP(SELF,FINAL).p());
|
||
} else if (expDTypep->isString() && !underp->dtypep()->isString()) {
|
||
underp = spliceCvtString(underp);
|
||
underp = underp->acceptSubtreeReturnEdits(*this,WidthVP(SELF,FINAL).p());
|
||
} else {
|
||
AstBasicDType* expBasicp = expDTypep->basicp();
|
||
AstBasicDType* underBasicp = underp->dtypep()->basicp();
|
||
if (expBasicp && underBasicp) {
|
||
AstNodeDType* subDTypep = expDTypep;
|
||
// We then iterate FINAL before width fixes, as if the under-operation
|
||
// is e.g. an ADD, the ADD will auto-adjust to the proper data type
|
||
// or if another operation e.g. ATOI will not.
|
||
if (determ == SELF) {
|
||
underp = underp->acceptSubtreeReturnEdits(*this,WidthVP(SELF,FINAL).p());
|
||
} else if (determ == ASSIGN) {
|
||
// IEEE: Signedness is solely determined by the RHS (underp), not by the LHS (expDTypep)
|
||
if (underp->isSigned() != subDTypep->isSigned()
|
||
|| underp->width() != subDTypep->width()) {
|
||
subDTypep = nodep->findLogicDType(max(subDTypep->width(), underp->width()),
|
||
max(subDTypep->widthMin(), underp->widthMin()),
|
||
AstNumeric::fromBool(underp->isSigned()));
|
||
UINFO(9,"Assignment of opposite-signed RHS to LHS: "<<nodep<<endl);
|
||
}
|
||
underp = underp->acceptSubtreeReturnEdits(*this,WidthVP(subDTypep,FINAL).p());
|
||
} else {
|
||
underp = underp->acceptSubtreeReturnEdits(*this,WidthVP(subDTypep,FINAL).p());
|
||
}
|
||
// Note the check uses the expected size, not the child's subDTypep as we want the
|
||
// child node's width to end up correct for the assignment (etc)
|
||
widthCheckSized(nodep,side,underp,expDTypep,extendRule,warnOn);
|
||
}
|
||
else {
|
||
// Hope it just works out
|
||
}
|
||
}
|
||
return underp;
|
||
}
|
||
|
||
void widthCheckSized (AstNode* nodep, const char* side,
|
||
AstNode* underp, // Node to be checked or have typecast added in front of
|
||
AstNodeDType* expDTypep,
|
||
ExtendRule extendRule,
|
||
bool warnOn=true) {
|
||
// Issue warnings on sized number width mismatches, then do appropriate size extension
|
||
// Generally iterateCheck is what is wanted instead of this
|
||
//UINFO(9,"wchk "<<side<<endl<<" "<<nodep<<endl<<" "<<underp<<endl<<" e="<<expDTypep<<" i"<<warnOn<<endl);
|
||
AstBasicDType* expBasicp = expDTypep->basicp();
|
||
AstBasicDType* underBasicp = underp->dtypep()->basicp();
|
||
if (expDTypep == underp->dtypep()) {
|
||
return; // Same type must match
|
||
} else if (!expBasicp || expBasicp->isDouble()
|
||
|| !underBasicp || underBasicp->isDouble()) {
|
||
// This is perhaps a v3fatalSrc as we should have checked the types before calling widthCheck,
|
||
// but we may have missed a non-sized check in earlier code, so might as well assume it is the users' fault.
|
||
nodep->v3error(ucfirst(nodep->prettyOperatorName())<<" expected non-complex non-double "<<side<<" in width check");
|
||
#if VL_DEBUG
|
||
nodep->v3fatalSrc("widthCheckSized should not be called on doubles/complex types");
|
||
#endif
|
||
return;
|
||
} else {
|
||
int expWidth = expDTypep->width();
|
||
int expWidthMin = expDTypep->widthMin();
|
||
if (expWidthMin==0) expWidthMin = expWidth;
|
||
bool bad = widthBad(underp,expDTypep);
|
||
if ((bad || underp->width() != expWidth)
|
||
&& fixAutoExtend(underp/*ref*/,expWidth)) {
|
||
underp=NULL; // Changes underp
|
||
return;
|
||
}
|
||
if (underp->castConst() && underp->castConst()->num().isFromString()
|
||
&& expWidth > underp->width()
|
||
&& (((expWidth - underp->width()) % 8) == 0)) { // At least it's character sized
|
||
// reg [31:0] == "foo" we'll consider probably fine.
|
||
// Maybe this should be a special warning? Not for now.
|
||
warnOn = false;
|
||
}
|
||
if ((nodep->castAdd() && underp->width()==1 && underp->isOne())
|
||
|| (nodep->castSub() && underp->width()==1 && underp->isOne() && 0==strcmp(side,"RHS"))) {
|
||
// "foo + 1'b1", or "foo - 1'b1" are very common, people assume they extend correctly
|
||
warnOn = false;
|
||
}
|
||
if (bad && warnOn) {
|
||
if (debug()>4) nodep->backp()->dumpTree(cout," back: ");
|
||
nodep->v3warn(WIDTH,ucfirst(nodep->prettyOperatorName())
|
||
<<" expects "<<expWidth
|
||
<<(expWidth!=expWidthMin?" or "+cvtToStr(expWidthMin):"")
|
||
<<" bits on the "<<side<<", but "<<side<<"'s "
|
||
<<underp->prettyTypeName()<<" generates "<<underp->width()
|
||
<<(underp->width()!=underp->widthMin()
|
||
?" or "+cvtToStr(underp->widthMin()):"")
|
||
<<" bits.");
|
||
}
|
||
if (bad || underp->width()!=expWidth) {
|
||
// If we're in an NodeAssign, don't truncate the RHS if the LHS is
|
||
// a NodeStream. The streaming operator changes the rules regarding
|
||
// which bits to truncate.
|
||
AstNodeAssign* assignp = nodep->castNodeAssign();
|
||
AstPin* pinp = nodep->castPin();
|
||
if (assignp && assignp->lhsp()->castNodeStream()) {
|
||
} else if (pinp && !pinp->modVarp()->isInput()) { // V3Inst::pinReconnectSimple must deal
|
||
UINFO(5,"pinInSizeMismatch: "<<pinp);
|
||
} else {
|
||
fixWidthExtend(underp, expDTypep, extendRule); VL_DANGLING(underp);//Changed
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
//----------------------------------------------------------------------
|
||
// SIGNED/DOUBLE METHODS
|
||
|
||
AstNode* checkCvtUS(AstNode* nodep) {
|
||
if (nodep && nodep->isDouble()) {
|
||
nodep->v3error("Expected integral (non-real) input to "<<nodep->backp()->prettyTypeName());
|
||
nodep = spliceCvtS(nodep, true);
|
||
}
|
||
return nodep;
|
||
}
|
||
|
||
AstNode* spliceCvtD(AstNode* nodep) {
|
||
// For integer used in REAL context, convert to real
|
||
// We don't warn here, "2.0 * 2" is common and reasonable
|
||
if (nodep && !nodep->isDouble()) {
|
||
UINFO(6," spliceCvtD: "<<nodep<<endl);
|
||
AstNRelinker linker;
|
||
nodep->unlinkFrBack(&linker);
|
||
AstNode* newp = new AstIToRD(nodep->fileline(), nodep);
|
||
linker.relink(newp);
|
||
return newp;
|
||
} else {
|
||
return nodep;
|
||
}
|
||
}
|
||
AstNode* spliceCvtS(AstNode* nodep, bool warnOn) {
|
||
// IEEE-2012 11.8.1: Signed: Type coercion creates signed
|
||
// 11.8.2: Argument to convert is self-determined
|
||
if (nodep && nodep->isDouble()) {
|
||
UINFO(6," spliceCvtS: "<<nodep<<endl);
|
||
AstNRelinker linker;
|
||
nodep->unlinkFrBack(&linker);
|
||
if (warnOn) nodep->v3warn(REALCVT,"Implicit conversion of real to integer");
|
||
AstNode* newp = new AstRToIRoundS(nodep->fileline(), nodep);
|
||
linker.relink(newp);
|
||
return newp;
|
||
} else {
|
||
return nodep;
|
||
}
|
||
}
|
||
AstNode* spliceCvtString(AstNode* nodep) {
|
||
// IEEE-2012 11.8.1: Signed: Type coercion creates signed
|
||
// 11.8.2: Argument to convert is self-determined
|
||
if (nodep && !nodep->dtypep()->basicp()->isString()) {
|
||
UINFO(6," spliceCvtString: "<<nodep<<endl);
|
||
AstNRelinker linker;
|
||
nodep->unlinkFrBack(&linker);
|
||
AstNode* newp = new AstCvtPackString(nodep->fileline(), nodep);
|
||
linker.relink(newp);
|
||
return newp;
|
||
} else {
|
||
return nodep;
|
||
}
|
||
}
|
||
AstNodeBiop* replaceWithUOrSVersion(AstNodeBiop* nodep, bool signedFlavorNeeded) {
|
||
// Given a signed/unsigned node type, create the opposite type
|
||
// Return new node or NULL if nothing
|
||
if (signedFlavorNeeded == nodep->signedFlavor()) {
|
||
return NULL;
|
||
}
|
||
if (!nodep->dtypep()) nodep->dtypeFrom(nodep->lhsp());
|
||
// To simplify callers, some node types don't need to change
|
||
switch (nodep->type()) {
|
||
case AstType::atEQ: nodep->dtypeChgSigned(signedFlavorNeeded); return NULL;
|
||
case AstType::atNEQ: nodep->dtypeChgSigned(signedFlavorNeeded); return NULL;
|
||
case AstType::atEQCASE: nodep->dtypeChgSigned(signedFlavorNeeded); return NULL;
|
||
case AstType::atNEQCASE: nodep->dtypeChgSigned(signedFlavorNeeded); return NULL;
|
||
case AstType::atADD: nodep->dtypeChgSigned(signedFlavorNeeded); return NULL;
|
||
case AstType::atSUB: nodep->dtypeChgSigned(signedFlavorNeeded); return NULL;
|
||
case AstType::atSHIFTL: nodep->dtypeChgSigned(signedFlavorNeeded); return NULL;
|
||
default: break;
|
||
}
|
||
FileLine* fl = nodep->fileline();
|
||
AstNode* lhsp = nodep->lhsp()->unlinkFrBack();
|
||
AstNode* rhsp = nodep->rhsp()->unlinkFrBack();
|
||
AstNodeBiop* newp = NULL;
|
||
switch (nodep->type()) {
|
||
case AstType::atGT: newp = new AstGtS (fl,lhsp,rhsp); break;
|
||
case AstType::atGTS: newp = new AstGt (fl,lhsp,rhsp); break;
|
||
case AstType::atGTE: newp = new AstGteS (fl,lhsp,rhsp); break;
|
||
case AstType::atGTES: newp = new AstGte (fl,lhsp,rhsp); break;
|
||
case AstType::atLT: newp = new AstLtS (fl,lhsp,rhsp); break;
|
||
case AstType::atLTS: newp = new AstLt (fl,lhsp,rhsp); break;
|
||
case AstType::atLTE: newp = new AstLteS (fl,lhsp,rhsp); break;
|
||
case AstType::atLTES: newp = new AstLte (fl,lhsp,rhsp); break;
|
||
case AstType::atDIV: newp = new AstDivS (fl,lhsp,rhsp); break;
|
||
case AstType::atDIVS: newp = new AstDiv (fl,lhsp,rhsp); break;
|
||
case AstType::atMODDIV: newp = new AstModDivS (fl,lhsp,rhsp); break;
|
||
case AstType::atMODDIVS: newp = new AstModDiv (fl,lhsp,rhsp); break;
|
||
case AstType::atMUL: newp = new AstMulS (fl,lhsp,rhsp); break;
|
||
case AstType::atMULS: newp = new AstMul (fl,lhsp,rhsp); break;
|
||
case AstType::atSHIFTR: newp = new AstShiftRS (fl,lhsp,rhsp); break;
|
||
case AstType::atSHIFTRS: newp = new AstShiftR (fl,lhsp,rhsp); break;
|
||
default:
|
||
nodep->v3fatalSrc("Node needs sign change, but bad case: "<<nodep<<endl);
|
||
break;
|
||
}
|
||
UINFO(6," ReplaceWithUOrSVersion: "<<nodep<<" w/ "<<newp<<endl);
|
||
nodep->replaceWith(newp);
|
||
newp->dtypeFrom(nodep);
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
return newp;
|
||
}
|
||
AstNodeBiop* replaceWithDVersion(AstNodeBiop* nodep) {
|
||
// Given a signed/unsigned node type, create the opposite type
|
||
// Return new node or NULL if nothing
|
||
if (nodep->doubleFlavor()) {
|
||
return NULL;
|
||
}
|
||
FileLine* fl = nodep->fileline();
|
||
AstNode* lhsp = nodep->lhsp()->unlinkFrBack();
|
||
AstNode* rhsp = nodep->rhsp()->unlinkFrBack();
|
||
AstNodeBiop* newp = NULL;
|
||
// No width change on output;... // All below have bool or double outputs
|
||
switch (nodep->type()) {
|
||
case AstType::atADD: newp = new AstAddD (fl,lhsp,rhsp); break;
|
||
case AstType::atSUB: newp = new AstSubD (fl,lhsp,rhsp); break;
|
||
case AstType::atPOW: newp = new AstPowD (fl,lhsp,rhsp); break;
|
||
case AstType::atEQ: case AstType::atEQCASE: newp = new AstEqD (fl,lhsp,rhsp); break;
|
||
case AstType::atNEQ: case AstType::atNEQCASE: newp = new AstNeqD (fl,lhsp,rhsp); break;
|
||
case AstType::atGT: case AstType::atGTS: newp = new AstGtD (fl,lhsp,rhsp); break;
|
||
case AstType::atGTE: case AstType::atGTES: newp = new AstGteD (fl,lhsp,rhsp); break;
|
||
case AstType::atLT: case AstType::atLTS: newp = new AstLtD (fl,lhsp,rhsp); break;
|
||
case AstType::atLTE: case AstType::atLTES: newp = new AstLteD (fl,lhsp,rhsp); break;
|
||
case AstType::atDIV: case AstType::atDIVS: newp = new AstDivD (fl,lhsp,rhsp); break;
|
||
case AstType::atMUL: case AstType::atMULS: newp = new AstMulD (fl,lhsp,rhsp); break;
|
||
default:
|
||
nodep->v3fatalSrc("Node needs conversion to double, but bad case: "<<nodep<<endl);
|
||
break;
|
||
}
|
||
UINFO(6," ReplaceWithDVersion: "<<nodep<<" w/ "<<newp<<endl);
|
||
nodep->replaceWith(newp);
|
||
// No width change; the default created type (bool or double) is correct
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
return newp;
|
||
}
|
||
AstNodeBiop* replaceWithNVersion(AstNodeBiop* nodep) {
|
||
// Given a signed/unsigned node type, replace with string version
|
||
// Return new node or NULL if nothing
|
||
if (nodep->stringFlavor()) {
|
||
return NULL;
|
||
}
|
||
FileLine* fl = nodep->fileline();
|
||
AstNode* lhsp = nodep->lhsp()->unlinkFrBack();
|
||
AstNode* rhsp = nodep->rhsp()->unlinkFrBack();
|
||
AstNodeBiop* newp = NULL;
|
||
// No width change on output;... // All below have bool or double outputs
|
||
switch (nodep->type()) {
|
||
case AstType::atEQ: case AstType::atEQCASE: newp = new AstEqN (fl,lhsp,rhsp); break;
|
||
case AstType::atNEQ: case AstType::atNEQCASE: newp = new AstNeqN (fl,lhsp,rhsp); break;
|
||
case AstType::atGT: case AstType::atGTS: newp = new AstGtN (fl,lhsp,rhsp); break;
|
||
case AstType::atGTE: case AstType::atGTES: newp = new AstGteN (fl,lhsp,rhsp); break;
|
||
case AstType::atLT: case AstType::atLTS: newp = new AstLtN (fl,lhsp,rhsp); break;
|
||
case AstType::atLTE: case AstType::atLTES: newp = new AstLteN (fl,lhsp,rhsp); break;
|
||
default:
|
||
nodep->v3fatalSrc("Node needs conversion to string, but bad case: "<<nodep<<endl);
|
||
break;
|
||
}
|
||
UINFO(6," ReplaceWithNVersion: "<<nodep<<" w/ "<<newp<<endl);
|
||
nodep->replaceWith(newp);
|
||
// No width change; the default created type (bool or string) is correct
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
return newp;
|
||
}
|
||
AstNodeUniop* replaceWithDVersion(AstNodeUniop* nodep) {
|
||
// Given a signed/unsigned node type, create the opposite type
|
||
// Return new node or NULL if nothing
|
||
if (nodep->doubleFlavor()) {
|
||
return NULL;
|
||
}
|
||
FileLine* fl = nodep->fileline();
|
||
AstNode* lhsp = nodep->lhsp()->unlinkFrBack();
|
||
AstNodeUniop* newp = NULL;
|
||
switch (nodep->type()) {
|
||
case AstType::atNEGATE: newp = new AstNegateD (fl,lhsp); break;
|
||
default:
|
||
nodep->v3fatalSrc("Node needs conversion to double, but bad case: "<<nodep<<endl);
|
||
break;
|
||
}
|
||
UINFO(6," ReplaceWithDVersion: "<<nodep<<" w/ "<<newp<<endl);
|
||
nodep->replaceWith(newp);
|
||
newp->dtypeFrom(nodep);
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
return newp;
|
||
}
|
||
|
||
//----------------------------------------------------------------------
|
||
// METHODS - data types
|
||
|
||
AstNodeDType* moveChildDTypeEdit(AstNode* nodep) {
|
||
// DTypes at parse time get added as a childDType to some node types such as AstVars.
|
||
// We move them to global scope, so removing/changing a variable won't lose the dtype.
|
||
AstNodeDType* dtnodep = nodep->getChildDTypep();
|
||
if (!dtnodep) nodep->v3fatalSrc("Caller should check for NULL before calling moveChild");
|
||
UINFO(9,"moveChildDTypeEdit "<<dtnodep<<endl);
|
||
dtnodep->unlinkFrBack(); // Make non-child
|
||
v3Global.rootp()->typeTablep()->addTypesp(dtnodep);
|
||
return dtnodep;
|
||
}
|
||
AstNodeDType* iterateEditDTypep(AstNode* parentp, AstNodeDType* nodep) {
|
||
// Iterate into a data type to resolve that type. This process
|
||
// may eventually create a new data type, but not today
|
||
// it may make a new datatype, need subChildDType() to point to it;
|
||
// maybe we have user5p indicate a "replace me with" pointer.
|
||
// Need to be careful with "implicit" types though.
|
||
//
|
||
// Alternative is to have WidthVP return information.
|
||
// or have a call outside of normal visitor land.
|
||
// or have a m_return type (but need to return if width called multiple times)
|
||
if (!nodep) parentp->v3fatalSrc("Null dtype when widthing dtype");
|
||
nodep->iterate(*this);
|
||
return nodep;
|
||
}
|
||
|
||
AstConst* dimensionValue(AstNodeDType* nodep, AstAttrType attrType, int dim) {
|
||
// Return the dimension value for the specified attribute and constant dimension
|
||
AstNodeDType* dtypep = nodep->skipRefp();
|
||
VNumRange declRange; // ranged() set false
|
||
for (int i = 1; i <= dim; ++i) {
|
||
//UINFO(9, " dim at "<<dim<<" "<<dtypep<<endl);
|
||
declRange = VNumRange(); // ranged() set false
|
||
if (AstNodeArrayDType* adtypep = dtypep->castNodeArrayDType()) {
|
||
declRange = adtypep->declRange();
|
||
if (i<dim) dtypep = adtypep->subDTypep()->skipRefp();
|
||
continue;
|
||
} else if (AstNodeClassDType* adtypep = dtypep->castNodeClassDType()) {
|
||
declRange = adtypep->declRange();
|
||
if (adtypep) {} // UNUSED
|
||
break; // Sub elements don't look like arrays and can't iterate into
|
||
} else if (AstBasicDType* adtypep = dtypep->castBasicDType()) {
|
||
if (adtypep->isRanged()) declRange = adtypep->declRange();
|
||
break;
|
||
}
|
||
break;
|
||
}
|
||
AstConst* valp = NULL; // If NULL, construct from val
|
||
int val = 0;
|
||
switch (attrType) {
|
||
case AstAttrType::DIM_BITS: {
|
||
int bits = 1;
|
||
while (dtypep) {
|
||
//UINFO(9, " bits at "<<bits<<" "<<dtypep<<endl);
|
||
if (AstNodeArrayDType* adtypep = dtypep->castNodeArrayDType()) {
|
||
bits *= adtypep->declRange().elements();
|
||
dtypep = adtypep->subDTypep()->skipRefp();
|
||
continue;
|
||
} else if (AstNodeClassDType* adtypep = dtypep->castNodeClassDType()) {
|
||
bits *= adtypep->width();
|
||
break;
|
||
} else if (AstBasicDType* adtypep = dtypep->castBasicDType()) {
|
||
bits *= adtypep->width();
|
||
break;
|
||
}
|
||
break;
|
||
}
|
||
if (dim == 0) {
|
||
val = 0;
|
||
} else if (dim == 1 && !declRange.ranged() && bits==1) { // $bits should be sane for non-arrays
|
||
val = nodep->width();
|
||
} else {
|
||
val = bits;
|
||
}
|
||
break; }
|
||
case AstAttrType::DIM_HIGH:
|
||
val = !declRange.ranged() ? 0 : declRange.hi();
|
||
break;
|
||
case AstAttrType::DIM_LEFT:
|
||
val = !declRange.ranged() ? 0 : declRange.left();
|
||
break;
|
||
case AstAttrType::DIM_LOW:
|
||
val = !declRange.ranged() ? 0 : declRange.lo();
|
||
break;
|
||
case AstAttrType::DIM_RIGHT:
|
||
val = !declRange.ranged() ? 0 : declRange.right();
|
||
break;
|
||
case AstAttrType::DIM_INCREMENT:
|
||
val = (declRange.ranged() && declRange.littleEndian()) ? -1 : 1;
|
||
break;
|
||
case AstAttrType::DIM_SIZE:
|
||
val = !declRange.ranged() ? 0 : declRange.elements();
|
||
break;
|
||
default:
|
||
nodep->v3fatalSrc("Missing DIM ATTR type case");
|
||
break;
|
||
}
|
||
if (!valp) valp = new AstConst(nodep->fileline(), AstConst::Signed32(), val);
|
||
UINFO(9," $dimension "<<attrType.ascii()<<"("<<((void*)dtypep)<<","<<dim<<")="<<valp<<endl);
|
||
return valp;
|
||
}
|
||
AstVar* dimensionVarp(AstNodeDType* nodep, AstAttrType attrType, uint32_t msbdim) {
|
||
// Return a variable table which has specified dimension properties for this variable
|
||
TableMap::iterator pos = m_tableMap.find(make_pair(nodep,attrType));
|
||
if (pos != m_tableMap.end()) {
|
||
return pos->second;
|
||
}
|
||
AstNodeArrayDType* vardtypep = new AstUnpackArrayDType(nodep->fileline(),
|
||
nodep->findSigned32DType(),
|
||
new AstRange(nodep->fileline(), msbdim, 0));
|
||
AstInitArray* initp = new AstInitArray (nodep->fileline(), vardtypep, NULL);
|
||
v3Global.rootp()->typeTablep()->addTypesp(vardtypep);
|
||
AstVar* varp = new AstVar (nodep->fileline(), AstVarType::MODULETEMP,
|
||
"__Vdimtab_" + VString::downcase(attrType.ascii()) + cvtToStr(m_dtTables++),
|
||
vardtypep);
|
||
varp->isConst(true);
|
||
varp->isStatic(true);
|
||
varp->valuep(initp);
|
||
// Add to root, as don't know module we are in, and aids later structure sharing
|
||
v3Global.rootp()->dollarUnitPkgAddp()->addStmtp(varp);
|
||
// Element 0 is a non-index and has speced values
|
||
initp->addInitsp(dimensionValue(nodep, attrType, 0));
|
||
for (unsigned i=1; i<msbdim+1; ++i) {
|
||
initp->addInitsp(dimensionValue(nodep, attrType, i));
|
||
}
|
||
varp->iterate(*this); // May have already done $unit so must do this var
|
||
m_tableMap.insert(make_pair(make_pair(nodep,attrType), varp));
|
||
return varp;
|
||
}
|
||
AstVar* enumVarp(AstEnumDType* nodep, AstAttrType attrType, uint32_t msbdim) {
|
||
// Return a variable table which has specified dimension properties for this variable
|
||
TableMap::iterator pos = m_tableMap.find(make_pair(nodep,attrType));
|
||
if (pos != m_tableMap.end()) {
|
||
return pos->second;
|
||
}
|
||
UINFO(9, "Construct Venumtab attr="<<attrType.ascii()<<" max="<<msbdim<<" for "<<nodep<<endl);
|
||
AstNodeDType* basep;
|
||
if (attrType == AstAttrType::ENUM_NAME) {
|
||
basep = nodep->findStringDType();
|
||
} else {
|
||
basep = nodep->findSigned32DType();
|
||
}
|
||
AstNodeArrayDType* vardtypep = new AstUnpackArrayDType(nodep->fileline(),
|
||
basep,
|
||
new AstRange(nodep->fileline(), msbdim, 0));
|
||
AstInitArray* initp = new AstInitArray (nodep->fileline(), vardtypep, NULL);
|
||
v3Global.rootp()->typeTablep()->addTypesp(vardtypep);
|
||
AstVar* varp = new AstVar (nodep->fileline(), AstVarType::MODULETEMP,
|
||
"__Venumtab_" + VString::downcase(attrType.ascii()) + cvtToStr(m_dtTables++),
|
||
vardtypep);
|
||
varp->isConst(true);
|
||
varp->isStatic(true);
|
||
varp->valuep(initp);
|
||
// Add to root, as don't know module we are in, and aids later structure sharing
|
||
v3Global.rootp()->dollarUnitPkgAddp()->addStmtp(varp);
|
||
|
||
// Find valid values and populate
|
||
if (!nodep->itemsp()) nodep->v3fatalSrc("enum without items");
|
||
vector<AstNode*> values;
|
||
values.reserve(msbdim+1);
|
||
for (unsigned i=0; i<(msbdim+1); ++i) {
|
||
values[i] = NULL;
|
||
}
|
||
{
|
||
AstEnumItem* firstp = nodep->itemsp();
|
||
AstEnumItem* prevp = firstp; // Prev must start with last item
|
||
while (prevp->nextp()) prevp = prevp->nextp()->castEnumItem();
|
||
for (AstEnumItem* itemp = firstp; itemp;) {
|
||
AstEnumItem* nextp = itemp->nextp()->castEnumItem();
|
||
AstConst* vconstp = itemp->valuep()->castConst();
|
||
if (!vconstp) nodep->v3fatalSrc("Enum item without constified value");
|
||
uint32_t i = vconstp->toUInt();
|
||
if (attrType == AstAttrType::ENUM_NAME) {
|
||
values[i] = new AstConst(nodep->fileline(), AstConst::String(), itemp->name());
|
||
} else if (attrType == AstAttrType::ENUM_NEXT) {
|
||
values[i] = (nextp ? nextp : firstp)->valuep()->cloneTree(false); // A const
|
||
} else if (attrType == AstAttrType::ENUM_PREV) {
|
||
values[i] = prevp->valuep()->cloneTree(false); // A const
|
||
} else {
|
||
nodep->v3fatalSrc("Bad case");
|
||
}
|
||
prevp = itemp;
|
||
itemp = nextp;
|
||
}
|
||
}
|
||
// Fill in all unspecified values and add to table
|
||
for (unsigned i=0; i<(msbdim+1); ++i) {
|
||
AstNode* valp = values[i];
|
||
if (!valp) {
|
||
if (attrType == AstAttrType::ENUM_NAME) {
|
||
valp = new AstConst(nodep->fileline(), AstConst::String(), "");
|
||
} else if (attrType == AstAttrType::ENUM_NEXT
|
||
|| attrType == AstAttrType::ENUM_PREV) {
|
||
valp = new AstConst(nodep->fileline(), V3Number(nodep->fileline(), nodep->width(), 0));
|
||
} else {
|
||
nodep->v3fatalSrc("Bad case");
|
||
}
|
||
}
|
||
initp->addInitsp(valp);
|
||
}
|
||
varp->iterate(*this); // May have already done $unit so must do this var
|
||
m_tableMap.insert(make_pair(make_pair(nodep,attrType), varp));
|
||
return varp;
|
||
}
|
||
|
||
PatVecMap patVectorMap(AstPattern* nodep, const VNumRange& range) {
|
||
PatVecMap patmap;
|
||
int element = range.left();
|
||
for (AstPatMember* patp = nodep->itemsp()->castPatMember();
|
||
patp; patp = patp->nextp()->castPatMember()) {
|
||
if (patp->keyp()) {
|
||
if (AstConst* constp = patp->keyp()->castConst()) {
|
||
element = constp->toSInt();
|
||
} else {
|
||
patp->keyp()->v3error("Assignment pattern key not supported/understood: "<<patp->keyp()->prettyTypeName());
|
||
}
|
||
}
|
||
if (patmap.find(element) != patmap.end()) {
|
||
patp->v3error("Assignment pattern key used multiple times: "<<element);
|
||
} else {
|
||
patmap.insert(make_pair(element, patp));
|
||
}
|
||
element += range.leftToRightInc();
|
||
}
|
||
return patmap;
|
||
}
|
||
|
||
|
||
//----------------------------------------------------------------------
|
||
// METHODS - special type detection
|
||
void assertAtStatement(AstNode* nodep, AstNUser* vup) {
|
||
if (VL_UNLIKELY(vup && !vup->c()->selfDtm())) {
|
||
UINFO(1,"-: "<<vup->c()<<endl);
|
||
nodep->v3fatalSrc("No dtype expected at statement "<<nodep->prettyTypeName());
|
||
}
|
||
}
|
||
void checkConstantOrReplace(AstNode* nodep, const string& message) {
|
||
// See also V3WidthSel::checkConstantOrReplace
|
||
// Note can't call V3Const::constifyParam(nodep) here, as constify may change nodep on us!
|
||
if (!nodep->castConst()) {
|
||
nodep->v3error(message);
|
||
nodep->replaceWith(new AstConst(nodep->fileline(), AstConst::Unsized32(), 1));
|
||
pushDeletep(nodep); VL_DANGLING(nodep);
|
||
}
|
||
}
|
||
|
||
public:
|
||
// CONSTUCTORS
|
||
WidthVisitor(bool paramsOnly, // [in] TRUE if we are considering parameters only.
|
||
bool doGenerate) { // [in] TRUE if we are inside a generate statement and
|
||
// // don't wish to trigger errors
|
||
m_paramsOnly = paramsOnly;
|
||
m_cellRangep = NULL;
|
||
m_funcp = NULL;
|
||
m_initialp = NULL;
|
||
m_attrp = NULL;
|
||
m_doGenerate = doGenerate;
|
||
m_dtTables = 0;
|
||
}
|
||
AstNode* mainAcceptEdit(AstNode* nodep) {
|
||
return nodep->acceptSubtreeReturnEdits(*this, WidthVP(SELF,BOTH).p());
|
||
}
|
||
virtual ~WidthVisitor() {}
|
||
};
|
||
|
||
//######################################################################
|
||
// Width class functions
|
||
|
||
int V3Width::debug() {
|
||
static int level = -1;
|
||
if (VL_UNLIKELY(level < 0)) level = v3Global.opt.debugSrcLevel(__FILE__);
|
||
return level;
|
||
}
|
||
|
||
void V3Width::width(AstNetlist* nodep) {
|
||
UINFO(2,__FUNCTION__<<": "<<endl);
|
||
// We should do it in bottom-up module order, but it works in any order.
|
||
WidthVisitor visitor (false, false);
|
||
(void)visitor.mainAcceptEdit(nodep);
|
||
WidthRemoveVisitor rvisitor;
|
||
(void)rvisitor.mainAcceptEdit(nodep);
|
||
V3Global::dumpCheckGlobalTree("width.tree", 0, v3Global.opt.dumpTreeLevel(__FILE__) >= 3);
|
||
}
|
||
|
||
//! Single node parameter propagation
|
||
//! Smaller step... Only do a single node for parameter propagation
|
||
AstNode* V3Width::widthParamsEdit (AstNode* nodep) {
|
||
UINFO(4,__FUNCTION__<<": "<<nodep<<endl);
|
||
// We should do it in bottom-up module order, but it works in any order.
|
||
WidthVisitor visitor (true, false);
|
||
nodep = visitor.mainAcceptEdit(nodep);
|
||
// No WidthRemoveVisitor, as don't want to drop $signed etc inside gen blocks
|
||
return nodep;
|
||
}
|
||
|
||
//! Single node parameter propagation for generate blocks.
|
||
//! Smaller step... Only do a single node for parameter propagation
|
||
//! If we are inside a generated "if", "case" or "for", we don't want to
|
||
//! trigger warnings when we deal with the width. It is possible that
|
||
//! these are spurious, existing within sub-expressions that will not
|
||
//! actually be generated. Since such occurrences, must be constant, in
|
||
//! order to be someting a generate block can depend on, we can wait until
|
||
//! later to do the width check.
|
||
//! @return Pointer to the edited node.
|
||
AstNode* V3Width::widthGenerateParamsEdit(
|
||
AstNode* nodep) { //!< [in] AST whose parameters widths are to be analysed.
|
||
UINFO(4,__FUNCTION__<<": "<<nodep<<endl);
|
||
// We should do it in bottom-up module order, but it works in any order.
|
||
WidthVisitor visitor (true, true);
|
||
nodep = visitor.mainAcceptEdit(nodep);
|
||
// No WidthRemoveVisitor, as don't want to drop $signed etc inside gen blocks
|
||
return nodep;
|
||
}
|
||
|
||
void V3Width::widthCommit(AstNetlist* nodep) {
|
||
UINFO(2,__FUNCTION__<<": "<<endl);
|
||
WidthCommitVisitor visitor (nodep);
|
||
V3Global::dumpCheckGlobalTree("widthcommit.tree", 0, v3Global.opt.dumpTreeLevel(__FILE__) >= 6);
|
||
}
|