// RUN: %clang_cc1 -analyze -analyzer-checker=core,alpha.core,debug.ExprInspection,unix.Malloc,unix.cstring,alpha.unix.cstring,unix.API,osx.API,osx.cocoa.RetainCount -Wno-null-dereference -Wno-tautological-compare -analyzer-store=region -fblocks -verify %s #define NULL 0 void clang_analyzer_eval(int); void myFunc(); void myWeakFunc() __attribute__((weak_import)); void testWeakFuncIsNull() { clang_analyzer_eval(myFunc == NULL); // expected-warning{{FALSE}} clang_analyzer_eval(myWeakFunc == NULL); // expected-warning{{UNKNOWN}} if (myWeakFunc == NULL) { clang_analyzer_eval(myWeakFunc == NULL); // expected-warning{{TRUE}} } else { clang_analyzer_eval(myWeakFunc == NULL); // expected-warning{{FALSE}} } } void testWeakFuncIsNot() { clang_analyzer_eval(myWeakFunc == NULL); // expected-warning{{UNKNOWN}} if (!myWeakFunc) { clang_analyzer_eval(myWeakFunc == NULL); // expected-warning{{TRUE}} } else { clang_analyzer_eval(myWeakFunc == NULL); // expected-warning{{FALSE}} } } void testWeakFuncIsTrue() { clang_analyzer_eval(myWeakFunc == NULL); // expected-warning{{UNKNOWN}} if (myWeakFunc) { clang_analyzer_eval(myWeakFunc == NULL); // expected-warning{{FALSE}} } else { clang_analyzer_eval(myWeakFunc == NULL); // expected-warning{{TRUE}} } } //===----------------------------------------------------------------------=== // func.c //===----------------------------------------------------------------------=== void f(void) __attribute__((weak_import)); void g(void (*fp)(void)) __attribute__((weak_import)); void f(void) { void (*p)(void); p = f; p = &f; p(); (*p)(); } void g(void (*fp)(void)); void f2() { g(f); } void f3(void (*f)(void), void (*g)(void)) { clang_analyzer_eval(!f); // expected-warning{{UNKNOWN}} f(); clang_analyzer_eval(!f); // expected-warning{{FALSE}} clang_analyzer_eval(!g); // expected-warning{{UNKNOWN}} (*g)(); clang_analyzer_eval(!g); // expected-warning{{FALSE}} } //===----------------------------------------------------------------------=== // free.c //===----------------------------------------------------------------------=== void free(void *) __attribute__((weak_import)); void t10 () { free((void*)&t10); // expected-warning {{Argument to free() is the address of the function 't10', which is not memory allocated by malloc()}} } //===----------------------------------------------------------------------=== // string.c : strnlen() //===----------------------------------------------------------------------=== typedef typeof(sizeof(int)) size_t; size_t strlen(const char *s) __attribute__((weak_import)); size_t strlen_fn() { return strlen((char*)&strlen_fn); // expected-warning{{Argument to string length function is the address of the function 'strlen_fn', which is not a null-terminated string}} } //===----------------------------------------------------------------------=== // unix-fns.c : dispatch_once //===----------------------------------------------------------------------=== typedef void (^dispatch_block_t)(void); typedef long dispatch_once_t; void dispatch_once(dispatch_once_t *predicate, dispatch_block_t block) __attribute__((weak_import)); void test_dispatch_once() { dispatch_once_t pred = 0; do { if (__builtin_expect(*(&pred), ~0l) != ~0l) dispatch_once((&pred), (^() {})); } while (0); // expected-warning{{Call to 'dispatch_once' uses the local variable 'pred' for the predicate value}} } void test_dispatch_once_neg() { static dispatch_once_t pred = 0; do { if (__builtin_expect(*(&pred), ~0l) != ~0l) dispatch_once((&pred), (^() {})); } while (0); // no-warning } //===----------------------------------------------------------------------=== // retain-release-path-notes.m //===----------------------------------------------------------------------=== typedef struct CFType *CFTypeRef; CFTypeRef CFCreateSomething() __attribute__((weak_import)); CFTypeRef CFGetSomething() __attribute__((weak_import)); CFTypeRef CFCopyRuleViolation () { CFTypeRef object = CFGetSomething(); return object; // expected-warning{{Object with a +0 retain count returned to caller where a +1 (owning) retain count is expected}} } CFTypeRef CFGetRuleViolation () { CFTypeRef object = CFCreateSomething(); // expected-warning{{Potential leak of an object stored into 'object'}} return object; }