10b57cec5SDimitry Andric //===--- TransRetainReleaseDealloc.cpp - Transformations to ARC mode ------===//
20b57cec5SDimitry Andric //
30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric //
70b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
80b57cec5SDimitry Andric //
90b57cec5SDimitry Andric // removeRetainReleaseDealloc:
100b57cec5SDimitry Andric //
110b57cec5SDimitry Andric // Removes retain/release/autorelease/dealloc messages.
120b57cec5SDimitry Andric //
130b57cec5SDimitry Andric // return [[foo retain] autorelease];
140b57cec5SDimitry Andric // ---->
150b57cec5SDimitry Andric // return foo;
160b57cec5SDimitry Andric //
170b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
180b57cec5SDimitry Andric
190b57cec5SDimitry Andric #include "Transforms.h"
200b57cec5SDimitry Andric #include "Internals.h"
210b57cec5SDimitry Andric #include "clang/AST/ASTContext.h"
220b57cec5SDimitry Andric #include "clang/AST/ParentMap.h"
230b57cec5SDimitry Andric #include "clang/Basic/SourceManager.h"
240b57cec5SDimitry Andric #include "clang/Lex/Lexer.h"
250b57cec5SDimitry Andric #include "clang/Sema/SemaDiagnostic.h"
260b57cec5SDimitry Andric #include "llvm/ADT/StringSwitch.h"
270b57cec5SDimitry Andric
280b57cec5SDimitry Andric using namespace clang;
290b57cec5SDimitry Andric using namespace arcmt;
300b57cec5SDimitry Andric using namespace trans;
310b57cec5SDimitry Andric
320b57cec5SDimitry Andric namespace {
330b57cec5SDimitry Andric
340b57cec5SDimitry Andric class RetainReleaseDeallocRemover :
350b57cec5SDimitry Andric public RecursiveASTVisitor<RetainReleaseDeallocRemover> {
360b57cec5SDimitry Andric Stmt *Body;
370b57cec5SDimitry Andric MigrationPass &Pass;
380b57cec5SDimitry Andric
390b57cec5SDimitry Andric ExprSet Removables;
400b57cec5SDimitry Andric std::unique_ptr<ParentMap> StmtMap;
410b57cec5SDimitry Andric
420b57cec5SDimitry Andric Selector DelegateSel, FinalizeSel;
430b57cec5SDimitry Andric
440b57cec5SDimitry Andric public:
RetainReleaseDeallocRemover(MigrationPass & pass)450b57cec5SDimitry Andric RetainReleaseDeallocRemover(MigrationPass &pass)
460b57cec5SDimitry Andric : Body(nullptr), Pass(pass) {
470b57cec5SDimitry Andric DelegateSel =
480b57cec5SDimitry Andric Pass.Ctx.Selectors.getNullarySelector(&Pass.Ctx.Idents.get("delegate"));
490b57cec5SDimitry Andric FinalizeSel =
500b57cec5SDimitry Andric Pass.Ctx.Selectors.getNullarySelector(&Pass.Ctx.Idents.get("finalize"));
510b57cec5SDimitry Andric }
520b57cec5SDimitry Andric
transformBody(Stmt * body,Decl * ParentD)530b57cec5SDimitry Andric void transformBody(Stmt *body, Decl *ParentD) {
540b57cec5SDimitry Andric Body = body;
550b57cec5SDimitry Andric collectRemovables(body, Removables);
560b57cec5SDimitry Andric StmtMap.reset(new ParentMap(body));
570b57cec5SDimitry Andric TraverseStmt(body);
580b57cec5SDimitry Andric }
590b57cec5SDimitry Andric
VisitObjCMessageExpr(ObjCMessageExpr * E)600b57cec5SDimitry Andric bool VisitObjCMessageExpr(ObjCMessageExpr *E) {
610b57cec5SDimitry Andric switch (E->getMethodFamily()) {
620b57cec5SDimitry Andric default:
630b57cec5SDimitry Andric if (E->isInstanceMessage() && E->getSelector() == FinalizeSel)
640b57cec5SDimitry Andric break;
650b57cec5SDimitry Andric return true;
660b57cec5SDimitry Andric case OMF_autorelease:
670b57cec5SDimitry Andric if (isRemovable(E)) {
680b57cec5SDimitry Andric if (!isCommonUnusedAutorelease(E)) {
690b57cec5SDimitry Andric // An unused autorelease is badness. If we remove it the receiver
700b57cec5SDimitry Andric // will likely die immediately while previously it was kept alive
710b57cec5SDimitry Andric // by the autorelease pool. This is bad practice in general, leave it
720b57cec5SDimitry Andric // and emit an error to force the user to restructure their code.
730b57cec5SDimitry Andric Pass.TA.reportError(
740b57cec5SDimitry Andric "it is not safe to remove an unused 'autorelease' "
750b57cec5SDimitry Andric "message; its receiver may be destroyed immediately",
760b57cec5SDimitry Andric E->getBeginLoc(), E->getSourceRange());
770b57cec5SDimitry Andric return true;
780b57cec5SDimitry Andric }
790b57cec5SDimitry Andric }
800b57cec5SDimitry Andric // Pass through.
81*bdd1243dSDimitry Andric [[fallthrough]];
820b57cec5SDimitry Andric case OMF_retain:
830b57cec5SDimitry Andric case OMF_release:
840b57cec5SDimitry Andric if (E->getReceiverKind() == ObjCMessageExpr::Instance)
850b57cec5SDimitry Andric if (Expr *rec = E->getInstanceReceiver()) {
860b57cec5SDimitry Andric rec = rec->IgnoreParenImpCasts();
870b57cec5SDimitry Andric if (rec->getType().getObjCLifetime() == Qualifiers::OCL_ExplicitNone &&
880b57cec5SDimitry Andric (E->getMethodFamily() != OMF_retain || isRemovable(E))) {
890b57cec5SDimitry Andric std::string err = "it is not safe to remove '";
900b57cec5SDimitry Andric err += E->getSelector().getAsString() + "' message on "
910b57cec5SDimitry Andric "an __unsafe_unretained type";
920b57cec5SDimitry Andric Pass.TA.reportError(err, rec->getBeginLoc());
930b57cec5SDimitry Andric return true;
940b57cec5SDimitry Andric }
950b57cec5SDimitry Andric
960b57cec5SDimitry Andric if (isGlobalVar(rec) &&
970b57cec5SDimitry Andric (E->getMethodFamily() != OMF_retain || isRemovable(E))) {
980b57cec5SDimitry Andric std::string err = "it is not safe to remove '";
990b57cec5SDimitry Andric err += E->getSelector().getAsString() + "' message on "
1000b57cec5SDimitry Andric "a global variable";
1010b57cec5SDimitry Andric Pass.TA.reportError(err, rec->getBeginLoc());
1020b57cec5SDimitry Andric return true;
1030b57cec5SDimitry Andric }
1040b57cec5SDimitry Andric
1050b57cec5SDimitry Andric if (E->getMethodFamily() == OMF_release && isDelegateMessage(rec)) {
1060b57cec5SDimitry Andric Pass.TA.reportError(
1070b57cec5SDimitry Andric "it is not safe to remove 'retain' "
1080b57cec5SDimitry Andric "message on the result of a 'delegate' message; "
1090b57cec5SDimitry Andric "the object that was passed to 'setDelegate:' may not be "
1100b57cec5SDimitry Andric "properly retained",
1110b57cec5SDimitry Andric rec->getBeginLoc());
1120b57cec5SDimitry Andric return true;
1130b57cec5SDimitry Andric }
1140b57cec5SDimitry Andric }
1150b57cec5SDimitry Andric break;
1160b57cec5SDimitry Andric case OMF_dealloc:
1170b57cec5SDimitry Andric break;
1180b57cec5SDimitry Andric }
1190b57cec5SDimitry Andric
1200b57cec5SDimitry Andric switch (E->getReceiverKind()) {
1210b57cec5SDimitry Andric default:
1220b57cec5SDimitry Andric return true;
1230b57cec5SDimitry Andric case ObjCMessageExpr::SuperInstance: {
1240b57cec5SDimitry Andric Transaction Trans(Pass.TA);
1250b57cec5SDimitry Andric clearDiagnostics(E->getSelectorLoc(0));
1260b57cec5SDimitry Andric if (tryRemoving(E))
1270b57cec5SDimitry Andric return true;
1280b57cec5SDimitry Andric Pass.TA.replace(E->getSourceRange(), "self");
1290b57cec5SDimitry Andric return true;
1300b57cec5SDimitry Andric }
1310b57cec5SDimitry Andric case ObjCMessageExpr::Instance:
1320b57cec5SDimitry Andric break;
1330b57cec5SDimitry Andric }
1340b57cec5SDimitry Andric
1350b57cec5SDimitry Andric Expr *rec = E->getInstanceReceiver();
1360b57cec5SDimitry Andric if (!rec) return true;
1370b57cec5SDimitry Andric
1380b57cec5SDimitry Andric Transaction Trans(Pass.TA);
1390b57cec5SDimitry Andric clearDiagnostics(E->getSelectorLoc(0));
1400b57cec5SDimitry Andric
1410b57cec5SDimitry Andric ObjCMessageExpr *Msg = E;
1420b57cec5SDimitry Andric Expr *RecContainer = Msg;
1430b57cec5SDimitry Andric SourceRange RecRange = rec->getSourceRange();
1440b57cec5SDimitry Andric checkForGCDOrXPC(Msg, RecContainer, rec, RecRange);
1450b57cec5SDimitry Andric
1460b57cec5SDimitry Andric if (Msg->getMethodFamily() == OMF_release &&
1470b57cec5SDimitry Andric isRemovable(RecContainer) && isInAtFinally(RecContainer)) {
1480b57cec5SDimitry Andric // Change the -release to "receiver = nil" in a finally to avoid a leak
1490b57cec5SDimitry Andric // when an exception is thrown.
1500b57cec5SDimitry Andric Pass.TA.replace(RecContainer->getSourceRange(), RecRange);
1510b57cec5SDimitry Andric std::string str = " = ";
1520b57cec5SDimitry Andric str += getNilString(Pass);
1530b57cec5SDimitry Andric Pass.TA.insertAfterToken(RecRange.getEnd(), str);
1540b57cec5SDimitry Andric return true;
1550b57cec5SDimitry Andric }
1560b57cec5SDimitry Andric
1570b57cec5SDimitry Andric if (hasSideEffects(rec, Pass.Ctx) || !tryRemoving(RecContainer))
1580b57cec5SDimitry Andric Pass.TA.replace(RecContainer->getSourceRange(), RecRange);
1590b57cec5SDimitry Andric
1600b57cec5SDimitry Andric return true;
1610b57cec5SDimitry Andric }
1620b57cec5SDimitry Andric
1630b57cec5SDimitry Andric private:
1640b57cec5SDimitry Andric /// Checks for idioms where an unused -autorelease is common.
1650b57cec5SDimitry Andric ///
1660b57cec5SDimitry Andric /// Returns true for this idiom which is common in property
1670b57cec5SDimitry Andric /// setters:
1680b57cec5SDimitry Andric ///
1690b57cec5SDimitry Andric /// [backingValue autorelease];
1700b57cec5SDimitry Andric /// backingValue = [newValue retain]; // in general a +1 assign
1710b57cec5SDimitry Andric ///
1720b57cec5SDimitry Andric /// For these as well:
1730b57cec5SDimitry Andric ///
1740b57cec5SDimitry Andric /// [[var retain] autorelease];
1750b57cec5SDimitry Andric /// return var;
1760b57cec5SDimitry Andric ///
isCommonUnusedAutorelease(ObjCMessageExpr * E)1770b57cec5SDimitry Andric bool isCommonUnusedAutorelease(ObjCMessageExpr *E) {
1780b57cec5SDimitry Andric return isPlusOneAssignBeforeOrAfterAutorelease(E) ||
1790b57cec5SDimitry Andric isReturnedAfterAutorelease(E);
1800b57cec5SDimitry Andric }
1810b57cec5SDimitry Andric
isReturnedAfterAutorelease(ObjCMessageExpr * E)1820b57cec5SDimitry Andric bool isReturnedAfterAutorelease(ObjCMessageExpr *E) {
1830b57cec5SDimitry Andric Expr *Rec = E->getInstanceReceiver();
1840b57cec5SDimitry Andric if (!Rec)
1850b57cec5SDimitry Andric return false;
1860b57cec5SDimitry Andric
1870b57cec5SDimitry Andric Decl *RefD = getReferencedDecl(Rec);
1880b57cec5SDimitry Andric if (!RefD)
1890b57cec5SDimitry Andric return false;
1900b57cec5SDimitry Andric
1910b57cec5SDimitry Andric Stmt *nextStmt = getNextStmt(E);
1920b57cec5SDimitry Andric if (!nextStmt)
1930b57cec5SDimitry Andric return false;
1940b57cec5SDimitry Andric
1950b57cec5SDimitry Andric // Check for "return <variable>;".
1960b57cec5SDimitry Andric
1970b57cec5SDimitry Andric if (ReturnStmt *RetS = dyn_cast<ReturnStmt>(nextStmt))
1980b57cec5SDimitry Andric return RefD == getReferencedDecl(RetS->getRetValue());
1990b57cec5SDimitry Andric
2000b57cec5SDimitry Andric return false;
2010b57cec5SDimitry Andric }
2020b57cec5SDimitry Andric
isPlusOneAssignBeforeOrAfterAutorelease(ObjCMessageExpr * E)2030b57cec5SDimitry Andric bool isPlusOneAssignBeforeOrAfterAutorelease(ObjCMessageExpr *E) {
2040b57cec5SDimitry Andric Expr *Rec = E->getInstanceReceiver();
2050b57cec5SDimitry Andric if (!Rec)
2060b57cec5SDimitry Andric return false;
2070b57cec5SDimitry Andric
2080b57cec5SDimitry Andric Decl *RefD = getReferencedDecl(Rec);
2090b57cec5SDimitry Andric if (!RefD)
2100b57cec5SDimitry Andric return false;
2110b57cec5SDimitry Andric
2120b57cec5SDimitry Andric Stmt *prevStmt, *nextStmt;
2130b57cec5SDimitry Andric std::tie(prevStmt, nextStmt) = getPreviousAndNextStmt(E);
2140b57cec5SDimitry Andric
2150b57cec5SDimitry Andric return isPlusOneAssignToVar(prevStmt, RefD) ||
2160b57cec5SDimitry Andric isPlusOneAssignToVar(nextStmt, RefD);
2170b57cec5SDimitry Andric }
2180b57cec5SDimitry Andric
isPlusOneAssignToVar(Stmt * S,Decl * RefD)2190b57cec5SDimitry Andric bool isPlusOneAssignToVar(Stmt *S, Decl *RefD) {
2200b57cec5SDimitry Andric if (!S)
2210b57cec5SDimitry Andric return false;
2220b57cec5SDimitry Andric
2230b57cec5SDimitry Andric // Check for "RefD = [+1 retained object];".
2240b57cec5SDimitry Andric
2250b57cec5SDimitry Andric if (BinaryOperator *Bop = dyn_cast<BinaryOperator>(S)) {
2260b57cec5SDimitry Andric return (RefD == getReferencedDecl(Bop->getLHS())) && isPlusOneAssign(Bop);
2270b57cec5SDimitry Andric }
2280b57cec5SDimitry Andric
2290b57cec5SDimitry Andric if (DeclStmt *DS = dyn_cast<DeclStmt>(S)) {
2300b57cec5SDimitry Andric if (DS->isSingleDecl() && DS->getSingleDecl() == RefD) {
2310b57cec5SDimitry Andric if (VarDecl *VD = dyn_cast<VarDecl>(RefD))
2320b57cec5SDimitry Andric return isPlusOne(VD->getInit());
2330b57cec5SDimitry Andric }
2340b57cec5SDimitry Andric return false;
2350b57cec5SDimitry Andric }
2360b57cec5SDimitry Andric
2370b57cec5SDimitry Andric return false;
2380b57cec5SDimitry Andric }
2390b57cec5SDimitry Andric
getNextStmt(Expr * E)2400b57cec5SDimitry Andric Stmt *getNextStmt(Expr *E) {
2410b57cec5SDimitry Andric return getPreviousAndNextStmt(E).second;
2420b57cec5SDimitry Andric }
2430b57cec5SDimitry Andric
getPreviousAndNextStmt(Expr * E)2440b57cec5SDimitry Andric std::pair<Stmt *, Stmt *> getPreviousAndNextStmt(Expr *E) {
2450b57cec5SDimitry Andric Stmt *prevStmt = nullptr, *nextStmt = nullptr;
2460b57cec5SDimitry Andric if (!E)
2470b57cec5SDimitry Andric return std::make_pair(prevStmt, nextStmt);
2480b57cec5SDimitry Andric
2490b57cec5SDimitry Andric Stmt *OuterS = E, *InnerS;
2500b57cec5SDimitry Andric do {
2510b57cec5SDimitry Andric InnerS = OuterS;
2520b57cec5SDimitry Andric OuterS = StmtMap->getParent(InnerS);
2530b57cec5SDimitry Andric }
2540b57cec5SDimitry Andric while (OuterS && (isa<ParenExpr>(OuterS) ||
2550b57cec5SDimitry Andric isa<CastExpr>(OuterS) ||
2560b57cec5SDimitry Andric isa<FullExpr>(OuterS)));
2570b57cec5SDimitry Andric
2580b57cec5SDimitry Andric if (!OuterS)
2590b57cec5SDimitry Andric return std::make_pair(prevStmt, nextStmt);
2600b57cec5SDimitry Andric
2610b57cec5SDimitry Andric Stmt::child_iterator currChildS = OuterS->child_begin();
2620b57cec5SDimitry Andric Stmt::child_iterator childE = OuterS->child_end();
2630b57cec5SDimitry Andric Stmt::child_iterator prevChildS = childE;
2640b57cec5SDimitry Andric for (; currChildS != childE; ++currChildS) {
2650b57cec5SDimitry Andric if (*currChildS == InnerS)
2660b57cec5SDimitry Andric break;
2670b57cec5SDimitry Andric prevChildS = currChildS;
2680b57cec5SDimitry Andric }
2690b57cec5SDimitry Andric
2700b57cec5SDimitry Andric if (prevChildS != childE) {
2710b57cec5SDimitry Andric prevStmt = *prevChildS;
2720b57cec5SDimitry Andric if (auto *E = dyn_cast_or_null<Expr>(prevStmt))
2730b57cec5SDimitry Andric prevStmt = E->IgnoreImplicit();
2740b57cec5SDimitry Andric }
2750b57cec5SDimitry Andric
2760b57cec5SDimitry Andric if (currChildS == childE)
2770b57cec5SDimitry Andric return std::make_pair(prevStmt, nextStmt);
2780b57cec5SDimitry Andric ++currChildS;
2790b57cec5SDimitry Andric if (currChildS == childE)
2800b57cec5SDimitry Andric return std::make_pair(prevStmt, nextStmt);
2810b57cec5SDimitry Andric
2820b57cec5SDimitry Andric nextStmt = *currChildS;
2830b57cec5SDimitry Andric if (auto *E = dyn_cast_or_null<Expr>(nextStmt))
2840b57cec5SDimitry Andric nextStmt = E->IgnoreImplicit();
2850b57cec5SDimitry Andric
2860b57cec5SDimitry Andric return std::make_pair(prevStmt, nextStmt);
2870b57cec5SDimitry Andric }
2880b57cec5SDimitry Andric
getReferencedDecl(Expr * E)2890b57cec5SDimitry Andric Decl *getReferencedDecl(Expr *E) {
2900b57cec5SDimitry Andric if (!E)
2910b57cec5SDimitry Andric return nullptr;
2920b57cec5SDimitry Andric
2930b57cec5SDimitry Andric E = E->IgnoreParenCasts();
2940b57cec5SDimitry Andric if (ObjCMessageExpr *ME = dyn_cast<ObjCMessageExpr>(E)) {
2950b57cec5SDimitry Andric switch (ME->getMethodFamily()) {
2960b57cec5SDimitry Andric case OMF_copy:
2970b57cec5SDimitry Andric case OMF_autorelease:
2980b57cec5SDimitry Andric case OMF_release:
2990b57cec5SDimitry Andric case OMF_retain:
3000b57cec5SDimitry Andric return getReferencedDecl(ME->getInstanceReceiver());
3010b57cec5SDimitry Andric default:
3020b57cec5SDimitry Andric return nullptr;
3030b57cec5SDimitry Andric }
3040b57cec5SDimitry Andric }
3050b57cec5SDimitry Andric if (DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(E))
3060b57cec5SDimitry Andric return DRE->getDecl();
3070b57cec5SDimitry Andric if (MemberExpr *ME = dyn_cast<MemberExpr>(E))
3080b57cec5SDimitry Andric return ME->getMemberDecl();
3090b57cec5SDimitry Andric if (ObjCIvarRefExpr *IRE = dyn_cast<ObjCIvarRefExpr>(E))
3100b57cec5SDimitry Andric return IRE->getDecl();
3110b57cec5SDimitry Andric
3120b57cec5SDimitry Andric return nullptr;
3130b57cec5SDimitry Andric }
3140b57cec5SDimitry Andric
3150b57cec5SDimitry Andric /// Check if the retain/release is due to a GCD/XPC macro that are
3160b57cec5SDimitry Andric /// defined as:
3170b57cec5SDimitry Andric ///
3180b57cec5SDimitry Andric /// #define dispatch_retain(object) ({ dispatch_object_t _o = (object); _dispatch_object_validate(_o); (void)[_o retain]; })
3190b57cec5SDimitry Andric /// #define dispatch_release(object) ({ dispatch_object_t _o = (object); _dispatch_object_validate(_o); [_o release]; })
3200b57cec5SDimitry Andric /// #define xpc_retain(object) ({ xpc_object_t _o = (object); _xpc_object_validate(_o); [_o retain]; })
3210b57cec5SDimitry Andric /// #define xpc_release(object) ({ xpc_object_t _o = (object); _xpc_object_validate(_o); [_o release]; })
3220b57cec5SDimitry Andric ///
3230b57cec5SDimitry Andric /// and return the top container which is the StmtExpr and the macro argument
3240b57cec5SDimitry Andric /// expression.
checkForGCDOrXPC(ObjCMessageExpr * Msg,Expr * & RecContainer,Expr * & Rec,SourceRange & RecRange)3250b57cec5SDimitry Andric void checkForGCDOrXPC(ObjCMessageExpr *Msg, Expr *&RecContainer,
3260b57cec5SDimitry Andric Expr *&Rec, SourceRange &RecRange) {
3270b57cec5SDimitry Andric SourceLocation Loc = Msg->getExprLoc();
3280b57cec5SDimitry Andric if (!Loc.isMacroID())
3290b57cec5SDimitry Andric return;
3300b57cec5SDimitry Andric SourceManager &SM = Pass.Ctx.getSourceManager();
3310b57cec5SDimitry Andric StringRef MacroName = Lexer::getImmediateMacroName(Loc, SM,
3320b57cec5SDimitry Andric Pass.Ctx.getLangOpts());
3330b57cec5SDimitry Andric bool isGCDOrXPC = llvm::StringSwitch<bool>(MacroName)
3340b57cec5SDimitry Andric .Case("dispatch_retain", true)
3350b57cec5SDimitry Andric .Case("dispatch_release", true)
3360b57cec5SDimitry Andric .Case("xpc_retain", true)
3370b57cec5SDimitry Andric .Case("xpc_release", true)
3380b57cec5SDimitry Andric .Default(false);
3390b57cec5SDimitry Andric if (!isGCDOrXPC)
3400b57cec5SDimitry Andric return;
3410b57cec5SDimitry Andric
3420b57cec5SDimitry Andric StmtExpr *StmtE = nullptr;
3430b57cec5SDimitry Andric Stmt *S = Msg;
3440b57cec5SDimitry Andric while (S) {
3450b57cec5SDimitry Andric if (StmtExpr *SE = dyn_cast<StmtExpr>(S)) {
3460b57cec5SDimitry Andric StmtE = SE;
3470b57cec5SDimitry Andric break;
3480b57cec5SDimitry Andric }
3490b57cec5SDimitry Andric S = StmtMap->getParent(S);
3500b57cec5SDimitry Andric }
3510b57cec5SDimitry Andric
3520b57cec5SDimitry Andric if (!StmtE)
3530b57cec5SDimitry Andric return;
3540b57cec5SDimitry Andric
3550b57cec5SDimitry Andric Stmt::child_range StmtExprChild = StmtE->children();
3560b57cec5SDimitry Andric if (StmtExprChild.begin() == StmtExprChild.end())
3570b57cec5SDimitry Andric return;
3580b57cec5SDimitry Andric auto *CompS = dyn_cast_or_null<CompoundStmt>(*StmtExprChild.begin());
3590b57cec5SDimitry Andric if (!CompS)
3600b57cec5SDimitry Andric return;
3610b57cec5SDimitry Andric
3620b57cec5SDimitry Andric Stmt::child_range CompStmtChild = CompS->children();
3630b57cec5SDimitry Andric if (CompStmtChild.begin() == CompStmtChild.end())
3640b57cec5SDimitry Andric return;
3650b57cec5SDimitry Andric auto *DeclS = dyn_cast_or_null<DeclStmt>(*CompStmtChild.begin());
3660b57cec5SDimitry Andric if (!DeclS)
3670b57cec5SDimitry Andric return;
3680b57cec5SDimitry Andric if (!DeclS->isSingleDecl())
3690b57cec5SDimitry Andric return;
3700b57cec5SDimitry Andric VarDecl *VD = dyn_cast_or_null<VarDecl>(DeclS->getSingleDecl());
3710b57cec5SDimitry Andric if (!VD)
3720b57cec5SDimitry Andric return;
3730b57cec5SDimitry Andric Expr *Init = VD->getInit();
3740b57cec5SDimitry Andric if (!Init)
3750b57cec5SDimitry Andric return;
3760b57cec5SDimitry Andric
3770b57cec5SDimitry Andric RecContainer = StmtE;
3780b57cec5SDimitry Andric Rec = Init->IgnoreParenImpCasts();
3790b57cec5SDimitry Andric if (FullExpr *FE = dyn_cast<FullExpr>(Rec))
3800b57cec5SDimitry Andric Rec = FE->getSubExpr()->IgnoreParenImpCasts();
3810b57cec5SDimitry Andric RecRange = Rec->getSourceRange();
3820b57cec5SDimitry Andric if (SM.isMacroArgExpansion(RecRange.getBegin()))
3830b57cec5SDimitry Andric RecRange.setBegin(SM.getImmediateSpellingLoc(RecRange.getBegin()));
3840b57cec5SDimitry Andric if (SM.isMacroArgExpansion(RecRange.getEnd()))
3850b57cec5SDimitry Andric RecRange.setEnd(SM.getImmediateSpellingLoc(RecRange.getEnd()));
3860b57cec5SDimitry Andric }
3870b57cec5SDimitry Andric
clearDiagnostics(SourceLocation loc) const3880b57cec5SDimitry Andric void clearDiagnostics(SourceLocation loc) const {
3890b57cec5SDimitry Andric Pass.TA.clearDiagnostic(diag::err_arc_illegal_explicit_message,
3900b57cec5SDimitry Andric diag::err_unavailable,
3910b57cec5SDimitry Andric diag::err_unavailable_message,
3920b57cec5SDimitry Andric loc);
3930b57cec5SDimitry Andric }
3940b57cec5SDimitry Andric
isDelegateMessage(Expr * E) const3950b57cec5SDimitry Andric bool isDelegateMessage(Expr *E) const {
3960b57cec5SDimitry Andric if (!E) return false;
3970b57cec5SDimitry Andric
3980b57cec5SDimitry Andric E = E->IgnoreParenCasts();
3990b57cec5SDimitry Andric
4000b57cec5SDimitry Andric // Also look through property-getter sugar.
4010b57cec5SDimitry Andric if (PseudoObjectExpr *pseudoOp = dyn_cast<PseudoObjectExpr>(E))
4020b57cec5SDimitry Andric E = pseudoOp->getResultExpr()->IgnoreImplicit();
4030b57cec5SDimitry Andric
4040b57cec5SDimitry Andric if (ObjCMessageExpr *ME = dyn_cast<ObjCMessageExpr>(E))
4050b57cec5SDimitry Andric return (ME->isInstanceMessage() && ME->getSelector() == DelegateSel);
4060b57cec5SDimitry Andric
4070b57cec5SDimitry Andric return false;
4080b57cec5SDimitry Andric }
4090b57cec5SDimitry Andric
isInAtFinally(Expr * E) const4100b57cec5SDimitry Andric bool isInAtFinally(Expr *E) const {
4110b57cec5SDimitry Andric assert(E);
4120b57cec5SDimitry Andric Stmt *S = E;
4130b57cec5SDimitry Andric while (S) {
4140b57cec5SDimitry Andric if (isa<ObjCAtFinallyStmt>(S))
4150b57cec5SDimitry Andric return true;
4160b57cec5SDimitry Andric S = StmtMap->getParent(S);
4170b57cec5SDimitry Andric }
4180b57cec5SDimitry Andric
4190b57cec5SDimitry Andric return false;
4200b57cec5SDimitry Andric }
4210b57cec5SDimitry Andric
isRemovable(Expr * E) const4220b57cec5SDimitry Andric bool isRemovable(Expr *E) const {
4230b57cec5SDimitry Andric return Removables.count(E);
4240b57cec5SDimitry Andric }
4250b57cec5SDimitry Andric
tryRemoving(Expr * E) const4260b57cec5SDimitry Andric bool tryRemoving(Expr *E) const {
4270b57cec5SDimitry Andric if (isRemovable(E)) {
4280b57cec5SDimitry Andric Pass.TA.removeStmt(E);
4290b57cec5SDimitry Andric return true;
4300b57cec5SDimitry Andric }
4310b57cec5SDimitry Andric
4320b57cec5SDimitry Andric Stmt *parent = StmtMap->getParent(E);
4330b57cec5SDimitry Andric
4340b57cec5SDimitry Andric if (ImplicitCastExpr *castE = dyn_cast_or_null<ImplicitCastExpr>(parent))
4350b57cec5SDimitry Andric return tryRemoving(castE);
4360b57cec5SDimitry Andric
4370b57cec5SDimitry Andric if (ParenExpr *parenE = dyn_cast_or_null<ParenExpr>(parent))
4380b57cec5SDimitry Andric return tryRemoving(parenE);
4390b57cec5SDimitry Andric
4400b57cec5SDimitry Andric if (BinaryOperator *
4410b57cec5SDimitry Andric bopE = dyn_cast_or_null<BinaryOperator>(parent)) {
4420b57cec5SDimitry Andric if (bopE->getOpcode() == BO_Comma && bopE->getLHS() == E &&
4430b57cec5SDimitry Andric isRemovable(bopE)) {
4440b57cec5SDimitry Andric Pass.TA.replace(bopE->getSourceRange(), bopE->getRHS()->getSourceRange());
4450b57cec5SDimitry Andric return true;
4460b57cec5SDimitry Andric }
4470b57cec5SDimitry Andric }
4480b57cec5SDimitry Andric
4490b57cec5SDimitry Andric return false;
4500b57cec5SDimitry Andric }
4510b57cec5SDimitry Andric
4520b57cec5SDimitry Andric };
4530b57cec5SDimitry Andric
4540b57cec5SDimitry Andric } // anonymous namespace
4550b57cec5SDimitry Andric
removeRetainReleaseDeallocFinalize(MigrationPass & pass)4560b57cec5SDimitry Andric void trans::removeRetainReleaseDeallocFinalize(MigrationPass &pass) {
4570b57cec5SDimitry Andric BodyTransform<RetainReleaseDeallocRemover> trans(pass);
4580b57cec5SDimitry Andric trans.TraverseDecl(pass.Ctx.getTranslationUnitDecl());
4590b57cec5SDimitry Andric }
460