summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGeorge Karpenkov <ekarpenkov@apple.com>2018-10-11 22:59:16 +0000
committerGeorge Karpenkov <ekarpenkov@apple.com>2018-10-11 22:59:16 +0000
commit41dc8de6ae9f5ac1d39e5b8a86c2990a5c1feb9c (patch)
tree37676dece6954d1a545c2f1f2ac6b66f76407caf
parent4733be6e7bbde89e4c10bfdaeeb542fe297e0150 (diff)
downloadbcm5719-llvm-41dc8de6ae9f5ac1d39e5b8a86c2990a5c1feb9c.tar.gz
bcm5719-llvm-41dc8de6ae9f5ac1d39e5b8a86c2990a5c1feb9c.zip
[analyzer] Retain count checker for OSObject: recognize OSDynamicCast
For now, tresting the cast as a no-op, and disregarding the case where the output becomes null due to the type mismatch. rdar://45174557 Differential Revision: https://reviews.llvm.org/D53156 llvm-svn: 344311
-rw-r--r--clang/lib/StaticAnalyzer/Checkers/RetainCountChecker/RetainCountChecker.cpp13
-rw-r--r--clang/test/Analysis/osobject-retain-release.cpp30
2 files changed, 41 insertions, 2 deletions
diff --git a/clang/lib/StaticAnalyzer/Checkers/RetainCountChecker/RetainCountChecker.cpp b/clang/lib/StaticAnalyzer/Checkers/RetainCountChecker/RetainCountChecker.cpp
index e5d27f577d1..ca58f14985c 100644
--- a/clang/lib/StaticAnalyzer/Checkers/RetainCountChecker/RetainCountChecker.cpp
+++ b/clang/lib/StaticAnalyzer/Checkers/RetainCountChecker/RetainCountChecker.cpp
@@ -774,12 +774,23 @@ bool RetainCountChecker::evalCall(const CallExpr *CE, CheckerContext &C) const {
// annotate attribute. If it does, we will not inline it.
bool hasTrustedImplementationAnnotation = false;
+ const LocationContext *LCtx = C.getLocationContext();
+
+ // Process OSDynamicCast: should just return the first argument.
+ // For now, tresting the cast as a no-op, and disregarding the case where
+ // the output becomes null due to the type mismatch.
+ if (FD->getNameAsString() == "safeMetaCast") {
+ state = state->BindExpr(CE, LCtx,
+ state->getSVal(CE->getArg(0), LCtx));
+ C.addTransition(state);
+ return true;
+ }
+
// See if it's one of the specific functions we know how to eval.
if (!SmrMgr.canEval(CE, FD, hasTrustedImplementationAnnotation))
return false;
// Bind the return value.
- const LocationContext *LCtx = C.getLocationContext();
SVal RetVal = state->getSVal(CE->getArg(0), LCtx);
if (RetVal.isUnknown() ||
(hasTrustedImplementationAnnotation && !ResultTy.isNull())) {
diff --git a/clang/test/Analysis/osobject-retain-release.cpp b/clang/test/Analysis/osobject-retain-release.cpp
index 18bdf0bafc5..c1b93fce50f 100644
--- a/clang/test/Analysis/osobject-retain-release.cpp
+++ b/clang/test/Analysis/osobject-retain-release.cpp
@@ -1,18 +1,46 @@
// RUN: %clang_analyze_cc1 -analyze -analyzer-checker=core,osx.cocoa.RetainCount -analyzer-config osx.cocoa.RetainCount:CheckOSObject=true -analyzer-output=text -verify %s
+struct OSMetaClass;
+
+#define OSTypeID(type) (type::metaClass)
+
+#define OSDynamicCast(type, inst) \
+ ((type *) OSMetaClassBase::safeMetaCast((inst), OSTypeID(type)))
+
struct OSObject {
virtual void retain();
virtual void release();
-
virtual ~OSObject(){}
+
+ static OSObject *generateObject(int);
+
+ static const OSMetaClass * const metaClass;
};
struct OSArray : public OSObject {
unsigned int getCount();
static OSArray *withCapacity(unsigned int capacity);
+
+ static const OSMetaClass * const metaClass;
};
+struct OSMetaClassBase {
+ static OSObject *safeMetaCast(const OSObject *inst, const OSMetaClass *meta);
+};
+
+void check_dynamic_cast() {
+ OSArray *arr = OSDynamicCast(OSArray, OSObject::generateObject(1));
+ arr->release();
+}
+
+void check_dynamic_cast_null_check() {
+ OSArray *arr = OSDynamicCast(OSArray, OSObject::generateObject(1));
+ if (!arr)
+ return;
+ arr->release();
+}
+
void use_after_release() {
OSArray *arr = OSArray::withCapacity(10); // expected-note{{Call to function 'withCapacity' returns an OSObject of type struct OSArray * with a +1 retain count}}
arr->release(); // expected-note{{Object released}}
OpenPOWER on IntegriCloud