forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			449 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Objective-C
		
	
	
	
			
		
		
	
	
			449 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Objective-C
		
	
	
	
| // RUN: %clang_analyze_cc1 -triple i386-apple-darwin10 -analyzer-checker=core,osx.cocoa.NilArg,osx.cocoa.RetainCount,alpha.core -analyzer-store=region -verify -Wno-objc-root-class %s
 | |
| // RUN: %clang_analyze_cc1 -triple i386-apple-darwin10 -analyzer-checker=core,osx.cocoa.NilArg,osx.cocoa.RetainCount,alpha.core -analyzer-store=region -analyzer-config mode=shallow -verify -Wno-objc-root-class %s
 | |
| // RUN: %clang_analyze_cc1 -DTEST_64 -triple x86_64-apple-darwin10 -analyzer-checker=core,osx.cocoa.NilArg,osx.cocoa.RetainCount,alpha.core -analyzer-store=region -verify -Wno-objc-root-class %s
 | |
| // RUN: %clang_analyze_cc1 -DOSATOMIC_USE_INLINED -triple i386-apple-darwin10 -analyzer-checker=core,osx.cocoa.NilArg,osx.cocoa.RetainCount,alpha.core -analyzer-store=region -verify -Wno-objc-root-class %s
 | |
| 
 | |
| //===----------------------------------------------------------------------===//
 | |
| // The following code is reduced using delta-debugging from
 | |
| // Foundation.h (Mac OS X).
 | |
| //
 | |
| // It includes the basic definitions for the test cases below.
 | |
| // Not directly including Foundation.h directly makes this test case 
 | |
| // both svelte and portable to non-Mac platforms.
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #ifdef TEST_64
 | |
| typedef long long int64_t;
 | |
| _Bool OSAtomicCompareAndSwap64Barrier( int64_t __oldValue, int64_t __newValue, volatile int64_t *__theValue );
 | |
| #define COMPARE_SWAP_BARRIER OSAtomicCompareAndSwap64Barrier
 | |
| typedef int64_t intptr_t;
 | |
| #else
 | |
| typedef int int32_t;
 | |
| _Bool OSAtomicCompareAndSwap32Barrier( int32_t __oldValue, int32_t __newValue, volatile int32_t *__theValue );
 | |
| #define COMPARE_SWAP_BARRIER OSAtomicCompareAndSwap32Barrier
 | |
| typedef int32_t intptr_t;
 | |
| #endif
 | |
| 
 | |
| typedef const void * CFTypeRef;
 | |
| typedef const struct __CFString * CFStringRef;
 | |
| typedef const struct __CFAllocator * CFAllocatorRef;
 | |
| extern const CFAllocatorRef kCFAllocatorDefault;
 | |
| extern CFTypeRef CFRetain(CFTypeRef cf);
 | |
| void CFRelease(CFTypeRef cf);
 | |
| typedef const struct __CFDictionary * CFDictionaryRef;
 | |
| const void *CFDictionaryGetValue(CFDictionaryRef theDict, const void *key);
 | |
| extern CFStringRef CFStringCreateWithFormat(CFAllocatorRef alloc, CFDictionaryRef formatOptions, CFStringRef format, ...);
 | |
| typedef signed char BOOL;
 | |
| typedef int NSInteger;
 | |
| typedef unsigned int NSUInteger;
 | |
| @class NSString, Protocol;
 | |
| extern void NSLog(NSString *format, ...) __attribute__((format(__NSString__, 1, 2)));
 | |
| typedef NSInteger NSComparisonResult;
 | |
| typedef struct _NSZone NSZone;
 | |
| @class NSInvocation, NSMethodSignature, NSCoder, NSString, NSEnumerator;
 | |
| @protocol NSObject
 | |
| - (BOOL)isEqual:(id)object;
 | |
| - (oneway void)release;
 | |
| - (id)retain;
 | |
| - (id)autorelease;
 | |
| @end
 | |
| @protocol NSCopying
 | |
| - (id)copyWithZone:(NSZone *)zone;
 | |
| @end
 | |
| @protocol NSMutableCopying
 | |
| - (id)mutableCopyWithZone:(NSZone *)zone;
 | |
| @end
 | |
| @protocol NSCoding
 | |
| - (void)encodeWithCoder:(NSCoder *)aCoder;
 | |
| @end
 | |
| @interface NSObject <NSObject> {}
 | |
| - (id)init;
 | |
| + (id)alloc;
 | |
| @end
 | |
| extern id NSAllocateObject(Class aClass, NSUInteger extraBytes, NSZone *zone);
 | |
