xref: /freebsd/contrib/llvm-project/llvm/lib/Transforms/ObjCARC/ObjCARC.cpp (revision 13ec1e3155c7e9bf037b12af186351b7fa9b9450)
1 //===-- ObjCARC.cpp -------------------------------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // This file implements common infrastructure for libLLVMObjCARCOpts.a, which
10 // implements several scalar transformations over the LLVM intermediate
11 // representation, including the C bindings for that library.
12 //
13 //===----------------------------------------------------------------------===//
14 
15 #include "ObjCARC.h"
16 #include "llvm-c/Initialization.h"
17 #include "llvm/Analysis/ObjCARCUtil.h"
18 #include "llvm/IR/IRBuilder.h"
19 #include "llvm/IR/InlineAsm.h"
20 #include "llvm/IR/Instructions.h"
21 #include "llvm/InitializePasses.h"
22 #include "llvm/Transforms/Utils/BasicBlockUtils.h"
23 
24 namespace llvm {
25   class PassRegistry;
26 }
27 
28 using namespace llvm;
29 using namespace llvm::objcarc;
30 
31 /// initializeObjCARCOptsPasses - Initialize all passes linked into the
32 /// ObjCARCOpts library.
33 void llvm::initializeObjCARCOpts(PassRegistry &Registry) {
34   initializeObjCARCAAWrapperPassPass(Registry);
35   initializeObjCARCAPElimPass(Registry);
36   initializeObjCARCExpandPass(Registry);
37   initializeObjCARCContractLegacyPassPass(Registry);
38   initializeObjCARCOptLegacyPassPass(Registry);
39   initializePAEvalPass(Registry);
40 }
41 
42 void LLVMInitializeObjCARCOpts(LLVMPassRegistryRef R) {
43   initializeObjCARCOpts(*unwrap(R));
44 }
45 
46 CallInst *objcarc::createCallInstWithColors(
47     FunctionCallee Func, ArrayRef<Value *> Args, const Twine &NameStr,
48     Instruction *InsertBefore,
49     const DenseMap<BasicBlock *, ColorVector> &BlockColors) {
50   FunctionType *FTy = Func.getFunctionType();
51   Value *Callee = Func.getCallee();
52   SmallVector<OperandBundleDef, 1> OpBundles;
53 
54   if (!BlockColors.empty()) {
55     const ColorVector &CV = BlockColors.find(InsertBefore->getParent())->second;
56     assert(CV.size() == 1 && "non-unique color for block!");
57     Instruction *EHPad = CV.front()->getFirstNonPHI();
58     if (EHPad->isEHPad())
59       OpBundles.emplace_back("funclet", EHPad);
60   }
61 
62   return CallInst::Create(FTy, Callee, Args, OpBundles, NameStr, InsertBefore);
63 }
64 
65 std::pair<bool, bool>
66 BundledRetainClaimRVs::insertAfterInvokes(Function &F, DominatorTree *DT) {
67   bool Changed = false, CFGChanged = false;
68 
69   for (BasicBlock &BB : F) {
70     auto *I = dyn_cast<InvokeInst>(BB.getTerminator());
71 
72     if (!I)
73       continue;
74 
75     if (!objcarc::hasAttachedCallOpBundle(I))
76       continue;
77 
78     BasicBlock *DestBB = I->getNormalDest();
79 
80     if (!DestBB->getSinglePredecessor()) {
81       assert(I->getSuccessor(0) == DestBB &&
82              "the normal dest is expected to be the first successor");
83       DestBB = SplitCriticalEdge(I, 0, CriticalEdgeSplittingOptions(DT));
84       CFGChanged = true;
85     }
86 
87     // We don't have to call insertRVCallWithColors since DestBB is the normal
88     // destination of the invoke.
89     insertRVCall(&*DestBB->getFirstInsertionPt(), I);
90     Changed = true;
91   }
92 
93   return std::make_pair(Changed, CFGChanged);
94 }
95 
96 CallInst *BundledRetainClaimRVs::insertRVCall(Instruction *InsertPt,
97                                               CallBase *AnnotatedCall) {
98   DenseMap<BasicBlock *, ColorVector> BlockColors;
99   return insertRVCallWithColors(InsertPt, AnnotatedCall, BlockColors);
100 }
101 
102 CallInst *BundledRetainClaimRVs::insertRVCallWithColors(
103     Instruction *InsertPt, CallBase *AnnotatedCall,
104     const DenseMap<BasicBlock *, ColorVector> &BlockColors) {
105   IRBuilder<> Builder(InsertPt);
106   bool IsRetainRV = objcarc::hasAttachedCallOpBundle(AnnotatedCall, true);
107   Function *Func = EP.get(IsRetainRV ? ARCRuntimeEntryPointKind::RetainRV
108                                      : ARCRuntimeEntryPointKind::ClaimRV);
109   Type *ParamTy = Func->getArg(0)->getType();
110   Value *CallArg = Builder.CreateBitCast(AnnotatedCall, ParamTy);
111   auto *Call =
112       createCallInstWithColors(Func, CallArg, "", InsertPt, BlockColors);
113   RVCalls[Call] = AnnotatedCall;
114   return Call;
115 }
116 
117 BundledRetainClaimRVs::~BundledRetainClaimRVs() {
118   if (ContractPass) {
119     // At this point, we know that the annotated calls can't be tail calls as
120     // they are followed by marker instructions and retainRV/claimRV calls. Mark
121     // them as notail, so that the backend knows these calls can't be tail
122     // calls.
123     for (auto P : RVCalls)
124       if (auto *CI = dyn_cast<CallInst>(P.second))
125         CI->setTailCallKind(CallInst::TCK_NoTail);
126   } else {
127     for (auto P : RVCalls)
128       EraseInstruction(P.first);
129   }
130 
131   RVCalls.clear();
132 }
133