forked from github/verilator
* Tests: Add a test to reproduce #3470 * Update LSB during return path of traversal. No functional change is intended. * Introduce LeafInfo::m_msb * Update LeafInfo::m_msb when visitin AstCCast * Internals: Add comment, reorder. No functional change is intended. * Delete explicit from copy constructor to fix build error. * Update Changes * Internals: Remove unused parameter. No functional change is intended. * Tests: Add explanation to t_const_opt.
This commit is contained in:
parent
0de1bbc85b
commit
9f37cef1bb
2
Changes
2
Changes
@ -13,7 +13,7 @@ Verilator 4.225 devel
|
|||||||
|
|
||||||
**Minor:**
|
**Minor:**
|
||||||
|
|
||||||
|
* Fix incorrect bit op tree optimization (#3470). [algrobman]
|
||||||
|
|
||||||
Verilator 4.224 2022-06-19
|
Verilator 4.224 2022-06-19
|
||||||
==========================
|
==========================
|
||||||
|
@ -80,30 +80,48 @@ class ConstBitOpTreeVisitor final : public VNVisitor {
|
|||||||
using ResultTerm = std::tuple<AstNode*, unsigned, bool>;
|
using ResultTerm = std::tuple<AstNode*, unsigned, bool>;
|
||||||
|
|
||||||
class LeafInfo final { // Leaf node (either AstConst or AstVarRef)
|
class LeafInfo final { // Leaf node (either AstConst or AstVarRef)
|
||||||
|
// MEMBERS
|
||||||
bool m_polarity = true;
|
bool m_polarity = true;
|
||||||
int m_lsb = 0;
|
int m_lsb = 0; // LSB of actually used bit of m_refp->varp()
|
||||||
|
int m_msb = 0; // MSB of actually used bit of m_refp->varp()
|
||||||
int m_wordIdx = -1; // -1 means AstWordSel is not used.
|
int m_wordIdx = -1; // -1 means AstWordSel is not used.
|
||||||
AstVarRef* m_refp = nullptr;
|
AstVarRef* m_refp = nullptr;
|
||||||
const AstConst* m_constp = nullptr;
|
const AstConst* m_constp = nullptr;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
// CONSTRUCTORS
|
||||||
|
LeafInfo() = default;
|
||||||
|
LeafInfo(const LeafInfo& other) = default;
|
||||||
|
explicit LeafInfo(int lsb)
|
||||||
|
: m_lsb{lsb} {}
|
||||||
|
|
||||||
|
// METHODS
|
||||||
void setLeaf(AstVarRef* refp) {
|
void setLeaf(AstVarRef* refp) {
|
||||||
UASSERT(!m_refp && !m_constp, "Must be called just once");
|
UASSERT(!m_refp && !m_constp, "Must be called just once");
|
||||||
m_refp = refp;
|
m_refp = refp;
|
||||||
|
m_msb = refp->varp()->widthMin() - 1;
|
||||||
}
|
}
|
||||||
void setLeaf(const AstConst* constp) {
|
void setLeaf(const AstConst* constp) {
|
||||||
UASSERT(!m_refp && !m_constp, "Must be called just once");
|
UASSERT(!m_refp && !m_constp, "Must be called just once");
|
||||||
m_constp = constp;
|
m_constp = constp;
|
||||||
|
m_msb = constp->widthMin() - 1;
|
||||||
}
|
}
|
||||||
|
void updateBitRange(const AstCCast* castp) {
|
||||||
|
m_msb = std::min(m_msb, m_lsb + castp->width() - 1);
|
||||||
|
}
|
||||||
|
void updateBitRange(const AstShiftR* shiftp) {
|
||||||
|
m_lsb += VN_AS(shiftp->rhsp(), Const)->toUInt();
|
||||||
|
}
|
||||||
|
void wordIdx(int i) { m_wordIdx = i; }
|
||||||
|
void polarity(bool p) { m_polarity = p; }
|
||||||
|
|
||||||
AstVarRef* refp() const { return m_refp; }
|
AstVarRef* refp() const { return m_refp; }
|
||||||
const AstConst* constp() const { return m_constp; }
|
const AstConst* constp() const { return m_constp; }
|
||||||
int wordIdx() const { return m_wordIdx; }
|
int wordIdx() const { return m_wordIdx; }
|
||||||
bool polarity() const { return m_polarity; }
|
bool polarity() const { return m_polarity; }
|
||||||
int lsb() const { return m_lsb; }
|
int lsb() const { return m_lsb; }
|
||||||
|
|
||||||
void wordIdx(int i) { m_wordIdx = i; }
|
int msb() const { return std::min(m_msb, varWidth() - 1); }
|
||||||
void lsb(int l) { m_lsb = l; }
|
|
||||||
void polarity(bool p) { m_polarity = p; }
|
|
||||||
int varWidth() const {
|
int varWidth() const {
|
||||||
UASSERT(m_refp, "m_refp should be set");
|
UASSERT(m_refp, "m_refp should be set");
|
||||||
const int width = m_refp->varp()->widthMin();
|
const int width = m_refp->varp()->widthMin();
|
||||||
@ -382,7 +400,7 @@ class ConstBitOpTreeVisitor final : public VNVisitor {
|
|||||||
|
|
||||||
// Traverse down to see AstConst or AstVarRef
|
// Traverse down to see AstConst or AstVarRef
|
||||||
LeafInfo findLeaf(AstNode* nodep, bool expectConst) {
|
LeafInfo findLeaf(AstNode* nodep, bool expectConst) {
|
||||||
LeafInfo info;
|
LeafInfo info{m_lsb};
|
||||||
{
|
{
|
||||||
VL_RESTORER(m_leafp);
|
VL_RESTORER(m_leafp);
|
||||||
m_leafp = &info;
|
m_leafp = &info;
|
||||||
@ -402,7 +420,10 @@ class ConstBitOpTreeVisitor final : public VNVisitor {
|
|||||||
virtual void visit(AstNode* nodep) override {
|
virtual void visit(AstNode* nodep) override {
|
||||||
CONST_BITOP_SET_FAILED("Hit unexpected op", nodep);
|
CONST_BITOP_SET_FAILED("Hit unexpected op", nodep);
|
||||||
}
|
}
|
||||||
virtual void visit(AstCCast* nodep) override { iterateChildren(nodep); }
|
virtual void visit(AstCCast* nodep) override {
|
||||||
|
iterateChildren(nodep);
|
||||||
|
if (m_leafp) m_leafp->updateBitRange(nodep);
|
||||||
|
}
|
||||||
virtual void visit(AstShiftR* nodep) override {
|
virtual void visit(AstShiftR* nodep) override {
|
||||||
CONST_BITOP_RETURN_IF(!m_leafp, nodep);
|
CONST_BITOP_RETURN_IF(!m_leafp, nodep);
|
||||||
AstConst* const constp = VN_CAST(nodep->rhsp(), Const);
|
AstConst* const constp = VN_CAST(nodep->rhsp(), Const);
|
||||||
@ -410,12 +431,14 @@ class ConstBitOpTreeVisitor final : public VNVisitor {
|
|||||||
m_lsb += constp->toUInt();
|
m_lsb += constp->toUInt();
|
||||||
incrOps(nodep, __LINE__);
|
incrOps(nodep, __LINE__);
|
||||||
iterate(nodep->lhsp());
|
iterate(nodep->lhsp());
|
||||||
|
m_leafp->updateBitRange(nodep);
|
||||||
m_lsb -= constp->toUInt();
|
m_lsb -= constp->toUInt();
|
||||||
}
|
}
|
||||||
virtual void visit(AstNot* nodep) override {
|
virtual void visit(AstNot* nodep) override {
|
||||||
CONST_BITOP_RETURN_IF(nodep->widthMin() != 1, nodep);
|
CONST_BITOP_RETURN_IF(nodep->widthMin() != 1, nodep);
|
||||||
AstNode* lhsp = nodep->lhsp();
|
AstNode* lhsp = nodep->lhsp();
|
||||||
if (AstCCast* const castp = VN_CAST(lhsp, CCast)) lhsp = castp->lhsp();
|
AstCCast* const castp = VN_CAST(lhsp, CCast);
|
||||||
|
if (castp) lhsp = castp->lhsp();
|
||||||
CONST_BITOP_RETURN_IF(!VN_IS(lhsp, VarRef) && !VN_IS(lhsp, Xor) && !VN_IS(lhsp, RedXor)
|
CONST_BITOP_RETURN_IF(!VN_IS(lhsp, VarRef) && !VN_IS(lhsp, Xor) && !VN_IS(lhsp, RedXor)
|
||||||
&& !VN_IS(lhsp, ShiftR),
|
&& !VN_IS(lhsp, ShiftR),
|
||||||
lhsp);
|
lhsp);
|
||||||
@ -424,6 +447,7 @@ class ConstBitOpTreeVisitor final : public VNVisitor {
|
|||||||
iterateChildren(nodep);
|
iterateChildren(nodep);
|
||||||
// Don't restore m_polarity for Xor as it counts parity of the entire tree
|
// Don't restore m_polarity for Xor as it counts parity of the entire tree
|
||||||
if (!isXorTree()) m_polarity = !m_polarity;
|
if (!isXorTree()) m_polarity = !m_polarity;
|
||||||
|
if (m_leafp && castp) m_leafp->updateBitRange(castp);
|
||||||
}
|
}
|
||||||
virtual void visit(AstWordSel* nodep) override {
|
virtual void visit(AstWordSel* nodep) override {
|
||||||
CONST_BITOP_RETURN_IF(!m_leafp, nodep);
|
CONST_BITOP_RETURN_IF(!m_leafp, nodep);
|
||||||
@ -437,27 +461,27 @@ class ConstBitOpTreeVisitor final : public VNVisitor {
|
|||||||
CONST_BITOP_RETURN_IF(!m_leafp, nodep);
|
CONST_BITOP_RETURN_IF(!m_leafp, nodep);
|
||||||
m_leafp->setLeaf(nodep);
|
m_leafp->setLeaf(nodep);
|
||||||
m_leafp->polarity(m_polarity);
|
m_leafp->polarity(m_polarity);
|
||||||
m_leafp->lsb(m_lsb);
|
|
||||||
}
|
}
|
||||||
virtual void visit(AstConst* nodep) override {
|
virtual void visit(AstConst* nodep) override {
|
||||||
CONST_BITOP_RETURN_IF(!m_leafp, nodep);
|
CONST_BITOP_RETURN_IF(!m_leafp, nodep);
|
||||||
m_leafp->setLeaf(nodep);
|
m_leafp->setLeaf(nodep);
|
||||||
m_leafp->lsb(m_lsb);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void visit(AstRedXor* nodep) override {
|
virtual void visit(AstRedXor* nodep) override {
|
||||||
Restorer restorer{*this};
|
Restorer restorer{*this};
|
||||||
CONST_BITOP_RETURN_IF(!VN_IS(m_rootp, Xor), nodep);
|
CONST_BITOP_RETURN_IF(!VN_IS(m_rootp, Xor), nodep);
|
||||||
AstNode* lhsp = nodep->lhsp();
|
AstNode* lhsp = nodep->lhsp();
|
||||||
if (const AstCCast* const castp = VN_CAST(lhsp, CCast)) lhsp = castp->lhsp();
|
const AstCCast* const castp = VN_CAST(lhsp, CCast);
|
||||||
|
if (castp) lhsp = castp->lhsp();
|
||||||
if (const AstAnd* const andp = VN_CAST(lhsp, And)) { // '^(mask & leaf)'
|
if (const AstAnd* const andp = VN_CAST(lhsp, And)) { // '^(mask & leaf)'
|
||||||
CONST_BITOP_RETURN_IF(!andp, lhsp);
|
CONST_BITOP_RETURN_IF(!andp, lhsp);
|
||||||
|
|
||||||
const LeafInfo& mask = findLeaf(andp->lhsp(), true);
|
const LeafInfo& mask = findLeaf(andp->lhsp(), true);
|
||||||
CONST_BITOP_RETURN_IF(!mask.constp() || mask.lsb() != 0, andp->lhsp());
|
CONST_BITOP_RETURN_IF(!mask.constp() || mask.lsb() != 0, andp->lhsp());
|
||||||
|
|
||||||
const LeafInfo& ref = findLeaf(andp->rhsp(), false);
|
LeafInfo ref = findLeaf(andp->rhsp(), false);
|
||||||
CONST_BITOP_RETURN_IF(!ref.refp(), andp->rhsp());
|
CONST_BITOP_RETURN_IF(!ref.refp(), andp->rhsp());
|
||||||
|
if (castp) ref.updateBitRange(castp);
|
||||||
|
|
||||||
restorer.disableRestore(); // Now all subtree succeeded
|
restorer.disableRestore(); // Now all subtree succeeded
|
||||||
|
|
||||||
@ -467,7 +491,7 @@ class ConstBitOpTreeVisitor final : public VNVisitor {
|
|||||||
incrOps(andp, __LINE__);
|
incrOps(andp, __LINE__);
|
||||||
|
|
||||||
// Mark all bits checked in this reduction
|
// Mark all bits checked in this reduction
|
||||||
const int maxBitIdx = std::min(ref.lsb() + maskNum.width(), ref.varWidth());
|
const int maxBitIdx = std::min(ref.lsb() + maskNum.width(), ref.msb() + 1);
|
||||||
for (int bitIdx = ref.lsb(); bitIdx < maxBitIdx; ++bitIdx) {
|
for (int bitIdx = ref.lsb(); bitIdx < maxBitIdx; ++bitIdx) {
|
||||||
const int maskIdx = bitIdx - ref.lsb();
|
const int maskIdx = bitIdx - ref.lsb();
|
||||||
if (maskNum.bitIs0(maskIdx)) continue;
|
if (maskNum.bitIs0(maskIdx)) continue;
|
||||||
@ -475,15 +499,16 @@ class ConstBitOpTreeVisitor final : public VNVisitor {
|
|||||||
m_bitPolarities.emplace_back(ref, true, bitIdx);
|
m_bitPolarities.emplace_back(ref, true, bitIdx);
|
||||||
}
|
}
|
||||||
} else { // '^leaf'
|
} else { // '^leaf'
|
||||||
const LeafInfo& ref = findLeaf(lhsp, false);
|
LeafInfo ref = findLeaf(lhsp, false);
|
||||||
CONST_BITOP_RETURN_IF(!ref.refp(), lhsp);
|
CONST_BITOP_RETURN_IF(!ref.refp(), lhsp);
|
||||||
|
if (castp) ref.updateBitRange(castp);
|
||||||
|
|
||||||
restorer.disableRestore(); // Now all checks passed
|
restorer.disableRestore(); // Now all checks passed
|
||||||
|
|
||||||
incrOps(nodep, __LINE__);
|
incrOps(nodep, __LINE__);
|
||||||
|
|
||||||
// Mark all bits checked by this comparison
|
// Mark all bits checked by this comparison
|
||||||
for (int bitIdx = ref.lsb(); bitIdx < ref.varWidth(); ++bitIdx) {
|
for (int bitIdx = ref.lsb(); bitIdx <= ref.msb(); ++bitIdx) {
|
||||||
m_bitPolarities.emplace_back(ref, true, bitIdx);
|
m_bitPolarities.emplace_back(ref, true, bitIdx);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -503,7 +528,7 @@ class ConstBitOpTreeVisitor final : public VNVisitor {
|
|||||||
|
|
||||||
for (const bool right : {false, true}) {
|
for (const bool right : {false, true}) {
|
||||||
Restorer restorer{*this};
|
Restorer restorer{*this};
|
||||||
LeafInfo leafInfo;
|
LeafInfo leafInfo{m_lsb};
|
||||||
m_leafp = &leafInfo;
|
m_leafp = &leafInfo;
|
||||||
AstNode* opp = right ? nodep->rhsp() : nodep->lhsp();
|
AstNode* opp = right ? nodep->rhsp() : nodep->lhsp();
|
||||||
const bool origFailed = m_failed;
|
const bool origFailed = m_failed;
|
||||||
@ -522,7 +547,7 @@ class ConstBitOpTreeVisitor final : public VNVisitor {
|
|||||||
// The conditional on the lsb being in range is necessary for some degenerate
|
// The conditional on the lsb being in range is necessary for some degenerate
|
||||||
// case, e.g.: (IData)((QData)wide[0] >> 32), or <1-bit-var> >> 1, which is
|
// case, e.g.: (IData)((QData)wide[0] >> 32), or <1-bit-var> >> 1, which is
|
||||||
// just zero
|
// just zero
|
||||||
if (leafInfo.lsb() < leafInfo.varWidth()) {
|
if (leafInfo.lsb() <= leafInfo.msb()) {
|
||||||
m_bitPolarities.emplace_back(leafInfo, isXorTree() || leafInfo.polarity(),
|
m_bitPolarities.emplace_back(leafInfo, isXorTree() || leafInfo.polarity(),
|
||||||
leafInfo.lsb());
|
leafInfo.lsb());
|
||||||
} else if (isAndTree() && leafInfo.polarity()) {
|
} else if (isAndTree() && leafInfo.polarity()) {
|
||||||
@ -559,7 +584,7 @@ class ConstBitOpTreeVisitor final : public VNVisitor {
|
|||||||
incrOps(andp, __LINE__);
|
incrOps(andp, __LINE__);
|
||||||
|
|
||||||
// Mark all bits checked by this comparison
|
// Mark all bits checked by this comparison
|
||||||
const int maxBitIdx = std::min(ref.lsb() + maskNum.width(), ref.varWidth());
|
const int maxBitIdx = std::min(ref.lsb() + maskNum.width(), ref.msb() + 1);
|
||||||
for (int bitIdx = ref.lsb(); bitIdx < maxBitIdx; ++bitIdx) {
|
for (int bitIdx = ref.lsb(); bitIdx < maxBitIdx; ++bitIdx) {
|
||||||
const int maskIdx = bitIdx - ref.lsb();
|
const int maskIdx = bitIdx - ref.lsb();
|
||||||
if (maskNum.bitIs0(maskIdx)) continue;
|
if (maskNum.bitIs0(maskIdx)) continue;
|
||||||
@ -575,7 +600,7 @@ class ConstBitOpTreeVisitor final : public VNVisitor {
|
|||||||
incrOps(nodep, __LINE__);
|
incrOps(nodep, __LINE__);
|
||||||
|
|
||||||
// Mark all bits checked by this comparison
|
// Mark all bits checked by this comparison
|
||||||
const int maxBitIdx = std::min(ref.lsb() + compNum.width(), ref.varWidth());
|
const int maxBitIdx = std::min(ref.lsb() + compNum.width(), ref.msb() + 1);
|
||||||
for (int bitIdx = ref.lsb(); bitIdx < maxBitIdx; ++bitIdx) {
|
for (int bitIdx = ref.lsb(); bitIdx < maxBitIdx; ++bitIdx) {
|
||||||
const int maskIdx = bitIdx - ref.lsb();
|
const int maskIdx = bitIdx - ref.lsb();
|
||||||
const bool polarity = compNum.bitIs1(maskIdx) != maskFlip;
|
const bool polarity = compNum.bitIs1(maskIdx) != maskFlip;
|
||||||
|
@ -19,7 +19,7 @@ execute(
|
|||||||
);
|
);
|
||||||
|
|
||||||
if ($Self->{vlt}) {
|
if ($Self->{vlt}) {
|
||||||
file_grep($Self->{stats}, qr/Optimizations, Const bit op reduction\s+(\d+)/i, 11);
|
file_grep($Self->{stats}, qr/Optimizations, Const bit op reduction\s+(\d+)/i, 14);
|
||||||
}
|
}
|
||||||
ok(1);
|
ok(1);
|
||||||
1;
|
1;
|
||||||
|
@ -62,7 +62,7 @@ module t(/*AUTOARG*/
|
|||||||
$write("[%0t] cyc==%0d crc=%x sum=%x\n", $time, cyc, crc, sum);
|
$write("[%0t] cyc==%0d crc=%x sum=%x\n", $time, cyc, crc, sum);
|
||||||
if (crc !== 64'hc77bb9b3784ea091) $stop;
|
if (crc !== 64'hc77bb9b3784ea091) $stop;
|
||||||
// What checksum will we end up with (above print should match)
|
// What checksum will we end up with (above print should match)
|
||||||
`define EXPECTED_SUM 64'hdccb9e7b8b638233
|
`define EXPECTED_SUM 64'hde21e019a3e12039
|
||||||
|
|
||||||
if (sum !== `EXPECTED_SUM) $stop;
|
if (sum !== `EXPECTED_SUM) $stop;
|
||||||
$write("*-* All Finished *-*\n");
|
$write("*-* All Finished *-*\n");
|
||||||
@ -86,10 +86,11 @@ module Test(/*AUTOARG*/
|
|||||||
logic bug3182_out;
|
logic bug3182_out;
|
||||||
logic bug3197_out;
|
logic bug3197_out;
|
||||||
logic bug3445_out;
|
logic bug3445_out;
|
||||||
|
logic bug3470_out;
|
||||||
|
|
||||||
output logic o;
|
output logic o;
|
||||||
|
|
||||||
logic [7:0] tmp;
|
logic [8:0] tmp;
|
||||||
assign o = ^tmp;
|
assign o = ^tmp;
|
||||||
|
|
||||||
always_ff @(posedge clk) begin
|
always_ff @(posedge clk) begin
|
||||||
@ -113,11 +114,13 @@ module Test(/*AUTOARG*/
|
|||||||
tmp[5] <= bug3182_out;
|
tmp[5] <= bug3182_out;
|
||||||
tmp[6] <= bug3197_out;
|
tmp[6] <= bug3197_out;
|
||||||
tmp[7] <= bug3445_out;
|
tmp[7] <= bug3445_out;
|
||||||
|
tmp[8] <= bug3470_out;
|
||||||
end
|
end
|
||||||
|
|
||||||
bug3182 i_bug3182(.in(d[4:0]), .out(bug3182_out));
|
bug3182 i_bug3182(.in(d[4:0]), .out(bug3182_out));
|
||||||
bug3197 i_bug3197(.clk(clk), .in(d), .out(bug3197_out));
|
bug3197 i_bug3197(.clk(clk), .in(d), .out(bug3197_out));
|
||||||
bug3445 i_bug3445(.clk(clk), .in(d), .out(bug3445_out));
|
bug3445 i_bug3445(.clk(clk), .in(d), .out(bug3445_out));
|
||||||
|
bug3470 i_bug3470(.clk(clk), .in(d), .out(bug3470_out));
|
||||||
|
|
||||||
endmodule
|
endmodule
|
||||||
|
|
||||||
@ -203,3 +206,32 @@ module bug3445(input wire clk, input wire [31:0] in, output wire out);
|
|||||||
|
|
||||||
assign out = result0 ^ result1 ^ (result2 | result3);
|
assign out = result0 ^ result1 ^ (result2 | result3);
|
||||||
endmodule
|
endmodule
|
||||||
|
|
||||||
|
// Bug3470
|
||||||
|
// CCast had been ignored in bit op tree optimization
|
||||||
|
// Assume the following HDL input:
|
||||||
|
// (^d[38:32]) ^ (^d[31:0])
|
||||||
|
// where d is logic [38:0]
|
||||||
|
// ^d[31:0] becomes REDXOR(CCast(uint32_t, d)),
|
||||||
|
// but CCast was ignored and interpreted as ^d[38:0].
|
||||||
|
// Finally (^d[38:32]) ^ (^d31:0]) was wrongly transformed to
|
||||||
|
// (^d[38:32]) ^ (^d[38:0])
|
||||||
|
// -> (^d[38:32]) ^ ((^d[38:32]) ^ (^d[31:0]))
|
||||||
|
// -> ^d[31:0]
|
||||||
|
// Of course the correct result is ^d[38:0] = ^d
|
||||||
|
module bug3470(input wire clk, input wire [31:0] in, output wire out);
|
||||||
|
logic [38:0] d;
|
||||||
|
always_ff @(posedge clk)
|
||||||
|
d <= {d[6:0], in};
|
||||||
|
|
||||||
|
logic tmp, expected;
|
||||||
|
always_ff @(posedge clk) begin
|
||||||
|
tmp <= ^(d >> 32) ^ (^d[31:0]);
|
||||||
|
expected <= ^d;
|
||||||
|
end
|
||||||
|
|
||||||
|
always @(posedge clk)
|
||||||
|
if (tmp != expected) $stop;
|
||||||
|
|
||||||
|
assign out = tmp;
|
||||||
|
endmodule
|
||||||
|
Loading…
Reference in New Issue
Block a user