summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRafael Espindola <rafael.espindola@gmail.com>2014-06-02 19:19:04 +0000
committerRafael Espindola <rafael.espindola@gmail.com>2014-06-02 19:19:04 +0000
commit83223f52e72534b3c6cd491cffa72a4642262967 (patch)
tree8b97132a3f3b92affbebddf4c492bf7e23f9df19
parentf50e3953084cff910450ada6babf672b65ef082d (diff)
downloadllvm-83223f52e72534b3c6cd491cffa72a4642262967.tar.gz
llvm-83223f52e72534b3c6cd491cffa72a4642262967.tar.bz2
llvm-83223f52e72534b3c6cd491cffa72a4642262967.tar.xz
Added support to optimize comparisons with "lshr exact" of a constant.
Patch by Rahul Jain. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@210040 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r--lib/Transforms/InstCombine/InstCombineCompares.cpp35
-rw-r--r--test/Transforms/InstCombine/icmp.ll8
2 files changed, 37 insertions, 6 deletions
diff --git a/lib/Transforms/InstCombine/InstCombineCompares.cpp b/lib/Transforms/InstCombine/InstCombineCompares.cpp
index 3d90a37ecb..13a5cd96f7 100644
--- a/lib/Transforms/InstCombine/InstCombineCompares.cpp
+++ b/lib/Transforms/InstCombine/InstCombineCompares.cpp
@@ -2318,6 +2318,32 @@ static bool swapMayExposeCSEOpportunities(const Value * Op0,
return GlobalSwapBenefits > 0;
}
+// Helper function to check whether Op represents a lshr/ashr exact
+// instruction. For example:
+// (icmp (ashr exact const2, A), const1) -> icmp A, Log2(const2/const1)
+// Here if Op represents -> (ashr exact const2, A), and CI represents
+// const1, we compute Quotient as const2/const1.
+
+static bool checkShrExact(Value *Op, APInt &Quotient, const ConstantInt *CI,
+ Value *&A) {
+
+ ConstantInt *CI2;
+ if (match(Op, m_AShr(m_ConstantInt(CI2), m_Value(A))) &&
+ (cast<BinaryOperator>(Op)->isExact())) {
+ Quotient = CI2->getValue().sdiv(CI->getValue());
+ return true;
+ }
+
+ // Handle the case for lhsr.
+ if (match(Op, m_LShr(m_ConstantInt(CI2), m_Value(A))) &&
+ (cast<BinaryOperator>(Op)->isExact())) {
+ Quotient = CI2->getValue().udiv(CI->getValue());
+ return true;
+ }
+
+ return false;
+}
+
Instruction *InstCombiner::visitICmpInst(ICmpInst &I) {
bool Changed = false;
Value *Op0 = I.getOperand(0), *Op1 = I.getOperand(1);
@@ -2443,13 +2469,10 @@ Instruction *InstCombiner::visitICmpInst(ICmpInst &I) {
// (icmp (ashr exact const2, A), const1) -> icmp A, Log2(const2/const1)
// Cases where const1 doesn't divide const2 exactly or Quotient is not
// exact of log2 are handled by SimplifyICmpInst call above where we
- // return false.
- // TODO: Handle this for lshr exact with udiv.
+ // return false. Similar for lshr.
{
- ConstantInt *CI2;
- if (match(Op0, m_AShr(m_ConstantInt(CI2), m_Value(A))) &&
- (cast<BinaryOperator>(Op0)->isExact())) {
- APInt Quotient = CI2->getValue().sdiv(CI->getValue());
+ APInt Quotient;
+ if (checkShrExact(Op0, Quotient, CI, A)) {
unsigned shift = Quotient.logBase2();
return new ICmpInst(I.getPredicate(), A,
ConstantInt::get(A->getType(), shift));
diff --git a/test/Transforms/InstCombine/icmp.ll b/test/Transforms/InstCombine/icmp.ll
index dcfff99136..8e107edd76 100644
--- a/test/Transforms/InstCombine/icmp.ll
+++ b/test/Transforms/InstCombine/icmp.ll
@@ -1390,3 +1390,11 @@ define i1 @exact_ashr_eq_false(i32 %a) {
%cmp = icmp eq i32 %shr, -15
ret i1 %cmp
}
+
+; CHECK-LABEL: @exact_lhsr
+; CHECK-NEXT: icmp eq i32 %a, 3
+define i1 @exact_lhsr(i32 %a) {
+ %shr = lshr exact i32 80, %a
+ %cmp = icmp eq i32 %shr, 10
+ ret i1 %cmp
+}