2009-11-03 16:03:59 +08:00
|
|
|
//=- NSAutoreleasePoolChecker.cpp --------------------------------*- C++ -*-==//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file defines a NSAutoreleasePoolChecker, a small checker that warns
|
|
|
|
// about subpar uses of NSAutoreleasePool. Note that while the check itself
|
|
|
|
// (in it's current form) could be written as a flow-insensitive check, in
|
|
|
|
// can be potentially enhanced in the future with flow-sensitive information.
|
|
|
|
// It is also a good example of the CheckerVisitor interface.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2010-12-23 02:51:49 +08:00
|
|
|
#include "clang/GR/BugReporter/BugReporter.h"
|
2010-12-23 02:53:44 +08:00
|
|
|
#include "clang/GR/PathSensitive/ExprEngine.h"
|
2010-12-23 02:51:49 +08:00
|
|
|
#include "clang/GR/PathSensitive/CheckerVisitor.h"
|
2009-11-03 16:03:59 +08:00
|
|
|
#include "BasicObjCFoundationChecks.h"
|
|
|
|
#include "clang/AST/DeclObjC.h"
|
|
|
|
#include "clang/AST/Decl.h"
|
|
|
|
|
|
|
|
using namespace clang;
|
2010-12-23 15:20:52 +08:00
|
|
|
using namespace ento;
|
2009-11-03 16:03:59 +08:00
|
|
|
|
|
|
|
namespace {
|
2009-11-28 14:07:30 +08:00
|
|
|
class NSAutoreleasePoolChecker
|
2009-11-03 16:03:59 +08:00
|
|
|
: public CheckerVisitor<NSAutoreleasePoolChecker> {
|
|
|
|
|
|
|
|
Selector releaseS;
|
|
|
|
|
|
|
|
public:
|
|
|
|
NSAutoreleasePoolChecker(Selector release_s) : releaseS(release_s) {}
|
|
|
|
|
|
|
|
static void *getTag() {
|
|
|
|
static int x = 0;
|
|
|
|
return &x;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PreVisitObjCMessageExpr(CheckerContext &C, const ObjCMessageExpr *ME);
|
|
|
|
};
|
|
|
|
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
|
2010-12-23 15:20:52 +08:00
|
|
|
void ento::RegisterNSAutoreleasePoolChecks(ExprEngine &Eng) {
|
2009-11-03 16:03:59 +08:00
|
|
|
ASTContext &Ctx = Eng.getContext();
|
|
|
|
if (Ctx.getLangOptions().getGCMode() != LangOptions::NonGC) {
|
|
|
|
Eng.registerCheck(new NSAutoreleasePoolChecker(GetNullarySelector("release",
|
|
|
|
Ctx)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
NSAutoreleasePoolChecker::PreVisitObjCMessageExpr(CheckerContext &C,
|
|
|
|
const ObjCMessageExpr *ME) {
|
|
|
|
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
const Expr *receiver = ME->getInstanceReceiver();
|
2009-11-03 16:03:59 +08:00
|
|
|
if (!receiver)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// FIXME: Enhance with value-tracking information instead of consulting
|
|
|
|
// the type of the expression.
|
|
|
|
const ObjCObjectPointerType* PT =
|
|
|
|
receiver->getType()->getAs<ObjCObjectPointerType>();
|
2009-11-20 08:12:36 +08:00
|
|
|
|
|
|
|
if (!PT)
|
|
|
|
return;
|
2009-11-03 16:03:59 +08:00
|
|
|
const ObjCInterfaceDecl* OD = PT->getInterfaceDecl();
|
|
|
|
if (!OD)
|
|
|
|
return;
|
|
|
|
if (!OD->getIdentifier()->getName().equals("NSAutoreleasePool"))
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Sending 'release' message?
|
|
|
|
if (ME->getSelector() != releaseS)
|
|
|
|
return;
|
|
|
|
|
|
|
|
SourceRange R = ME->getSourceRange();
|
|
|
|
|
|
|
|
C.getBugReporter().EmitBasicReport("Use -drain instead of -release",
|
|
|
|
"API Upgrade (Apple)",
|
|
|
|
"Use -drain instead of -release when using NSAutoreleasePool "
|
|
|
|
"and garbage collection", ME->getLocStart(), &R, 1);
|
|
|
|
}
|