mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2025-01-31 20:51:52 +01:00
[InstSimplify] use ConstantRange to simplify or-of-icmps
We can simplify (or (icmp X, C1), (icmp X, C2)) to 'true' or one of the icmps in many cases. I had to check some of these with Alive to prove to myself it's right, but everything seems to check out. Eg, the deleted code in instcombine was completely ignoring predicates with mismatched signedness. This is a follow-up to: https://reviews.llvm.org/rL301260 https://reviews.llvm.org/D32143 llvm-svn: 302370
This commit is contained in:
parent
22df982bb3
commit
41b5979dcf
@ -1519,6 +1519,46 @@ static Value *simplifyOrOfICmpsWithSameOperands(ICmpInst *Op0, ICmpInst *Op1) {
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
/// Test if a pair of compares with a shared operand and 2 constants has an
|
||||
/// empty set intersection, full set union, or if one compare is a superset of
|
||||
/// the other.
|
||||
static Value *simplifyAndOrOfICmpsWithConstants(ICmpInst *Cmp0, ICmpInst *Cmp1,
|
||||
bool IsAnd) {
|
||||
// Look for this pattern: {and/or} (icmp X, C0), (icmp X, C1)).
|
||||
if (Cmp0->getOperand(0) != Cmp1->getOperand(0))
|
||||
return nullptr;
|
||||
|
||||
const APInt *C0, *C1;
|
||||
if (!match(Cmp0->getOperand(1), m_APInt(C0)) ||
|
||||
!match(Cmp1->getOperand(1), m_APInt(C1)))
|
||||
return nullptr;
|
||||
|
||||
auto Range0 = ConstantRange::makeExactICmpRegion(Cmp0->getPredicate(), *C0);
|
||||
auto Range1 = ConstantRange::makeExactICmpRegion(Cmp1->getPredicate(), *C1);
|
||||
|
||||
// For and-of-comapares, check if the intersection is empty:
|
||||
// (icmp X, C0) && (icmp X, C1) --> empty set --> false
|
||||
if (IsAnd && Range0.intersectWith(Range1).isEmptySet())
|
||||
return getFalse(Cmp0->getType());
|
||||
|
||||
// For or-of-compares, check if the union is full:
|
||||
// (icmp X, C0) || (icmp X, C1) --> full set --> true
|
||||
if (!IsAnd && Range0.unionWith(Range1).isFullSet())
|
||||
return getTrue(Cmp0->getType());
|
||||
|
||||
// Is one range a superset of the other?
|
||||
// If this is and-of-compares, take the smaller set:
|
||||
// (icmp sgt X, 4) && (icmp sgt X, 42) --> icmp sgt X, 42
|
||||
// If this is or-of-compares, take the larger set:
|
||||
// (icmp sgt X, 4) || (icmp sgt X, 42) --> icmp sgt X, 4
|
||||
if (Range0.contains(Range1))
|
||||
return IsAnd ? Cmp1 : Cmp0;
|
||||
if (Range1.contains(Range0))
|
||||
return IsAnd ? Cmp0 : Cmp1;
|
||||
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
/// Commuted variants are assumed to be handled by calling this function again
|
||||
/// with the parameters swapped.
|
||||
static Value *simplifyAndOfICmps(ICmpInst *Op0, ICmpInst *Op1) {
|
||||
@ -1528,29 +1568,14 @@ static Value *simplifyAndOfICmps(ICmpInst *Op0, ICmpInst *Op1) {
|
||||
if (Value *X = simplifyAndOfICmpsWithSameOperands(Op0, Op1))
|
||||
return X;
|
||||
|
||||
// FIXME: This should be shared with or-of-icmps.
|
||||
// Look for this pattern: (icmp V, C0) & (icmp V, C1)).
|
||||
if (Value *X = simplifyAndOrOfICmpsWithConstants(Op0, Op1, true))
|
||||
return X;
|
||||
|
||||
// (icmp (add V, C0), C1) & (icmp V, C0)
|
||||
Type *ITy = Op0->getType();
|
||||
ICmpInst::Predicate Pred0, Pred1;
|
||||
const APInt *C0, *C1;
|
||||
Value *V;
|
||||
if (match(Op0, m_ICmp(Pred0, m_Value(V), m_APInt(C0))) &&
|
||||
match(Op1, m_ICmp(Pred1, m_Specific(V), m_APInt(C1)))) {
|
||||
// Make a constant range that's the intersection of the two icmp ranges.
|
||||
// If the intersection is empty, we know that the result is false.
|
||||
auto Range0 = ConstantRange::makeExactICmpRegion(Pred0, *C0);
|
||||
auto Range1 = ConstantRange::makeExactICmpRegion(Pred1, *C1);
|
||||
if (Range0.intersectWith(Range1).isEmptySet())
|
||||
return getFalse(ITy);
|
||||
|
||||
// If a range is a superset of the other, the smaller set is all we need.
|
||||
if (Range0.contains(Range1))
|
||||
return Op1;
|
||||
if (Range1.contains(Range0))
|
||||
return Op0;
|
||||
}
|
||||
|
||||
// (icmp (add V, C0), C1) & (icmp V, C0)
|
||||
if (!match(Op0, m_ICmp(Pred0, m_Add(m_Value(V), m_APInt(C0)), m_APInt(C1))))
|
||||
return nullptr;
|
||||
|
||||
@ -1600,6 +1625,9 @@ static Value *simplifyOrOfICmps(ICmpInst *Op0, ICmpInst *Op1) {
|
||||
if (Value *X = simplifyOrOfICmpsWithSameOperands(Op0, Op1))
|
||||
return X;
|
||||
|
||||
if (Value *X = simplifyAndOrOfICmpsWithConstants(Op0, Op1, false))
|
||||
return X;
|
||||
|
||||
// (icmp (add V, C0), C1) | (icmp V, C0)
|
||||
ICmpInst::Predicate Pred0, Pred1;
|
||||
const APInt *C0, *C1;
|
||||
|
@ -1834,25 +1834,8 @@ Value *InstCombiner::FoldOrOfICmps(ICmpInst *LHS, ICmpInst *RHS,
|
||||
case ICmpInst::ICMP_UGT: // (X == 13 | X u> 14) -> no change
|
||||
case ICmpInst::ICMP_SGT: // (X == 13 | X s> 14) -> no change
|
||||
break;
|
||||
case ICmpInst::ICMP_NE: // (X == 13 | X != 15) -> X != 15
|
||||
case ICmpInst::ICMP_ULT: // (X == 13 | X u< 15) -> X u< 15
|
||||
case ICmpInst::ICMP_SLT: // (X == 13 | X s< 15) -> X s< 15
|
||||
return RHS;
|
||||
}
|
||||
break;
|
||||
case ICmpInst::ICMP_NE:
|
||||
switch (PredR) {
|
||||
default:
|
||||
llvm_unreachable("Unknown integer condition code!");
|
||||
case ICmpInst::ICMP_EQ: // (X != 13 | X == 15) -> X != 13
|
||||
case ICmpInst::ICMP_UGT: // (X != 13 | X u> 15) -> X != 13
|
||||
case ICmpInst::ICMP_SGT: // (X != 13 | X s> 15) -> X != 13
|
||||
return LHS;
|
||||
case ICmpInst::ICMP_NE: // (X != 13 | X != 15) -> true
|
||||
case ICmpInst::ICMP_ULT: // (X != 13 | X u< 15) -> true
|
||||
case ICmpInst::ICMP_SLT: // (X != 13 | X s< 15) -> true
|
||||
return Builder->getTrue();
|
||||
}
|
||||
case ICmpInst::ICMP_ULT:
|
||||
switch (PredR) {
|
||||
default:
|
||||
@ -1860,15 +1843,9 @@ Value *InstCombiner::FoldOrOfICmps(ICmpInst *LHS, ICmpInst *RHS,
|
||||
case ICmpInst::ICMP_EQ: // (X u< 13 | X == 14) -> no change
|
||||
break;
|
||||
case ICmpInst::ICMP_UGT: // (X u< 13 | X u> 15) -> (X-13) u> 2
|
||||
// If RHSC is [us]MAXINT, it is always false. Not handling
|
||||
// this can cause overflow.
|
||||
if (RHSC->isMaxValue(false))
|
||||
return LHS;
|
||||
assert(!RHSC->isMaxValue(false) && "Missed icmp simplification");
|
||||
return insertRangeTest(LHS0, LHSC->getValue(), RHSC->getValue() + 1,
|
||||
false, false);
|
||||
case ICmpInst::ICMP_NE: // (X u< 13 | X != 15) -> X != 15
|
||||
case ICmpInst::ICMP_ULT: // (X u< 13 | X u< 15) -> X u< 15
|
||||
return RHS;
|
||||
}
|
||||
break;
|
||||
case ICmpInst::ICMP_SLT:
|
||||
@ -1878,39 +1855,9 @@ Value *InstCombiner::FoldOrOfICmps(ICmpInst *LHS, ICmpInst *RHS,
|
||||
case ICmpInst::ICMP_EQ: // (X s< 13 | X == 14) -> no change
|
||||
break;
|
||||
case ICmpInst::ICMP_SGT: // (X s< 13 | X s> 15) -> (X-13) s> 2
|
||||
// If RHSC is [us]MAXINT, it is always false. Not handling
|
||||
// this can cause overflow.
|
||||
if (RHSC->isMaxValue(true))
|
||||
return LHS;
|
||||
assert(!RHSC->isMaxValue(true) && "Missed icmp simplification");
|
||||
return insertRangeTest(LHS0, LHSC->getValue(), RHSC->getValue() + 1, true,
|
||||
false);
|
||||
case ICmpInst::ICMP_NE: // (X s< 13 | X != 15) -> X != 15
|
||||
case ICmpInst::ICMP_SLT: // (X s< 13 | X s< 15) -> X s< 15
|
||||
return RHS;
|
||||
}
|
||||
break;
|
||||
case ICmpInst::ICMP_UGT:
|
||||
switch (PredR) {
|
||||
default:
|
||||
llvm_unreachable("Unknown integer condition code!");
|
||||
case ICmpInst::ICMP_EQ: // (X u> 13 | X == 15) -> X u> 13
|
||||
case ICmpInst::ICMP_UGT: // (X u> 13 | X u> 15) -> X u> 13
|
||||
return LHS;
|
||||
case ICmpInst::ICMP_NE: // (X u> 13 | X != 15) -> true
|
||||
case ICmpInst::ICMP_ULT: // (X u> 13 | X u< 15) -> true
|
||||
return Builder->getTrue();
|
||||
}
|
||||
break;
|
||||
case ICmpInst::ICMP_SGT:
|
||||
switch (PredR) {
|
||||
default:
|
||||
llvm_unreachable("Unknown integer condition code!");
|
||||
case ICmpInst::ICMP_EQ: // (X s> 13 | X == 15) -> X > 13
|
||||
case ICmpInst::ICMP_SGT: // (X s> 13 | X s> 15) -> X > 13
|
||||
return LHS;
|
||||
case ICmpInst::ICMP_NE: // (X s> 13 | X != 15) -> true
|
||||
case ICmpInst::ICMP_SLT: // (X s> 13 | X s< 15) -> true
|
||||
return Builder->getTrue();
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
@ -1,14 +0,0 @@
|
||||
; RUN: opt < %s -instcombine -S | \
|
||||
; RUN: grep "ret i1 true"
|
||||
; PR586
|
||||
|
||||
@g_07918478 = external global i32 ; <i32*> [#uses=1]
|
||||
|
||||
define i1 @test() {
|
||||
%tmp.0 = load i32, i32* @g_07918478 ; <i32> [#uses=2]
|
||||
%tmp.1 = icmp ne i32 %tmp.0, 0 ; <i1> [#uses=1]
|
||||
%tmp.4 = icmp ult i32 %tmp.0, 4111 ; <i1> [#uses=1]
|
||||
%bothcond = or i1 %tmp.1, %tmp.4 ; <i1> [#uses=1]
|
||||
ret i1 %bothcond
|
||||
}
|
||||
|
@ -15,9 +15,7 @@ define i1 @PR1817_1(i32 %X) {
|
||||
define i1 @PR1817_2(i32 %X) {
|
||||
; CHECK-LABEL: @PR1817_2(
|
||||
; CHECK-NEXT: [[A:%.*]] = icmp slt i32 %X, 10
|
||||
; CHECK-NEXT: [[B:%.*]] = icmp ult i32 %X, 10
|
||||
; CHECK-NEXT: [[C:%.*]] = or i1 [[A]], [[B]]
|
||||
; CHECK-NEXT: ret i1 [[C]]
|
||||
; CHECK-NEXT: ret i1 [[A]]
|
||||
;
|
||||
%A = icmp slt i32 %X, 10
|
||||
%B = icmp ult i32 %X, 10
|
||||
|
@ -661,17 +661,6 @@ define i1 @test47(i8 signext %c) {
|
||||
ret i1 %or
|
||||
}
|
||||
|
||||
define i1 @test48(i64 %x, i1 %b) {
|
||||
; CHECK-LABEL: @test48(
|
||||
; CHECK-NEXT: ret i1 true
|
||||
;
|
||||
%1 = icmp ult i64 %x, 2305843009213693952
|
||||
%2 = icmp ugt i64 %x, 2305843009213693951
|
||||
%.b = or i1 %2, %b
|
||||
%3 = or i1 %1, %.b
|
||||
ret i1 %3
|
||||
}
|
||||
|
||||
define i32 @test49(i1 %C) {
|
||||
; CHECK-LABEL: @test49(
|
||||
; CHECK-NEXT: [[V:%.*]] = select i1 [[C:%.*]], i32 1019, i32 123
|
||||
|
File diff suppressed because it is too large
Load Diff
Loading…
x
Reference in New Issue
Block a user