[llvm] 8a60814 - [ValueTracking] Use `KnownBits` functions for `computeKnownBits` of saturating add/sub functions

Noah Goldstein via llvm-commits llvm-commits at lists.llvm.org
Tue May 23 11:57:05 PDT 2023


Author: Noah Goldstein
Date: 2023-05-23T13:52:40-05:00
New Revision: 8a60814ed56c45bdc469fc215bca0aacb0e5b457

URL: https://github.com/llvm/llvm-project/commit/8a60814ed56c45bdc469fc215bca0aacb0e5b457
DIFF: https://github.com/llvm/llvm-project/commit/8a60814ed56c45bdc469fc215bca0aacb0e5b457.diff

LOG: [ValueTracking] Use `KnownBits` functions for `computeKnownBits` of saturating add/sub functions

The knownbits implementation covers all the cases previously handled
by `uadd.sat`/`usub.sat` as well some additional ones. We previously
were not handling the `ssub.sat`/`sadd.sat` cases at all.

Reviewed By: RKSimon

Differential Revision: https://reviews.llvm.org/D150103

Added: 
    

Modified: 
    llvm/lib/Analysis/ValueTracking.cpp
    llvm/test/Analysis/ValueTracking/knownbits-sat-addsub.ll

Removed: 
    


################################################################################
diff  --git a/llvm/lib/Analysis/ValueTracking.cpp b/llvm/lib/Analysis/ValueTracking.cpp
index a70b4cf907a9..39c665f43f2f 100644
--- a/llvm/lib/Analysis/ValueTracking.cpp
+++ b/llvm/lib/Analysis/ValueTracking.cpp
@@ -1774,36 +1774,25 @@ static void computeKnownBitsFromOperator(const Operator *I,
         break;
       }
       case Intrinsic::uadd_sat:
-      case Intrinsic::usub_sat: {
-        bool IsAdd = II->getIntrinsicID() == Intrinsic::uadd_sat;
         computeKnownBits(I->getOperand(0), Known, Depth + 1, Q);
         computeKnownBits(I->getOperand(1), Known2, Depth + 1, Q);
-
-        // Add: Leading ones of either operand are preserved.
-        // Sub: Leading zeros of LHS and leading ones of RHS are preserved
-        // as leading zeros in the result.
-        unsigned LeadingKnown;
-        if (IsAdd)
-          LeadingKnown = std::max(Known.countMinLeadingOnes(),
-                                  Known2.countMinLeadingOnes());
-        else
-          LeadingKnown = std::max(Known.countMinLeadingZeros(),
-                                  Known2.countMinLeadingOnes());
-
-        Known = KnownBits::computeForAddSub(
-            IsAdd, /* NSW */ false, Known, Known2);
-
-        // We select between the operation result and all-ones/zero
-        // respectively, so we can preserve known ones/zeros.
-        if (IsAdd) {
-          Known.One.setHighBits(LeadingKnown);
-          Known.Zero.clearAllBits();
-        } else {
-          Known.Zero.setHighBits(LeadingKnown);
-          Known.One.clearAllBits();
-        }
+        Known = KnownBits::uadd_sat(Known, Known2);
+        break;
+      case Intrinsic::usub_sat:
+        computeKnownBits(I->getOperand(0), Known, Depth + 1, Q);
+        computeKnownBits(I->getOperand(1), Known2, Depth + 1, Q);
+        Known = KnownBits::usub_sat(Known, Known2);
+        break;
+      case Intrinsic::sadd_sat:
+        computeKnownBits(I->getOperand(0), Known, Depth + 1, Q);
+        computeKnownBits(I->getOperand(1), Known2, Depth + 1, Q);
+        Known = KnownBits::sadd_sat(Known, Known2);
+        break;
+      case Intrinsic::ssub_sat:
+        computeKnownBits(I->getOperand(0), Known, Depth + 1, Q);
+        computeKnownBits(I->getOperand(1), Known2, Depth + 1, Q);
+        Known = KnownBits::ssub_sat(Known, Known2);
         break;
-      }
       case Intrinsic::umin:
         computeKnownBits(I->getOperand(0), Known, Depth + 1, Q);
         computeKnownBits(I->getOperand(1), Known2, Depth + 1, Q);

