xref: /aosp_15_r20/external/clang/lib/ARCMigrate/TransRetainReleaseDealloc.cpp (revision 67e74705e28f6214e480b399dd47ea732279e315)
1*67e74705SXin Li //===--- TransRetainReleaseDealloc.cpp - Transformations to ARC mode ------===//
2*67e74705SXin Li //
3*67e74705SXin Li //                     The LLVM Compiler Infrastructure
4*67e74705SXin Li //
5*67e74705SXin Li // This file is distributed under the University of Illinois Open Source
6*67e74705SXin Li // License. See LICENSE.TXT for details.
7*67e74705SXin Li //
8*67e74705SXin Li //===----------------------------------------------------------------------===//
9*67e74705SXin Li //
10*67e74705SXin Li // removeRetainReleaseDealloc:
11*67e74705SXin Li //
12*67e74705SXin Li // Removes retain/release/autorelease/dealloc messages.
13*67e74705SXin Li //
14*67e74705SXin Li //  return [[foo retain] autorelease];
15*67e74705SXin Li // ---->
16*67e74705SXin Li //  return foo;
17*67e74705SXin Li //
18*67e74705SXin Li //===----------------------------------------------------------------------===//
19*67e74705SXin Li 
20*67e74705SXin Li #include "Transforms.h"
21*67e74705SXin Li #include "Internals.h"
22*67e74705SXin Li #include "clang/AST/ASTContext.h"
23*67e74705SXin Li #include "clang/AST/ParentMap.h"
24*67e74705SXin Li #include "clang/Basic/SourceManager.h"
25*67e74705SXin Li #include "clang/Lex/Lexer.h"
26*67e74705SXin Li #include "clang/Sema/SemaDiagnostic.h"
27*67e74705SXin Li #include "llvm/ADT/StringSwitch.h"
28*67e74705SXin Li 
29*67e74705SXin Li using namespace clang;
30*67e74705SXin Li using namespace arcmt;
31*67e74705SXin Li using namespace trans;
32*67e74705SXin Li 
33*67e74705SXin Li namespace {
34*67e74705SXin Li 
35*67e74705SXin Li class RetainReleaseDeallocRemover :
36*67e74705SXin Li                        public RecursiveASTVisitor<RetainReleaseDeallocRemover> {
37*67e74705SXin Li   Stmt *Body;
38*67e74705SXin Li   MigrationPass &Pass;
39*67e74705SXin Li 
40*67e74705SXin Li   ExprSet Removables;
41*67e74705SXin Li   std::unique_ptr<ParentMap> StmtMap;
42*67e74705SXin Li 
43*67e74705SXin Li   Selector DelegateSel, FinalizeSel;
44*67e74705SXin Li 
45*67e74705SXin Li public:
RetainReleaseDeallocRemover(MigrationPass & pass)46*67e74705SXin Li   RetainReleaseDeallocRemover(MigrationPass &pass)
47*67e74705SXin Li     : Body(nullptr), Pass(pass) {
48*67e74705SXin Li     DelegateSel =
49*67e74705SXin Li         Pass.Ctx.Selectors.getNullarySelector(&Pass.Ctx.Idents.get("delegate"));
50*67e74705SXin Li     FinalizeSel =
51*67e74705SXin Li         Pass.Ctx.Selectors.getNullarySelector(&Pass.Ctx.Idents.get("finalize"));
52*67e74705SXin Li   }
53*67e74705SXin Li 
transformBody(Stmt * body,Decl * ParentD)54*67e74705SXin Li   void transformBody(Stmt *body, Decl *ParentD) {
55*67e74705SXin Li     Body = body;
56*67e74705SXin Li     collectRemovables(body, Removables);
57*67e74705SXin Li     StmtMap.reset(new ParentMap(body));
58*67e74705SXin Li     TraverseStmt(body);
59*67e74705SXin Li   }
60*67e74705SXin Li 
VisitObjCMessageExpr(ObjCMessageExpr * E)61*67e74705SXin Li   bool VisitObjCMessageExpr(ObjCMessageExpr *E) {
62*67e74705SXin Li     switch (E->getMethodFamily()) {
63*67e74705SXin Li     default:
64*67e74705SXin Li       if (E->isInstanceMessage() && E->getSelector() == FinalizeSel)
65*67e74705SXin Li         break;
66*67e74705SXin Li       return true;
67*67e74705SXin Li     case OMF_autorelease:
68*67e74705SXin Li       if (isRemovable(E)) {
69*67e74705SXin Li         if (!isCommonUnusedAutorelease(E)) {
70*67e74705SXin Li           // An unused autorelease is badness. If we remove it the receiver
71*67e74705SXin Li           // will likely die immediately while previously it was kept alive
72*67e74705SXin Li           // by the autorelease pool. This is bad practice in general, leave it
73*67e74705SXin Li           // and emit an error to force the user to restructure their code.
74*67e74705SXin Li           Pass.TA.reportError("it is not safe to remove an unused 'autorelease' "
75*67e74705SXin Li               "message; its receiver may be destroyed immediately",
76*67e74705SXin Li               E->getLocStart(), E->getSourceRange());
77*67e74705SXin Li           return true;
78*67e74705SXin Li         }
79*67e74705SXin Li       }
80*67e74705SXin Li       // Pass through.
81*67e74705SXin Li     case OMF_retain:
82*67e74705SXin Li     case OMF_release:
83*67e74705SXin Li       if (E->getReceiverKind() == ObjCMessageExpr::Instance)
84*67e74705SXin Li         if (Expr *rec = E->getInstanceReceiver()) {
85*67e74705SXin Li           rec = rec->IgnoreParenImpCasts();
86*67e74705SXin Li           if (rec->getType().getObjCLifetime() == Qualifiers::OCL_ExplicitNone &&
87*67e74705SXin Li               (E->getMethodFamily() != OMF_retain || isRemovable(E))) {
88*67e74705SXin Li             std::string err = "it is not safe to remove '";
89*67e74705SXin Li             err += E->getSelector().getAsString() + "' message on "
90*67e74705SXin Li                 "an __unsafe_unretained type";
91*67e74705SXin Li             Pass.TA.reportError(err, rec->getLocStart());
92*67e74705SXin Li             return true;
93*67e74705SXin Li           }
94*67e74705SXin Li 
95*67e74705SXin Li           if (isGlobalVar(rec) &&
96*67e74705SXin Li               (E->getMethodFamily() != OMF_retain || isRemovable(E))) {
97*67e74705SXin Li             std::string err = "it is not safe to remove '";
98*67e74705SXin Li             err += E->getSelector().getAsString() + "' message on "
99*67e74705SXin Li                 "a global variable";
100*67e74705SXin Li             Pass.TA.reportError(err, rec->getLocStart());
101*67e74705SXin Li             return true;
102*67e74705SXin Li           }
103*67e74705SXin Li 
104*67e74705SXin Li           if (E->getMethodFamily() == OMF_release && isDelegateMessage(rec)) {
105*67e74705SXin Li             Pass.TA.reportError("it is not safe to remove 'retain' "
106*67e74705SXin Li                 "message on the result of a 'delegate' message; "
107*67e74705SXin Li                 "the object that was passed to 'setDelegate:' may not be "
108*67e74705SXin Li                 "properly retained", rec->getLocStart());
109*67e74705SXin Li             return true;
110*67e74705SXin Li           }
111*67e74705SXin Li         }
112*67e74705SXin Li     case OMF_dealloc:
113*67e74705SXin Li       break;
114*67e74705SXin Li     }
115*67e74705SXin Li 
116*67e74705SXin Li     switch (E->getReceiverKind()) {
117*67e74705SXin Li     default:
118*67e74705SXin Li       return true;
119*67e74705SXin Li     case ObjCMessageExpr::SuperInstance: {
120*67e74705SXin Li       Transaction Trans(Pass.TA);
121*67e74705SXin Li       clearDiagnostics(E->getSelectorLoc(0));
122*67e74705SXin Li       if (tryRemoving(E))
123*67e74705SXin Li         return true;
124*67e74705SXin Li       Pass.TA.replace(E->getSourceRange(), "self");
125*67e74705SXin Li       return true;
126*67e74705SXin Li     }
127*67e74705SXin Li     case ObjCMessageExpr::Instance:
128*67e74705SXin Li       break;
129*67e74705SXin Li     }
130*67e74705SXin Li 
131*67e74705SXin Li     Expr *rec = E->getInstanceReceiver();
132*67e74705SXin Li     if (!rec) return true;
133*67e74705SXin Li 
134*67e74705SXin Li     Transaction Trans(Pass.TA);
135*67e74705SXin Li     clearDiagnostics(E->getSelectorLoc(0));
136*67e74705SXin Li 
137*67e74705SXin Li     ObjCMessageExpr *Msg = E;
138*67e74705SXin Li     Expr *RecContainer = Msg;
139*67e74705SXin Li     SourceRange RecRange = rec->getSourceRange();
140*67e74705SXin Li     checkForGCDOrXPC(Msg, RecContainer, rec, RecRange);
141*67e74705SXin Li 
142*67e74705SXin Li     if (Msg->getMethodFamily() == OMF_release &&
143*67e74705SXin Li         isRemovable(RecContainer) && isInAtFinally(RecContainer)) {
144*67e74705SXin Li       // Change the -release to "receiver = nil" in a finally to avoid a leak
145*67e74705SXin Li       // when an exception is thrown.
146*67e74705SXin Li       Pass.TA.replace(RecContainer->getSourceRange(), RecRange);
147*67e74705SXin Li       std::string str = " = ";
148*67e74705SXin Li       str += getNilString(Pass);
149*67e74705SXin Li       Pass.TA.insertAfterToken(RecRange.getEnd(), str);
150*67e74705SXin Li       return true;
151*67e74705SXin Li     }
152*67e74705SXin Li 
153*67e74705SXin Li     if (hasSideEffects(rec, Pass.Ctx) || !tryRemoving(RecContainer))
154*67e74705SXin Li       Pass.TA.replace(RecContainer->getSourceRange(), RecRange);
155*67e74705SXin Li 
156*67e74705SXin Li     return true;
157*67e74705SXin Li   }
158*67e74705SXin Li 
159*67e74705SXin Li private:
160*67e74705SXin Li   /// \brief Checks for idioms where an unused -autorelease is common.
161*67e74705SXin Li   ///
162*67e74705SXin Li   /// Returns true for this idiom which is common in property
163*67e74705SXin Li   /// setters:
164*67e74705SXin Li   ///
165*67e74705SXin Li   ///   [backingValue autorelease];
166*67e74705SXin Li   ///   backingValue = [newValue retain]; // in general a +1 assign
167*67e74705SXin Li   ///
168*67e74705SXin Li   /// For these as well:
169*67e74705SXin Li   ///
170*67e74705SXin Li   ///   [[var retain] autorelease];
171*67e74705SXin Li   ///   return var;
172*67e74705SXin Li   ///
isCommonUnusedAutorelease(ObjCMessageExpr * E)173*67e74705SXin Li   bool isCommonUnusedAutorelease(ObjCMessageExpr *E) {
174*67e74705SXin Li     return isPlusOneAssignBeforeOrAfterAutorelease(E) ||
175*67e74705SXin Li            isReturnedAfterAutorelease(E);
176*67e74705SXin Li   }
177*67e74705SXin Li 
isReturnedAfterAutorelease(ObjCMessageExpr * E)178*67e74705SXin Li   bool isReturnedAfterAutorelease(ObjCMessageExpr *E) {
179*67e74705SXin Li     Expr *Rec = E->getInstanceReceiver();
180*67e74705SXin Li     if (!Rec)
181*67e74705SXin Li       return false;
182*67e74705SXin Li 
183*67e74705SXin Li     Decl *RefD = getReferencedDecl(Rec);
184*67e74705SXin Li     if (!RefD)
185*67e74705SXin Li       return false;
186*67e74705SXin Li 
187*67e74705SXin Li     Stmt *nextStmt = getNextStmt(E);
188*67e74705SXin Li     if (!nextStmt)
189*67e74705SXin Li       return false;
190*67e74705SXin Li 
191*67e74705SXin Li     // Check for "return <variable>;".
192*67e74705SXin Li 
193*67e74705SXin Li     if (ReturnStmt *RetS = dyn_cast<ReturnStmt>(nextStmt))
194*67e74705SXin Li       return RefD == getReferencedDecl(RetS->getRetValue());
195*67e74705SXin Li 
196*67e74705SXin Li     return false;
197*67e74705SXin Li   }
198*67e74705SXin Li 
isPlusOneAssignBeforeOrAfterAutorelease(ObjCMessageExpr * E)199*67e74705SXin Li   bool isPlusOneAssignBeforeOrAfterAutorelease(ObjCMessageExpr *E) {
200*67e74705SXin Li     Expr *Rec = E->getInstanceReceiver();
201*67e74705SXin Li     if (!Rec)
202*67e74705SXin Li       return false;
203*67e74705SXin Li 
204*67e74705SXin Li     Decl *RefD = getReferencedDecl(Rec);
205*67e74705SXin Li     if (!RefD)
206*67e74705SXin Li       return false;
207*67e74705SXin Li 
208*67e74705SXin Li     Stmt *prevStmt, *nextStmt;
209*67e74705SXin Li     std::tie(prevStmt, nextStmt) = getPreviousAndNextStmt(E);
210*67e74705SXin Li 
211*67e74705SXin Li     return isPlusOneAssignToVar(prevStmt, RefD) ||
212*67e74705SXin Li            isPlusOneAssignToVar(nextStmt, RefD);
213*67e74705SXin Li   }
214*67e74705SXin Li 
isPlusOneAssignToVar(Stmt * S,Decl * RefD)215*67e74705SXin Li   bool isPlusOneAssignToVar(Stmt *S, Decl *RefD) {
216*67e74705SXin Li     if (!S)
217*67e74705SXin Li       return false;
218*67e74705SXin Li 
219*67e74705SXin Li     // Check for "RefD = [+1 retained object];".
220*67e74705SXin Li 
221*67e74705SXin Li     if (BinaryOperator *Bop = dyn_cast<BinaryOperator>(S)) {
222*67e74705SXin Li       return (RefD == getReferencedDecl(Bop->getLHS())) && isPlusOneAssign(Bop);
223*67e74705SXin Li     }
224*67e74705SXin Li 
225*67e74705SXin Li     if (DeclStmt *DS = dyn_cast<DeclStmt>(S)) {
226*67e74705SXin Li       if (DS->isSingleDecl() && DS->getSingleDecl() == RefD) {
227*67e74705SXin Li         if (VarDecl *VD = dyn_cast<VarDecl>(RefD))
228*67e74705SXin Li           return isPlusOne(VD->getInit());
229*67e74705SXin Li       }
230*67e74705SXin Li       return false;
231*67e74705SXin Li     }
232*67e74705SXin Li 
233*67e74705SXin Li     return false;
234*67e74705SXin Li   }
235*67e74705SXin Li 
getNextStmt(Expr * E)236*67e74705SXin Li   Stmt *getNextStmt(Expr *E) {
237*67e74705SXin Li     return getPreviousAndNextStmt(E).second;
238*67e74705SXin Li   }
239*67e74705SXin Li 
getPreviousAndNextStmt(Expr * E)240*67e74705SXin Li   std::pair<Stmt *, Stmt *> getPreviousAndNextStmt(Expr *E) {
241*67e74705SXin Li     Stmt *prevStmt = nullptr, *nextStmt = nullptr;
242*67e74705SXin Li     if (!E)
243*67e74705SXin Li       return std::make_pair(prevStmt, nextStmt);
244*67e74705SXin Li 
245*67e74705SXin Li     Stmt *OuterS = E, *InnerS;
246*67e74705SXin Li     do {
247*67e74705SXin Li       InnerS = OuterS;
248*67e74705SXin Li       OuterS = StmtMap->getParent(InnerS);
249*67e74705SXin Li     }
250*67e74705SXin Li     while (OuterS && (isa<ParenExpr>(OuterS) ||
251*67e74705SXin Li                       isa<CastExpr>(OuterS) ||
252*67e74705SXin Li                       isa<ExprWithCleanups>(OuterS)));
253*67e74705SXin Li 
254*67e74705SXin Li     if (!OuterS)
255*67e74705SXin Li       return std::make_pair(prevStmt, nextStmt);
256*67e74705SXin Li 
257*67e74705SXin Li     Stmt::child_iterator currChildS = OuterS->child_begin();
258*67e74705SXin Li     Stmt::child_iterator childE = OuterS->child_end();
259*67e74705SXin Li     Stmt::child_iterator prevChildS = childE;
260*67e74705SXin Li     for (; currChildS != childE; ++currChildS) {
261*67e74705SXin Li       if (*currChildS == InnerS)
262*67e74705SXin Li         break;
263*67e74705SXin Li       prevChildS = currChildS;
264*67e74705SXin Li     }
265*67e74705SXin Li 
266*67e74705SXin Li     if (prevChildS != childE) {
267*67e74705SXin Li       prevStmt = *prevChildS;
268*67e74705SXin Li       if (prevStmt)
269*67e74705SXin Li         prevStmt = prevStmt->IgnoreImplicit();
270*67e74705SXin Li     }
271*67e74705SXin Li 
272*67e74705SXin Li     if (currChildS == childE)
273*67e74705SXin Li       return std::make_pair(prevStmt, nextStmt);
274*67e74705SXin Li     ++currChildS;
275*67e74705SXin Li     if (currChildS == childE)
276*67e74705SXin Li       return std::make_pair(prevStmt, nextStmt);
277*67e74705SXin Li 
278*67e74705SXin Li     nextStmt = *currChildS;
279*67e74705SXin Li     if (nextStmt)
280*67e74705SXin Li       nextStmt = nextStmt->IgnoreImplicit();
281*67e74705SXin Li 
282*67e74705SXin Li     return std::make_pair(prevStmt, nextStmt);
283*67e74705SXin Li   }
284*67e74705SXin Li 
getReferencedDecl(Expr * E)285*67e74705SXin Li   Decl *getReferencedDecl(Expr *E) {
286*67e74705SXin Li     if (!E)
287*67e74705SXin Li       return nullptr;
288*67e74705SXin Li 
289*67e74705SXin Li     E = E->IgnoreParenCasts();
290*67e74705SXin Li     if (ObjCMessageExpr *ME = dyn_cast<ObjCMessageExpr>(E)) {
291*67e74705SXin Li       switch (ME->getMethodFamily()) {
292*67e74705SXin Li       case OMF_copy:
293*67e74705SXin Li       case OMF_autorelease:
294*67e74705SXin Li       case OMF_release:
295*67e74705SXin Li       case OMF_retain:
296*67e74705SXin Li         return getReferencedDecl(ME->getInstanceReceiver());
297*67e74705SXin Li       default:
298*67e74705SXin Li         return nullptr;
299*67e74705SXin Li       }
300*67e74705SXin Li     }
301*67e74705SXin Li     if (DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(E))
302*67e74705SXin Li       return DRE->getDecl();
303*67e74705SXin Li     if (MemberExpr *ME = dyn_cast<MemberExpr>(E))
304*67e74705SXin Li       return ME->getMemberDecl();
305*67e74705SXin Li     if (ObjCIvarRefExpr *IRE = dyn_cast<ObjCIvarRefExpr>(E))
306*67e74705SXin Li       return IRE->getDecl();
307*67e74705SXin Li 
308*67e74705SXin Li     return nullptr;
309*67e74705SXin Li   }
310*67e74705SXin Li 
311*67e74705SXin Li   /// \brief Check if the retain/release is due to a GCD/XPC macro that are
312*67e74705SXin Li   /// defined as:
313*67e74705SXin Li   ///
314*67e74705SXin Li   /// #define dispatch_retain(object) ({ dispatch_object_t _o = (object); _dispatch_object_validate(_o); (void)[_o retain]; })
315*67e74705SXin Li   /// #define dispatch_release(object) ({ dispatch_object_t _o = (object); _dispatch_object_validate(_o); [_o release]; })
316*67e74705SXin Li   /// #define xpc_retain(object) ({ xpc_object_t _o = (object); _xpc_object_validate(_o); [_o retain]; })
317*67e74705SXin Li   /// #define xpc_release(object) ({ xpc_object_t _o = (object); _xpc_object_validate(_o); [_o release]; })
318*67e74705SXin Li   ///
319*67e74705SXin Li   /// and return the top container which is the StmtExpr and the macro argument
320*67e74705SXin Li   /// expression.
checkForGCDOrXPC(ObjCMessageExpr * Msg,Expr * & RecContainer,Expr * & Rec,SourceRange & RecRange)321*67e74705SXin Li   void checkForGCDOrXPC(ObjCMessageExpr *Msg, Expr *&RecContainer,
322*67e74705SXin Li                         Expr *&Rec, SourceRange &RecRange) {
323*67e74705SXin Li     SourceLocation Loc = Msg->getExprLoc();
324*67e74705SXin Li     if (!Loc.isMacroID())
325*67e74705SXin Li       return;
326*67e74705SXin Li     SourceManager &SM = Pass.Ctx.getSourceManager();
327*67e74705SXin Li     StringRef MacroName = Lexer::getImmediateMacroName(Loc, SM,
328*67e74705SXin Li                                                      Pass.Ctx.getLangOpts());
329*67e74705SXin Li     bool isGCDOrXPC = llvm::StringSwitch<bool>(MacroName)
330*67e74705SXin Li         .Case("dispatch_retain", true)
331*67e74705SXin Li         .Case("dispatch_release", true)
332*67e74705SXin Li         .Case("xpc_retain", true)
333*67e74705SXin Li         .Case("xpc_release", true)
334*67e74705SXin Li         .Default(false);
335*67e74705SXin Li     if (!isGCDOrXPC)
336*67e74705SXin Li       return;
337*67e74705SXin Li 
338*67e74705SXin Li     StmtExpr *StmtE = nullptr;
339*67e74705SXin Li     Stmt *S = Msg;
340*67e74705SXin Li     while (S) {
341*67e74705SXin Li       if (StmtExpr *SE = dyn_cast<StmtExpr>(S)) {
342*67e74705SXin Li         StmtE = SE;
343*67e74705SXin Li         break;
344*67e74705SXin Li       }
345*67e74705SXin Li       S = StmtMap->getParent(S);
346*67e74705SXin Li     }
347*67e74705SXin Li 
348*67e74705SXin Li     if (!StmtE)
349*67e74705SXin Li       return;
350*67e74705SXin Li 
351*67e74705SXin Li     Stmt::child_range StmtExprChild = StmtE->children();
352*67e74705SXin Li     if (StmtExprChild.begin() == StmtExprChild.end())
353*67e74705SXin Li       return;
354*67e74705SXin Li     auto *CompS = dyn_cast_or_null<CompoundStmt>(*StmtExprChild.begin());
355*67e74705SXin Li     if (!CompS)
356*67e74705SXin Li       return;
357*67e74705SXin Li 
358*67e74705SXin Li     Stmt::child_range CompStmtChild = CompS->children();
359*67e74705SXin Li     if (CompStmtChild.begin() == CompStmtChild.end())
360*67e74705SXin Li       return;
361*67e74705SXin Li     auto *DeclS = dyn_cast_or_null<DeclStmt>(*CompStmtChild.begin());
362*67e74705SXin Li     if (!DeclS)
363*67e74705SXin Li       return;
364*67e74705SXin Li     if (!DeclS->isSingleDecl())
365*67e74705SXin Li       return;
366*67e74705SXin Li     VarDecl *VD = dyn_cast_or_null<VarDecl>(DeclS->getSingleDecl());
367*67e74705SXin Li     if (!VD)
368*67e74705SXin Li       return;
369*67e74705SXin Li     Expr *Init = VD->getInit();
370*67e74705SXin Li     if (!Init)
371*67e74705SXin Li       return;
372*67e74705SXin Li 
373*67e74705SXin Li     RecContainer = StmtE;
374*67e74705SXin Li     Rec = Init->IgnoreParenImpCasts();
375*67e74705SXin Li     if (ExprWithCleanups *EWC = dyn_cast<ExprWithCleanups>(Rec))
376*67e74705SXin Li       Rec = EWC->getSubExpr()->IgnoreParenImpCasts();
377*67e74705SXin Li     RecRange = Rec->getSourceRange();
378*67e74705SXin Li     if (SM.isMacroArgExpansion(RecRange.getBegin()))
379*67e74705SXin Li       RecRange.setBegin(SM.getImmediateSpellingLoc(RecRange.getBegin()));
380*67e74705SXin Li     if (SM.isMacroArgExpansion(RecRange.getEnd()))
381*67e74705SXin Li       RecRange.setEnd(SM.getImmediateSpellingLoc(RecRange.getEnd()));
382*67e74705SXin Li   }
383*67e74705SXin Li 
clearDiagnostics(SourceLocation loc) const384*67e74705SXin Li   void clearDiagnostics(SourceLocation loc) const {
385*67e74705SXin Li     Pass.TA.clearDiagnostic(diag::err_arc_illegal_explicit_message,
386*67e74705SXin Li                             diag::err_unavailable,
387*67e74705SXin Li                             diag::err_unavailable_message,
388*67e74705SXin Li                             loc);
389*67e74705SXin Li   }
390*67e74705SXin Li 
isDelegateMessage(Expr * E) const391*67e74705SXin Li   bool isDelegateMessage(Expr *E) const {
392*67e74705SXin Li     if (!E) return false;
393*67e74705SXin Li 
394*67e74705SXin Li     E = E->IgnoreParenCasts();
395*67e74705SXin Li 
396*67e74705SXin Li     // Also look through property-getter sugar.
397*67e74705SXin Li     if (PseudoObjectExpr *pseudoOp = dyn_cast<PseudoObjectExpr>(E))
398*67e74705SXin Li       E = pseudoOp->getResultExpr()->IgnoreImplicit();
399*67e74705SXin Li 
400*67e74705SXin Li     if (ObjCMessageExpr *ME = dyn_cast<ObjCMessageExpr>(E))
401*67e74705SXin Li       return (ME->isInstanceMessage() && ME->getSelector() == DelegateSel);
402*67e74705SXin Li 
403*67e74705SXin Li     return false;
404*67e74705SXin Li   }
405*67e74705SXin Li 
isInAtFinally(Expr * E) const406*67e74705SXin Li   bool isInAtFinally(Expr *E) const {
407*67e74705SXin Li     assert(E);
408*67e74705SXin Li     Stmt *S = E;
409*67e74705SXin Li     while (S) {
410*67e74705SXin Li       if (isa<ObjCAtFinallyStmt>(S))
411*67e74705SXin Li         return true;
412*67e74705SXin Li       S = StmtMap->getParent(S);
413*67e74705SXin Li     }
414*67e74705SXin Li 
415*67e74705SXin Li     return false;
416*67e74705SXin Li   }
417*67e74705SXin Li 
isRemovable(Expr * E) const418*67e74705SXin Li   bool isRemovable(Expr *E) const {
419*67e74705SXin Li     return Removables.count(E);
420*67e74705SXin Li   }
421*67e74705SXin Li 
tryRemoving(Expr * E) const422*67e74705SXin Li   bool tryRemoving(Expr *E) const {
423*67e74705SXin Li     if (isRemovable(E)) {
424*67e74705SXin Li       Pass.TA.removeStmt(E);
425*67e74705SXin Li       return true;
426*67e74705SXin Li     }
427*67e74705SXin Li 
428*67e74705SXin Li     Stmt *parent = StmtMap->getParent(E);
429*67e74705SXin Li 
430*67e74705SXin Li     if (ImplicitCastExpr *castE = dyn_cast_or_null<ImplicitCastExpr>(parent))
431*67e74705SXin Li       return tryRemoving(castE);
432*67e74705SXin Li 
433*67e74705SXin Li     if (ParenExpr *parenE = dyn_cast_or_null<ParenExpr>(parent))
434*67e74705SXin Li       return tryRemoving(parenE);
435*67e74705SXin Li 
436*67e74705SXin Li     if (BinaryOperator *
437*67e74705SXin Li           bopE = dyn_cast_or_null<BinaryOperator>(parent)) {
438*67e74705SXin Li       if (bopE->getOpcode() == BO_Comma && bopE->getLHS() == E &&
439*67e74705SXin Li           isRemovable(bopE)) {
440*67e74705SXin Li         Pass.TA.replace(bopE->getSourceRange(), bopE->getRHS()->getSourceRange());
441*67e74705SXin Li         return true;
442*67e74705SXin Li       }
443*67e74705SXin Li     }
444*67e74705SXin Li 
445*67e74705SXin Li     return false;
446*67e74705SXin Li   }
447*67e74705SXin Li 
448*67e74705SXin Li };
449*67e74705SXin Li 
450*67e74705SXin Li } // anonymous namespace
451*67e74705SXin Li 
removeRetainReleaseDeallocFinalize(MigrationPass & pass)452*67e74705SXin Li void trans::removeRetainReleaseDeallocFinalize(MigrationPass &pass) {
453*67e74705SXin Li   BodyTransform<RetainReleaseDeallocRemover> trans(pass);
454*67e74705SXin Li   trans.TraverseDecl(pass.Ctx.getTranslationUnitDecl());
455*67e74705SXin Li }
456