blob: 6ff4bb1e554fba9296fa96825e02e04c15f9fef5 [file] [log] [blame]
Ted Kremenek51885072011-03-24 00:28:47 +00001// RUN: %clang_cc1 -triple i386-apple-darwin10 -analyze -analyzer-checker=core,osx.cocoa.NilArg,osx.AtomicCAS,core.experimental -analyzer-store=region -analyzer-constraints=basic -verify %s
2// RUN: %clang_cc1 -triple i386-apple-darwin10 -analyze -analyzer-checker=core,osx.cocoa.NilArg,osx.AtomicCAS,core.experimental -analyzer-store=region -analyzer-constraints=range -verify %s
3// RUN: %clang_cc1 -DTEST_64 -triple x86_64-apple-darwin10 -analyze -analyzer-checker=core,osx.cocoa.NilArg,osx.AtomicCAS,core.experimental -analyzer-store=region -analyzer-constraints=basic -verify %s
4// RUN: %clang_cc1 -DTEST_64 -triple x86_64-apple-darwin10 -analyze -analyzer-checker=core,osx.cocoa.NilArg,osx.AtomicCAS,core.experimental -analyzer-store=region -analyzer-constraints=range -verify %s
Ted Kremenek45f7c272009-10-15 01:40:34 +00005
6// ==-- FIXME: -analyzer-store=basic fails on this file (false negatives). --==
Ted Kremenek51885072011-03-24 00:28:47 +00007// NOTWORK: %clang_cc1 -triple i386-apple-darwin10 -analyze -analyzer-checker=core,osx.cocoa.NilArg,osx.AtomicCAS,core.experimental -analyzer-store=basic -analyzer-constraints=range -verify %s &&
8// NOTWORK: %clang_cc1 -triple i386-apple-darwin10 -analyze -analyzer-checker=core,osx.cocoa.NilArg,osx.AtomicCAS,core.experimental -analyzer-store=basic -analyzer-constraints=basic -verify %s &&
9// NOTWORK: %clang_cc1 -DTEST_64 -triple x86_64-apple-darwin10 -analyze -analyzer-checker=core,osx.cocoa.NilArg,osx.AtomicCAS,core.experimental -analyzer-store=basic -analyzer-constraints=basic -verify %s &&
10// NOTWORK: %clang_cc1 -DTEST_64 -triple x86_64-apple-darwin10 -analyze -analyzer-checker=core,osx.cocoa.NilArg,osx.AtomicCAS,core.experimental -analyzer-store=basic -analyzer-constraints=range -verify %s
Ted Kremenek859be3b2008-06-16 20:37:30 +000011
12//===----------------------------------------------------------------------===//
13// The following code is reduced using delta-debugging from
14// Foundation.h (Mac OS X).
15//
16// It includes the basic definitions for the test cases below.
17// Not directly including Foundation.h directly makes this test case
18// both svelte and portable to non-Mac platforms.
19//===----------------------------------------------------------------------===//
20
Ted Kremenek45f7c272009-10-15 01:40:34 +000021#ifdef TEST_64
22typedef long long int64_t;
23_Bool OSAtomicCompareAndSwap64Barrier( int64_t __oldValue, int64_t __newValue, volatile int64_t *__theValue );
24#define COMPARE_SWAP_BARRIER OSAtomicCompareAndSwap64Barrier
25typedef int64_t intptr_t;
26#else
Ted Kremeneka3f45402009-04-29 16:03:59 +000027typedef int int32_t;
Ted Kremenek45f7c272009-10-15 01:40:34 +000028_Bool OSAtomicCompareAndSwap32Barrier( int32_t __oldValue, int32_t __newValue, volatile int32_t *__theValue );
29#define COMPARE_SWAP_BARRIER OSAtomicCompareAndSwap32Barrier
30typedef int32_t intptr_t;
31#endif
32
Ted Kremenek859be3b2008-06-16 20:37:30 +000033typedef const void * CFTypeRef;
34typedef const struct __CFString * CFStringRef;
35typedef const struct __CFAllocator * CFAllocatorRef;
36extern const CFAllocatorRef kCFAllocatorDefault;
37extern CFTypeRef CFRetain(CFTypeRef cf);
Ted Kremenek09f14192009-04-21 20:01:03 +000038void CFRelease(CFTypeRef cf);
Ted Kremenek859be3b2008-06-16 20:37:30 +000039typedef const struct __CFDictionary * CFDictionaryRef;
Ted Kremenek2fb78a72008-12-17 21:50:35 +000040const void *CFDictionaryGetValue(CFDictionaryRef theDict, const void *key);
Ted Kremenek859be3b2008-06-16 20:37:30 +000041extern CFStringRef CFStringCreateWithFormat(CFAllocatorRef alloc, CFDictionaryRef formatOptions, CFStringRef format, ...);
42typedef signed char BOOL;
43typedef int NSInteger;
44typedef unsigned int NSUInteger;
45@class NSString, Protocol;
46extern void NSLog(NSString *format, ...) __attribute__((format(__NSString__, 1, 2)));
47typedef NSInteger NSComparisonResult;
48typedef struct _NSZone NSZone;
49@class NSInvocation, NSMethodSignature, NSCoder, NSString, NSEnumerator;
50@protocol NSObject
51- (BOOL)isEqual:(id)object;
52- (oneway void)release;
Ted Kremenek2fb78a72008-12-17 21:50:35 +000053- (id)retain;
Ted Kremenek95d3b902009-05-11 15:26:06 +000054- (id)autorelease;
Ted Kremenek859be3b2008-06-16 20:37:30 +000055@end
56@protocol NSCopying
57- (id)copyWithZone:(NSZone *)zone;
58@end
59@protocol NSMutableCopying
60- (id)mutableCopyWithZone:(NSZone *)zone;
61@end
62@protocol NSCoding
63- (void)encodeWithCoder:(NSCoder *)aCoder;
64@end
65@interface NSObject <NSObject> {}
66- (id)init;
67+ (id)alloc;
68@end
69extern id NSAllocateObject(Class aClass, NSUInteger extraBytes, NSZone *zone);
70typedef struct {} NSFastEnumerationState;
71@protocol NSFastEnumeration
72- (NSUInteger)countByEnumeratingWithState:(NSFastEnumerationState *)state objects:(id *)stackbuf count:(NSUInteger)len;
73@end
74@class NSString;
75typedef struct _NSRange {} NSRange;
76@interface NSArray : NSObject <NSCopying, NSMutableCopying, NSCoding, NSFastEnumeration>
77- (NSUInteger)count;
78@end
79@interface NSMutableArray : NSArray
80- (void)addObject:(id)anObject;
81- (id)initWithCapacity:(NSUInteger)numItems;
82@end
83typedef unsigned short unichar;
84@class NSData, NSArray, NSDictionary, NSCharacterSet, NSData, NSURL, NSError, NSLocale;
85typedef NSUInteger NSStringCompareOptions;
86@interface NSString : NSObject <NSCopying, NSMutableCopying, NSCoding> - (NSUInteger)length;
87- (NSComparisonResult)compare:(NSString *)string;
88- (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask;
89- (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask range:(NSRange)compareRange;
90- (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask range:(NSRange)compareRange locale:(id)locale;
91- (NSComparisonResult)caseInsensitiveCompare:(NSString *)string;
92- (NSArray *)componentsSeparatedByCharactersInSet:(NSCharacterSet *)separator;
Ted Kremeneke87450e2009-04-23 19:11:35 +000093+ (id)stringWithFormat:(NSString *)format, ... __attribute__((format(__NSString__, 1, 2)));
Ted Kremenek859be3b2008-06-16 20:37:30 +000094@end
95@interface NSSimpleCString : NSString {} @end
96@interface NSConstantString : NSSimpleCString @end
97extern void *_NSConstantStringClassReference;
98
99//===----------------------------------------------------------------------===//
100// Test cases.
101//===----------------------------------------------------------------------===//
102
103NSComparisonResult f1(NSString* s) {
104 NSString *aString = 0;
105 return [s compare:aString]; // expected-warning {{Argument to 'NSString' method 'compare:' cannot be nil.}}
106}
107
108NSComparisonResult f2(NSString* s) {
109 NSString *aString = 0;
110 return [s caseInsensitiveCompare:aString]; // expected-warning {{Argument to 'NSString' method 'caseInsensitiveCompare:' cannot be nil.}}
111}
112
113NSComparisonResult f3(NSString* s, NSStringCompareOptions op) {
114 NSString *aString = 0;
115 return [s compare:aString options:op]; // expected-warning {{Argument to 'NSString' method 'compare:options:' cannot be nil.}}
116}
117
118NSComparisonResult f4(NSString* s, NSStringCompareOptions op, NSRange R) {
119 NSString *aString = 0;
120 return [s compare:aString options:op range:R]; // expected-warning {{Argument to 'NSString' method 'compare:options:range:' cannot be nil.}}
121}
122
123NSComparisonResult f5(NSString* s, NSStringCompareOptions op, NSRange R) {
124 NSString *aString = 0;
125 return [s compare:aString options:op range:R locale:0]; // expected-warning {{Argument to 'NSString' method 'compare:options:range:locale:' cannot be nil.}}
126}
127
Daniel Dunbar4489fe12008-08-05 00:07:51 +0000128NSArray *f6(NSString* s) {
Ted Kremenek859be3b2008-06-16 20:37:30 +0000129 return [s componentsSeparatedByCharactersInSet:0]; // expected-warning {{Argument to 'NSString' method 'componentsSeparatedByCharactersInSet:' cannot be nil.}}
130}
131
132NSString* f7(NSString* s1, NSString* s2, NSString* s3) {
133
Daniel Dunbar4489fe12008-08-05 00:07:51 +0000134 NSString* s4 = (NSString*)
Ted Kremenekcf118d42009-02-04 23:49:09 +0000135 CFStringCreateWithFormat(kCFAllocatorDefault, 0, // expected-warning{{leak}}
Daniel Dunbar4489fe12008-08-05 00:07:51 +0000136 (CFStringRef) __builtin___CFStringMakeConstantString("%@ %@ (%@)"),
137 s1, s2, s3);
Ted Kremenek859be3b2008-06-16 20:37:30 +0000138
139 CFRetain(s4);
Ted Kremenekcf118d42009-02-04 23:49:09 +0000140 return s4;
Ted Kremenek859be3b2008-06-16 20:37:30 +0000141}
142
143NSMutableArray* f8() {
144
145 NSString* s = [[NSString alloc] init];
146 NSMutableArray* a = [[NSMutableArray alloc] initWithCapacity:2];
147 [a addObject:s];
148 [s release]; // no-warning
149 return a;
150}
151
152void f9() {
153
154 NSString* s = [[NSString alloc] init];
155 NSString* q = s;
156 [s release];
157 [q release]; // expected-warning {{used after it is released}}
158}
159
160NSString* f10() {
Ted Kremenek859be3b2008-06-16 20:37:30 +0000161 static NSString* s = 0;
Ted Kremenek859be3b2008-06-16 20:37:30 +0000162 if (!s) s = [[NSString alloc] init];
Ted Kremenek859be3b2008-06-16 20:37:30 +0000163 return s; // no-warning
164}
165
Ted Kremenek2fb78a72008-12-17 21:50:35 +0000166// Test case for regression reported in <rdar://problem/6452745>.
167// Essentially 's' should not be considered allocated on the false branch.
168// This exercises the 'EvalAssume' logic in GRTransferFuncs (CFRefCount.cpp).
169NSString* f11(CFDictionaryRef dict, const char* key) {
170 NSString* s = (NSString*) CFDictionaryGetValue(dict, key);
171 [s retain];
172 if (s) {
173 [s release];
174 }
Mike Stump4393b3f2009-07-21 18:46:15 +0000175 return 0;
Ted Kremenek2fb78a72008-12-17 21:50:35 +0000176}
177
Ted Kremenek784606f2008-12-18 23:40:58 +0000178// Test case for passing a tracked object by-reference to a function we
Ted Kremenek09f14192009-04-21 20:01:03 +0000179// don't understand.
Ted Kremenek784606f2008-12-18 23:40:58 +0000180void unknown_function_f12(NSString** s);
181void f12() {
182 NSString *string = [[NSString alloc] init];
183 unknown_function_f12(&string); // no-warning
184}
185
Ted Kremenek09f14192009-04-21 20:01:03 +0000186// Test double release of CFString (PR 4014).
187void f13(void) {
188 CFStringRef ref = CFStringCreateWithFormat(kCFAllocatorDefault, ((void*)0), ((CFStringRef) __builtin___CFStringMakeConstantString ("" "%d" "")), 100);
189 CFRelease(ref);
190 CFRelease(ref); // expected-warning{{Reference-counted object is used after it is released}}
191}
Ted Kremenek784606f2008-12-18 23:40:58 +0000192
Anders Carlssonb62bdce2011-03-08 20:05:26 +0000193@interface MyString : NSString
194@end
195
196void f14(MyString *s) {
197 [s compare:0]; // expected-warning {{Argument to 'MyString' method 'compare:' cannot be nil.}}
198}
199
Ted Kremenek95d3b902009-05-11 15:26:06 +0000200// Test regular use of -autorelease
201@interface TestAutorelease
202-(NSString*) getString;
203@end
204@implementation TestAutorelease
205-(NSString*) getString {
206 NSString *str = [[NSString alloc] init];
207 return [str autorelease]; // no-warning
208}
Ted Kremenek2033a952009-05-13 07:12:33 +0000209- (void)m1
210{
211 NSString *s = [[NSString alloc] init]; // expected-warning{{leak}}
212 [s retain];
213 [s autorelease];
214}
215- (void)m2
216{
217 NSString *s = [[[NSString alloc] init] autorelease]; // expected-warning{{leak}}
218 [s retain];
219}
220- (void)m3
221{
222 NSString *s = [[[NSString alloc] init] autorelease];
223 [s retain];
224 [s autorelease];
225}
226- (void)m4
227{
228 NSString *s = [[NSString alloc] init]; // expected-warning{{leak}}
229 [s retain];
230}
231- (void)m5
232{
233 NSString *s = [[NSString alloc] init];
234 [s autorelease];
235}
Ted Kremenek95d3b902009-05-11 15:26:06 +0000236@end
237
Ted Kremenek859be3b2008-06-16 20:37:30 +0000238@interface C1 : NSObject {}
239- (NSString*) getShared;
240+ (C1*) sharedInstance;
241@end
Ted Kremenekaeca9632008-07-03 15:37:02 +0000242@implementation C1 : NSObject {}
Ted Kremenek859be3b2008-06-16 20:37:30 +0000243- (NSString*) getShared {
244 static NSString* s = 0;
245 if (!s) s = [[NSString alloc] init];
246 return s; // no-warning
247}
248+ (C1 *)sharedInstance {
249 static C1 *sharedInstance = 0;
250 if (!sharedInstance) {
251 sharedInstance = [[C1 alloc] init];
252 }
253 return sharedInstance; // no-warning
254}
255@end
256
257@interface SharedClass : NSObject
258+ (id)sharedInstance;
Ted Kremenek234a4c22009-01-07 00:39:56 +0000259- (id)notShared;
Ted Kremenek859be3b2008-06-16 20:37:30 +0000260@end
Ted Kremenek234a4c22009-01-07 00:39:56 +0000261
Ted Kremenekaeca9632008-07-03 15:37:02 +0000262@implementation SharedClass
Ted Kremenek859be3b2008-06-16 20:37:30 +0000263
264- (id)_init {
265 if ((self = [super init])) {
266 NSLog(@"Bar");
267 }
268 return self;
269}
270
Ted Kremenek234a4c22009-01-07 00:39:56 +0000271- (id)notShared {
Ted Kremenek043254a2009-02-07 22:55:48 +0000272 return [[SharedClass alloc] _init]; // expected-warning{{leak}}
Ted Kremenek234a4c22009-01-07 00:39:56 +0000273}
274
Ted Kremenek859be3b2008-06-16 20:37:30 +0000275+ (id)sharedInstance {
276 static SharedClass *_sharedInstance = 0;
277 if (!_sharedInstance) {
278 _sharedInstance = [[SharedClass alloc] _init];
279 }
280 return _sharedInstance; // no-warning
281}
282@end
Ted Kremenek234a4c22009-01-07 00:39:56 +0000283
284id testSharedClassFromFunction() {
285 return [[SharedClass alloc] _init]; // no-warning
286}
287
Ted Kremenek1670e402009-04-11 00:11:10 +0000288// Test OSCompareAndSwap
289_Bool OSAtomicCompareAndSwapPtr( void *__oldValue, void *__newValue, void * volatile *__theValue );
Ted Kremenekb3bf76f2009-04-11 00:54:13 +0000290extern BOOL objc_atomicCompareAndSwapPtr(id predicate, id replacement, volatile id *objectLocation);
Ted Kremenek1670e402009-04-11 00:11:10 +0000291
292void testOSCompareAndSwap() {
293 NSString *old = 0;
Ted Kremenekb3bf76f2009-04-11 00:54:13 +0000294 NSString *s = [[NSString alloc] init]; // no-warning
Ted Kremenek1670e402009-04-11 00:11:10 +0000295 if (!OSAtomicCompareAndSwapPtr(0, s, (void**) &old))
296 [s release];
297 else
298 [old release];
299}
300
Ted Kremenek6bcd5a02009-12-09 23:29:55 +0000301void testOSCompareAndSwapXXBarrier_local() {
Ted Kremeneka3f45402009-04-29 16:03:59 +0000302 NSString *old = 0;
303 NSString *s = [[NSString alloc] init]; // no-warning
Ted Kremenek45f7c272009-10-15 01:40:34 +0000304 if (!COMPARE_SWAP_BARRIER((intptr_t) 0, (intptr_t) s, (intptr_t*) &old))
Ted Kremeneka3f45402009-04-29 16:03:59 +0000305 [s release];
306 else
307 [old release];
308}
309
Ted Kremenek6bcd5a02009-12-09 23:29:55 +0000310void testOSCompareAndSwapXXBarrier_local_no_direct_release() {
Ted Kremenek45f7c272009-10-15 01:40:34 +0000311 NSString *old = 0;
Zhongxing Xu2f4a6b22009-12-09 08:32:57 +0000312 NSString *s = [[NSString alloc] init]; // no-warning
Ted Kremenek45f7c272009-10-15 01:40:34 +0000313 if (!COMPARE_SWAP_BARRIER((intptr_t) 0, (intptr_t) s, (intptr_t*) &old))
314 return;
315 else
316 [old release];
317}
318
319int testOSCompareAndSwapXXBarrier_id(Class myclass, id xclass) {
320 if (COMPARE_SWAP_BARRIER(0, (intptr_t) myclass, (intptr_t*) &xclass))
Ted Kremenek0aeaf5a2009-07-29 18:18:25 +0000321 return 1;
322 return 0;
Ted Kremenek45f7c272009-10-15 01:40:34 +0000323}
Ted Kremenek0aeaf5a2009-07-29 18:18:25 +0000324
Ted Kremenek6bcd5a02009-12-09 23:29:55 +0000325void test_objc_atomicCompareAndSwap_local() {
Ted Kremenekb3bf76f2009-04-11 00:54:13 +0000326 NSString *old = 0;
327 NSString *s = [[NSString alloc] init]; // no-warning
328 if (!objc_atomicCompareAndSwapPtr(0, s, &old))
329 [s release];
330 else
331 [old release];
332}
333
Ted Kremenek6bcd5a02009-12-09 23:29:55 +0000334void test_objc_atomicCompareAndSwap_local_no_direct_release() {
Ted Kremenek45f7c272009-10-15 01:40:34 +0000335 NSString *old = 0;
Zhongxing Xu2f4a6b22009-12-09 08:32:57 +0000336 NSString *s = [[NSString alloc] init]; // no-warning
Ted Kremenek45f7c272009-10-15 01:40:34 +0000337 if (!objc_atomicCompareAndSwapPtr(0, s, &old))
338 return;
339 else
340 [old release];
341}
342
Ted Kremenek6bcd5a02009-12-09 23:29:55 +0000343void test_objc_atomicCompareAndSwap_parameter(NSString **old) {
344 NSString *s = [[NSString alloc] init]; // no-warning
345 if (!objc_atomicCompareAndSwapPtr(0, s, old))
346 [s release];
347 else
348 [*old release];
349}
350
351void test_objc_atomicCompareAndSwap_parameter_no_direct_release(NSString **old) {
352 NSString *s = [[NSString alloc] init]; // expected-warning{{leak}}
353 if (!objc_atomicCompareAndSwapPtr(0, s, old))
354 return;
355 else
356 [*old release];
357}
358
Ted Kremenek45f7c272009-10-15 01:40:34 +0000359
Ted Kremeneke87450e2009-04-23 19:11:35 +0000360// Test stringWithFormat (<rdar://problem/6815234>)
361void test_stringWithFormat() {
362 NSString *string = [[NSString stringWithFormat:@"%ld", (long) 100] retain];
363 [string release];
Ted Kremenekf9df1362009-04-23 21:25:57 +0000364 [string release]; // expected-warning{{Incorrect decrement of the reference count}}
Ted Kremeneke87450e2009-04-23 19:11:35 +0000365}
366
Ted Kremenek92511432009-05-03 06:08:32 +0000367// Test isTrackedObjectType().
Ted Kremenek97d095f2009-04-23 22:11:07 +0000368typedef NSString* WonkyTypedef;
369@interface TestIsTracked
370+ (WonkyTypedef)newString;
371@end
372
373void test_isTrackedObjectType(void) {
374 NSString *str = [TestIsTracked newString]; // expected-warning{{Potential leak}}
375}
Ted Kremenekb3b0b362009-05-02 01:49:13 +0000376
Ted Kremenek92511432009-05-03 06:08:32 +0000377// Test isTrackedCFObjectType().
378@interface TestIsCFTracked
379+ (CFStringRef) badNewCFString;
380+ (CFStringRef) newCFString;
381@end
382
383@implementation TestIsCFTracked
384+ (CFStringRef) newCFString {
385 return CFStringCreateWithFormat(kCFAllocatorDefault, ((void*)0), ((CFStringRef) __builtin___CFStringMakeConstantString ("" "%d" "")), 100); // no-warning
386}
387+ (CFStringRef) badNewCFString {
388 return CFStringCreateWithFormat(kCFAllocatorDefault, ((void*)0), ((CFStringRef) __builtin___CFStringMakeConstantString ("" "%d" "")), 100); // expected-warning{{leak}}
389}
390
Ted Kremenekb3b0b362009-05-02 01:49:13 +0000391// Test @synchronized
392void test_synchronized(id x) {
393 @synchronized(x) {
394 NSString *string = [[NSString stringWithFormat:@"%ld", (long) 100] retain]; // expected-warning {{leak}}
395 }
396}
Fariborz Jahanian63e963c2009-11-16 18:57:01 +0000397@end
Zhongxing Xu604848a2010-01-11 06:52:53 +0000398
399void testOSCompareAndSwapXXBarrier_parameter(NSString **old) {
400 NSString *s = [[NSString alloc] init]; // no-warning
401 if (!COMPARE_SWAP_BARRIER((intptr_t) 0, (intptr_t) s, (intptr_t*) old))
402 [s release];
403 else
404 [*old release];
405}
Zhongxing Xu951b3342010-01-11 07:40:00 +0000406
407void testOSCompareAndSwapXXBarrier_parameter_no_direct_release(NSString **old) {
408 NSString *s = [[NSString alloc] init]; // no-warning
409 if (!COMPARE_SWAP_BARRIER((intptr_t) 0, (intptr_t) s, (intptr_t*) old))
410 [s release];
411 else
412 return;
413}