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