Skip to content

Commit 60a7211

Browse files
allightcopybara-github
authored andcommitted
Recognise OR(X, Not(X)) and Add(X, Not(X)) in arith_simp
These can't be recognised by Ternary or Range due to losing the connection between the bits. The BDD can see this but we don't run that very often. PiperOrigin-RevId: 597983344
1 parent bbc9d9e commit 60a7211

File tree

3 files changed

+70
-1
lines changed

3 files changed

+70
-1
lines changed

xls/passes/BUILD

+1
Original file line numberDiff line numberDiff line change
@@ -153,6 +153,7 @@ cc_test(
153153
"//xls/ir:op",
154154
"//xls/ir:value",
155155
"//xls/solvers:z3_ir_equivalence",
156+
"//xls/solvers:z3_ir_equivalence_testutils",
156157
],
157158
)
158159

xls/passes/arith_simplification_pass.cc

+26
Original file line numberDiff line numberDiff line change
@@ -705,6 +705,32 @@ absl::StatusOr<bool> MatchArithPatterns(int64_t opt_level, Node* n) {
705705
return true;
706706
}
707707

708+
// Or(X, Not(X), ...) => 1...
709+
//
710+
// Note that this won't be found through the ternary query engine because
711+
// conservatively it determines `not(UNKNOWN) = UNKNOWN`.
712+
if (n->op() == Op::kOr && has_inverted_operand()) {
713+
XLS_VLOG(2) << "FOUND: replace Or(x, not(x)) with 1...";
714+
XLS_RETURN_IF_ERROR(
715+
n->ReplaceUsesWithNew<Literal>(Value(Bits::AllOnes(n->BitCountOrDie())))
716+
.status());
717+
return true;
718+
}
719+
720+
// Add(X, Not(X)) => 1...
721+
// Add(Not(X), X) => 1...
722+
//
723+
// Note that this won't be found through the ternary query engine because
724+
// conservatively it determines `not(UNKNOWN) = UNKNOWN`.
725+
if (n->op() == Op::kAdd && n->operand_count() == 2 &&
726+
has_inverted_operand()) {
727+
XLS_VLOG(2) << "FOUND: replace Add(x, not(x)) with 1...";
728+
XLS_RETURN_IF_ERROR(
729+
n->ReplaceUsesWithNew<Literal>(Value(Bits::AllOnes(n->BitCountOrDie())))
730+
.status());
731+
return true;
732+
}
733+
708734
// Xor(x, -1) => Not(x)
709735
if (n->op() == Op::kXor && n->operand_count() == 2 &&
710736
IsLiteralAllOnes(n->operand(1))) {

xls/passes/arith_simplification_pass_test.cc

+43-1
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,7 @@
3939
#include "xls/passes/optimization_pass.h"
4040
#include "xls/passes/pass_base.h"
4141
#include "xls/solvers/z3_ir_equivalence.h"
42+
#include "xls/solvers/z3_ir_equivalence_testutils.h"
4243

4344
namespace m = ::xls::op_matchers;
4445

@@ -48,6 +49,7 @@ namespace {
4849
constexpr absl::Duration kProverTimeout = absl::Seconds(10);
4950

5051
using status_testing::IsOkAndHolds;
52+
using ::xls::solvers::z3::ScopedVerifyEquivalence;
5153
using ::xls::solvers::z3::TryProveEquivalence;
5254

5355
using ::testing::AllOf;
@@ -329,7 +331,7 @@ TEST_F(ArithSimplificationPassTest, SubFallsToZero) {
329331
FunctionBuilder fb(TestName(), p.get());
330332
auto param = fb.Param("x", p->GetBitsType(32));
331333
fb.Subtract(param, param);
332-
XLS_ASSERT_OK_AND_ASSIGN(Function* f, fb.Build());
334+
XLS_ASSERT_OK_AND_ASSIGN(Function * f, fb.Build());
333335
ASSERT_THAT(Run(p.get()), IsOkAndHolds(true));
334336
EXPECT_THAT(f->return_value(), m::Literal(0));
335337
}
@@ -2177,5 +2179,45 @@ TEST_F(ArithSimplificationPassTest, ShiftLeftOfPowerOfTwo) {
21772179
IsOkAndHolds(true));
21782180
}
21792181

2182+
TEST_F(ArithSimplificationPassTest, AddInverses) {
2183+
auto p = CreatePackage();
2184+
FunctionBuilder fb(TestName(), p.get());
2185+
BValue p1 = fb.Param("p1", p->GetBitsType(32));
2186+
fb.Tuple({
2187+
fb.Add(p1, fb.Not(p1)),
2188+
fb.Add(fb.Not(p1), p1),
2189+
});
2190+
XLS_ASSERT_OK_AND_ASSIGN(Function * f, fb.Build());
2191+
2192+
ScopedVerifyEquivalence sve(f);
2193+
ASSERT_THAT(Run(p.get()), IsOkAndHolds(true));
2194+
2195+
EXPECT_THAT(
2196+
f->return_value(),
2197+
m::Tuple(m::Literal(Bits::AllOnes(32)), m::Literal(Bits::AllOnes(32))));
2198+
}
2199+
2200+
TEST_F(ArithSimplificationPassTest, OrInverses) {
2201+
auto p = CreatePackage();
2202+
FunctionBuilder fb(TestName(), p.get());
2203+
BValue p1 = fb.Param("p1", p->GetBitsType(32));
2204+
BValue p2 = fb.Param("p2", p->GetBitsType(32));
2205+
fb.Tuple({
2206+
fb.Or(p1, fb.Not(p1)),
2207+
fb.Or(fb.Not(p1), p1),
2208+
fb.Or({p1, p2, fb.Not(p1)}),
2209+
fb.Or({p1, p2, fb.Not(p2)}),
2210+
});
2211+
XLS_ASSERT_OK_AND_ASSIGN(Function * f, fb.Build());
2212+
2213+
ScopedVerifyEquivalence sve(f);
2214+
ASSERT_THAT(Run(p.get()), IsOkAndHolds(true));
2215+
2216+
EXPECT_THAT(
2217+
f->return_value(),
2218+
m::Tuple(m::Literal(Bits::AllOnes(32)), m::Literal(Bits::AllOnes(32)),
2219+
m::Literal(Bits::AllOnes(32)), m::Literal(Bits::AllOnes(32))));
2220+
}
2221+
21802222
} // namespace
21812223
} // namespace xls

0 commit comments

Comments
 (0)