NSString.m revision be1fe1eb12a1cb91c8e3a9fcc2db4dfe989def6c
1// RUN: clang -analyze -checker-cfref -analyzer-store=basic -verify %s && 2// RUN: clang -analyze -checker-cfref -analyzer-store=basic -analyzer-constraints=range -verify %s && 3// RUN: clang -analyze -checker-cfref -analyzer-store=region -verify %s 4 5//===----------------------------------------------------------------------===// 6// The following code is reduced using delta-debugging from 7// Foundation.h (Mac OS X). 8// 9// It includes the basic definitions for the test cases below. 10// Not directly including Foundation.h directly makes this test case 11// both svelte and portable to non-Mac platforms. 12//===----------------------------------------------------------------------===// 13 14typedef const void * CFTypeRef; 15typedef const struct __CFString * CFStringRef; 16typedef const struct __CFAllocator * CFAllocatorRef; 17extern const CFAllocatorRef kCFAllocatorDefault; 18extern CFTypeRef CFRetain(CFTypeRef cf); 19typedef const struct __CFDictionary * CFDictionaryRef; 20const void *CFDictionaryGetValue(CFDictionaryRef theDict, const void *key); 21extern CFStringRef CFStringCreateWithFormat(CFAllocatorRef alloc, CFDictionaryRef formatOptions, CFStringRef format, ...); 22typedef signed char BOOL; 23typedef int NSInteger; 24typedef unsigned int NSUInteger; 25@class NSString, Protocol; 26extern void NSLog(NSString *format, ...) __attribute__((format(__NSString__, 1, 2))); 27typedef NSInteger NSComparisonResult; 28typedef struct _NSZone NSZone; 29@class NSInvocation, NSMethodSignature, NSCoder, NSString, NSEnumerator; 30@protocol NSObject 31- (BOOL)isEqual:(id)object; 32- (oneway void)release; 33- (id)retain; 34@end 35@protocol NSCopying 36- (id)copyWithZone:(NSZone *)zone; 37@end 38@protocol NSMutableCopying 39- (id)mutableCopyWithZone:(NSZone *)zone; 40@end 41@protocol NSCoding 42- (void)encodeWithCoder:(NSCoder *)aCoder; 43@end 44@interface NSObject <NSObject> {} 45- (id)init; 46+ (id)alloc; 47@end 48extern id NSAllocateObject(Class aClass, NSUInteger extraBytes, NSZone *zone); 49typedef struct {} NSFastEnumerationState; 50@protocol NSFastEnumeration 51- (NSUInteger)countByEnumeratingWithState:(NSFastEnumerationState *)state objects:(id *)stackbuf count:(NSUInteger)len; 52@end 53@class NSString; 54typedef struct _NSRange {} NSRange; 55@interface NSArray : NSObject <NSCopying, NSMutableCopying, NSCoding, NSFastEnumeration> 56- (NSUInteger)count; 57@end 58@interface NSMutableArray : NSArray 59- (void)addObject:(id)anObject; 60- (id)initWithCapacity:(NSUInteger)numItems; 61@end 62typedef unsigned short unichar; 63@class NSData, NSArray, NSDictionary, NSCharacterSet, NSData, NSURL, NSError, NSLocale; 64typedef NSUInteger NSStringCompareOptions; 65@interface NSString : NSObject <NSCopying, NSMutableCopying, NSCoding> - (NSUInteger)length; 66- (NSComparisonResult)compare:(NSString *)string; 67- (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask; 68- (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask range:(NSRange)compareRange; 69- (NSComparisonResult)compare:(NSString *)string options:(NSStringCompareOptions)mask range:(NSRange)compareRange locale:(id)locale; 70- (NSComparisonResult)caseInsensitiveCompare:(NSString *)string; 71- (NSArray *)componentsSeparatedByCharactersInSet:(NSCharacterSet *)separator; 72@end 73@interface NSSimpleCString : NSString {} @end 74@interface NSConstantString : NSSimpleCString @end 75extern void *_NSConstantStringClassReference; 76 77//===----------------------------------------------------------------------===// 78// Test cases. 79//===----------------------------------------------------------------------===// 80 81NSComparisonResult f1(NSString* s) { 82 NSString *aString = 0; 83 return [s compare:aString]; // expected-warning {{Argument to 'NSString' method 'compare:' cannot be nil.}} 84} 85 86NSComparisonResult f2(NSString* s) { 87 NSString *aString = 0; 88 return [s caseInsensitiveCompare:aString]; // expected-warning {{Argument to 'NSString' method 'caseInsensitiveCompare:' cannot be nil.}} 89} 90 91NSComparisonResult f3(NSString* s, NSStringCompareOptions op) { 92 NSString *aString = 0; 93 return [s compare:aString options:op]; // expected-warning {{Argument to 'NSString' method 'compare:options:' cannot be nil.}} 94} 95 96NSComparisonResult f4(NSString* s, NSStringCompareOptions op, NSRange R) { 97 NSString *aString = 0; 98 return [s compare:aString options:op range:R]; // expected-warning {{Argument to 'NSString' method 'compare:options:range:' cannot be nil.}} 99} 100 101NSComparisonResult f5(NSString* s, NSStringCompareOptions op, NSRange R) { 102 NSString *aString = 0; 103 return [s compare:aString options:op range:R locale:0]; // expected-warning {{Argument to 'NSString' method 'compare:options:range:locale:' cannot be nil.}} 104} 105 106NSArray *f6(NSString* s) { 107 return [s componentsSeparatedByCharactersInSet:0]; // expected-warning {{Argument to 'NSString' method 'componentsSeparatedByCharactersInSet:' cannot be nil.}} 108} 109 110NSString* f7(NSString* s1, NSString* s2, NSString* s3) { 111 112 NSString* s4 = (NSString*) 113 CFStringCreateWithFormat(kCFAllocatorDefault, 0, // expected-warning{{leak}} 114 (CFStringRef) __builtin___CFStringMakeConstantString("%@ %@ (%@)"), 115 s1, s2, s3); 116 117 CFRetain(s4); 118 return s4; 119} 120 121NSMutableArray* f8() { 122 123 NSString* s = [[NSString alloc] init]; 124 NSMutableArray* a = [[NSMutableArray alloc] initWithCapacity:2]; 125 [a addObject:s]; 126 [s release]; // no-warning 127 return a; 128} 129 130void f9() { 131 132 NSString* s = [[NSString alloc] init]; 133 NSString* q = s; 134 [s release]; 135 [q release]; // expected-warning {{used after it is released}} 136} 137 138NSString* f10() { 139 static NSString* s = 0; 140 if (!s) s = [[NSString alloc] init]; 141 return s; // no-warning 142} 143 144// Test case for regression reported in <rdar://problem/6452745>. 145// Essentially 's' should not be considered allocated on the false branch. 146// This exercises the 'EvalAssume' logic in GRTransferFuncs (CFRefCount.cpp). 147NSString* f11(CFDictionaryRef dict, const char* key) { 148 NSString* s = (NSString*) CFDictionaryGetValue(dict, key); 149 [s retain]; 150 if (s) { 151 [s release]; 152 } 153} 154 155// Test case for passing a tracked object by-reference to a function we 156// don't undersand. 157void unknown_function_f12(NSString** s); 158void f12() { 159 NSString *string = [[NSString alloc] init]; 160 unknown_function_f12(&string); // no-warning 161} 162 163 164@interface C1 : NSObject {} 165- (NSString*) getShared; 166+ (C1*) sharedInstance; 167@end 168@implementation C1 : NSObject {} 169- (NSString*) getShared { 170 static NSString* s = 0; 171 if (!s) s = [[NSString alloc] init]; 172 return s; // no-warning 173} 174+ (C1 *)sharedInstance { 175 static C1 *sharedInstance = 0; 176 if (!sharedInstance) { 177 sharedInstance = [[C1 alloc] init]; 178 } 179 return sharedInstance; // no-warning 180} 181@end 182 183@interface SharedClass : NSObject 184+ (id)sharedInstance; 185- (id)notShared; 186@end 187 188@implementation SharedClass 189 190- (id)_init { 191 if ((self = [super init])) { 192 NSLog(@"Bar"); 193 } 194 return self; 195} 196 197- (id)notShared { 198 return [[SharedClass alloc] _init]; // expected-warning{{leak}} 199} 200 201+ (id)sharedInstance { 202 static SharedClass *_sharedInstance = 0; 203 if (!_sharedInstance) { 204 _sharedInstance = [[SharedClass alloc] _init]; 205 } 206 return _sharedInstance; // no-warning 207} 208@end 209 210id testSharedClassFromFunction() { 211 return [[SharedClass alloc] _init]; // no-warning 212} 213 214