annotate clang/lib/ARCMigrate/TransRetainReleaseDealloc.cpp @ 176:de4ac79aef9d

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