| Ted Kremenek | 3b28f49 | 2008-07-23 00:45:26 +0000 | [diff] [blame] | 1 | //==- CheckObjCUnusedIVars.cpp - Check for unused ivars ----------*- C++ -*-==// | 
|  | 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 | //  This file defines a CheckObjCUnusedIvars, a checker that | 
|  | 11 | //  analyzes an Objective-C class's interface/implementation to determine if it | 
|  | 12 | //  has any ivars that are never accessed. | 
|  | 13 | // | 
|  | 14 | //===----------------------------------------------------------------------===// | 
|  | 15 |  | 
|  | 16 | #include "clang/Analysis/LocalCheckers.h" | 
|  | 17 | #include "clang/Analysis/PathDiagnostic.h" | 
|  | 18 | #include "clang/Analysis/PathSensitive/BugReporter.h" | 
|  | 19 | #include "clang/AST/ExprObjC.h" | 
|  | 20 | #include "clang/AST/Expr.h" | 
|  | 21 | #include "clang/AST/DeclObjC.h" | 
|  | 22 | #include "clang/Basic/LangOptions.h" | 
|  | 23 | #include <sstream> | 
|  | 24 |  | 
|  | 25 | using namespace clang; | 
|  | 26 |  | 
|  | 27 | enum IVarState { Unused, Used }; | 
|  | 28 | typedef llvm::DenseMap<ObjCIvarDecl*,IVarState> IvarUsageMap; | 
|  | 29 |  | 
|  | 30 | static void Scan(IvarUsageMap& M, Stmt* S) { | 
|  | 31 | if (!S) | 
|  | 32 | return; | 
|  | 33 |  | 
|  | 34 | if (ObjCIvarRefExpr* Ex = dyn_cast<ObjCIvarRefExpr>(S)) { | 
|  | 35 | ObjCIvarDecl* D = Ex->getDecl(); | 
|  | 36 | IvarUsageMap::iterator I = M.find(D); | 
|  | 37 | if (I != M.end()) I->second = Used; | 
| Ted Kremenek | d074ce4 | 2008-07-25 20:28:02 +0000 | [diff] [blame] | 38 | return; | 
| Ted Kremenek | 3b28f49 | 2008-07-23 00:45:26 +0000 | [diff] [blame] | 39 | } | 
| Ted Kremenek | d074ce4 | 2008-07-25 20:28:02 +0000 | [diff] [blame] | 40 |  | 
|  | 41 | for (Stmt::child_iterator I=S->child_begin(), E=S->child_end(); I!=E;++I) | 
|  | 42 | Scan(M, *I); | 
|  | 43 | } | 
|  | 44 |  | 
|  | 45 | static void Scan(IvarUsageMap& M, ObjCPropertyImplDecl* D) { | 
|  | 46 | if (!D) | 
|  | 47 | return; | 
|  | 48 |  | 
|  | 49 | ObjCIvarDecl* ID = D->getPropertyIvarDecl(); | 
|  | 50 |  | 
|  | 51 | if (!ID) | 
|  | 52 | return; | 
|  | 53 |  | 
|  | 54 | IvarUsageMap::iterator I = M.find(ID); | 
|  | 55 | if (I != M.end()) I->second = Used; | 
| Ted Kremenek | 3b28f49 | 2008-07-23 00:45:26 +0000 | [diff] [blame] | 56 | } | 
|  | 57 |  | 
|  | 58 | void clang::CheckObjCUnusedIvar(ObjCImplementationDecl* D, BugReporter& BR) { | 
|  | 59 |  | 
|  | 60 | ObjCInterfaceDecl* ID = D->getClassInterface(); | 
|  | 61 | IvarUsageMap M; | 
|  | 62 |  | 
|  | 63 |  | 
| Douglas Gregor | 29bd76f | 2009-04-23 01:02:12 +0000 | [diff] [blame] | 64 | ASTContext &Ctx = BR.getContext(); | 
|  | 65 |  | 
| Ted Kremenek | 3b28f49 | 2008-07-23 00:45:26 +0000 | [diff] [blame] | 66 | // Iterate over the ivars. | 
|  | 67 | for (ObjCInterfaceDecl::ivar_iterator I=ID->ivar_begin(), E=ID->ivar_end(); | 
|  | 68 | I!=E; ++I) { | 
|  | 69 |  | 
|  | 70 | ObjCIvarDecl* ID = *I; | 
|  | 71 |  | 
|  | 72 | // Ignore ivars that aren't private. | 
| Ted Kremenek | 6b6a4b6 | 2008-07-23 17:14:39 +0000 | [diff] [blame] | 73 | if (ID->getAccessControl() != ObjCIvarDecl::Private) | 
| Ted Kremenek | 3b28f49 | 2008-07-23 00:45:26 +0000 | [diff] [blame] | 74 | continue; | 
| Ted Kremenek | 46abc7d | 2008-07-23 18:21:36 +0000 | [diff] [blame] | 75 |  | 
|  | 76 | // Skip IB Outlets. | 
| Douglas Gregor | 78bd61f | 2009-06-18 16:11:24 +0000 | [diff] [blame] | 77 | if (ID->getAttr<IBOutletAttr>(Ctx)) | 
| Ted Kremenek | 3b28f49 | 2008-07-23 00:45:26 +0000 | [diff] [blame] | 78 | continue; | 
|  | 79 |  | 
|  | 80 | M[ID] = Unused; | 
|  | 81 | } | 
|  | 82 |  | 
|  | 83 | if (M.empty()) | 
|  | 84 | return; | 
|  | 85 |  | 
|  | 86 | // Now scan the methods for accesses. | 
| Douglas Gregor | 29bd76f | 2009-04-23 01:02:12 +0000 | [diff] [blame] | 87 | for (ObjCImplementationDecl::instmeth_iterator I = D->instmeth_begin(Ctx), | 
|  | 88 | E = D->instmeth_end(Ctx); I!=E; ++I) | 
|  | 89 | Scan(M, (*I)->getBody(Ctx)); | 
| Ted Kremenek | 3b28f49 | 2008-07-23 00:45:26 +0000 | [diff] [blame] | 90 |  | 
| Ted Kremenek | d074ce4 | 2008-07-25 20:28:02 +0000 | [diff] [blame] | 91 | // Scan for @synthesized property methods that act as setters/getters | 
|  | 92 | // to an ivar. | 
| Douglas Gregor | 29bd76f | 2009-04-23 01:02:12 +0000 | [diff] [blame] | 93 | for (ObjCImplementationDecl::propimpl_iterator I = D->propimpl_begin(Ctx), | 
|  | 94 | E = D->propimpl_end(Ctx); I!=E; ++I) | 
| Ted Kremenek | d074ce4 | 2008-07-25 20:28:02 +0000 | [diff] [blame] | 95 | Scan(M, *I); | 
|  | 96 |  | 
| Ted Kremenek | 3b28f49 | 2008-07-23 00:45:26 +0000 | [diff] [blame] | 97 | // Find ivars that are unused. | 
|  | 98 | for (IvarUsageMap::iterator I = M.begin(), E = M.end(); I!=E; ++I) | 
|  | 99 | if (I->second == Unused) { | 
|  | 100 |  | 
|  | 101 | std::ostringstream os; | 
| Chris Lattner | f3d3fae | 2008-11-24 05:29:24 +0000 | [diff] [blame] | 102 | os << "Instance variable '" << I->first->getNameAsString() | 
|  | 103 | << "' in class '" << ID->getNameAsString() | 
| Ted Kremenek | d074ce4 | 2008-07-25 20:28:02 +0000 | [diff] [blame] | 104 | << "' is never used by the methods in its @implementation " | 
|  | 105 | "(although it may be used by category methods)."; | 
| Ted Kremenek | 46abc7d | 2008-07-23 18:21:36 +0000 | [diff] [blame] | 106 |  | 
| Ted Kremenek | 6c3413c | 2009-04-02 02:44:03 +0000 | [diff] [blame] | 107 | BR.EmitBasicReport("Unused instance variable", "Optimization", | 
| Ted Kremenek | 3b28f49 | 2008-07-23 00:45:26 +0000 | [diff] [blame] | 108 | os.str().c_str(), I->first->getLocation()); | 
|  | 109 | } | 
|  | 110 | } | 
|  | 111 |  |