blob: f03ab5a6f4bc064812f6a974229d4aab4fa05f3d [file] [log] [blame]
Argyrios Kyrtzidis7196d062011-06-21 20:20:39 +00001//===--- TransRetainReleaseDealloc.cpp - Tranformations to ARC mode -------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// removeRetainReleaseDealloc:
11//
12// Removes retain/release/autorelease/dealloc messages.
13//
14// return [[foo retain] autorelease];
15// ---->
16// return foo;
17//
18//===----------------------------------------------------------------------===//
19
20#include "Transforms.h"
21#include "Internals.h"
22#include "clang/Sema/SemaDiagnostic.h"
23#include "clang/AST/ParentMap.h"
24
25using namespace clang;
26using namespace arcmt;
27using namespace trans;
28using llvm::StringRef;
29
30namespace {
31
32class RetainReleaseDeallocRemover :
33 public RecursiveASTVisitor<RetainReleaseDeallocRemover> {
34 Decl *Dcl;
35 Stmt *Body;
36 MigrationPass &Pass;
37
38 ExprSet Removables;
39 llvm::OwningPtr<ParentMap> StmtMap;
40
41public:
42 RetainReleaseDeallocRemover(Decl *D, MigrationPass &pass)
43 : Dcl(D), Body(0), Pass(pass) { }
44
45 void transformBody(Stmt *body) {
46 Body = body;
47 collectRemovables(body, Removables);
48 StmtMap.reset(new ParentMap(body));
49 TraverseStmt(body);
50 }
51
52 bool VisitObjCMessageExpr(ObjCMessageExpr *E) {
53 switch (E->getMethodFamily()) {
54 default:
55 return true;
56 case OMF_retain:
57 case OMF_release:
58 case OMF_autorelease:
59 if (E->getReceiverKind() == ObjCMessageExpr::Instance)
60 if (Expr *rec = E->getInstanceReceiver()) {
61 rec = rec->IgnoreParenImpCasts();
62 if (rec->getType().getObjCLifetime() == Qualifiers::OCL_ExplicitNone){
63 std::string err = "It is not safe to remove '";
64 err += E->getSelector().getAsString() + "' message on "
65 "an __unsafe_unretained type";
66 Pass.TA.reportError(err, rec->getLocStart());
67 return true;
68 }
69 }
70 case OMF_dealloc:
71 break;
72 }
73
74 switch (E->getReceiverKind()) {
75 default:
76 return true;
77 case ObjCMessageExpr::SuperInstance: {
78 Transaction Trans(Pass.TA);
79 Pass.TA.clearDiagnostic(diag::err_arc_illegal_explicit_message,
80 diag::err_unavailable,
81 diag::err_unavailable_message,
82 E->getSuperLoc());
83 if (tryRemoving(E))
84 return true;
85 Pass.TA.replace(E->getSourceRange(), "self");
86 return true;
87 }
88 case ObjCMessageExpr::Instance:
89 break;
90 }
91
92 Expr *rec = E->getInstanceReceiver();
93 if (!rec) return true;
94
95 Transaction Trans(Pass.TA);
96 Pass.TA.clearDiagnostic(diag::err_arc_illegal_explicit_message,
97 diag::err_unavailable,
98 diag::err_unavailable_message,
99 rec->getExprLoc());
100 if (!hasSideEffects(E, Pass.Ctx)) {
101 if (tryRemoving(E))
102 return true;
103 }
104 Pass.TA.replace(E->getSourceRange(), rec->getSourceRange());
105
106 return true;
107 }
108
109private:
110 bool isRemovable(Expr *E) const {
111 return Removables.count(E);
112 }
113
114 bool tryRemoving(Expr *E) const {
115 if (isRemovable(E)) {
116 Pass.TA.removeStmt(E);
117 return true;
118 }
119
120 if (ParenExpr *parenE = dyn_cast_or_null<ParenExpr>(StmtMap->getParent(E)))
121 return tryRemoving(parenE);
122
123 if (BinaryOperator *
124 bopE = dyn_cast_or_null<BinaryOperator>(StmtMap->getParent(E))) {
125 if (bopE->getOpcode() == BO_Comma && bopE->getLHS() == E &&
126 isRemovable(bopE)) {
127 Pass.TA.replace(bopE->getSourceRange(), bopE->getRHS()->getSourceRange());
128 return true;
129 }
130 }
131
132 return false;
133 }
134
135};
136
137} // anonymous namespace
138
139void trans::removeRetainReleaseDealloc(MigrationPass &pass) {
140 BodyTransform<RetainReleaseDeallocRemover> trans(pass);
141 trans.TraverseDecl(pass.Ctx.getTranslationUnitDecl());
142}