retain-release-path-notes.m revision 70fdbc366da85880aae5baebd3351e993ca05603
1// RUN: %clang_cc1 -triple x86_64-apple-darwin10 -analyze -analyzer-checker=core,osx.coreFoundation.CFRetainRelease,osx.cocoa.ClassRelease,osx.cocoa.RetainCount -analyzer-store=region -analyzer-output=text -verify %s
2
3// This actually still works after the pseudo-object refactor, it just
4// uses messages that say 'method' instead of 'property'.  Ted wanted
5// this xfailed and filed as a bug.  rdar://problem/10402993
6
7/***
8This file is for testing the path-sensitive notes for retain/release errors.
9Its goal is to have simple branch coverage of any path-based diagnostics,
10not to actually check all possible retain/release errors.
11
12This file includes notes that only appear in a ref-counted analysis. 
13GC-specific notes should go in retain-release-path-notes-gc.m.
14***/
15
16@interface NSObject
17+ (id)alloc;
18- (id)init;
19- (void)dealloc;
20
21- (Class)class;
22
23- (id)retain;
24- (void)release;
25- (void)autorelease;
26@end
27
28@interface Foo : NSObject
29- (id)methodWithValue;
30@property(retain) id propertyValue;
31@end
32
33typedef struct CFType *CFTypeRef;
34CFTypeRef CFRetain(CFTypeRef);
35void CFRelease(CFTypeRef);
36
37id NSMakeCollectable(CFTypeRef);
38CFTypeRef CFMakeCollectable(CFTypeRef);
39
40CFTypeRef CFCreateSomething();
41CFTypeRef CFGetSomething();
42
43
44void creationViaAlloc () {
45  id leaked = [[NSObject alloc] init]; // expected-warning{{leak}} expected-note{{Method returns an Objective-C object with a +1 retain count}}
46  return; // expected-note{{Object leaked: object allocated and stored into 'leaked' is not referenced later in this execution path and has a retain count of +1}}
47}
48
49void creationViaCFCreate () {
50  CFTypeRef leaked = CFCreateSomething(); // expected-warning{{leak}} expected-note{{Call to function 'CFCreateSomething' returns a Core Foundation object with a +1 retain count}}
51  return; // expected-note{{Object leaked: object allocated and stored into 'leaked' is not referenced later in this execution path and has a retain count of +1}}
52}
53
54void acquisitionViaMethod (Foo *foo) {
55  id leaked = [foo methodWithValue]; // expected-warning{{leak}} expected-note{{Method returns an Objective-C object with a +0 retain count}}
56  [leaked retain]; // expected-note{{Reference count incremented. The object now has a +1 retain count}}
57  [leaked retain]; // expected-note{{Reference count incremented. The object now has a +2 retain count}}
58  [leaked release]; // expected-note{{Reference count decremented. The object now has a +1 retain count}}
59  return; // expected-note{{Object leaked: object allocated and stored into 'leaked' is not referenced later in this execution path and has a retain count of +1}}
60}
61
62void acquisitionViaProperty (Foo *foo) {
63  id leaked = foo.propertyValue; // expected-warning{{leak}} expected-note{{Property returns an Objective-C object with a +0 retain count}}
64  [leaked retain]; // expected-note{{Reference count incremented. The object now has a +1 retain count}}
65  return; // expected-note{{Object leaked: object allocated and stored into 'leaked' is not referenced later in this execution path and has a retain count of +1}}
66}
67
68void acquisitionViaCFFunction () {
69  CFTypeRef leaked = CFGetSomething(); // expected-warning{{leak}} expected-note{{Call to function 'CFGetSomething' returns a Core Foundation object with a +0 retain count}}
70  CFRetain(leaked); // expected-note{{Reference count incremented. The object now has a +1 retain count}}
71  return; // expected-note{{Object leaked: object allocated and stored into 'leaked' is not referenced later in this execution path and has a retain count of +1}}
72}
73
74void explicitDealloc () {
75  id object = [[NSObject alloc] init]; // expected-note{{Method returns an Objective-C object with a +1 retain count}}
76  [object dealloc]; // expected-note{{Object released by directly sending the '-dealloc' message}}
77  [object class]; // expected-warning{{Reference-counted object is used after it is released}} // expected-note{{Reference-counted object is used after it is released}}
78}
79
80void implicitDealloc () {
81  id object = [[NSObject alloc] init]; // expected-note{{Method returns an Objective-C object with a +1 retain count}}
82  [object release]; // expected-note{{Object released}}
83  [object class]; // expected-warning{{Reference-counted object is used after it is released}} // expected-note{{Reference-counted object is used after it is released}}
84}
85
86void overAutorelease () {
87  id object = [[NSObject alloc] init]; // expected-note{{Method returns an Objective-C object with a +1 retain count}}
88  [object autorelease]; // expected-note{{Object sent -autorelease message}}
89  [object autorelease]; // expected-note{{Object sent -autorelease message}} 
90  return; // expected-warning{{Object sent -autorelease too many times}} expected-note{{Object over-autoreleased: object was sent -autorelease 2 times but the object has a +1 retain count}} 
91}
92
93void autoreleaseUnowned (Foo *foo) {
94  id object = foo.propertyValue; // expected-note{{Property returns an Objective-C object with a +0 retain count}}
95  [object autorelease]; // expected-note{{Object sent -autorelease message}} 
96  return; // expected-warning{{Object sent -autorelease too many times}} expected-note{{Object over-autoreleased: object was sent -autorelease but the object has a +0 retain count}}
97}
98
99void makeCollectableIgnored () {
100  CFTypeRef leaked = CFCreateSomething(); // expected-warning{{leak}} expected-note{{Call to function 'CFCreateSomething' returns a Core Foundation object with a +1 retain count}}
101  CFMakeCollectable(leaked); // expected-note{{When GC is not enabled a call to 'CFMakeCollectable' has no effect on its argument}}
102  NSMakeCollectable(leaked); // expected-note{{When GC is not enabled a call to 'NSMakeCollectable' has no effect on its argument}}
103  return; // expected-note{{Object leaked: object allocated and stored into 'leaked' is not referenced later in this execution path and has a retain count of +1}}
104}
105
106CFTypeRef CFCopyRuleViolation () {
107  CFTypeRef object = CFGetSomething(); // expected-note{{Call to function 'CFGetSomething' returns a Core Foundation object with a +0 retain count}}
108  return object; // expected-warning{{Object with a +0 retain count returned to caller where a +1 (owning) retain count is expected}} expected-note{{Object returned to caller with a +0 retain count}} expected-note{{Object with a +0 retain count returned to caller where a +1 (owning) retain count is expected}}
109}
110
111CFTypeRef CFGetRuleViolation () {
112  CFTypeRef object = CFCreateSomething(); // expected-warning{{leak}} expected-note{{Call to function 'CFCreateSomething' returns a Core Foundation object with a +1 retain count}}
113  return object; // expected-note{{Object returned to caller as an owning reference (single retain count transferred to caller)}} expected-note{{Object leaked: object allocated and stored into 'object' is returned from a function whose name ('CFGetRuleViolation') does not contain 'Copy' or 'Create'.  This violates the naming convention rules given in the Memory Management Guide for Core Foundation}}
114}
115
116@implementation Foo (FundamentalMemoryManagementRules)
117- (id)copyViolation {
118  id result = self.propertyValue; // expected-note{{Property returns an Objective-C object with a +0 retain count}}
119  return result; // expected-warning{{Object with a +0 retain count returned to caller where a +1 (owning) retain count is expected}} expected-note{{Object returned to caller with a +0 retain count}} expected-note{{Object with a +0 retain count returned to caller where a +1 (owning) retain count is expected}}
120}
121
122- (id)getViolation {
123  id result = [[Foo alloc] init]; // expected-warning{{leak}} expected-note{{Method returns an Objective-C object with a +1 retain count}}
124  return result; // expected-note{{Object returned to caller as an owning reference (single retain count transferred to caller)}} expected-note{{Object leaked: object allocated and stored into 'result' is returned from a method whose name ('getViolation') does not start with 'copy', 'mutableCopy', 'alloc' or 'new'.  This violates the naming convention rules given in the Memory Management Guide for Cocoa}}
125}
126
127- (id)copyAutorelease {
128  id result = [[Foo alloc] init]; // expected-note{{Method returns an Objective-C object with a +1 retain count}}
129  [result autorelease]; // expected-note{{Object sent -autorelease message}}
130  return result; // expected-warning{{Object with a +0 retain count returned to caller where a +1 (owning) retain count is expected}} expected-note{{Object returned to caller with a +0 retain count}} expected-note{{Object with a +0 retain count returned to caller where a +1 (owning) retain count is expected}}
131}
132@end
133
134
135typedef unsigned long NSUInteger;
136
137@interface NSValue : NSObject
138@end
139
140@interface NSNumber : NSValue
141+ (NSNumber *)numberWithInt:(int)i;
142@end
143
144@interface NSString : NSObject
145+ (NSString *)stringWithUTF8String:(const char *)str;
146@end
147
148@interface NSArray : NSObject
149+ (NSArray *)arrayWithObjects:(const id [])objects count:(NSUInteger)count;
150@end
151
152@interface NSDictionary : NSObject
153+ (id)dictionaryWithObjects:(const id [])objects forKeys:(const id /* <NSCopying> */ [])keys count:(NSUInteger)count;
154@end
155
156
157void testNumericLiteral() {
158  id result = @1; // expected-note{{NSNumber literal is an object with a +0 retain count}}
159  [result release]; // expected-warning{{decrement}} expected-note{{Incorrect decrement of the reference count of an object that is not owned at this point by the caller}}
160}
161
162void testBoxedInt(int x) {
163  id result = @(x); // expected-note{{NSNumber boxed expression produces an object with a +0 retain count}}
164  [result release]; // expected-warning{{decrement}} expected-note{{Incorrect decrement of the reference count of an object that is not owned at this point by the caller}}
165}
166
167void testBoxedString(const char *str) {
168  id result = @(str); // expected-note{{NSString boxed expression produces an object with a +0 retain count}}
169  [result release]; // expected-warning{{decrement}} expected-note{{Incorrect decrement of the reference count of an object that is not owned at this point by the caller}}
170}
171
172void testArray(id obj) {
173  id result = @[obj]; // expected-note{{NSArray literal is an object with a +0 retain count}}
174  [result release]; // expected-warning{{decrement}} expected-note{{Incorrect decrement of the reference count of an object that is not owned at this point by the caller}}
175}
176
177void testDictionary(id key, id value) {
178  id result = @{key: value}; // expected-note{{NSDictionary literal is an object with a +0 retain count}}
179  [result release]; // expected-warning{{decrement}} expected-note{{Incorrect decrement of the reference count of an object that is not owned at this point by the caller}}
180}
181