blob: 24009ddeccd7c5b226b5e14766b929987a2cd160 [file] [log] [blame]
Eli Friedmanc219a152009-05-19 11:12:40 +00001// RUN: clang-cc -triple i386-pc-linux-gnu -analyze -checker-cfref -analyzer-store=basic -analyzer-constraints=basic -verify %s &&
Ted Kremeneke1cea752009-07-06 21:58:46 +00002// RUN: clang-cc -triple i386-pc-linux-gnu -analyze -checker-cfref -analyzer-store=basic-new-cast -analyzer-constraints=basic -verify %s &&
3// RUN: clang-cc -triple i386-pc-linux-gnu -analyze -checker-cfref -analyzer-store=basic -analyzer-constraints=range -verify %s &&
4// RUN: clang-cc -triple i386-pc-linux-gnu -analyze -checker-cfref -analyzer-store=basic-new-cast -analyzer-constraints=range -verify %s
Ted Kremenek1670e402009-04-11 00:11:10 +00005
6
Eli Friedmanc219a152009-05-19 11:12:40 +00007// NOTWORK: clang-cc -triple i386-pc-linux-gnu -analyze -checker-cfref -analyzer-store=region -analyzer-constraints=basic -verify %s &&
8// NOTWORK: clang-cc -triple i386-pc-linux-gnu -analyze -checker-cfref -analyzer-store=region -analyzer-constraints=range -verify %s
Ted Kremenek859be3b2008-06-16 20:37:30 +00009
10//===----------------------------------------------------------------------===//
11// The following code is reduced using delta-debugging from
12// Foundation.h (Mac OS X).
13//
14// It includes the basic definitions for the test cases below.
15// Not directly including Foundation.h directly makes this test case
16// both svelte and portable to non-Mac platforms.
17//===----------------------------------------------------------------------===//
18
Ted Kremeneka3f45402009-04-29 16:03:59 +000019typedef int int32_t;
Ted Kremenek859be3b2008-06-16 20:37:30 +000020typedef const void * CFTypeRef;
21typedef const struct __CFString * CFStringRef;
22typedef const struct __CFAllocator * CFAllocatorRef;
23extern const CFAllocatorRef kCFAllocatorDefault;
24extern CFTypeRef CFRetain(CFTypeRef cf);
Ted Kremenek09f14192009-04-21 20:01:03 +000025void CFRelease(CFTypeRef cf);
Ted Kremenek859be3b2008-06-16 20:37:30 +000026typedef const struct __CFDictionary * CFDictionaryRef;
Ted Kremenek2fb78a72008-12-17 21:50:35 +000027const void *CFDictionaryGetValue(CFDictionaryRef theDict, const void *key);
Ted Kremenek859be3b2008-06-16 20:37:30 +000028extern CFStringRef CFStringCreateWithFormat(CFAllocatorRef alloc, CFDictionaryRef formatOptions, CFStringRef format, ...);
29typedef signed char BOOL;
30typedef int NSInteger;
31typedef unsigned int NSUInteger;
32@class NSString, Protocol;
33extern void NSLog(NSString *format, ...) __attribute__((format(__NSString__, 1, 2)));
34typedef NSInteger NSComparisonResult;
35typedef struct _NSZone NSZone;
36@class NSInvocation, NSMethodSignature, NSCoder, NSString, NSEnumerator;
37@protocol NSObject
38- (BOOL)isEqual:(id)object;
39- (oneway void)release;
Ted Kremenek2fb78a72008-12-17 21:50:35 +000040- (id)retain;
Ted Kremenek95d3b902009-05-11 15:26:06 +000041- (id)autorelease;
Ted Kremenek859be3b2008-06-16 20:37:30 +000042@end
43@protocol NSCopying
44- (id)copyWithZone:(NSZone *)zone;
45@end
46@protocol NSMutableCopying
47- (id)mutableCopyWithZone:(NSZone *)zone;
48@end
49@protocol NSCoding
50- (void)encodeWithCoder:(NSCoder *)aCoder;
51@end
52@interface NSObject <NSObject> {}
53- (id)init;
54+ (id)alloc;
55@end
56extern id NSAllocateObject(Class aClass, NSUInteger extraBytes, NSZone *zone);
57typedef struct {} NSFastEnumerationState;
58@protocol NSFastEnumeration
59- (NSUInteger)countByEnumeratingWithState:(NSFastEnumerationState *)state objects:(id *)stackbuf count:(NSUInteger)len;
60@end
61@class NSString;
62typedef struct _NSRange {} NSRange;
63@interface NSArray : NSObject <NSCopying, NSMutableCopying, NSCoding, NSFastEnumeration>
64- (NSUInteger)count;
65@end
66@interface NSMutableArray : NSArray
67- (void)addObject:(id)anObject;
68- (id)initWithCapacity:(NSUInteger)numItems;
69@end
70typedef unsigned short unichar;
71@class NSData, NSArray, NSDictionary, NSCharacterSet, NSData, NSURL, NSError, NSLocale;
72typedef NSUInteger NSStringCompareOptions;
73@interface NSString : NSObject <NSCopying, NSMutableCopying, NSCoding> - (NSUInteger)length;
74- (NSComparisonResult)compare:(NSString *)string;
75- (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask;
76- (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask range:(NSRange)compareRange;
77- (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask range:(NSRange)compareRange locale:(id)locale;
78- (NSComparisonResult)caseInsensitiveCompare:(NSString *)string;
79- (NSArray *)componentsSeparatedByCharactersInSet:(NSCharacterSet *)separator;
Ted Kremeneke87450e2009-04-23 19:11:35 +000080+ (id)stringWithFormat:(NSString *)format, ... __attribute__((format(__NSString__, 1, 2)));
Ted Kremenek859be3b2008-06-16 20:37:30 +000081@end
82@interface NSSimpleCString : NSString {} @end
83@interface NSConstantString : NSSimpleCString @end
84extern void *_NSConstantStringClassReference;
85
86//===----------------------------------------------------------------------===//
87// Test cases.
88//===----------------------------------------------------------------------===//
89
90NSComparisonResult f1(NSString* s) {
91 NSString *aString = 0;
92 return [s compare:aString]; // expected-warning {{Argument to 'NSString' method 'compare:' cannot be nil.}}
93}
94
95NSComparisonResult f2(NSString* s) {
96 NSString *aString = 0;
97 return [s caseInsensitiveCompare:aString]; // expected-warning {{Argument to 'NSString' method 'caseInsensitiveCompare:' cannot be nil.}}
98}
99
100NSComparisonResult f3(NSString* s, NSStringCompareOptions op) {
101 NSString *aString = 0;
102 return [s compare:aString options:op]; // expected-warning {{Argument to 'NSString' method 'compare:options:' cannot be nil.}}
103}
104
105NSComparisonResult f4(NSString* s, NSStringCompareOptions op, NSRange R) {
106 NSString *aString = 0;
107 return [s compare:aString options:op range:R]; // expected-warning {{Argument to 'NSString' method 'compare:options:range:' cannot be nil.}}
108}
109
110NSComparisonResult f5(NSString* s, NSStringCompareOptions op, NSRange R) {
111 NSString *aString = 0;
112 return [s compare:aString options:op range:R locale:0]; // expected-warning {{Argument to 'NSString' method 'compare:options:range:locale:' cannot be nil.}}
113}
114
Daniel Dunbar4489fe12008-08-05 00:07:51 +0000115NSArray *f6(NSString* s) {
Ted Kremenek859be3b2008-06-16 20:37:30 +0000116 return [s componentsSeparatedByCharactersInSet:0]; // expected-warning {{Argument to 'NSString' method 'componentsSeparatedByCharactersInSet:' cannot be nil.}}
117}
118
119NSString* f7(NSString* s1, NSString* s2, NSString* s3) {
120
Daniel Dunbar4489fe12008-08-05 00:07:51 +0000121 NSString* s4 = (NSString*)
Ted Kremenekcf118d42009-02-04 23:49:09 +0000122 CFStringCreateWithFormat(kCFAllocatorDefault, 0, // expected-warning{{leak}}
Daniel Dunbar4489fe12008-08-05 00:07:51 +0000123 (CFStringRef) __builtin___CFStringMakeConstantString("%@ %@ (%@)"),
124 s1, s2, s3);
Ted Kremenek859be3b2008-06-16 20:37:30 +0000125
126 CFRetain(s4);
Ted Kremenekcf118d42009-02-04 23:49:09 +0000127 return s4;
Ted Kremenek859be3b2008-06-16 20:37:30 +0000128}
129
130NSMutableArray* f8() {
131
132 NSString* s = [[NSString alloc] init];
133 NSMutableArray* a = [[NSMutableArray alloc] initWithCapacity:2];
134 [a addObject:s];
135 [s release]; // no-warning
136 return a;
137}
138
139void f9() {
140
141 NSString* s = [[NSString alloc] init];
142 NSString* q = s;
143 [s release];
144 [q release]; // expected-warning {{used after it is released}}
145}
146
147NSString* f10() {
Ted Kremenek859be3b2008-06-16 20:37:30 +0000148 static NSString* s = 0;
Ted Kremenek859be3b2008-06-16 20:37:30 +0000149 if (!s) s = [[NSString alloc] init];
Ted Kremenek859be3b2008-06-16 20:37:30 +0000150 return s; // no-warning
151}
152
Ted Kremenek2fb78a72008-12-17 21:50:35 +0000153// Test case for regression reported in <rdar://problem/6452745>.
154// Essentially 's' should not be considered allocated on the false branch.
155// This exercises the 'EvalAssume' logic in GRTransferFuncs (CFRefCount.cpp).
156NSString* f11(CFDictionaryRef dict, const char* key) {
157 NSString* s = (NSString*) CFDictionaryGetValue(dict, key);
158 [s retain];
159 if (s) {
160 [s release];
161 }
162}
163
Ted Kremenek784606f2008-12-18 23:40:58 +0000164// Test case for passing a tracked object by-reference to a function we
Ted Kremenek09f14192009-04-21 20:01:03 +0000165// don't understand.
Ted Kremenek784606f2008-12-18 23:40:58 +0000166void unknown_function_f12(NSString** s);
167void f12() {
168 NSString *string = [[NSString alloc] init];
169 unknown_function_f12(&string); // no-warning
170}
171
Ted Kremenek09f14192009-04-21 20:01:03 +0000172// Test double release of CFString (PR 4014).
173void f13(void) {
174 CFStringRef ref = CFStringCreateWithFormat(kCFAllocatorDefault, ((void*)0), ((CFStringRef) __builtin___CFStringMakeConstantString ("" "%d" "")), 100);
175 CFRelease(ref);
176 CFRelease(ref); // expected-warning{{Reference-counted object is used after it is released}}
177}
Ted Kremenek784606f2008-12-18 23:40:58 +0000178
Ted Kremenek95d3b902009-05-11 15:26:06 +0000179// Test regular use of -autorelease
180@interface TestAutorelease
181-(NSString*) getString;
182@end
183@implementation TestAutorelease
184-(NSString*) getString {
185 NSString *str = [[NSString alloc] init];
186 return [str autorelease]; // no-warning
187}
Ted Kremenek2033a952009-05-13 07:12:33 +0000188- (void)m1
189{
190 NSString *s = [[NSString alloc] init]; // expected-warning{{leak}}
191 [s retain];
192 [s autorelease];
193}
194- (void)m2
195{
196 NSString *s = [[[NSString alloc] init] autorelease]; // expected-warning{{leak}}
197 [s retain];
198}
199- (void)m3
200{
201 NSString *s = [[[NSString alloc] init] autorelease];
202 [s retain];
203 [s autorelease];
204}
205- (void)m4
206{
207 NSString *s = [[NSString alloc] init]; // expected-warning{{leak}}
208 [s retain];
209}
210- (void)m5
211{
212 NSString *s = [[NSString alloc] init];
213 [s autorelease];
214}
Ted Kremenek95d3b902009-05-11 15:26:06 +0000215@end
216
Ted Kremenek859be3b2008-06-16 20:37:30 +0000217@interface C1 : NSObject {}
218- (NSString*) getShared;
219+ (C1*) sharedInstance;
220@end
Ted Kremenekaeca9632008-07-03 15:37:02 +0000221@implementation C1 : NSObject {}
Ted Kremenek859be3b2008-06-16 20:37:30 +0000222- (NSString*) getShared {
223 static NSString* s = 0;
224 if (!s) s = [[NSString alloc] init];
225 return s; // no-warning
226}
227+ (C1 *)sharedInstance {
228 static C1 *sharedInstance = 0;
229 if (!sharedInstance) {
230 sharedInstance = [[C1 alloc] init];
231 }
232 return sharedInstance; // no-warning
233}
234@end
235
236@interface SharedClass : NSObject
237+ (id)sharedInstance;
Ted Kremenek234a4c22009-01-07 00:39:56 +0000238- (id)notShared;
Ted Kremenek859be3b2008-06-16 20:37:30 +0000239@end
Ted Kremenek234a4c22009-01-07 00:39:56 +0000240
Ted Kremenekaeca9632008-07-03 15:37:02 +0000241@implementation SharedClass
Ted Kremenek859be3b2008-06-16 20:37:30 +0000242
243- (id)_init {
244 if ((self = [super init])) {
245 NSLog(@"Bar");
246 }
247 return self;
248}
249
Ted Kremenek234a4c22009-01-07 00:39:56 +0000250- (id)notShared {
Ted Kremenek043254a2009-02-07 22:55:48 +0000251 return [[SharedClass alloc] _init]; // expected-warning{{leak}}
Ted Kremenek234a4c22009-01-07 00:39:56 +0000252}
253
Ted Kremenek859be3b2008-06-16 20:37:30 +0000254+ (id)sharedInstance {
255 static SharedClass *_sharedInstance = 0;
256 if (!_sharedInstance) {
257 _sharedInstance = [[SharedClass alloc] _init];
258 }
259 return _sharedInstance; // no-warning
260}
261@end
Ted Kremenek234a4c22009-01-07 00:39:56 +0000262
263id testSharedClassFromFunction() {
264 return [[SharedClass alloc] _init]; // no-warning
265}
266
Ted Kremenek1670e402009-04-11 00:11:10 +0000267// Test OSCompareAndSwap
268_Bool OSAtomicCompareAndSwapPtr( void *__oldValue, void *__newValue, void * volatile *__theValue );
Ted Kremenek98104722009-05-01 17:37:31 +0000269_Bool OSAtomicCompareAndSwap32Barrier( int32_t __oldValue, int32_t __newValue, volatile int32_t *__theValue );
Ted Kremenekb3bf76f2009-04-11 00:54:13 +0000270extern BOOL objc_atomicCompareAndSwapPtr(id predicate, id replacement, volatile id *objectLocation);
Ted Kremenek1670e402009-04-11 00:11:10 +0000271
272void testOSCompareAndSwap() {
273 NSString *old = 0;
Ted Kremenekb3bf76f2009-04-11 00:54:13 +0000274 NSString *s = [[NSString alloc] init]; // no-warning
Ted Kremenek1670e402009-04-11 00:11:10 +0000275 if (!OSAtomicCompareAndSwapPtr(0, s, (void**) &old))
276 [s release];
277 else
278 [old release];
279}
280
Ted Kremeneka3f45402009-04-29 16:03:59 +0000281void testOSCompareAndSwap32Barrier() {
282 NSString *old = 0;
283 NSString *s = [[NSString alloc] init]; // no-warning
284 if (!OSAtomicCompareAndSwap32Barrier((int32_t) 0, (int32_t) s, (int32_t*) &old))
285 [s release];
286 else
287 [old release];
288}
289
Ted Kremenekb3bf76f2009-04-11 00:54:13 +0000290void test_objc_atomicCompareAndSwap() {
291 NSString *old = 0;
292 NSString *s = [[NSString alloc] init]; // no-warning
293 if (!objc_atomicCompareAndSwapPtr(0, s, &old))
294 [s release];
295 else
296 [old release];
297}
298
Ted Kremeneke87450e2009-04-23 19:11:35 +0000299// Test stringWithFormat (<rdar://problem/6815234>)
300void test_stringWithFormat() {
301 NSString *string = [[NSString stringWithFormat:@"%ld", (long) 100] retain];
302 [string release];
Ted Kremenekf9df1362009-04-23 21:25:57 +0000303 [string release]; // expected-warning{{Incorrect decrement of the reference count}}
Ted Kremeneke87450e2009-04-23 19:11:35 +0000304}
305
Ted Kremenek92511432009-05-03 06:08:32 +0000306// Test isTrackedObjectType().
Ted Kremenek97d095f2009-04-23 22:11:07 +0000307typedef NSString* WonkyTypedef;
308@interface TestIsTracked
309+ (WonkyTypedef)newString;
310@end
311
312void test_isTrackedObjectType(void) {
313 NSString *str = [TestIsTracked newString]; // expected-warning{{Potential leak}}
314}
Ted Kremenekb3b0b362009-05-02 01:49:13 +0000315
Ted Kremenek92511432009-05-03 06:08:32 +0000316// Test isTrackedCFObjectType().
317@interface TestIsCFTracked
318+ (CFStringRef) badNewCFString;
319+ (CFStringRef) newCFString;
320@end
321
322@implementation TestIsCFTracked
323+ (CFStringRef) newCFString {
324 return CFStringCreateWithFormat(kCFAllocatorDefault, ((void*)0), ((CFStringRef) __builtin___CFStringMakeConstantString ("" "%d" "")), 100); // no-warning
325}
326+ (CFStringRef) badNewCFString {
327 return CFStringCreateWithFormat(kCFAllocatorDefault, ((void*)0), ((CFStringRef) __builtin___CFStringMakeConstantString ("" "%d" "")), 100); // expected-warning{{leak}}
328}
329
Ted Kremenekb3b0b362009-05-02 01:49:13 +0000330// Test @synchronized
331void test_synchronized(id x) {
332 @synchronized(x) {
333 NSString *string = [[NSString stringWithFormat:@"%ld", (long) 100] retain]; // expected-warning {{leak}}
334 }
335}
336
Ted Kremenek2033a952009-05-13 07:12:33 +0000337