1*e5dd7070Spatrick //===--- TransUnusedInitDelegate.cpp - Transformations to ARC mode --------===//
2*e5dd7070Spatrick //
3*e5dd7070Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*e5dd7070Spatrick // See https://llvm.org/LICENSE.txt for license information.
5*e5dd7070Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*e5dd7070Spatrick //
7*e5dd7070Spatrick //===----------------------------------------------------------------------===//
8*e5dd7070Spatrick // Transformations:
9*e5dd7070Spatrick //===----------------------------------------------------------------------===//
10*e5dd7070Spatrick //
11*e5dd7070Spatrick // rewriteUnusedInitDelegate:
12*e5dd7070Spatrick //
13*e5dd7070Spatrick // Rewrites an unused result of calling a delegate initialization, to assigning
14*e5dd7070Spatrick // the result to self.
15*e5dd7070Spatrick // e.g
16*e5dd7070Spatrick // [self init];
17*e5dd7070Spatrick // ---->
18*e5dd7070Spatrick // self = [self init];
19*e5dd7070Spatrick //
20*e5dd7070Spatrick //===----------------------------------------------------------------------===//
21*e5dd7070Spatrick
22*e5dd7070Spatrick #include "Transforms.h"
23*e5dd7070Spatrick #include "Internals.h"
24*e5dd7070Spatrick #include "clang/AST/ASTContext.h"
25*e5dd7070Spatrick #include "clang/Sema/SemaDiagnostic.h"
26*e5dd7070Spatrick
27*e5dd7070Spatrick using namespace clang;
28*e5dd7070Spatrick using namespace arcmt;
29*e5dd7070Spatrick using namespace trans;
30*e5dd7070Spatrick
31*e5dd7070Spatrick namespace {
32*e5dd7070Spatrick
33*e5dd7070Spatrick class UnusedInitRewriter : public RecursiveASTVisitor<UnusedInitRewriter> {
34*e5dd7070Spatrick Stmt *Body;
35*e5dd7070Spatrick MigrationPass &Pass;
36*e5dd7070Spatrick
37*e5dd7070Spatrick ExprSet Removables;
38*e5dd7070Spatrick
39*e5dd7070Spatrick public:
UnusedInitRewriter(MigrationPass & pass)40*e5dd7070Spatrick UnusedInitRewriter(MigrationPass &pass)
41*e5dd7070Spatrick : Body(nullptr), Pass(pass) { }
42*e5dd7070Spatrick
transformBody(Stmt * body,Decl * ParentD)43*e5dd7070Spatrick void transformBody(Stmt *body, Decl *ParentD) {
44*e5dd7070Spatrick Body = body;
45*e5dd7070Spatrick collectRemovables(body, Removables);
46*e5dd7070Spatrick TraverseStmt(body);
47*e5dd7070Spatrick }
48*e5dd7070Spatrick
VisitObjCMessageExpr(ObjCMessageExpr * ME)49*e5dd7070Spatrick bool VisitObjCMessageExpr(ObjCMessageExpr *ME) {
50*e5dd7070Spatrick if (ME->isDelegateInitCall() &&
51*e5dd7070Spatrick isRemovable(ME) &&
52*e5dd7070Spatrick Pass.TA.hasDiagnostic(diag::err_arc_unused_init_message,
53*e5dd7070Spatrick ME->getExprLoc())) {
54*e5dd7070Spatrick Transaction Trans(Pass.TA);
55*e5dd7070Spatrick Pass.TA.clearDiagnostic(diag::err_arc_unused_init_message,
56*e5dd7070Spatrick ME->getExprLoc());
57*e5dd7070Spatrick SourceRange ExprRange = ME->getSourceRange();
58*e5dd7070Spatrick Pass.TA.insert(ExprRange.getBegin(), "if (!(self = ");
59*e5dd7070Spatrick std::string retStr = ")) return ";
60*e5dd7070Spatrick retStr += getNilString(Pass);
61*e5dd7070Spatrick Pass.TA.insertAfterToken(ExprRange.getEnd(), retStr);
62*e5dd7070Spatrick }
63*e5dd7070Spatrick return true;
64*e5dd7070Spatrick }
65*e5dd7070Spatrick
66*e5dd7070Spatrick private:
isRemovable(Expr * E) const67*e5dd7070Spatrick bool isRemovable(Expr *E) const {
68*e5dd7070Spatrick return Removables.count(E);
69*e5dd7070Spatrick }
70*e5dd7070Spatrick };
71*e5dd7070Spatrick
72*e5dd7070Spatrick } // anonymous namespace
73*e5dd7070Spatrick
rewriteUnusedInitDelegate(MigrationPass & pass)74*e5dd7070Spatrick void trans::rewriteUnusedInitDelegate(MigrationPass &pass) {
75*e5dd7070Spatrick BodyTransform<UnusedInitRewriter> trans(pass);
76*e5dd7070Spatrick trans.TraverseDecl(pass.Ctx.getTranslationUnitDecl());
77*e5dd7070Spatrick }
78