[llvm] Redesign Straight-Line Strength Reduction (SLSR) (PR #162930)
    Matt Arsenault via llvm-commits 
    llvm-commits at lists.llvm.org
       
    Fri Oct 10 17:52:11 PDT 2025
    
    
  
================
@@ -269,17 +590,284 @@ FunctionPass *llvm::createStraightLineStrengthReducePass() {
   return new StraightLineStrengthReduceLegacyPass();
 }
 
-bool StraightLineStrengthReduce::isBasisFor(const Candidate &Basis,
-                                            const Candidate &C) {
-  return (Basis.Ins != C.Ins && // skip the same instruction
-          // They must have the same type too. Basis.Base == C.Base
-          // doesn't guarantee their types are the same (PR23975).
-          Basis.Ins->getType() == C.Ins->getType() &&
-          // Basis must dominate C in order to rewrite C with respect to Basis.
-          DT->dominates(Basis.Ins->getParent(), C.Ins->getParent()) &&
-          // They share the same base, stride, and candidate kind.
-          Basis.Base == C.Base && Basis.Stride == C.Stride &&
-          Basis.CandidateKind == C.CandidateKind);
+// A helper function that unifies the bitwidth of A and B.
+static void unifyBitWidth(APInt &A, APInt &B) {
+  if (A.getBitWidth() < B.getBitWidth())
+    A = A.sext(B.getBitWidth());
+  else if (A.getBitWidth() > B.getBitWidth())
+    B = B.sext(A.getBitWidth());
+}
+
+Constant *StraightLineStrengthReduce::getIndexDelta(Candidate &C,
+                                                    Candidate &Basis) {
+  APInt Idx = C.Index->getValue(), BasisIdx = Basis.Index->getValue();
+  unifyBitWidth(Idx, BasisIdx);
+  APInt IndexOffset = Idx - BasisIdx;
+  IntegerType *DeltaType =
+      IntegerType::get(C.Ins->getContext(), IndexOffset.getBitWidth());
+  return ConstantInt::get(DeltaType, IndexOffset);
+}
+
+bool StraightLineStrengthReduce::isSimilar(Candidate &C, Candidate &Basis,
+                                           Candidate::DKind K) {
+  bool SameType = false;
+  switch (K) {
+  case Candidate::StrideDelta:
+    SameType = C.StrideSCEV->getType() == Basis.StrideSCEV->getType();
+    break;
+  case Candidate::BaseDelta:
+    SameType = C.Base->getType() == Basis.Base->getType();
+    break;
+  case Candidate::IndexDelta:
+    SameType = true;
+    break;
+  default:;
+  }
+  return SameType && Basis.Ins != C.Ins &&
+         Basis.CandidateKind == C.CandidateKind;
+}
+
+void StraightLineStrengthReduce::setBasisAndDeltaFor(Candidate &C) {
+  auto SearchFrom = [this, &C](const CandidateDictTy::BBToCandsTy &BBToCands,
+                               auto IsTarget) -> bool {
+    // Search dominating candidates by walking the immediate-dominator chain
+    // from the candidate's defining block upward. Visiting blocks in this
+    // order ensures we prefer the closest dominating basis.
+    const BasicBlock *BB = C.Ins->getParent();
+    while (BB) {
+      auto It = BBToCands.find(BB);
+      if (It != BBToCands.end())
+        for (Candidate *Basis : reverse(It->second))
+          if (IsTarget(Basis))
+            return true;
+
+      const DomTreeNode *Node = DT->getNode(BB);
+      if (!Node)
+        break;
+      Node = Node->getIDom();
+      BB = Node ? Node->getBlock() : nullptr;
+    }
+    return false;
+  };
+
+  // Priority:
+  // Constant Delta from Index > Constant Delta from Base >
+  // Constant Delta from Stride > Variable Delta from Base or Stride
+  // TODO: Change the priority to align with the cost model.
+
+  // First, look for a constant index-diff basis
+  if (const auto *IndexDeltaCandidates =
+          CandidateDict.getCandidatesWithDeltaKind(C, Candidate::IndexDelta)) {
+    bool FoundConstDelta =
+        SearchFrom(*IndexDeltaCandidates, [&DT = DT, &C](Candidate *Basis) {
+          if (isSimilar(C, *Basis, Candidate::IndexDelta)) {
+            assert(DT->dominates(Basis->Ins, C.Ins));
+            auto *Delta = getIndexDelta(C, *Basis);
+            if (!C.isProfitableRewrite(Delta, Candidate::IndexDelta))
+              return false;
+            C.Basis = Basis;
+            C.DeltaKind = Candidate::IndexDelta;
+            C.Delta = Delta;
+            LLVM_DEBUG(dbgs() << "Found delta from Index " << *C.Delta << "\n");
+            return true;
+          }
+          return false;
+        });
+    if (FoundConstDelta)
+      return;
+  }
+
+  // No constant-index-diff basis found. look for the best possible base-diff
+  // or stride-diff basis
+  // Base/Stride diffs not supported for form (B + i) * S
+  if (C.CandidateKind == Candidate::Mul)
+    return;
+
+  auto For = [this, &C](Candidate::DKind K) {
+    // return true if find a Basis with constant delta and stop searching,
+    // return false if did not find a Basis or the delta is not a constant
+    // and continue searching for a Basis with constant delta
+    return [K, this, &C](Candidate *Basis) -> bool {
+      if (!isSimilar(C, *Basis, K))
+        return false;
+
+      assert(DT->dominates(Basis->Ins, C.Ins));
+      const SCEV *BasisPart =
+          (K == Candidate::BaseDelta) ? Basis->Base : Basis->StrideSCEV;
+      const SCEV *CandPart =
+          (K == Candidate::BaseDelta) ? C.Base : C.StrideSCEV;
+      const SCEV *Diff = SE->getMinusSCEV(CandPart, BasisPart);
+      Value *AvailableVal = getNearestValueOfSCEV(Diff, C.Ins);
+      if (!AvailableVal)
+        return false;
+
+      // Record delta if none has been found yet, or the new delta is
+      // a constant that is better than the existing delta.
+      if (!C.Delta || isa<ConstantInt>(AvailableVal)) {
+        C.Delta = AvailableVal;
+        C.Basis = Basis;
+        C.DeltaKind = K;
+      }
+      return isa<ConstantInt>(C.Delta);
+    };
+  };
+
+  if (const auto *BaseDeltaCandidates =
+          CandidateDict.getCandidatesWithDeltaKind(C, Candidate::BaseDelta)) {
+    if (SearchFrom(*BaseDeltaCandidates, For(Candidate::BaseDelta))) {
+      LLVM_DEBUG(dbgs() << "Found delta from Base: " << *C.Delta << "\n");
+      return;
+    }
+  }
+
+  if (const auto *StrideDeltaCandidates =
+          CandidateDict.getCandidatesWithDeltaKind(C, Candidate::StrideDelta)) {
+    if (SearchFrom(*StrideDeltaCandidates, For(Candidate::StrideDelta))) {
+      LLVM_DEBUG(dbgs() << "Found delta from Stride: " << *C.Delta << "\n");
+      return;
+    }
+  }
+
+  // If we did not find a constant delta, we might have found a variable delta
+  if (C.Delta) {
+    LLVM_DEBUG(dbgs() << "Found delta from ";
+               if (C.DeltaKind == Candidate::BaseDelta) dbgs() << "Base: ";
+               else dbgs() << "Stride: "; dbgs() << *C.Delta << "\n");
+    assert(C.DeltaKind != Candidate::InvalidDelta && C.Basis);
+  }
+}
+
+// Compress the path from `Basis` to the deepest Basis in the Basis chain
+// to avoid non-profitable data dependency and improve ILP.
+// X = A + 1
+// Y = X + 1
+// Z = Y + 1
+// ->
+// X = A + 1
+// Y = A + 2
+// Z = A + 3
+// Return the delta info for C aginst the new Basis
+auto StraightLineStrengthReduce::compressPath(
+    Candidate &C, Candidate *Basis) const -> DeltaInfo {
+  if (!Basis || !Basis->Basis || C.CandidateKind == Candidate::Mul)
+    return {};
+  Candidate *Root = Basis;
+  Value *NewDelta = nullptr;
+  auto NewKind = Candidate::InvalidDelta;
+
+  while (Root->Basis) {
+    Candidate *NextRoot = Root->Basis;
+    if (C.Base == NextRoot->Base && C.StrideSCEV == NextRoot->StrideSCEV &&
+        isSimilar(C, *NextRoot, Candidate::IndexDelta)) {
+      ConstantInt *CI = cast<ConstantInt>(getIndexDelta(C, *NextRoot));
+      if (CI->isZero() || CI->isOne() || isa<SCEVConstant>(C.StrideSCEV)) {
+        Root = NextRoot;
+        NewKind = Candidate::IndexDelta;
+        NewDelta = CI;
+        continue;
+      }
+    }
+
+    const SCEV *CandPart = nullptr;
+    const SCEV *BasisPart = nullptr;
+    auto CurrKind = Candidate::InvalidDelta;
+    if (C.Base == NextRoot->Base && C.Index == NextRoot->Index) {
+      CandPart = C.StrideSCEV;
+      BasisPart = NextRoot->StrideSCEV;
+      CurrKind = Candidate::StrideDelta;
+    } else if (C.StrideSCEV == NextRoot->StrideSCEV &&
+               C.Index == NextRoot->Index) {
+      CandPart = C.Base;
+      BasisPart = NextRoot->Base;
+      CurrKind = Candidate::BaseDelta;
+    } else
+      break;
+
+    if (!isSimilar(C, *NextRoot, CurrKind))
+      break;
+
+    if (auto DeltaVal =
+            dyn_cast<SCEVConstant>(SE->getMinusSCEV(CandPart, BasisPart))) {
+      Root = NextRoot;
+      NewDelta = DeltaVal->getValue();
+      NewKind = CurrKind;
+    } else
+      break;
+  }
+
+  if (Root != Basis) {
+    assert(NewKind != Candidate::InvalidDelta && NewDelta);
+    LLVM_DEBUG(dbgs() << "Found new Basis with " << *NewDelta
+                      << " from path compression.\n");
+    return {Root, NewKind, NewDelta};
+  }
+
+  return {};
+}
+
+// Topologically sort candidate instructions based on their relationship in
+// dependency graph.
+void StraightLineStrengthReduce::sortCandidateInstructions() {
+  SortedCandidateInsts.clear();
+  // An instruction may have multiple candidates that get different Basis
+  // instructions, and each candidate can get dependencies from Basis and
+  // Stride when Stride will also be rewritten by SLSR. Hence, an instruction
+  // may have multiple dependencies. Use InDegree to ensure all dependencies
+  // processed before processing itself.
+  DenseMap<Instruction *, int> InDegree;
+  for (auto &KV : DependencyGraph) {
+    if (InDegree.find(KV.first) == InDegree.end())
+      InDegree[KV.first] = 0;
+
+    for (auto *Child : KV.second) {
+      InDegree[Child]++;
+    }
+  }
+  std::queue<Instruction *> WorkList;
+  DenseSet<Instruction *> Visited;
+
+  for (auto &KV : DependencyGraph)
+    if (InDegree[KV.first] == 0)
+      WorkList.push(KV.first);
+
+  while (!WorkList.empty()) {
+    Instruction *I = WorkList.front();
+    WorkList.pop();
+    if (!Visited.insert(I).second)
+      continue;
+
+    SortedCandidateInsts.push_back(I);
+
+    for (auto *Next : DependencyGraph[I]) {
+      InDegree[Next]--;
+      if (InDegree[Next] == 0)
----------------
arsenm wrote:
Avoid double map lookup 
https://github.com/llvm/llvm-project/pull/162930
    
    
More information about the llvm-commits
mailing list