[ValueTracking] De-pessimize isImpliedCondition around unsigned compares
authorSanjoy Das <sanjoy@playingwithpointers.com>
Fri, 6 Nov 2015 19:01:03 +0000 (19:01 +0000)
committerSanjoy Das <sanjoy@playingwithpointers.com>
Fri, 6 Nov 2015 19:01:03 +0000 (19:01 +0000)
Summary:
Currently `isImpliedCondition` will optimize "I +_nuw C < L ==> I < L"
only if C is positive.  This is an unnecessary restriction -- the
implication holds even if `C` is negative.

Reviewers: reames, majnemer

Subscribers: llvm-commits

Differential Revision: http://reviews.llvm.org/D14369

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@252332 91177308-0d34-0410-b5e6-96231b3b80d8

lib/Analysis/ValueTracking.cpp
test/Transforms/InstSimplify/implies.ll

index 3dc9f3a10370d19d02b84272af162f826147d965..f4824aebe525610a5197aa5bc102e0f88bc8193b 100644 (file)
@@ -4109,12 +4109,12 @@ static bool isTruePredicate(CmpInst::Predicate Pred, Value *LHS, Value *RHS) {
   case CmpInst::ICMP_ULE: {
     ConstantInt *CI;
 
-    // LHS u<  LHS +_{nuw} C   if C > 0
-    // LHS u<= LHS +_{nuw} C   if C >= 0
+    // LHS u<  LHS +_{nuw} C   if C != 0
+    // LHS u<= LHS +_{nuw} C
     if (match(RHS, m_NUWAdd(m_Specific(LHS), m_ConstantInt(CI)))) {
       if (Pred == CmpInst::ICMP_ULT)
-        return CI->getValue().isStrictlyPositive();
-      return !CI->isNegative();
+        return !CI->isZero();
+      return true;
     }
     return false;
   }
index 8e5bbf2c89709603b67dda3ae13f29cdce1dc161..eb8db4224faf0040cfd6b420d30d29ed7dade9f9 100644 (file)
@@ -65,7 +65,7 @@ define i1 @test3(i32 %length.i, i32 %i) {
   ret i1 %res
 }
 
-; i +_{nuw} C_{>0} <u L ==> i <u L
+; i +_{nuw} C <u L ==> i <u L
 define i1 @test4(i32 %length.i, i32 %i) {
 ; CHECK-LABEL: @test4
 ; CHECK: ret i1 true
@@ -116,6 +116,17 @@ define i1 @test8(i32 %length.i, i32 %i) {
   ret i1 %res
 }
 
+; i +_{nuw} C <s L ==> i < L, even if C is negative
+define i1 @test9(i32 %length.i, i32 %i) {
+; CHECK-LABEL: @test9(
+; CHECK: ret i1 true
+  %iplus1 = add nuw i32 %i, -100
+  %var29 = icmp ult i32 %i, %length.i
+  %var30 = icmp ult i32 %iplus1, %length.i
+  %res = icmp ule i1 %var30, %var29
+  ret i1 %res
+}
+
 ; X >=(s) Y == X ==> Y (i1 1 becomes -1 for reasoning)
 define i1 @test_sge(i32 %length.i, i32 %i) {
 ; CHECK-LABEL: @test_sge