[llvm] Think in terms of blocks for ARC (PR #130860)
via llvm-commits
llvm-commits at lists.llvm.org
Tue Mar 11 15:51:24 PDT 2025
https://github.com/AZero13 created https://github.com/llvm/llvm-project/pull/130860
Handle critical edges for ObjCARC
>From 50c7e464f8f1e8f9028563856732869199e07cb8 Mon Sep 17 00:00:00 2001
From: Rose <gfunni234 at gmail.com>
Date: Mon, 10 Mar 2025 14:24:09 -0400
Subject: [PATCH] Think in terms of blocks
Handle critical edges for ObjCARC
---
llvm/lib/Transforms/ObjCARC/ObjCARCOpts.cpp | 102 ++++++++++++++++++--
1 file changed, 94 insertions(+), 8 deletions(-)
diff --git a/llvm/lib/Transforms/ObjCARC/ObjCARCOpts.cpp b/llvm/lib/Transforms/ObjCARC/ObjCARCOpts.cpp
index 9d7f5e64f9868..1c19743880fc4 100644
--- a/llvm/lib/Transforms/ObjCARC/ObjCARCOpts.cpp
+++ b/llvm/lib/Transforms/ObjCARC/ObjCARCOpts.cpp
@@ -138,12 +138,6 @@ static const Value *FindSingleUseIdentifiedObject(const Value *Arg) {
// calls followed by objc_autoreleasePoolPop calls (perhaps in ObjC++ code
// after inlining) can be turned into plain release calls.
-// TODO: Critical-edge splitting. If the optimial insertion point is
-// a critical edge, the current algorithm has to fail, because it doesn't
-// know how to split edges. It should be possible to make the optimizer
-// think in terms of edges, rather than blocks, and then split critical
-// edges on demand.
-
// TODO: OptimizeSequences could generalized to be Interprocedural.
// TODO: Recognize that a bunch of other objc runtime calls have
@@ -599,6 +593,7 @@ class ObjCARCOpt {
void init(Function &F);
bool run(Function &F, AAResults &AA);
bool hasCFGChanged() const { return CFGChanged; }
+ BasicBlock *SplitCriticalEdge(BasicBlock *Pred, BasicBlock *Succ);
};
} // end anonymous namespace
@@ -1765,8 +1760,50 @@ void ObjCARCOpt::MoveCalls(Value *Arg, RRInfo &RetainsToMove,
Module *M) {
LLVM_DEBUG(dbgs() << "== ObjCARCOpt::MoveCalls ==\n");
- // Insert the new retain and release calls.
+ // First, handle critical edges for retain insertion points
+ SmallVector<Instruction *, 4> NewRetainInsertPts;
+ for (Instruction *InsertPt : RetainsToMove.ReverseInsertPts) {
+ BasicBlock *BB = InsertPt->getParent();
+ BasicBlock *Pred = BB->getUniquePredecessor();
+
+ // If this is a critical edge, split it
+ if (!Pred && BB->hasNPredecessors(1)) {
+ for (BasicBlock *PredBB : predecessors(BB)) {
+ if (PredBB->getTerminator()->getNumSuccessors() > 1) {
+ if (BasicBlock *NewBB = SplitCriticalEdge(PredBB, BB)) {
+ // Add the new block as an insertion point
+ NewRetainInsertPts.push_back(NewBB->getTerminator());
+ }
+ }
+ }
+ } else {
+ NewRetainInsertPts.push_back(InsertPt);
+ }
+ }
+
+ // Then handle critical edges for release insertion points
+ SmallVector<Instruction *, 4> NewReleaseInsertPts;
for (Instruction *InsertPt : ReleasesToMove.ReverseInsertPts) {
+ BasicBlock *BB = InsertPt->getParent();
+ BasicBlock *Pred = BB->getUniquePredecessor();
+
+ // If this is a critical edge, split it
+ if (!Pred && BB->hasNPredecessors(1)) {
+ for (BasicBlock *PredBB : predecessors(BB)) {
+ if (PredBB->getTerminator()->getNumSuccessors() > 1) {
+ if (BasicBlock *NewBB = SplitCriticalEdge(PredBB, BB)) {
+ // Add the new block as an insertion point
+ NewReleaseInsertPts.push_back(NewBB->getTerminator());
+ }
+ }
+ }
+ } else {
+ NewReleaseInsertPts.push_back(InsertPt);
+ }
+ }
+
+ // Now insert the new retain calls at the split points
+ for (Instruction *InsertPt : NewRetainInsertPts) {
Function *Decl = EP.get(ARCRuntimeEntryPointKind::Retain);
SmallVector<OperandBundleDef, 1> BundleList;
addOpBundleForFunclet(InsertPt->getParent(), BundleList);
@@ -1780,7 +1817,9 @@ void ObjCARCOpt::MoveCalls(Value *Arg, RRInfo &RetainsToMove,
"At insertion point: "
<< *InsertPt << "\n");
}
- for (Instruction *InsertPt : RetainsToMove.ReverseInsertPts) {
+
+ // And insert the new release calls at the split points
+ for (Instruction *InsertPt : NewReleaseInsertPts) {
Function *Decl = EP.get(ARCRuntimeEntryPointKind::Release);
SmallVector<OperandBundleDef, 1> BundleList;
addOpBundleForFunclet(InsertPt->getParent(), BundleList);
@@ -2488,6 +2527,53 @@ bool ObjCARCOpt::run(Function &F, AAResults &AA) {
return Changed;
}
+/// Split critical edges where we need to insert retain/release calls
+BasicBlock *ObjCARCOpt::SplitCriticalEdge(BasicBlock *Pred, BasicBlock *Succ) {
+ // Don't split if either block is a landing pad - we want to maintain
+ // the property that landing pads have exactly one predecessor
+ if (Succ->isLandingPad() || isa<InvokeInst>(Pred->getTerminator()))
+ return nullptr;
+
+ // We need both multiple successors in predecessor and
+ // multiple predecessors in successor
+ if (Pred->getTerminator()->getNumSuccessors() <= 1 ||
+ Succ->getUniquePredecessor())
+ return nullptr;
+
+ // Create a new basic block for the split edge
+ BasicBlock *NewBB = BasicBlock::Create(Pred->getContext(),
+ Pred->getName() + "." + Succ->getName() + ".arc",
+ Pred->getParent());
+
+ // Update the terminator of Pred to branch to NewBB instead of Succ
+ BranchInst *Term = cast<BranchInst>(Pred->getTerminator());
+ for (unsigned i = 0, e = Term->getNumSuccessors(); i != e; ++i) {
+ if (Term->getSuccessor(i) == Succ) {
+ Term->setSuccessor(i, NewBB);
+ break;
+ }
+ }
+
+ // Create an unconditional branch from NewBB to Succ
+ BranchInst::Create(Succ, NewBB);
+
+ // Update PHI nodes in Succ
+ for (PHINode &PHI : Succ->phis()) {
+ Value *V = PHI.getIncomingValueForBlock(Pred);
+ PHI.setIncomingBlock(PHI.getBasicBlockIndex(Pred), NewBB);
+ PHI.addIncoming(V, Pred);
+ }
+
+ // Update any funclet bundles
+ if (!BlockEHColors.empty()) {
+ const ColorVector &CV = BlockEHColors.find(Pred)->second;
+ BlockEHColors[NewBB] = CV;
+ }
+
+ CFGChanged = true;
+ return NewBB;
+}
+
/// @}
///
More information about the llvm-commits
mailing list