aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--lib/Analysis/CFRefCount.cpp16
-rw-r--r--test/Analysis/NSString.m9
2 files changed, 19 insertions, 6 deletions
diff --git a/lib/Analysis/CFRefCount.cpp b/lib/Analysis/CFRefCount.cpp
index 575a47b776..052a610db1 100644
--- a/lib/Analysis/CFRefCount.cpp
+++ b/lib/Analysis/CFRefCount.cpp
@@ -784,18 +784,22 @@ RetainSummaryManager::getPersistentSummary(ArgEffects* AE, RetEffect RetEff,
// Predicates.
//===----------------------------------------------------------------------===//
-bool RetainSummaryManager::isTrackedObjectType(QualType T) {
- if (!Ctx.isObjCObjectPointerType(T))
+bool RetainSummaryManager::isTrackedObjectType(QualType Ty) {
+ if (!Ctx.isObjCObjectPointerType(Ty))
return false;
- // Does it subclass NSObject?
- ObjCInterfaceType* OT = dyn_cast<ObjCInterfaceType>(T.getTypePtr());
+ // We assume that id<..>, id, and "Class" all represent tracked objects.
+ const PointerType *PT = Ty->getAsPointerType();
+ if (PT == 0)
+ return true;
+
+ const ObjCInterfaceType *OT = PT->getPointeeType()->getAsObjCInterfaceType();
// We assume that id<..>, id, and "Class" all represent tracked objects.
if (!OT)
return true;
-
- // Does the object type subclass NSObject?
+
+ // Does the interface subclass NSObject?
// FIXME: We can memoize here if this gets too expensive.
IdentifierInfo* NSObjectII = &Ctx.Idents.get("NSObject");
ObjCInterfaceDecl* ID = OT->getDecl();
diff --git a/test/Analysis/NSString.m b/test/Analysis/NSString.m
index a95e8868c3..3f2b09189e 100644
--- a/test/Analysis/NSString.m
+++ b/test/Analysis/NSString.m
@@ -251,3 +251,12 @@ void test_stringWithFormat() {
[string release]; // expected-warning{{Incorrect decrement of the reference count}}
}
+// Test isTrackedObjectType()
+typedef NSString* WonkyTypedef;
+@interface TestIsTracked
++ (WonkyTypedef)newString;
+@end
+
+void test_isTrackedObjectType(void) {
+ NSString *str = [TestIsTracked newString]; // expected-warning{{Potential leak}}
+}