forked from github/verilator
Support up to 64 bit enums for .next/.prev/.name (#3244).
This commit is contained in:
parent
9a722ccf62
commit
59d170c6f8
1
Changes
1
Changes
@ -19,6 +19,7 @@ Verilator 4.217 devel
|
|||||||
|
|
||||||
**Minor:**
|
**Minor:**
|
||||||
|
|
||||||
|
* Support up to 64 bit enums for .next/.prev/.name (#3244). [Alexander Grobman]
|
||||||
* Fix MSWIN compile error (#2681). [Unai Martinez-Corral]
|
* Fix MSWIN compile error (#2681). [Unai Martinez-Corral]
|
||||||
* Fix VL_STREAML_FAST_QQI with 64 bit left-hand-side (#3232) (#3235)
|
* Fix VL_STREAML_FAST_QQI with 64 bit left-hand-side (#3232) (#3235)
|
||||||
* Fix $sformat of inputs/outputs (#3236). [Adrien Le Masle]
|
* Fix $sformat of inputs/outputs (#3236). [Adrien Le Masle]
|
||||||
|
@ -1017,21 +1017,41 @@ static bool sameInit(const AstInitArray* ap, const AstInitArray* bp) {
|
|||||||
// - the default/inititem children might be in different order yet still yield the same table
|
// - the default/inititem children might be in different order yet still yield the same table
|
||||||
// See note in AstInitArray::same about the same. This function instead compares by initializer
|
// See note in AstInitArray::same about the same. This function instead compares by initializer
|
||||||
// value, rather than by tree structure.
|
// value, rather than by tree structure.
|
||||||
const AstUnpackArrayDType* const aDTypep = VN_AS(ap->dtypep(), UnpackArrayDType);
|
if (const AstAssocArrayDType* const aDTypep = VN_CAST(ap->dtypep(), AssocArrayDType)) {
|
||||||
const AstUnpackArrayDType* const bDTypep = VN_AS(bp->dtypep(), UnpackArrayDType);
|
const AstAssocArrayDType* const bDTypep = VN_CAST(bp->dtypep(), AssocArrayDType);
|
||||||
if (!aDTypep->subDTypep()->sameTree(bDTypep->subDTypep())) { // Element types differ
|
if (!bDTypep) return false;
|
||||||
return false;
|
if (!aDTypep->subDTypep()->sameTree(bDTypep->subDTypep())) return false;
|
||||||
}
|
if (!aDTypep->keyDTypep()->sameTree(bDTypep->keyDTypep())) return false;
|
||||||
if (!aDTypep->rangep()->sameTree(bDTypep->rangep())) { // Ranges differ
|
UASSERT_OBJ(ap->defaultp(), ap, "Assoc InitArray should have a default");
|
||||||
return false;
|
UASSERT_OBJ(bp->defaultp(), bp, "Assoc InitArray should have a default");
|
||||||
}
|
if (!ap->defaultp()->sameTree(bp->defaultp())) return false;
|
||||||
// Compare initializer arrays by value. Note this is only called when they hash the same,
|
// Compare initializer arrays by value. Note this is only called when they hash the same,
|
||||||
// so they likely run at most once per call to 'AstConstPool::findTable'.
|
// so they likely run at most once per call to 'AstConstPool::findTable'.
|
||||||
const uint32_t size = aDTypep->elementsConst();
|
// This assumes that the defaults are used in the same way.
|
||||||
for (uint32_t n = 0; n < size; ++n) {
|
// TODO when buinding the AstInitArray, remove any values matching the default
|
||||||
|
const auto& amapr = ap->map();
|
||||||
|
const auto& bmapr = bp->map();
|
||||||
|
const auto ait = amapr.cbegin();
|
||||||
|
const auto bit = bmapr.cbegin();
|
||||||
|
while (ait != amapr.cend() || bit != bmapr.cend()) {
|
||||||
|
if (ait == amapr.cend() || bit == bmapr.cend()) return false; // Different size
|
||||||
|
if (ait->first != bit->first) return false; // Different key
|
||||||
|
if (ait->second->sameTree(bit->second)) return false; // Different value
|
||||||
|
}
|
||||||
|
} else if (const AstUnpackArrayDType* const aDTypep
|
||||||
|
= VN_CAST(ap->dtypep(), UnpackArrayDType)) {
|
||||||
|
const AstUnpackArrayDType* const bDTypep = VN_CAST(bp->dtypep(), UnpackArrayDType);
|
||||||
|
if (!bDTypep) return false;
|
||||||
|
if (!aDTypep->subDTypep()->sameTree(bDTypep->subDTypep())) return false;
|
||||||
|
if (!aDTypep->rangep()->sameTree(bDTypep->rangep())) return false;
|
||||||
|
// Compare initializer arrays by value. Note this is only called when they hash the same,
|
||||||
|
// so they likely run at most once per call to 'AstConstPool::findTable'.
|
||||||
|
const vluint64_t size = aDTypep->elementsConst();
|
||||||
|
for (vluint64_t n = 0; n < size; ++n) {
|
||||||
const AstNode* const valAp = ap->getIndexDefaultedValuep(n);
|
const AstNode* const valAp = ap->getIndexDefaultedValuep(n);
|
||||||
const AstNode* const valBp = bp->getIndexDefaultedValuep(n);
|
const AstNode* const valBp = bp->getIndexDefaultedValuep(n);
|
||||||
if (!valAp->sameTree(valBp)) { return false; }
|
if (!valAp->sameTree(valBp)) return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -1039,8 +1059,9 @@ static bool sameInit(const AstInitArray* ap, const AstInitArray* bp) {
|
|||||||
AstVarScope* AstConstPool::findTable(AstInitArray* initp) {
|
AstVarScope* AstConstPool::findTable(AstInitArray* initp) {
|
||||||
const AstNode* const defaultp = initp->defaultp();
|
const AstNode* const defaultp = initp->defaultp();
|
||||||
// Verify initializer is well formed
|
// Verify initializer is well formed
|
||||||
UASSERT_OBJ(VN_IS(initp->dtypep(), UnpackArrayDType), initp,
|
UASSERT_OBJ(VN_IS(initp->dtypep(), AssocArrayDType)
|
||||||
"Const pool table must have AstUnpackArrayDType dtype");
|
|| VN_IS(initp->dtypep(), UnpackArrayDType),
|
||||||
|
initp, "Const pool table must have array dtype");
|
||||||
UASSERT_OBJ(!defaultp || VN_IS(defaultp, Const), initp,
|
UASSERT_OBJ(!defaultp || VN_IS(defaultp, Const), initp,
|
||||||
"Const pool table default must be Const");
|
"Const pool table default must be Const");
|
||||||
for (AstNode* nodep = initp->initsp(); nodep; nodep = nodep->nextp()) {
|
for (AstNode* nodep = initp->initsp(); nodep; nodep = nodep->nextp()) {
|
||||||
|
@ -538,6 +538,12 @@ public:
|
|||||||
keyDTypep(nullptr);
|
keyDTypep(nullptr);
|
||||||
dtypep(nullptr); // V3Width will resolve
|
dtypep(nullptr); // V3Width will resolve
|
||||||
}
|
}
|
||||||
|
AstAssocArrayDType(FileLine* fl, AstNodeDType* dtp, AstNodeDType* keyDtp)
|
||||||
|
: ASTGEN_SUPER_AssocArrayDType(fl) {
|
||||||
|
refDTypep(dtp);
|
||||||
|
keyDTypep(keyDtp);
|
||||||
|
dtypep(dtp);
|
||||||
|
}
|
||||||
ASTNODE_NODE_FUNCS(AssocArrayDType)
|
ASTNODE_NODE_FUNCS(AssocArrayDType)
|
||||||
virtual const char* broken() const override {
|
virtual const char* broken() const override {
|
||||||
BROKEN_RTN(!((m_refDTypep && !childDTypep() && m_refDTypep->brokeExists())
|
BROKEN_RTN(!((m_refDTypep && !childDTypep() && m_refDTypep->brokeExists())
|
||||||
@ -4953,12 +4959,12 @@ class AstInitArray final : public AstNode {
|
|||||||
// Parents: ASTVAR::init()
|
// Parents: ASTVAR::init()
|
||||||
// Children: AstInitItem
|
// Children: AstInitItem
|
||||||
public:
|
public:
|
||||||
using KeyItemMap = std::map<uint32_t, AstInitItem*>;
|
using KeyItemMap = std::map<vluint64_t, AstInitItem*>;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
KeyItemMap m_map; // Node value for each array index
|
KeyItemMap m_map; // Node value for each array index
|
||||||
public:
|
public:
|
||||||
AstInitArray(FileLine* fl, AstNodeArrayDType* newDTypep, AstNode* defaultp)
|
AstInitArray(FileLine* fl, AstNodeDType* newDTypep, AstNode* defaultp)
|
||||||
: ASTGEN_SUPER_InitArray(fl) {
|
: ASTGEN_SUPER_InitArray(fl) {
|
||||||
dtypep(newDTypep);
|
dtypep(newDTypep);
|
||||||
addNOp1p(defaultp);
|
addNOp1p(defaultp);
|
||||||
@ -4988,7 +4994,7 @@ public:
|
|||||||
AstNode* initsp() const { return op2p(); } // op2 = Initial value expressions
|
AstNode* initsp() const { return op2p(); } // op2 = Initial value expressions
|
||||||
void addValuep(AstNode* newp) { addIndexValuep(m_map.size(), newp); }
|
void addValuep(AstNode* newp) { addIndexValuep(m_map.size(), newp); }
|
||||||
const KeyItemMap& map() const { return m_map; }
|
const KeyItemMap& map() const { return m_map; }
|
||||||
AstNode* addIndexValuep(uint32_t index, AstNode* newp) {
|
AstNode* addIndexValuep(vluint64_t index, AstNode* newp) {
|
||||||
// Returns old value, caller must garbage collect
|
// Returns old value, caller must garbage collect
|
||||||
AstNode* oldp = nullptr;
|
AstNode* oldp = nullptr;
|
||||||
const auto it = m_map.find(index);
|
const auto it = m_map.find(index);
|
||||||
@ -5002,7 +5008,7 @@ public:
|
|||||||
}
|
}
|
||||||
return oldp;
|
return oldp;
|
||||||
}
|
}
|
||||||
AstNode* getIndexValuep(uint32_t index) const {
|
AstNode* getIndexValuep(vluint64_t index) const {
|
||||||
const auto it = m_map.find(index);
|
const auto it = m_map.find(index);
|
||||||
if (it == m_map.end()) {
|
if (it == m_map.end()) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
@ -5010,7 +5016,7 @@ public:
|
|||||||
return it->second->valuep();
|
return it->second->valuep();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
AstNode* getIndexDefaultedValuep(uint32_t index) const {
|
AstNode* getIndexDefaultedValuep(vluint64_t index) const {
|
||||||
AstNode* valuep = getIndexValuep(index);
|
AstNode* valuep = getIndexValuep(index);
|
||||||
if (!valuep) valuep = defaultp();
|
if (!valuep) valuep = defaultp();
|
||||||
return valuep;
|
return valuep;
|
||||||
|
@ -46,20 +46,42 @@ class EmitCConstInit VL_NOT_FINAL : public EmitCBaseVisitor {
|
|||||||
protected:
|
protected:
|
||||||
// VISITORS
|
// VISITORS
|
||||||
virtual void visit(AstInitArray* nodep) override {
|
virtual void visit(AstInitArray* nodep) override {
|
||||||
const AstUnpackArrayDType* const dtypep
|
|
||||||
= VN_AS(nodep->dtypep()->skipRefp(), UnpackArrayDType);
|
|
||||||
UASSERT_OBJ(dtypep, nodep, "Array initializer has non-array dtype");
|
|
||||||
const uint32_t size = dtypep->elementsConst();
|
|
||||||
const uint32_t tabMod = tabModulus(dtypep->subDTypep());
|
|
||||||
VL_RESTORER(m_inUnpacked);
|
VL_RESTORER(m_inUnpacked);
|
||||||
VL_RESTORER(m_unpackedWord);
|
VL_RESTORER(m_unpackedWord);
|
||||||
m_inUnpacked = true;
|
m_inUnpacked = true;
|
||||||
// Note the double {{ initializer. The first { starts the initializer of the VlUnpacked,
|
if (const AstAssocArrayDType* const dtypep
|
||||||
// and the second starts the initializer of m_storage within the VlUnpacked.
|
= VN_CAST(nodep->dtypep()->skipRefp(), AssocArrayDType)) {
|
||||||
|
// Note the double {{ initializer. The first { starts the initializer of the
|
||||||
|
// VlUnpacked, and the second starts the initializer of m_storage within the
|
||||||
|
// VlUnpacked.
|
||||||
puts("{");
|
puts("{");
|
||||||
ofp()->putsNoTracking("{");
|
ofp()->putsNoTracking("{");
|
||||||
puts("\n");
|
puts("\n");
|
||||||
for (uint32_t n = 0; n < size; ++n) {
|
int comma = 0;
|
||||||
|
const auto& mapr = nodep->map();
|
||||||
|
for (const auto& itr : mapr) {
|
||||||
|
if (comma++) putbs(",\n");
|
||||||
|
puts(cvtToStr(itr.first));
|
||||||
|
ofp()->printf("%" PRIx64 "ULL", itr.first);
|
||||||
|
ofp()->putsNoTracking(":");
|
||||||
|
ofp()->putsNoTracking("{");
|
||||||
|
iterate(nodep->getIndexValuep(itr.first));
|
||||||
|
ofp()->putsNoTracking("}");
|
||||||
|
}
|
||||||
|
puts("\n");
|
||||||
|
puts("}");
|
||||||
|
ofp()->putsNoTracking("}");
|
||||||
|
} else if (const AstUnpackArrayDType* const dtypep
|
||||||
|
= VN_CAST(nodep->dtypep()->skipRefp(), UnpackArrayDType)) {
|
||||||
|
const vluint64_t size = dtypep->elementsConst();
|
||||||
|
const uint32_t tabMod = tabModulus(dtypep->subDTypep());
|
||||||
|
// Note the double {{ initializer. The first { starts the initializer of the
|
||||||
|
// VlUnpacked, and the second starts the initializer of m_storage within the
|
||||||
|
// VlUnpacked.
|
||||||
|
puts("{");
|
||||||
|
ofp()->putsNoTracking("{");
|
||||||
|
puts("\n");
|
||||||
|
for (vluint64_t n = 0; n < size; ++n) {
|
||||||
m_unpackedWord = n;
|
m_unpackedWord = n;
|
||||||
if (n) puts((n % tabMod) ? ", " : ",\n");
|
if (n) puts((n % tabMod) ? ", " : ",\n");
|
||||||
iterate(nodep->getIndexDefaultedValuep(n));
|
iterate(nodep->getIndexDefaultedValuep(n));
|
||||||
@ -67,6 +89,9 @@ protected:
|
|||||||
puts("\n");
|
puts("\n");
|
||||||
puts("}");
|
puts("}");
|
||||||
ofp()->putsNoTracking("}");
|
ofp()->putsNoTracking("}");
|
||||||
|
} else {
|
||||||
|
nodep->v3fatalSrc("Array initializer has non-array dtype");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void visit(AstInitItem* nodep) override { // LCOV_EXCL_START
|
virtual void visit(AstInitItem* nodep) override { // LCOV_EXCL_START
|
||||||
|
@ -622,7 +622,18 @@ void EmitCFunc::emitVarReset(AstVar* varp) {
|
|||||||
// If an ARRAYINIT we initialize it using an initial block similar to a signal
|
// If an ARRAYINIT we initialize it using an initial block similar to a signal
|
||||||
// puts("// parameter "+varp->nameProtect()+" = "+varp->valuep()->name()+"\n");
|
// puts("// parameter "+varp->nameProtect()+" = "+varp->valuep()->name()+"\n");
|
||||||
} else if (const AstInitArray* const initarp = VN_CAST(varp->valuep(), InitArray)) {
|
} else if (const AstInitArray* const initarp = VN_CAST(varp->valuep(), InitArray)) {
|
||||||
if (AstUnpackArrayDType* const adtypep = VN_CAST(dtypep, UnpackArrayDType)) {
|
if (AstAssocArrayDType* const adtypep = VN_CAST(dtypep, AssocArrayDType)) {
|
||||||
|
if (initarp->defaultp()) {
|
||||||
|
emitSetVarConstant(varNameProtected + ".atDefault()",
|
||||||
|
VN_AS(initarp->defaultp(), Const));
|
||||||
|
}
|
||||||
|
const auto& mapr = initarp->map();
|
||||||
|
for (const auto& itr : mapr) {
|
||||||
|
AstNode* const valuep = itr.second->valuep();
|
||||||
|
emitSetVarConstant(varNameProtected + ".at(" + cvtToStr(itr.first) + ")",
|
||||||
|
VN_AS(valuep, Const));
|
||||||
|
}
|
||||||
|
} else if (AstUnpackArrayDType* const adtypep = VN_CAST(dtypep, UnpackArrayDType)) {
|
||||||
if (initarp->defaultp()) {
|
if (initarp->defaultp()) {
|
||||||
puts("for (int __Vi=0; __Vi<" + cvtToStr(adtypep->elementsConst()));
|
puts("for (int __Vi=0; __Vi<" + cvtToStr(adtypep->elementsConst()));
|
||||||
puts("; ++__Vi) {\n");
|
puts("; ++__Vi) {\n");
|
||||||
|
@ -351,9 +351,21 @@ private:
|
|||||||
m_hash += hashNodeAndIterate(nodep, HASH_DTYPE, HASH_CHILDREN, [=]() {});
|
m_hash += hashNodeAndIterate(nodep, HASH_DTYPE, HASH_CHILDREN, [=]() {});
|
||||||
}
|
}
|
||||||
virtual void visit(AstInitArray* nodep) override {
|
virtual void visit(AstInitArray* nodep) override {
|
||||||
// Hash unpacked array initializers by value, as the order of initializer nodes does not
|
if (const AstAssocArrayDType* const dtypep = VN_CAST(nodep->dtypep(), AssocArrayDType)) {
|
||||||
// matter, and we want semantically equivalent initializers to map to the same hash.
|
if (nodep->defaultp()) {
|
||||||
const AstUnpackArrayDType* const dtypep = VN_CAST(nodep->dtypep(), UnpackArrayDType);
|
m_hash
|
||||||
|
+= hashNodeAndIterate(nodep->defaultp(), HASH_DTYPE, HASH_CHILDREN, [=]() {});
|
||||||
|
}
|
||||||
|
const auto& mapr = nodep->map();
|
||||||
|
for (const auto& itr : mapr) { // mapr is sorted, so hash should get stable results
|
||||||
|
m_hash += itr.first;
|
||||||
|
m_hash += hashNodeAndIterate(itr.second, HASH_DTYPE, HASH_CHILDREN, [=]() {});
|
||||||
|
}
|
||||||
|
} else if (const AstUnpackArrayDType* const dtypep
|
||||||
|
= VN_CAST(nodep->dtypep(), UnpackArrayDType)) {
|
||||||
|
// Hash unpacked array initializers by value, as the order of initializer nodes does
|
||||||
|
// not matter, and we want semantically equivalent initializers to map to the same
|
||||||
|
// hash.
|
||||||
m_hash += hashNodeAndIterate(nodep, HASH_DTYPE, /* hashChildren: */ !dtypep, [=]() {
|
m_hash += hashNodeAndIterate(nodep, HASH_DTYPE, /* hashChildren: */ !dtypep, [=]() {
|
||||||
if (dtypep) {
|
if (dtypep) {
|
||||||
const uint32_t size = dtypep->elementsConst();
|
const uint32_t size = dtypep->elementsConst();
|
||||||
@ -363,6 +375,7 @@ private:
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
virtual void visit(AstPragma* nodep) override {
|
virtual void visit(AstPragma* nodep) override {
|
||||||
m_hash += hashNodeAndIterate(nodep, HASH_DTYPE, HASH_CHILDREN, [=]() { //
|
m_hash += hashNodeAndIterate(nodep, HASH_DTYPE, HASH_CHILDREN, [=]() { //
|
||||||
m_hash += nodep->pragType();
|
m_hash += nodep->pragType();
|
||||||
|
@ -1675,20 +1675,30 @@ private:
|
|||||||
auto* const enumDtp = VN_AS(toDtp, EnumDType);
|
auto* const enumDtp = VN_AS(toDtp, EnumDType);
|
||||||
UASSERT_OBJ(enumDtp, nodep, "$cast determined as enum, but not enum type");
|
UASSERT_OBJ(enumDtp, nodep, "$cast determined as enum, but not enum type");
|
||||||
const uint64_t maxval = enumMaxValue(nodep, enumDtp);
|
const uint64_t maxval = enumMaxValue(nodep, enumDtp);
|
||||||
|
const bool assoc = maxval > ENUM_LOOKUP_BITS;
|
||||||
|
AstNode* testp = nullptr;
|
||||||
|
if (assoc) {
|
||||||
|
AstVar* const varp = enumVarp(enumDtp, AstAttrType::ENUM_VALID, true, 0);
|
||||||
|
AstVarRef* const varrefp = new AstVarRef{fl, varp, VAccess::READ};
|
||||||
|
varrefp->classOrPackagep(v3Global.rootp()->dollarUnitPkgAddp());
|
||||||
|
testp = new AstAssocSel{fl, varrefp, nodep->fromp()->cloneTree(false)};
|
||||||
|
} else {
|
||||||
const int selwidth = V3Number::log2b(maxval) + 1; // Width to address a bit
|
const int selwidth = V3Number::log2b(maxval) + 1; // Width to address a bit
|
||||||
AstVar* const varp
|
AstVar* const varp
|
||||||
= enumVarp(enumDtp, AstAttrType::ENUM_VALID, (1ULL << selwidth) - 1);
|
= enumVarp(enumDtp, AstAttrType::ENUM_VALID, false, (1ULL << selwidth) - 1);
|
||||||
AstVarRef* const varrefp = new AstVarRef(fl, varp, VAccess::READ);
|
AstVarRef* const varrefp = new AstVarRef(fl, varp, VAccess::READ);
|
||||||
varrefp->classOrPackagep(v3Global.rootp()->dollarUnitPkgAddp());
|
varrefp->classOrPackagep(v3Global.rootp()->dollarUnitPkgAddp());
|
||||||
FileLine* const fl_nowarn = new FileLine(fl);
|
FileLine* const fl_nowarn = new FileLine(fl);
|
||||||
fl_nowarn->warnOff(V3ErrorCode::WIDTH, true);
|
fl_nowarn->warnOff(V3ErrorCode::WIDTH, true);
|
||||||
auto* const testp = new AstCond{
|
testp = new AstCond{
|
||||||
fl,
|
fl,
|
||||||
new AstGt{fl_nowarn, nodep->fromp()->cloneTree(false),
|
new AstGt{fl_nowarn, nodep->fromp()->cloneTree(false),
|
||||||
new AstConst{fl_nowarn, AstConst::Unsized64{}, maxval}},
|
new AstConst{fl_nowarn, AstConst::Unsized64{}, maxval}},
|
||||||
new AstConst{fl, AstConst::BitFalse{}},
|
new AstConst{fl, AstConst::BitFalse{}},
|
||||||
new AstArraySel{fl, varrefp,
|
new AstArraySel{
|
||||||
|
fl, varrefp,
|
||||||
new AstSel{fl, nodep->fromp()->cloneTree(false), 0, selwidth}}};
|
new AstSel{fl, nodep->fromp()->cloneTree(false), 0, selwidth}}};
|
||||||
|
}
|
||||||
newp = new AstCond{fl, testp,
|
newp = new AstCond{fl, testp,
|
||||||
new AstExprStmt{fl,
|
new AstExprStmt{fl,
|
||||||
new AstAssign{fl, nodep->top()->unlinkFrBack(),
|
new AstAssign{fl, nodep->top()->unlinkFrBack(),
|
||||||
@ -2224,8 +2234,8 @@ private:
|
|||||||
AstNodeDType* const vdtypep = m_vup->dtypeNullp();
|
AstNodeDType* const vdtypep = m_vup->dtypeNullp();
|
||||||
UASSERT_OBJ(vdtypep, nodep, "InitArray type not assigned by AstPattern/Var visitor");
|
UASSERT_OBJ(vdtypep, nodep, "InitArray type not assigned by AstPattern/Var visitor");
|
||||||
nodep->dtypep(vdtypep);
|
nodep->dtypep(vdtypep);
|
||||||
if (const AstNodeArrayDType* const arrayp
|
const AstNodeDType* const arrayp = vdtypep->skipRefp();
|
||||||
= VN_CAST(vdtypep->skipRefp(), NodeArrayDType)) {
|
if (VN_IS(arrayp, NodeArrayDType) || VN_IS(arrayp, AssocArrayDType)) {
|
||||||
userIterateChildren(nodep, WidthVP(arrayp->subDTypep(), BOTH).p());
|
userIterateChildren(nodep, WidthVP(arrayp->subDTypep(), BOTH).p());
|
||||||
} else {
|
} else {
|
||||||
UINFO(1, "dtype object " << vdtypep->skipRefp() << endl);
|
UINFO(1, "dtype object " << vdtypep->skipRefp() << endl);
|
||||||
@ -2635,9 +2645,18 @@ private:
|
|||||||
}
|
}
|
||||||
// Need a runtime lookup table. Yuk.
|
// Need a runtime lookup table. Yuk.
|
||||||
const uint64_t msbdim = enumMaxValue(nodep, adtypep);
|
const uint64_t msbdim = enumMaxValue(nodep, adtypep);
|
||||||
|
const bool assoc = msbdim > ENUM_LOOKUP_BITS;
|
||||||
|
if (assoc) {
|
||||||
|
AstVar* const varp = enumVarp(adtypep, attrType, true, 0);
|
||||||
|
AstVarRef* const varrefp = new AstVarRef{nodep->fileline(), varp, VAccess::READ};
|
||||||
|
varrefp->classOrPackagep(v3Global.rootp()->dollarUnitPkgAddp());
|
||||||
|
AstNode* const newp
|
||||||
|
= new AstAssocSel{nodep->fileline(), varrefp, nodep->fromp()->unlinkFrBack()};
|
||||||
|
nodep->replaceWith(newp);
|
||||||
|
} else {
|
||||||
const int selwidth = V3Number::log2b(msbdim) + 1; // Width to address a bit
|
const int selwidth = V3Number::log2b(msbdim) + 1; // Width to address a bit
|
||||||
AstVar* const varp = enumVarp(adtypep, attrType, (1ULL << selwidth) - 1);
|
AstVar* const varp = enumVarp(adtypep, attrType, false, (1ULL << selwidth) - 1);
|
||||||
AstVarRef* const varrefp = new AstVarRef(nodep->fileline(), varp, VAccess::READ);
|
AstVarRef* const varrefp = new AstVarRef{nodep->fileline(), varp, VAccess::READ};
|
||||||
varrefp->classOrPackagep(v3Global.rootp()->dollarUnitPkgAddp());
|
varrefp->classOrPackagep(v3Global.rootp()->dollarUnitPkgAddp());
|
||||||
AstNode* const newp = new AstArraySel(
|
AstNode* const newp = new AstArraySel(
|
||||||
nodep->fileline(), varrefp,
|
nodep->fileline(), varrefp,
|
||||||
@ -2646,6 +2665,7 @@ private:
|
|||||||
// as next/previous on illegal values just need something good out
|
// as next/previous on illegal values just need something good out
|
||||||
new AstSel(nodep->fileline(), nodep->fromp()->unlinkFrBack(), 0, selwidth));
|
new AstSel(nodep->fileline(), nodep->fromp()->unlinkFrBack(), 0, selwidth));
|
||||||
nodep->replaceWith(newp);
|
nodep->replaceWith(newp);
|
||||||
|
}
|
||||||
VL_DO_DANGLING(nodep->deleteTree(), nodep);
|
VL_DO_DANGLING(nodep->deleteTree(), nodep);
|
||||||
} else {
|
} else {
|
||||||
nodep->v3error("Unknown built-in enum method " << nodep->prettyNameQ());
|
nodep->v3error("Unknown built-in enum method " << nodep->prettyNameQ());
|
||||||
@ -5912,19 +5932,19 @@ private:
|
|||||||
UASSERT_OBJ(vconstp, errNodep, "Enum item without constified value");
|
UASSERT_OBJ(vconstp, errNodep, "Enum item without constified value");
|
||||||
if (vconstp->toUQuad() >= maxval) maxval = vconstp->toUQuad();
|
if (vconstp->toUQuad() >= maxval) maxval = vconstp->toUQuad();
|
||||||
}
|
}
|
||||||
if (adtypep->itemsp()->width() > 64 || maxval >= (1 << ENUM_LOOKUP_BITS)) {
|
if (adtypep->itemsp()->width() > 64) {
|
||||||
errNodep->v3warn(E_UNSUPPORTED,
|
errNodep->v3warn(E_UNSUPPORTED,
|
||||||
"Unsupported: enum next/prev method on enum with > 10 bits");
|
"Unsupported: enum next/prev/name method on enum with > 64 bits");
|
||||||
return ENUM_LOOKUP_BITS;
|
return 64;
|
||||||
}
|
}
|
||||||
return maxval;
|
return maxval;
|
||||||
}
|
}
|
||||||
AstVar* enumVarp(AstEnumDType* nodep, AstAttrType attrType, uint32_t msbdim) {
|
AstVar* enumVarp(AstEnumDType* nodep, AstAttrType attrType, bool assoc, uint32_t msbdim) {
|
||||||
// Return a variable table which has specified dimension properties for this variable
|
// Return a variable table which has specified dimension properties for this variable
|
||||||
const auto pos = m_tableMap.find(std::make_pair(nodep, attrType));
|
const auto pos = m_tableMap.find(std::make_pair(nodep, attrType));
|
||||||
if (pos != m_tableMap.end()) return pos->second;
|
if (pos != m_tableMap.end()) return pos->second;
|
||||||
UINFO(9, "Construct Venumtab attr=" << attrType.ascii() << " max=" << msbdim << " for "
|
UINFO(9, "Construct Venumtab attr=" << attrType.ascii() << " assoc=" << assoc
|
||||||
<< nodep << endl);
|
<< " max=" << msbdim << " for " << nodep << endl);
|
||||||
AstNodeDType* basep;
|
AstNodeDType* basep;
|
||||||
if (attrType == AstAttrType::ENUM_NAME) {
|
if (attrType == AstAttrType::ENUM_NAME) {
|
||||||
basep = nodep->findStringDType();
|
basep = nodep->findStringDType();
|
||||||
@ -5937,8 +5957,13 @@ private:
|
|||||||
} else {
|
} else {
|
||||||
basep = nodep->dtypep();
|
basep = nodep->dtypep();
|
||||||
}
|
}
|
||||||
AstNodeArrayDType* const vardtypep = new AstUnpackArrayDType(
|
AstNodeDType* vardtypep;
|
||||||
nodep->fileline(), basep, new AstRange(nodep->fileline(), msbdim, 0));
|
if (assoc) {
|
||||||
|
vardtypep = new AstAssocArrayDType{nodep->fileline(), basep, nodep};
|
||||||
|
} else {
|
||||||
|
vardtypep = new AstUnpackArrayDType{nodep->fileline(), basep,
|
||||||
|
new AstRange(nodep->fileline(), msbdim, 0)};
|
||||||
|
}
|
||||||
AstInitArray* const initp = new AstInitArray(nodep->fileline(), vardtypep, nullptr);
|
AstInitArray* const initp = new AstInitArray(nodep->fileline(), vardtypep, nullptr);
|
||||||
v3Global.rootp()->typeTablep()->addTypesp(vardtypep);
|
v3Global.rootp()->typeTablep()->addTypesp(vardtypep);
|
||||||
AstVar* const varp = new AstVar(nodep->fileline(), AstVarType::MODULETEMP,
|
AstVar* const varp = new AstVar(nodep->fileline(), AstVarType::MODULETEMP,
|
||||||
@ -5990,9 +6015,13 @@ private:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Add all specified values to table
|
// Add all specified values to table
|
||||||
|
if (assoc) {
|
||||||
|
for (const auto& itr : values) initp->addIndexValuep(itr.first, itr.second);
|
||||||
|
} else {
|
||||||
for (vluint64_t i = 0; i < (msbdim + 1); ++i) {
|
for (vluint64_t i = 0; i < (msbdim + 1); ++i) {
|
||||||
if (values[i]) initp->addIndexValuep(i, values[i]);
|
if (values[i]) initp->addIndexValuep(i, values[i]);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
userIterate(varp, nullptr); // May have already done $unit so must do this var
|
userIterate(varp, nullptr); // May have already done $unit so must do this var
|
||||||
m_tableMap.emplace(std::make_pair(nodep, attrType), varp);
|
m_tableMap.emplace(std::make_pair(nodep, attrType), varp);
|
||||||
return varp;
|
return varp;
|
||||||
|
21
test_regress/t/t_enum_huge_methods.pl
Executable file
21
test_regress/t/t_enum_huge_methods.pl
Executable file
@ -0,0 +1,21 @@
|
|||||||
|
#!/usr/bin/env perl
|
||||||
|
if (!$::Driver) { use FindBin; exec("$FindBin::Bin/bootstrap.pl", @ARGV, $0); die; }
|
||||||
|
# DESCRIPTION: Verilator: Verilog Test driver/expect definition
|
||||||
|
#
|
||||||
|
# Copyright 2003 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.
|
||||||
|
# SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
|
||||||
|
|
||||||
|
scenarios(simulator => 1);
|
||||||
|
|
||||||
|
compile(
|
||||||
|
);
|
||||||
|
|
||||||
|
execute(
|
||||||
|
check_finished => 1,
|
||||||
|
);
|
||||||
|
|
||||||
|
ok(1);
|
||||||
|
1;
|
73
test_regress/t/t_enum_huge_methods.v
Normal file
73
test_regress/t/t_enum_huge_methods.v
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
// DESCRIPTION: Verilator: Verilog Test module
|
||||||
|
//
|
||||||
|
// This file ONLY is placed under the Creative Commons Public Domain, for
|
||||||
|
// any use, without warranty, 2014 by Wilson Snyder.
|
||||||
|
// SPDX-License-Identifier: CC0-1.0
|
||||||
|
|
||||||
|
`define checkh(gotv,expv) do if ((gotv) !== (expv)) begin $write("%%Error: %s:%0d: got='h%x exp='h%x\n", `__FILE__,`__LINE__, (gotv), (expv)); $stop; end while(0);
|
||||||
|
`define checks(gotv,expv) do if ((gotv) !== (expv)) begin $write("%%Error: %s:%0d: got='%s' exp='%s'\n", `__FILE__,`__LINE__, (gotv), (expv)); $stop; end while(0);
|
||||||
|
|
||||||
|
module t (/*AUTOARG*/
|
||||||
|
// Inputs
|
||||||
|
clk
|
||||||
|
);
|
||||||
|
input clk;
|
||||||
|
|
||||||
|
typedef enum logic [59:0] {
|
||||||
|
E01 = 60'h1,
|
||||||
|
ELARGE = 60'h1234_4567_abcd
|
||||||
|
} my_t;
|
||||||
|
|
||||||
|
integer cyc = 0;
|
||||||
|
my_t e;
|
||||||
|
|
||||||
|
string all;
|
||||||
|
int i_cast;
|
||||||
|
|
||||||
|
// Check runtime
|
||||||
|
always @ (posedge clk) begin
|
||||||
|
cyc <= cyc + 1;
|
||||||
|
if (cyc == 0) begin
|
||||||
|
// Setup
|
||||||
|
e <= E01;
|
||||||
|
end
|
||||||
|
else if (cyc == 1) begin
|
||||||
|
`checks(e.name, "E01");
|
||||||
|
`checkh(e.next, ELARGE);
|
||||||
|
e <= ELARGE;
|
||||||
|
end
|
||||||
|
else if (cyc == 3) begin
|
||||||
|
`checks(e.name, "ELARGE");
|
||||||
|
`checkh(e.next, E01);
|
||||||
|
`checkh(e.prev, E01);
|
||||||
|
e <= E01;
|
||||||
|
end
|
||||||
|
//
|
||||||
|
else if (cyc == 10) begin
|
||||||
|
i_cast <= $cast(e, 60'h1234);
|
||||||
|
end
|
||||||
|
else if (cyc == 11) begin
|
||||||
|
`checkh(i_cast, 0);
|
||||||
|
i_cast <= $cast(e, 60'h1);
|
||||||
|
end
|
||||||
|
else if (cyc == 12) begin
|
||||||
|
`checkh(i_cast, 1);
|
||||||
|
i_cast <= $cast(e, 60'h1234_4567_abcd);
|
||||||
|
end
|
||||||
|
else if (cyc == 13) begin
|
||||||
|
`checkh(i_cast, 1);
|
||||||
|
end
|
||||||
|
//
|
||||||
|
else if (cyc == 20) begin
|
||||||
|
e <= 'h11; // Unknown
|
||||||
|
end
|
||||||
|
else if (cyc == 21) begin
|
||||||
|
`checks(e.name, ""); // Unknown
|
||||||
|
end
|
||||||
|
else if (cyc == 99) begin
|
||||||
|
$write("*-* All Finished *-*\n");
|
||||||
|
$finish;
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
endmodule
|
@ -1,7 +1,7 @@
|
|||||||
%Error: t/t_enum_huge_methods_bad.v:15:11: Value too wide for 64-bits expected in this context 160'h12344567abcd12344567abcd
|
%Error: t/t_enum_huge_methods_bad.v:15:11: Value too wide for 64-bits expected in this context 160'h12344567abcd12344567abcd
|
||||||
15 | ELARGE = 160'h1234_4567_abcd_1234_4567_abcd
|
15 | ELARGE = 160'h1234_4567_abcd_1234_4567_abcd
|
||||||
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
%Error-UNSUPPORTED: t/t_enum_huge_methods_bad.v:30:14: Unsupported: enum next/prev method on enum with > 10 bits
|
%Error-UNSUPPORTED: t/t_enum_huge_methods_bad.v:30:14: Unsupported: enum next/prev/name method on enum with > 64 bits
|
||||||
: ... In instance t
|
: ... In instance t
|
||||||
30 | $display(e.name);
|
30 | $display(e.name);
|
||||||
| ^~~~
|
| ^~~~
|
||||||
|
Loading…
Reference in New Issue
Block a user