[llvm] [ValueTracking][WIP] Check non-null through ops (PR #126009)

via llvm-commits llvm-commits at lists.llvm.org
Wed Feb 5 21:38:40 PST 2025


https://github.com/goldsteinn created https://github.com/llvm/llvm-project/pull/126009

None

>From e29643a30f739d31574d4c6b33ba6ceeceb21edc Mon Sep 17 00:00:00 2001
From: Noah Goldstein <goldstein.w.n at gmail.com>
Date: Thu, 6 Feb 2025 00:38:15 -0500
Subject: [PATCH] [ValueTracking][WIP] Check non-null through ops

---
 llvm/lib/Analysis/ValueTracking.cpp | 67 +++++++++++++++++++++++++----
 1 file changed, 59 insertions(+), 8 deletions(-)

diff --git a/llvm/lib/Analysis/ValueTracking.cpp b/llvm/lib/Analysis/ValueTracking.cpp
index f4958162accfe33..2e6084f666ba73f 100644
--- a/llvm/lib/Analysis/ValueTracking.cpp
+++ b/llvm/lib/Analysis/ValueTracking.cpp
@@ -2625,6 +2625,47 @@ static bool isGEPKnownNonNull(const GEPOperator *GEP, unsigned Depth,
   return false;
 }
 
+static bool transferNonZero(const Value *Op, const Value *V, unsigned Depth) {
+  if (Op == V)
+    return true;
+  if (Depth++ > MaxAnalysisRecursionDepth / 2)
+    return false;
+  const Operator *I = dyn_cast<Operator>(V);
+  if (!I)
+    return false;
+  switch (I->getOpcode()) {
+  case Instruction::And:
+    return transferNonZero(I->getOperand(0), V, Depth) ||
+           transferNonZero(I->getOperand(1), V, Depth);
+  case Instruction::Select: {
+    const Value *Unused0, *Unused1;
+    return matchSelectPattern(I, Unused0, Unused1).Flavor == SPF_UMIN;
+  }
+  case Instruction::Call:
+  case Instruction::Invoke:
+    if (const IntrinsicInst *II = dyn_cast<IntrinsicInst>(I)) {
+      switch (II->getIntrinsicID()) {
+      default:
+        break;
+      case Intrinsic::ctpop:
+      case Intrinsic::bitreverse:
+      case Intrinsic::bswap:
+      case Intrinsic::abs:
+        return transferNonZero(II->getArgOperand(0), V, Depth);
+      case Intrinsic::umin:
+        return transferNonZero(II->getArgOperand(0), V, Depth) ||
+               transferNonZero(II->getArgOperand(1), V, Depth);
+      case Intrinsic::ptrmask:
+        return transferNonZero(II->getArgOperand(0), V, Depth);
+      }
+    }
+    break;
+  default:
+    break;
+  }
+  return false;
+}
+
 static bool isKnownNonNullFromDominatingCondition(const Value *V,
                                                   const Instruction *CtxI,
                                                   const DominatorTree *DT) {
@@ -2654,22 +2695,32 @@ static bool isKnownNonNullFromDominatingCondition(const Value *V,
     }
 
     // If the value is used as a load/store, then the pointer must be non null.
-    if (V == getLoadStorePointerOperand(UI)) {
-      if (!NullPointerIsDefined(UI->getFunction(),
-                                V->getType()->getPointerAddressSpace()) &&
-          DT->dominates(UI, CtxI))
-        return true;
+    if (auto *LV = getLoadStorePointerOperand(UI)) {
+      if (transferNonZero(LV, V, /*Depth=*/0)) {
+        if (!NullPointerIsDefined(UI->getFunction(),
+                                  V->getType()->getPointerAddressSpace()) &&
+            DT->dominates(UI, CtxI))
+          return true;
+      }
     }
 
-    if ((match(UI, m_IDiv(m_Value(), m_Specific(V))) ||
-         match(UI, m_IRem(m_Value(), m_Specific(V)))) &&
+    Value *X;
+    if ((match(UI, m_IDiv(m_Value(), m_Value(X))) ||
+         match(UI, m_IRem(m_Value(), m_Value(X)))) &&
+        transferNonZero(X, V, /*Depth=*/0) &&
         isValidAssumeForContext(UI, CtxI, DT))
       return true;
 
     // Consider only compare instructions uniquely controlling a branch
     Value *RHS;
     CmpPredicate Pred;
-    if (!match(UI, m_c_ICmp(Pred, m_Specific(V), m_Value(RHS))))
+    if (!match(UI, m_ICmp(Pred, m_Value(X), m_Value(RHS))))
+      continue;
+
+    if (transferNonZero(RHS, V, /*Depth=*/0)) {
+      Pred = CmpInst::getSwappedPredicate(Pred);
+      std::swap(RHS, X);
+    } else if (!transferNonZero(X, V, /*Depth=*/0))
       continue;
 
     bool NonNullIfTrue;



More information about the llvm-commits mailing list