| typedef struct {} NSFastEnumerationState;
 | |
| @protocol NSFastEnumeration
 | |
| - (NSUInteger)countByEnumeratingWithState:(NSFastEnumerationState *)state objects:(id *)stackbuf count:(NSUInteger)len;
 | |
| @end
 | |
| @class NSString;
 | |
| typedef struct _NSRange {} NSRange;
 | |
| @interface NSArray : NSObject <NSCopying, NSMutableCopying, NSCoding, NSFastEnumeration>
 | |
| - (NSUInteger)count;
 | |
| @end
 | |
| @interface NSMutableArray : NSArray
 | |
| - (void)addObject:(id)anObject;
 | |
| - (id)initWithCapacity:(NSUInteger)numItems;
 | |
| @end
 | |
| typedef unsigned short unichar;
 | |
| @class NSData, NSArray, NSDictionary, NSCharacterSet, NSData, NSURL, NSError, NSLocale;
 | |
| typedef NSUInteger NSStringCompareOptions;
 | |
| @interface NSString : NSObject <NSCopying, NSMutableCopying, NSCoding>    - (NSUInteger)length;
 | |
| - (NSComparisonResult)compare:(NSString *)string;
 | |
| - (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask;
 | |
| - (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask range:(NSRange)compareRange;
 | |
| - (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask range:(NSRange)compareRange locale:(id)locale;
 | |
| - (NSComparisonResult)caseInsensitiveCompare:(NSString *)string;
 | |
| - (NSArray *)componentsSeparatedByCharactersInSet:(NSCharacterSet *)separator;
 | |
| + (id)stringWithFormat:(NSString *)format, ... __attribute__((format(__NSString__, 1, 2)));
 | |
| @end
 | |
| @interface NSSimpleCString : NSString {} @end
 | |
| @interface NSConstantString : NSSimpleCString @end
 | |
| extern void *_NSConstantStringClassReference;
 | |
| 
 | |
| //===----------------------------------------------------------------------===//
 | |
| // Test cases.
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| NSComparisonResult f1(NSString* s) {
 | |
|   NSString *aString = 0;
 | |
|   return [s compare:aString]; // expected-warning {{Argument to 'NSString' method 'compare:' cannot be nil}}
 | |
| }
 | |
| 
 | |
| NSComparisonResult f2(NSString* s) {
 | |
|   NSString *aString = 0;
 | |
|   return [s caseInsensitiveCompare:aString]; // expected-warning {{Argument to 'NSString' method 'caseInsensitiveCompare:' cannot be nil}}
 | |
| }
 | |
| 
 | |
| NSComparisonResult f3(NSString* s, NSStringCompareOptions op) {
 | |
|   NSString *aString = 0;
 | |
|   return [s compare:aString options:op]; // expected-warning {{Argument to 'NSString' method 'compare:options:' cannot be nil}}
 | |
| }
 | |
| 
 | |
| NSComparisonResult f4(NSString* s, NSStringCompareOptions op, NSRange R) {
 | |
|   NSString *aString = 0;
 | |
|   return [s compare:aString options:op range:R]; // expected-warning {{Argument to 'NSString' method 'compare:options:range:' cannot be nil}}
 | |
| }
 | |
| 
 | |
| NSComparisonResult f5(NSString* s, NSStringCompareOptions op, NSRange R) {
 | |
|   NSString *aString = 0;
 | |
|   return [s compare:aString options:op range:R locale:0]; // expected-warning {{Argument to 'NSString' method 'compare:options:range:locale:' cannot be nil}}
 | |
| }
 | |
| 
 | |
| NSArray *f6(NSString* s) {
 | |
|   return [s componentsSeparatedByCharactersInSet:0]; // expected-warning {{Argument to 'NSString' method 'componentsSeparatedByCharactersInSet:' cannot be nil}}
 | |
| }
 | |
| 
 | |
| NSString* f7(NSString* s1, NSString* s2, NSString* s3) {
 | |
| 
 | |
|   NSString* s4 = (NSString*)
 | |
|     CFStringCreateWithFormat(kCFAllocatorDefault, 0,  // expected-warning{{leak}}
 | |
|                              (CFStringRef) __builtin___CFStringMakeConstantString("%@ %@ (%@)"), 
 | |
|                              s1, s2, s3);
 | |
| 
 | |
|   CFRetain(s4);
 | |
|   return s4;
 | |
| }
 | |
| 
 | |
| NSMutableArray* f8() {
 | |
|   
 | |
|   NSString* s = [[NSString alloc] init];
 | |
|   NSMutableArray* a = [[NSMutableArray alloc] initWithCapacity:2];
 | |
|   [a addObject:s];
 | |
|   [s release]; // no-warning
 | |
|   return a;
 | |
| }
 | |
| 
 | |
| void f9() {
 | |
|   
 | |
|   NSString* s = [[NSString alloc] init];
 | |
|   NSString* q = s;
 | |
|   [s release];
 | |
|   [q release]; // expected-warning {{used after it is released}}
 | |
| }
 | |
| 
 | |
| NSString* f10() {
 | |
|   static NSString* s = 0;
 | |
|   if (!s) s = [[NSString alloc] init];
 | |
|   return s; // no-warning
 | |
| }
 | |
| 
 | |
| // Test case for regression reported in <rdar://problem/6452745>.
 | |
| // Essentially 's' should not be considered allocated on the false branch.
 | |
| // This exercises the 'EvalAssume' logic in GRTransferFuncs (CFRefCount.cpp).
 | |
| NSString* f11(CFDictionaryRef dict, const char* key) {
 | |
|   NSString* s = (NSString*) CFDictionaryGetValue(dict, key);
 | |
|   [s retain];
 | |
|   if (s) {
 | |
|     [s release];
 | |
|   }
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| // Test case for passing a tracked object by-reference to a function we
 | |
| // don't understand.
 | |
| void unknown_function_f12(NSString** s);
 | |
| void f12() {
 | |
|   NSString *string = [[NSString alloc] init];
 | |
|   unknown_function_f12(&string); // no-warning
 | |
| }
 | |
| 
 | |
| // Test double release of CFString (PR 4014).
 | |
| void f13(void) {
 | |
|   CFStringRef ref = CFStringCreateWithFormat(kCFAllocatorDefault, ((void*)0), ((CFStringRef) __builtin___CFStringMakeConstantString ("" "%d" "")), 100);
 | |
|   CFRelease(ref);
 | |
|   CFRelease(ref); // expected-warning{{Reference-counted object is used after it is released}}
 | |
| }
 | |
| 
 | |
| @interface MyString : NSString
 | |
| @end
 | |
| 
 | |
| void f14(MyString *s) {
 | |
|   [s compare:0]; // expected-warning {{Argument to 'MyString' method 'compare:' cannot be nil}}
 | |
| }
 | |
| 
 | |
| // Test regular use of -autorelease
 | |
| @interface TestAutorelease
 | |
| -(NSString*) getString;
 | |
| @end
 | |
| @implementation TestAutorelease
 | |
| -(NSString*) getString {
 | |
|   NSString *str = [[NSString alloc] init];
 | |
|   return [str autorelease]; // no-warning
 | |
| }
 | |
| - (void)m1
 | |
| {
 | |
|  NSString *s = [[NSString alloc] init]; // expected-warning{{leak}}
 | |
|  [s retain];
 | |
|  [s autorelease];
 | |
| }
 | |
| - (void)m2
 | |
| {
 | |
|  NSString *s = [[[NSString alloc] init] autorelease]; // expected-warning{{leak}}
 | |
|  [s retain];
 | |
| }
 | |
| - (void)m3
 | |
| {
 | |
|  NSString *s = [[[NSString alloc] init] autorelease];
 | |
|  [s retain];
 | |
|  [s autorelease];
 | |
| }
 | |
| - (void)m4
 | |
| {
 | |
|  NSString *s = [[NSString alloc] init]; // expected-warning{{leak}}
 | |
|  [s retain];
 | |
| }
 | |
| - (void)m5
 | |
| {
 | |
|  NSString *s = [[NSString alloc] init];
 | |
|  [s autorelease];
 | |
| }
 | |
| @end
 | |
| 
 | |
| @interface C1 : NSObject {}
 | |
| - (NSString*) getShared;
 | |
| + (C1*) sharedInstance;
 | |
| @end
 | |
| @implementation C1 : NSObject {}
 | |
| - (NSString*) getShared {
 | |
|   static NSString* s = 0;
 | |
|   if (!s) s = [[NSString alloc] init];    
 | |
|   return s; // no-warning  
 | |
| }
 | |
| + (C1 *)sharedInstance {
 | |
|   static C1 *sharedInstance = 0;
 | |
|   if (!sharedInstance) {
 | |
|     sharedInstance = [[C1 alloc] init];
 | |
|   }
 | |
|   return sharedInstance; // no-warning
 | |
| }
 | |
| @end
 | |
| 
 | |
| @interface SharedClass : NSObject
 | |
| + (id)sharedInstance;
 | |
| - (id)notShared;
 | |
| @end
 | |
| 
 | |
| @implementation SharedClass
 | |
| 
 | |
| - (id)_init {
 | |
|     if ((self = [super init])) {
 | |
|         NSLog(@"Bar");
 | |
|     }
 | |
|     return self;
 | |
| }
 | |
| 
 | |
| - (id)notShared {
 | |
|   return [[SharedClass alloc] _init]; // expected-warning{{leak}}
 | |
| }
 | |
| 
 | |
| + (id)sharedInstance {
 | |
|     static SharedClass *_sharedInstance = 0;
 | |
|     if (!_sharedInstance) {
 | |
|         _sharedInstance = [[SharedClass alloc] _init];
 | |
|     }
 | |
|     return _sharedInstance; // no-warning
 | |
| }
 | |
| @end
 | |
| 
 | |
| id testSharedClassFromFunction() {
 | |
|   return [[SharedClass alloc] _init]; // no-warning
 | |
| }
 | |
| 
 | |
| #if !(defined(OSATOMIC_USE_INLINED) && OSATOMIC_USE_INLINED)
 | |
| // Test OSCompareAndSwap
 | |
| _Bool OSAtomicCompareAndSwapPtr( void *__oldValue, void *__newValue, void * volatile *__theValue );
 | |
| extern BOOL objc_atomicCompareAndSwapPtr(id predicate, id replacement, volatile id *objectLocation);
 | |
| #else
 | |
| // Test that the body farm models are still used even when a body is available.
 | |
| _Bool opaque_OSAtomicCompareAndSwapPtr( void *__oldValue, void *__newValue, void * volatile *__theValue );
 | |
| _Bool OSAtomicCompareAndSwapPtr( void *__oldValue, void *__newValue, void * volatile *__theValue ) {
 | |
|   return opaque_OSAtomicCompareAndSwapPtr(__oldValue, __newValue, __theValue);
 | |
| }
 | |
| // Test that the analyzer doesn't crash when the farm model is used. 
 | |
| // The analyzer ignores the autosynthesized code.
 | |
| _Bool OSAtomicCompareAndSwapEmptyFunction( void *__oldValue, void *__newValue, void * volatile *__theValue ) {
 | |
|   return 0;
 | |
| }
 | |
| extern BOOL opaque_objc_atomicCompareAndSwapPtr(id predicate, id replacement, volatile id *objectLocation);
 | |
| extern BOOL objc_atomicCompareAndSwapPtr(id predicate, id replacement, volatile id *objectLocation) {
 | |
|   return opaque_objc_atomicCompareAndSwapPtr(predicate, replacement, objectLocation);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| void testOSCompareAndSwap() {
 | |
|   NSString *old = 0;
 | |
|   NSString *s = [[NSString alloc] init]; // no-warning
 | |
|   if (!OSAtomicCompareAndSwapPtr(0, s, (void**) &old))
 | |
|     [s release];
 | |
|   else    
 | |
|     [old release];
 | |
| }
 | |
| 
 | |
| void testOSCompareAndSwapXXBarrier_local() {
 | |
|   NSString *old = 0;
 | |
|   NSString *s = [[NSString alloc] init]; // no-warning
 | |
|   if (!COMPARE_SWAP_BARRIER((intptr_t) 0, (intptr_t) s, (intptr_t*) &old))
 | |
|     [s release];
 | |
|   else    
 | |
|     [old release];
 | |
| }
 | |
| 
 | |
| void testOSCompareAndSwapXXBarrier_local_no_direct_release() {
 | |
|   NSString *old = 0;
 | |
|   NSString *s = [[NSString alloc] init]; // no-warning
 | |
|   if (!COMPARE_SWAP_BARRIER((intptr_t) 0, (intptr_t) s, (intptr_t*) &old))
 | |
|     return;
 | |
|   else    
 | |
|     [old release];
 | |
| }
 | |
| 
 | |
| int testOSCompareAndSwapXXBarrier_id(Class myclass, id xclass) {
 | |
|   if (COMPARE_SWAP_BARRIER(0, (intptr_t) myclass, (intptr_t*) &xclass))
 | |
|     return 1;
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| void test_objc_atomicCompareAndSwap_local() {
 | |
|   NSString *old = 0;
 | |
|   NSString *s = [[NSString alloc] init]; // no-warning
 | |
|   if (!objc_atomicCompareAndSwapPtr(0, s, &old))
 | |
|     [s release];
 | |
|   else    
 | |
|     [old release];
 | |
| }
 | |
| 
 | |
| void test_objc_atomicCompareAndSwap_local_no_direct_release() {
 | |
|   NSString *old = 0;
 | |
|   NSString *s = [[NSString alloc] init]; // no-warning
 | |
|   if (!objc_atomicCompareAndSwapPtr(0, s, &old))
 | |
|     return;
 | |
|   else    
 | |
|     [old release];
 | |
| }
 | |
| 
 | |
| void test_objc_atomicCompareAndSwap_parameter(NSString **old) {
 | |
|   NSString *s = [[NSString alloc] init]; // no-warning
 | |
|   if (!objc_atomicCompareAndSwapPtr(0, s, old))
 | |
|     [s release];
 | |
|   else    
 | |
|     [*old release];
 | |
| }
 | |
| 
 | |
| void test_objc_atomicCompareAndSwap_parameter_no_direct_release(NSString **old) {
 | |
|   NSString *s = [[NSString alloc] init]; // expected-warning{{leak}}
 | |
|   if (!objc_atomicCompareAndSwapPtr(0, s, old))
 | |
|     return;
 | |
|   else    
 | |
|     [*old release];
 | |
| }
 | |
| 
 | |
| 
 | |
| // Test stringWithFormat (<rdar://problem/6815234>)
 | |
| void test_stringWithFormat() {  
 | |
|   NSString *string = [[NSString stringWithFormat:@"%ld", (long) 100] retain];
 | |
|   [string release];
 | |
|   [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}}
 | |
| }
 | |
| 
 | |
| // Test isTrackedCFObjectType().
 | |
| @interface TestIsCFTracked
 | |
| + (CFStringRef) badNewCFString;
 | |
| + (CFStringRef) newCFString;
 | |
| @end
 | |
| 
 | |
| @implementation TestIsCFTracked
 | |
| + (CFStringRef) newCFString {
 | |
|   return CFStringCreateWithFormat(kCFAllocatorDefault, ((void*)0), ((CFStringRef) __builtin___CFStringMakeConstantString ("" "%d" "")), 100); // no-warning
 | |
| }
 | |
| + (CFStringRef) badNewCFString {
 | |
|   return CFStringCreateWithFormat(kCFAllocatorDefault, ((void*)0), ((CFStringRef) __builtin___CFStringMakeConstantString ("" "%d" "")), 100); // expected-warning{{leak}}
 | |
| }
 | |
| 
 | |
| // Test @synchronized
 | |
| void test_synchronized(id x) {
 | |
|   @synchronized(x) {
 | |
|     NSString *string = [[NSString stringWithFormat:@"%ld", (long) 100] retain]; // expected-warning {{leak}}
 | |
|   }
 | |
| }
 | |
| @end
 | |
| 
 | |
| void testOSCompareAndSwapXXBarrier_parameter(NSString **old) {
 | |
|   NSString *s = [[NSString alloc] init]; // no-warning
 | |
|   if (!COMPARE_SWAP_BARRIER((intptr_t) 0, (intptr_t) s, (intptr_t*) old))
 | |
|     [s release];
 | |
|   else    
 | |
|     [*old release];
 | |
| }
 | |
| 
 | |
| void testOSCompareAndSwapXXBarrier_parameter_no_direct_release(NSString **old) {
 | |
|   NSString *s = [[NSString alloc] init]; // no-warning
 | |
|   if (!COMPARE_SWAP_BARRIER((intptr_t) 0, (intptr_t) s, (intptr_t*) old))
 | |
|     [s release];
 | |
|   else    
 | |
|     return;
 | |
| }
 | |
| 
 | |
| @interface AlwaysInlineBodyFarmBodies : NSObject {
 | |
|   NSString *_value;
 | |
| }
 | |
|   - (NSString *)_value;
 | |
|   - (void)callValue;
 | |
| @end
 | |
| 
 | |
| @implementation AlwaysInlineBodyFarmBodies
 | |
| 
 | |
| - (NSString *)_value {
 | |
|   if (!_value) {
 | |
|     NSString *s = [[NSString alloc] init];
 | |
|     if (!OSAtomicCompareAndSwapPtr(0, s, (void**)&_value)) {
 | |
|       [s release];
 | |
|     }
 | |
|   }
 | |
|   return _value;
 | |
| }
 | |
| 
 | |
| - (void)callValue {
 | |
|   [self _value];
 | |
| }
 | |
| @end
 |