diff  --git a/llvm/test/Analysis/ValueTracking/knownbits-sat-addsub.ll b/llvm/test/Analysis/ValueTracking/knownbits-sat-addsub.ll
index 47e4adcd5591..c2926eaffa58 100644
--- a/llvm/test/Analysis/ValueTracking/knownbits-sat-addsub.ll
+++ b/llvm/test/Analysis/ValueTracking/knownbits-sat-addsub.ll
@@ -8,11 +8,7 @@ declare i8 @llvm.usub.sat.i8(i8, i8)
 
 define i1 @uadd_sat_overflow(i8 %x, i8 %y) {
 ; CHECK-LABEL: @uadd_sat_overflow(
-; CHECK-NEXT:    [[LHS:%.*]] = or i8 [[X:%.*]], -128
-; CHECK-NEXT:    [[RHS:%.*]] = or i8 [[Y:%.*]], -128
-; CHECK-NEXT:    [[EXP:%.*]] = call i8 @llvm.uadd.sat.i8(i8 [[LHS]], i8 [[RHS]])
-; CHECK-NEXT:    [[R:%.*]] = icmp eq i8 [[EXP]], -2
-; CHECK-NEXT:    ret i1 [[R]]
+; CHECK-NEXT:    ret i1 false
 ;
   %lhs = or i8 %x, 128
   %rhs = or i8 %y, 128
@@ -38,11 +34,7 @@ define i1 @uadd_sat_overflow_fail(i8 %x, i8 %y) {
 
 define i1 @usub_sat_overflow(i8 %x, i8 %y) {
 ; CHECK-LABEL: @usub_sat_overflow(
-; CHECK-NEXT:    [[LHS:%.*]] = and i8 [[X:%.*]], 127
-; CHECK-NEXT:    [[RHS:%.*]] = or i8 [[Y:%.*]], -128
-; CHECK-NEXT:    [[EXP:%.*]] = call i8 @llvm.usub.sat.i8(i8 [[LHS]], i8 [[RHS]])
-; CHECK-NEXT:    [[R:%.*]] = icmp eq i8 [[EXP]], 1
-; CHECK-NEXT:    ret i1 [[R]]
+; CHECK-NEXT:    ret i1 false
 ;
   %lhs = and i8 %x, 127
   %rhs = or i8 %y, 128
@@ -68,13 +60,7 @@ define i1 @usub_sat_overflow_fail(i8 %x, i8 %y) {
 
 define i1 @sadd_sat_overflow_pos(i8 %x, i8 %y) {
 ; CHECK-LABEL: @sadd_sat_overflow_pos(
-; CHECK-NEXT:    [[XX:%.*]] = and i8 [[X:%.*]], 127
-; CHECK-NEXT:    [[YY:%.*]] = and i8 [[Y:%.*]], 127
-; CHECK-NEXT:    [[LHS:%.*]] = or i8 [[XX]], 64
-; CHECK-NEXT:    [[RHS:%.*]] = or i8 [[YY]], 65
-; CHECK-NEXT:    [[EXP:%.*]] = call i8 @llvm.sadd.sat.i8(i8 [[LHS]], i8 [[RHS]])
-; CHECK-NEXT:    [[R:%.*]] = icmp eq i8 [[EXP]], -128
-; CHECK-NEXT:    ret i1 [[R]]
+; CHECK-NEXT:    ret i1 false
 ;
   %xx = and i8 %x, 127
   %yy = and i8 %y, 127
@@ -87,14 +73,7 @@ define i1 @sadd_sat_overflow_pos(i8 %x, i8 %y) {
 
 define i1 @sadd_sat_low_bits(i8 %x, i8 %y) {
 ; CHECK-LABEL: @sadd_sat_low_bits(
-; CHECK-NEXT:    [[XX:%.*]] = and i8 [[X:%.*]], 15
-; CHECK-NEXT:    [[YY:%.*]] = and i8 [[Y:%.*]], 15
-; CHECK-NEXT:    [[LHS:%.*]] = or i8 [[XX]], 1
-; CHECK-NEXT:    [[RHS:%.*]] = and i8 [[YY]], -2
-; CHECK-NEXT:    [[EXP:%.*]] = call i8 @llvm.sadd.sat.i8(i8 [[LHS]], i8 [[RHS]])
-; CHECK-NEXT:    [[AND:%.*]] = and i8 [[EXP]], 1
-; CHECK-NEXT:    [[R:%.*]] = icmp eq i8 [[AND]], 0
-; CHECK-NEXT:    ret i1 [[R]]
+; CHECK-NEXT:    ret i1 false
 ;
   %xx = and i8 %x, 15
   %yy = and i8 %y, 15
@@ -125,11 +104,7 @@ define i1 @sadd_sat_fail_may_overflow(i8 %x, i8 %y) {
 
 define i1 @sadd_sat_overflow_neg(i8 %x, i8 %y) {
 ; CHECK-LABEL: @sadd_sat_overflow_neg(
-; CHECK-NEXT:    [[LHS:%.*]] = or i8 [[X:%.*]], -64
-; CHECK-NEXT:    [[RHS:%.*]] = or i8 [[Y:%.*]], -65
-; CHECK-NEXT:    [[EXP:%.*]] = call i8 @llvm.sadd.sat.i8(i8 [[LHS]], i8 [[RHS]])
-; CHECK-NEXT:    [[R:%.*]] = icmp eq i8 [[EXP]], 127
-; CHECK-NEXT:    ret i1 [[R]]
+; CHECK-NEXT:    ret i1 false
 ;
   %lhs = or i8 %x, 192
   %rhs = or i8 %y, 191
@@ -140,13 +115,7 @@ define i1 @sadd_sat_overflow_neg(i8 %x, i8 %y) {
 
 define i1 @ssub_sat_overflow_neg(i8 %x, i8 %y) {
 ; CHECK-LABEL: @ssub_sat_overflow_neg(
-; CHECK-NEXT:    [[XX:%.*]] = and i8 [[X:%.*]], 112
-; CHECK-NEXT:    [[YY:%.*]] = and i8 [[Y:%.*]], 127
-; CHECK-NEXT:    [[LHS:%.*]] = or i8 [[XX]], -128
-; CHECK-NEXT:    [[RHS:%.*]] = or i8 [[YY]], 126
-; CHECK-NEXT:    [[EXP:%.*]] = call i8 @llvm.ssub.sat.i8(i8 [[LHS]], i8 [[RHS]])
-; CHECK-NEXT:    [[R:%.*]] = icmp eq i8 [[EXP]], 32
-; CHECK-NEXT:    ret i1 [[R]]
+; CHECK-NEXT:    ret i1 false
 ;
   %xx = and i8 %x, 112
   %yy = and i8 %y, 127
@@ -159,14 +128,7 @@ define i1 @ssub_sat_overflow_neg(i8 %x, i8 %y) {
 
 define i1 @ssub_sat_low_bits(i8 %x, i8 %y) {
 ; CHECK-LABEL: @ssub_sat_low_bits(
-; CHECK-NEXT:    [[XX:%.*]] = and i8 [[X:%.*]], 15
-; CHECK-NEXT:    [[YY:%.*]] = and i8 [[Y:%.*]], 15
-; CHECK-NEXT:    [[LHS:%.*]] = or i8 [[XX]], 17
-; CHECK-NEXT:    [[RHS:%.*]] = and i8 [[YY]], -2
-; CHECK-NEXT:    [[EXP:%.*]] = call i8 @llvm.ssub.sat.i8(i8 [[LHS]], i8 [[RHS]])
-; CHECK-NEXT:    [[AND:%.*]] = and i8 [[EXP]], 1
-; CHECK-NEXT:    [[R:%.*]] = icmp eq i8 [[AND]], 0
-; CHECK-NEXT:    ret i1 [[R]]
+; CHECK-NEXT:    ret i1 false
 ;
   %xx = and i8 %x, 15
   %yy = and i8 %y, 15
@@ -201,13 +163,7 @@ define i1 @ssub_sat_fail_may_overflow(i8 %x, i8 %y) {
 
 define i1 @ssub_sat_overflow_pos(i8 %x, i8 %y) {
 ; CHECK-LABEL: @ssub_sat_overflow_pos(
-; CHECK-NEXT:    [[XX:%.*]] = and i8 [[X:%.*]], 24
-; CHECK-NEXT:    [[YY:%.*]] = and i8 [[Y:%.*]], 3
-; CHECK-NEXT:    [[LHS:%.*]] = or i8 [[XX]], 8
-; CHECK-NEXT:    [[RHS:%.*]] = or i8 [[YY]], -128
-; CHECK-NEXT:    [[EXP:%.*]] = call i8 @llvm.ssub.sat.i8(i8 [[LHS]], i8 [[RHS]])
-; CHECK-NEXT:    [[R:%.*]] = icmp eq i8 [[EXP]], -128
-; CHECK-NEXT:    ret i1 [[R]]
+; CHECK-NEXT:    ret i1 false
 ;
   %xx = and i8 %x, 24
   %yy = and i8 %y, 3


        


More information about the llvm-commits mailing list