[llvm] 744a23f - [ValueTracking] Use select condition to help infer bits of arms
Noah Goldstein via llvm-commits
llvm-commits at lists.llvm.org
Wed Mar 13 12:27:22 PDT 2024
Author: Noah Goldstein
Date: 2024-03-13T14:27:05-05:00
New Revision: 744a23f24b08e8b988b176173c433d64761e66b3
URL: https://github.com/llvm/llvm-project/commit/744a23f24b08e8b988b176173c433d64761e66b3
DIFF: https://github.com/llvm/llvm-project/commit/744a23f24b08e8b988b176173c433d64761e66b3.diff
LOG: [ValueTracking] Use select condition to help infer bits of arms
If we have something like `(select (icmp ult x, 8), x, y)`, we can use
the `(icmp ult x, 8)` to help compute the knownbits of `x`.
Closes #84699
Added:
Modified:
llvm/lib/Analysis/ValueTracking.cpp
llvm/test/Analysis/ValueTracking/knownbits-select-from-cond.ll
Removed:
################################################################################
diff --git a/llvm/lib/Analysis/ValueTracking.cpp b/llvm/lib/Analysis/ValueTracking.cpp
index 8a4a2c4f92a0dc..edbeede910d7f7 100644
--- a/llvm/lib/Analysis/ValueTracking.cpp
+++ b/llvm/lib/Analysis/ValueTracking.cpp
@@ -1023,11 +1023,44 @@ static void computeKnownBitsFromOperator(const Operator *I,
break;
}
case Instruction::Select: {
- computeKnownBits(I->getOperand(2), Known, Depth + 1, Q);
- computeKnownBits(I->getOperand(1), Known2, Depth + 1, Q);
-
+ auto ComputeForArm = [&](Value *Arm, bool Invert) {
+ KnownBits Res(Known.getBitWidth());
+ computeKnownBits(Arm, Res, Depth + 1, Q);
+ // If we have a constant arm, we are done.
+ if (Res.isConstant())
+ return Res;
+
+ // See what condition implies about the bits of the two select arms.
+ KnownBits CondRes(Res.getBitWidth());
+ computeKnownBitsFromCond(Arm, I->getOperand(0), CondRes, Depth + 1, Q,
+ Invert);
+ // If we don't get any information from the condition, no reason to
+ // proceed.
+ if (CondRes.isUnknown())
+ return Res;
+
+ // We can have conflict if the condition is dead. I.e if we have
+ // (x | 64) < 32 ? (x | 64) : y
+ // we will have conflict at bit 6 from the condition/the `or`.
+ // In that case just return. Its not particularly important
+ // what we do, as this select is going to be simplified soon.
+ CondRes = CondRes.unionWith(Res);
+ if (CondRes.hasConflict())
+ return Res;
+
+ // Finally make sure the information we found is valid. This is relatively
+ // expensive so it's left for the very end.
+ if (!isGuaranteedNotToBeUndef(Arm, Q.AC, Q.CxtI, Q.DT, Depth + 1))
+ return Res;
+
+ // Finally, we know we get information from the condition and its valid,
+ // so return it.
+ return CondRes;
+ };
// Only known if known in both the LHS and RHS.
- Known = Known.intersectWith(Known2);
+ Known =
+ ComputeForArm(I->getOperand(1), /*Invert=*/false)
+ .intersectWith(ComputeForArm(I->getOperand(2), /*Invert=*/true));
break;
}
case Instruction::FPTrunc:
diff --git a/llvm/test/Analysis/ValueTracking/knownbits-select-from-cond.ll b/llvm/test/Analysis/ValueTracking/knownbits-select-from-cond.ll
index 0a1cccaca2bb2a..c3343edfb4c9b3 100644
--- a/llvm/test/Analysis/ValueTracking/knownbits-select-from-cond.ll
+++ b/llvm/test/Analysis/ValueTracking/knownbits-select-from-cond.ll
@@ -6,7 +6,7 @@ define i8 @select_condition_implies_highbits_op1(i8 %xx, i8 noundef %y) {
; CHECK-NEXT: [[X:%.*]] = and i8 [[XX:%.*]], 15
; CHECK-NEXT: [[COND:%.*]] = icmp ult i8 [[Y:%.*]], 3
; CHECK-NEXT: [[SEL:%.*]] = select i1 [[COND]], i8 [[Y]], i8 [[X]]
-; CHECK-NEXT: [[R:%.*]] = add i8 [[SEL]], 32
+; CHECK-NEXT: [[R:%.*]] = or disjoint i8 [[SEL]], 32
; CHECK-NEXT: ret i8 [[R]]
;
%x = and i8 %xx, 15
@@ -36,7 +36,7 @@ define i8 @select_condition_implies_highbits_op2(i8 %xx, i8 noundef %y) {
; CHECK-NEXT: [[X:%.*]] = and i8 [[XX:%.*]], 15
; CHECK-NEXT: [[COND:%.*]] = icmp ugt i8 [[Y:%.*]], 3
; CHECK-NEXT: [[SEL:%.*]] = select i1 [[COND]], i8 [[X]], i8 [[Y]]
-; CHECK-NEXT: [[R:%.*]] = add i8 [[SEL]], 32
+; CHECK-NEXT: [[R:%.*]] = or disjoint i8 [[SEL]], 32
; CHECK-NEXT: ret i8 [[R]]
;
%x = and i8 %xx, 15
@@ -52,7 +52,7 @@ define i8 @select_condition_implies_highbits_op1_and(i8 %xx, i8 noundef %y, i1 %
; CHECK-NEXT: [[COND0:%.*]] = icmp ult i8 [[Y:%.*]], 3
; CHECK-NEXT: [[COND:%.*]] = and i1 [[COND0]], [[OTHER_COND:%.*]]
; CHECK-NEXT: [[SEL:%.*]] = select i1 [[COND]], i8 [[Y]], i8 [[X]]
-; CHECK-NEXT: [[R:%.*]] = add i8 [[SEL]], 32
+; CHECK-NEXT: [[R:%.*]] = or disjoint i8 [[SEL]], 32
; CHECK-NEXT: ret i8 [[R]]
;
%x = and i8 %xx, 15
@@ -69,7 +69,7 @@ define i8 @select_condition_implies_highbits_op2_or(i8 %xx, i8 noundef %y, i1 %o
; CHECK-NEXT: [[COND0:%.*]] = icmp ugt i8 [[Y:%.*]], 3
; CHECK-NEXT: [[COND:%.*]] = or i1 [[COND0]], [[OTHER_COND:%.*]]
; CHECK-NEXT: [[SEL:%.*]] = select i1 [[COND]], i8 [[X]], i8 [[Y]]
-; CHECK-NEXT: [[R:%.*]] = add i8 [[SEL]], 32
+; CHECK-NEXT: [[R:%.*]] = or disjoint i8 [[SEL]], 32
; CHECK-NEXT: ret i8 [[R]]
;
%x = and i8 %xx, 15
More information about the llvm-commits
mailing list