blob: e3900334831f50353a991a83dda8339988ff41dc [file] [log] [blame]
Ted Kremenek42adacb2012-10-11 20:58:21 +00001// RUN: %clang_cc1 -triple i386-apple-darwin10 -analyze -analyzer-checker=core,osx.cocoa.NilArg,osx.cocoa.RetainCount,alpha.core -analyzer-store=region -analyzer-constraints=range -verify -Wno-objc-root-class %s
Anna Zaks8dadf152013-02-22 02:59:24 +00002// RUN: %clang_cc1 -triple i386-apple-darwin10 -analyze -analyzer-checker=core,osx.cocoa.NilArg,osx.cocoa.RetainCount,alpha.core -analyzer-store=region -analyzer-constraints=range -analyzer-config mode=shallow -verify -Wno-objc-root-class %s
Ted Kremenek42adacb2012-10-11 20:58:21 +00003// RUN: %clang_cc1 -DTEST_64 -triple x86_64-apple-darwin10 -analyze -analyzer-checker=core,osx.cocoa.NilArg,osx.cocoa.RetainCount,alpha.core -analyzer-store=region -analyzer-constraints=range -verify -Wno-objc-root-class %s
Ted Kremenek45f7c272009-10-15 01:40:34 +00004
Ted Kremenek859be3b2008-06-16 20:37:30 +00005
6//===----------------------------------------------------------------------===//
7// The following code is reduced using delta-debugging from
8// Foundation.h (Mac OS X).
9//
10// It includes the basic definitions for the test cases below.
11// Not directly including Foundation.h directly makes this test case
12// both svelte and portable to non-Mac platforms.
13//===----------------------------------------------------------------------===//
14
Ted Kremenek45f7c272009-10-15 01:40:34 +000015#ifdef TEST_64
16typedef long long int64_t;
17_Bool OSAtomicCompareAndSwap64Barrier( int64_t __oldValue, int64_t __newValue, volatile int64_t *__theValue );
18#define COMPARE_SWAP_BARRIER OSAtomicCompareAndSwap64Barrier
19typedef int64_t intptr_t;
20#else
Ted Kremeneka3f45402009-04-29 16:03:59 +000021typedef int int32_t;
Ted Kremenek45f7c272009-10-15 01:40:34 +000022_Bool OSAtomicCompareAndSwap32Barrier( int32_t __oldValue, int32_t __newValue, volatile int32_t *__theValue );
23#define COMPARE_SWAP_BARRIER OSAtomicCompareAndSwap32Barrier
24typedef int32_t intptr_t;
25#endif
26
Ted Kremenek859be3b2008-06-16 20:37:30 +000027typedef const void * CFTypeRef;
28typedef const struct __CFString * CFStringRef;
29typedef const struct __CFAllocator * CFAllocatorRef;
30extern const CFAllocatorRef kCFAllocatorDefault;
31extern CFTypeRef CFRetain(CFTypeRef cf);
Ted Kremenek09f14192009-04-21 20:01:03 +000032void CFRelease(CFTypeRef cf);
Ted Kremenek859be3b2008-06-16 20:37:30 +000033typedef const struct __CFDictionary * CFDictionaryRef;
Ted Kremenek2fb78a72008-12-17 21:50:35 +000034const void *CFDictionaryGetValue(CFDictionaryRef theDict, const void *key);
Ted Kremenek859be3b2008-06-16 20:37:30 +000035extern CFStringRef CFStringCreateWithFormat(CFAllocatorRef alloc, CFDictionaryRef formatOptions, CFStringRef format, ...);
36typedef signed char BOOL;
37typedef int NSInteger;
38typedef unsigned int NSUInteger;
39@class NSString, Protocol;
40extern void NSLog(NSString *format, ...) __attribute__((format(__NSString__, 1, 2)));
41typedef NSInteger NSComparisonResult;
42typedef struct _NSZone NSZone;
43@class NSInvocation, NSMethodSignature, NSCoder, NSString, NSEnumerator;
44@protocol NSObject
45- (BOOL)isEqual:(id)object;
46- (oneway void)release;
Ted Kremenek2fb78a72008-12-17 21:50:35 +000047- (id)retain;
Ted Kremenek95d3b902009-05-11 15:26:06 +000048- (id)autorelease;
Ted Kremenek859be3b2008-06-16 20:37:30 +000049@end
50@protocol NSCopying
51- (id)copyWithZone:(NSZone *)zone;
52@end
53@protocol NSMutableCopying
54- (id)mutableCopyWithZone:(NSZone *)zone;
55@end
56@protocol NSCoding
57- (void)encodeWithCoder:(NSCoder *)aCoder;
58@end
59@interface NSObject <NSObject> {}
60- (id)init;
61+ (id)alloc;
62@end
63extern id NSAllocateObject(Class aClass, NSUInteger extraBytes, NSZone *zone);
64typedef struct {} NSFastEnumerationState;
65@protocol NSFastEnumeration
66- (NSUInteger)countByEnumeratingWithState:(NSFastEnumerationState *)state objects:(id *)stackbuf count:(NSUInteger)len;
67@end
68@class NSString;
69typedef struct _NSRange {} NSRange;
70@interface NSArray : NSObject <NSCopying, NSMutableCopying, NSCoding, NSFastEnumeration>
71- (NSUInteger)count;
72@end
73@interface NSMutableArray : NSArray
74- (void)addObject:(id)anObject;
75- (id)initWithCapacity:(NSUInteger)numItems;
76@end
77typedef unsigned short unichar;
78@class NSData, NSArray, NSDictionary, NSCharacterSet, NSData, NSURL, NSError, NSLocale;
79typedef NSUInteger NSStringCompareOptions;
80@interface NSString : NSObject <NSCopying, NSMutableCopying, NSCoding> - (NSUInteger)length;
81- (NSComparisonResult)compare:(NSString *)string;
82- (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask;
83- (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask range:(NSRange)compareRange;
84- (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask range:(NSRange)compareRange locale:(id)locale;
85- (NSComparisonResult)caseInsensitiveCompare:(NSString *)string;
86- (NSArray *)componentsSeparatedByCharactersInSet:(NSCharacterSet *)separator;
Ted Kremeneke87450e2009-04-23 19:11:35 +000087+ (id)stringWithFormat:(NSString *)format, ... __attribute__((format(__NSString__, 1, 2)));
Ted Kremenek859be3b2008-06-16 20:37:30 +000088@end
89@interface NSSimpleCString : NSString {} @end
90@interface NSConstantString : NSSimpleCString @end
91extern void *_NSConstantStringClassReference;
92
93//===----------------------------------------------------------------------===//
94// Test cases.
95//===----------------------------------------------------------------------===//
96
97NSComparisonResult f1(NSString* s) {
98 NSString *aString = 0;
Richard Trieu2fe9b7f2011-12-15 00:38:15 +000099 return [s compare:aString]; // expected-warning {{Argument to 'NSString' method 'compare:' cannot be nil}}
Ted Kremenek859be3b2008-06-16 20:37:30 +0000100}
101
102NSComparisonResult f2(NSString* s) {
103 NSString *aString = 0;
Richard Trieu2fe9b7f2011-12-15 00:38:15 +0000104 return [s caseInsensitiveCompare:aString]; // expected-warning {{Argument to 'NSString' method 'caseInsensitiveCompare:' cannot be nil}}
Ted Kremenek859be3b2008-06-16 20:37:30 +0000105}
106
107NSComparisonResult f3(NSString* s, NSStringCompareOptions op) {
108 NSString *aString = 0;
Richard Trieu2fe9b7f2011-12-15 00:38:15 +0000109 return [s compare:aString options:op]; // expected-warning {{Argument to 'NSString' method 'compare:options:' cannot be nil}}
Ted Kremenek859be3b2008-06-16 20:37:30 +0000110}
111
112NSComparisonResult f4(NSString* s, NSStringCompareOptions op, NSRange R) {
113 NSString *aString = 0;
Richard Trieu2fe9b7f2011-12-15 00:38:15 +0000114 return [s compare:aString options:op range:R]; // expected-warning {{Argument to 'NSString' method 'compare:options:range:' cannot be nil}}
Ted Kremenek859be3b2008-06-16 20:37:30 +0000115}
116
117NSComparisonResult f5(NSString* s, NSStringCompareOptions op, NSRange R) {
118 NSString *aString = 0;
Richard Trieu2fe9b7f2011-12-15 00:38:15 +0000119 return [s compare:aString options:op range:R locale:0]; // expected-warning {{Argument to 'NSString' method 'compare:options:range:locale:' cannot be nil}}
Ted Kremenek859be3b2008-06-16 20:37:30 +0000120}
121
Daniel Dunbar4489fe12008-08-05 00:07:51 +0000122NSArray *f6(NSString* s) {
Richard Trieu2fe9b7f2011-12-15 00:38:15 +0000123 return [s componentsSeparatedByCharactersInSet:0]; // expected-warning {{Argument to 'NSString' method 'componentsSeparatedByCharactersInSet:' cannot be nil}}
Ted Kremenek859be3b2008-06-16 20:37:30 +0000124}
125
126NSString* f7(NSString* s1, NSString* s2, NSString* s3) {
127
Daniel Dunbar4489fe12008-08-05 00:07:51 +0000128 NSString* s4 = (NSString*)
Ted Kremenekcf118d42009-02-04 23:49:09 +0000129 CFStringCreateWithFormat(kCFAllocatorDefault, 0, // expected-warning{{leak}}
Daniel Dunbar4489fe12008-08-05 00:07:51 +0000130 (CFStringRef) __builtin___CFStringMakeConstantString("%@ %@ (%@)"),
131 s1, s2, s3);
Ted Kremenek859be3b2008-06-16 20:37:30 +0000132
133 CFRetain(s4);
Ted Kremenekcf118d42009-02-04 23:49:09 +0000134 return s4;
Ted Kremenek859be3b2008-06-16 20:37:30 +0000135}
136
137NSMutableArray* f8() {
138
139 NSString* s = [[NSString alloc] init];
140 NSMutableArray* a = [[NSMutableArray alloc] initWithCapacity:2];
141 [a addObject:s];
142 [s release]; // no-warning
143 return a;
144}
145
146void f9() {
147
148 NSString* s = [[NSString alloc] init];
149 NSString* q = s;
150 [s release];
151 [q release]; // expected-warning {{used after it is released}}
152}
153
154NSString* f10() {
Ted Kremenek859be3b2008-06-16 20:37:30 +0000155 static NSString* s = 0;
Ted Kremenek859be3b2008-06-16 20:37:30 +0000156 if (!s) s = [[NSString alloc] init];
Ted Kremenek859be3b2008-06-16 20:37:30 +0000157 return s; // no-warning
158}
159
Ted Kremenek2fb78a72008-12-17 21:50:35 +0000160// Test case for regression reported in <rdar://problem/6452745>.
161// Essentially 's' should not be considered allocated on the false branch.
162// This exercises the 'EvalAssume' logic in GRTransferFuncs (CFRefCount.cpp).
163NSString* f11(CFDictionaryRef dict, const char* key) {
164 NSString* s = (NSString*) CFDictionaryGetValue(dict, key);
165 [s retain];
166 if (s) {
167 [s release];
168 }
Mike Stump4393b3f2009-07-21 18:46:15 +0000169 return 0;
Ted Kremenek2fb78a72008-12-17 21:50:35 +0000170}
171
Ted Kremenek784606f2008-12-18 23:40:58 +0000172// Test case for passing a tracked object by-reference to a function we
Ted Kremenek09f14192009-04-21 20:01:03 +0000173// don't understand.
Ted Kremenek784606f2008-12-18 23:40:58 +0000174void unknown_function_f12(NSString** s);
175void f12() {
176 NSString *string = [[NSString alloc] init];
177 unknown_function_f12(&string); // no-warning
178}
179
Ted Kremenek09f14192009-04-21 20:01:03 +0000180// Test double release of CFString (PR 4014).
181void f13(void) {
182 CFStringRef ref = CFStringCreateWithFormat(kCFAllocatorDefault, ((void*)0), ((CFStringRef) __builtin___CFStringMakeConstantString ("" "%d" "")), 100);
183 CFRelease(ref);
184 CFRelease(ref); // expected-warning{{Reference-counted object is used after it is released}}
185}
Ted Kremenek784606f2008-12-18 23:40:58 +0000186
Anders Carlssonb62bdce2011-03-08 20:05:26 +0000187@interface MyString : NSString
188@end
189
190void f14(MyString *s) {
Richard Trieu2fe9b7f2011-12-15 00:38:15 +0000191 [s compare:0]; // expected-warning {{Argument to 'MyString' method 'compare:' cannot be nil}}
Anders Carlssonb62bdce2011-03-08 20:05:26 +0000192}
193
Ted Kremenek95d3b902009-05-11 15:26:06 +0000194// Test regular use of -autorelease
195@interface TestAutorelease
196-(NSString*) getString;
197@end
198@implementation TestAutorelease
199-(NSString*) getString {
200 NSString *str = [[NSString alloc] init];
201 return [str autorelease]; // no-warning
202}
Ted Kremenek2033a952009-05-13 07:12:33 +0000203- (void)m1
204{
205 NSString *s = [[NSString alloc] init]; // expected-warning{{leak}}
206 [s retain];
207 [s autorelease];
208}
209- (void)m2
210{
211 NSString *s = [[[NSString alloc] init] autorelease]; // expected-warning{{leak}}
212 [s retain];
213}
214- (void)m3
215{
216 NSString *s = [[[NSString alloc] init] autorelease];
217 [s retain];
218 [s autorelease];
219}
220- (void)m4
221{
222 NSString *s = [[NSString alloc] init]; // expected-warning{{leak}}
223 [s retain];
224}
225- (void)m5
226{
227 NSString *s = [[NSString alloc] init];
228 [s autorelease];
229}
Ted Kremenek95d3b902009-05-11 15:26:06 +0000230@end
231
Ted Kremenek859be3b2008-06-16 20:37:30 +0000232@interface C1 : NSObject {}
233- (NSString*) getShared;
234+ (C1*) sharedInstance;
235@end
Ted Kremenekaeca9632008-07-03 15:37:02 +0000236@implementation C1 : NSObject {}
Ted Kremenek859be3b2008-06-16 20:37:30 +0000237- (NSString*) getShared {
238 static NSString* s = 0;
239 if (!s) s = [[NSString alloc] init];
240 return s; // no-warning
241}
242+ (C1 *)sharedInstance {
243 static C1 *sharedInstance = 0;
244 if (!sharedInstance) {
245 sharedInstance = [[C1 alloc] init];
246 }
247 return sharedInstance; // no-warning
248}
249@end
250
251@interface SharedClass : NSObject
252+ (id)sharedInstance;
Ted Kremenek234a4c22009-01-07 00:39:56 +0000253- (id)notShared;
Ted Kremenek859be3b2008-06-16 20:37:30 +0000254@end
Ted Kremenek234a4c22009-01-07 00:39:56 +0000255
Ted Kremenekaeca9632008-07-03 15:37:02 +0000256@implementation SharedClass
Ted Kremenek859be3b2008-06-16 20:37:30 +0000257
258- (id)_init {
259 if ((self = [super init])) {
260 NSLog(@"Bar");
261 }
262 return self;
263}
264
Ted Kremenek234a4c22009-01-07 00:39:56 +0000265- (id)notShared {
Ted Kremenek043254a2009-02-07 22:55:48 +0000266 return [[SharedClass alloc] _init]; // expected-warning{{leak}}
Ted Kremenek234a4c22009-01-07 00:39:56 +0000267}
268
Ted Kremenek859be3b2008-06-16 20:37:30 +0000269+ (id)sharedInstance {
270 static SharedClass *_sharedInstance = 0;
271 if (!_sharedInstance) {
272 _sharedInstance = [[SharedClass alloc] _init];
273 }
274 return _sharedInstance; // no-warning
275}
276@end
Ted Kremenek234a4c22009-01-07 00:39:56 +0000277
278id testSharedClassFromFunction() {
279 return [[SharedClass alloc] _init]; // no-warning
280}
281
Ted Kremenek1670e402009-04-11 00:11:10 +0000282// Test OSCompareAndSwap
283_Bool OSAtomicCompareAndSwapPtr( void *__oldValue, void *__newValue, void * volatile *__theValue );
Ted Kremenekb3bf76f2009-04-11 00:54:13 +0000284extern BOOL objc_atomicCompareAndSwapPtr(id predicate, id replacement, volatile id *objectLocation);
Ted Kremenek1670e402009-04-11 00:11:10 +0000285
286void testOSCompareAndSwap() {
287 NSString *old = 0;
Ted Kremenekb3bf76f2009-04-11 00:54:13 +0000288 NSString *s = [[NSString alloc] init]; // no-warning
Ted Kremenek1670e402009-04-11 00:11:10 +0000289 if (!OSAtomicCompareAndSwapPtr(0, s, (void**) &old))
290 [s release];
291 else
292 [old release];
293}
294
Ted Kremenek6bcd5a02009-12-09 23:29:55 +0000295void testOSCompareAndSwapXXBarrier_local() {
Ted Kremeneka3f45402009-04-29 16:03:59 +0000296 NSString *old = 0;
297 NSString *s = [[NSString alloc] init]; // no-warning
Ted Kremenek45f7c272009-10-15 01:40:34 +0000298 if (!COMPARE_SWAP_BARRIER((intptr_t) 0, (intptr_t) s, (intptr_t*) &old))
Ted Kremeneka3f45402009-04-29 16:03:59 +0000299 [s release];
300 else
301 [old release];
302}
303
Ted Kremenek6bcd5a02009-12-09 23:29:55 +0000304void testOSCompareAndSwapXXBarrier_local_no_direct_release() {
Ted Kremenek45f7c272009-10-15 01:40:34 +0000305 NSString *old = 0;
Zhongxing Xu2f4a6b22009-12-09 08:32:57 +0000306 NSString *s = [[NSString alloc] init]; // no-warning
Ted Kremenek45f7c272009-10-15 01:40:34 +0000307 if (!COMPARE_SWAP_BARRIER((intptr_t) 0, (intptr_t) s, (intptr_t*) &old))
308 return;
309 else
310 [old release];
311}
312
313int testOSCompareAndSwapXXBarrier_id(Class myclass, id xclass) {
314 if (COMPARE_SWAP_BARRIER(0, (intptr_t) myclass, (intptr_t*) &xclass))
Ted Kremenek0aeaf5a2009-07-29 18:18:25 +0000315 return 1;
316 return 0;
Ted Kremenek45f7c272009-10-15 01:40:34 +0000317}
Ted Kremenek0aeaf5a2009-07-29 18:18:25 +0000318
Ted Kremenek6bcd5a02009-12-09 23:29:55 +0000319void test_objc_atomicCompareAndSwap_local() {
Ted Kremenekb3bf76f2009-04-11 00:54:13 +0000320 NSString *old = 0;
321 NSString *s = [[NSString alloc] init]; // no-warning
322 if (!objc_atomicCompareAndSwapPtr(0, s, &old))
323 [s release];
324 else
325 [old release];
326}
327
Ted Kremenek6bcd5a02009-12-09 23:29:55 +0000328void test_objc_atomicCompareAndSwap_local_no_direct_release() {
Ted Kremenek45f7c272009-10-15 01:40:34 +0000329 NSString *old = 0;
Zhongxing Xu2f4a6b22009-12-09 08:32:57 +0000330 NSString *s = [[NSString alloc] init]; // no-warning
Ted Kremenek45f7c272009-10-15 01:40:34 +0000331 if (!objc_atomicCompareAndSwapPtr(0, s, &old))
332 return;
333 else
334 [old release];
335}
336
Ted Kremenek6bcd5a02009-12-09 23:29:55 +0000337void test_objc_atomicCompareAndSwap_parameter(NSString **old) {
338 NSString *s = [[NSString alloc] init]; // no-warning
339 if (!objc_atomicCompareAndSwapPtr(0, s, old))
340 [s release];
341 else
342 [*old release];
343}
344
345void test_objc_atomicCompareAndSwap_parameter_no_direct_release(NSString **old) {
346 NSString *s = [[NSString alloc] init]; // expected-warning{{leak}}
347 if (!objc_atomicCompareAndSwapPtr(0, s, old))
348 return;
349 else
350 [*old release];
351}
352
Ted Kremenek45f7c272009-10-15 01:40:34 +0000353
Ted Kremeneke87450e2009-04-23 19:11:35 +0000354// Test stringWithFormat (<rdar://problem/6815234>)
355void test_stringWithFormat() {
356 NSString *string = [[NSString stringWithFormat:@"%ld", (long) 100] retain];
357 [string release];
Ted Kremenekf9df1362009-04-23 21:25:57 +0000358 [string release]; // expected-warning{{Incorrect decrement of the reference count}}
Ted Kremeneke87450e2009-04-23 19:11:35 +0000359}
360
Ted Kremenek92511432009-05-03 06:08:32 +0000361// Test isTrackedObjectType().
Ted Kremenek97d095f2009-04-23 22:11:07 +0000362typedef NSString* WonkyTypedef;
363@interface TestIsTracked
364+ (WonkyTypedef)newString;
365@end
366
367void test_isTrackedObjectType(void) {
368 NSString *str = [TestIsTracked newString]; // expected-warning{{Potential leak}}
369}
Ted Kremenekb3b0b362009-05-02 01:49:13 +0000370
Ted Kremenek92511432009-05-03 06:08:32 +0000371// Test isTrackedCFObjectType().
372@interface TestIsCFTracked
373+ (CFStringRef) badNewCFString;
374+ (CFStringRef) newCFString;
375@end
376
377@implementation TestIsCFTracked
378+ (CFStringRef) newCFString {
379 return CFStringCreateWithFormat(kCFAllocatorDefault, ((void*)0), ((CFStringRef) __builtin___CFStringMakeConstantString ("" "%d" "")), 100); // no-warning
380}
381+ (CFStringRef) badNewCFString {
382 return CFStringCreateWithFormat(kCFAllocatorDefault, ((void*)0), ((CFStringRef) __builtin___CFStringMakeConstantString ("" "%d" "")), 100); // expected-warning{{leak}}
383}
384
Ted Kremenekb3b0b362009-05-02 01:49:13 +0000385// Test @synchronized
386void test_synchronized(id x) {
387 @synchronized(x) {
388 NSString *string = [[NSString stringWithFormat:@"%ld", (long) 100] retain]; // expected-warning {{leak}}
389 }
390}
Fariborz Jahanian63e963c2009-11-16 18:57:01 +0000391@end
Zhongxing Xu604848a2010-01-11 06:52:53 +0000392
393void testOSCompareAndSwapXXBarrier_parameter(NSString **old) {
394 NSString *s = [[NSString alloc] init]; // no-warning
395 if (!COMPARE_SWAP_BARRIER((intptr_t) 0, (intptr_t) s, (intptr_t*) old))
396 [s release];
397 else
398 [*old release];
399}
Zhongxing Xu951b3342010-01-11 07:40:00 +0000400
401void testOSCompareAndSwapXXBarrier_parameter_no_direct_release(NSString **old) {
402 NSString *s = [[NSString alloc] init]; // no-warning
403 if (!COMPARE_SWAP_BARRIER((intptr_t) 0, (intptr_t) s, (intptr_t*) old))
404 [s release];
405 else
406 return;
407}
Anna Zaks453cb852013-02-02 00:30:04 +0000408
409@interface AlwaysInlineBodyFarmBodies : NSObject {
410 NSString *_value;
411}
412 - (NSString *)_value;
413 - (void)callValue;
414@end
415
416@implementation AlwaysInlineBodyFarmBodies
417
418- (NSString *)_value {
419 if (!_value) {
420 NSString *s = [[NSString alloc] init];
421 if (!OSAtomicCompareAndSwapPtr(0, s, (void**)&_value)) {
422 [s release];
423 }
424 }
425 return _value;
426}
427
428- (void)callValue {
429 [self _value];
430}
431@end