| // RUN: %clang_cc1 -analyze -analyzer-checker=core,osx.cocoa.RetainCount -analyzer-config ipa=dynamic-bifurcate -verify %s |
| |
| typedef signed char BOOL; |
| typedef struct objc_class *Class; |
| typedef struct objc_object { |
| Class isa; |
| } *id; |
| @protocol NSObject - (BOOL)isEqual:(id)object; @end |
| @interface NSObject <NSObject> {} |
| +(id)alloc; |
| +(id)new; |
| - (oneway void)release; |
| -(id)init; |
| -(id)autorelease; |
| -(id)copy; |
| - (Class)class; |
| -(id)retain; |
| - (oneway void)release; |
| @end |
| |
| @interface SelfStaysLive : NSObject |
| - (id)init; |
| @end |
| |
| @implementation SelfStaysLive |
| - (id)init { |
| return [super init]; |
| } |
| @end |
| |
| void selfStaysLive() { |
| SelfStaysLive *foo = [[SelfStaysLive alloc] init]; |
| [foo release]; |
| } |
| |
| // Test that retain release checker warns on leaks and use-after-frees when |
| // self init is not enabled. |
| // radar://12115830 |
| @interface ParentOfCell : NSObject |
| - (id)initWithInt: (int)inInt; |
| @end |
| @interface Cell : ParentOfCell{ |
| int x; |
| } |
| - (id)initWithInt: (int)inInt; |
| + (void)testOverRelease; |
| + (void)testLeak; |
| @property int x; |
| @end |
| @implementation Cell |
| @synthesize x; |
| - (id) initWithInt: (int)inInt { |
| [super initWithInt: inInt]; |
| self.x = inInt; // no-warning |
| return self; // Self Init checker would produce a warning here. |
| } |
| + (void) testOverRelease { |
| Cell *sharedCell3 = [[Cell alloc] initWithInt: 3]; |
| [sharedCell3 release]; |
| [sharedCell3 release]; // expected-warning {{Reference-counted object is used after it is released}} |
| } |
| + (void) testLeak { |
| Cell *sharedCell4 = [[Cell alloc] initWithInt: 3]; // expected-warning {{leak}} |
| } |
| @end |
| |
| // We should stop tracking some objects even when we inline the call. |
| // Specialically, the objects passed into calls with delegate and callback |
| // parameters. |
| @class DelegateTest; |
| typedef void (*ReleaseCallbackTy) (DelegateTest *c); |
| |
| @interface Delegate : NSObject |
| @end |
| |
| @interface DelegateTest : NSObject { |
| Delegate *myDel; |
| } |
| // Object initialized with a delagate which could potentially release it. |
| - (id)initWithDelegate: (id) d; |
| |
| - (void) setDelegate: (id) d; |
| |
| // Releases object through callback. |
| + (void)updateObject:(DelegateTest*)obj WithCallback:(ReleaseCallbackTy)rc; |
| |
| + (void)test: (Delegate *)d; |
| |
| @property (assign) Delegate* myDel; |
| @end |
| |
| void releaseObj(DelegateTest *c); |
| |
| // Releases object through callback. |
| void updateObject(DelegateTest *c, ReleaseCallbackTy rel) { |
| rel(c); |
| } |
| |
| @implementation DelegateTest |
| @synthesize myDel; |
| |
| - (id) initWithDelegate: (id) d { |
| if ((self = [super init])) |
| myDel = d; |
| return self; |
| } |
| |
| - (void) setDelegate: (id) d { |
| myDel = d; |
| } |
| |
| + (void)updateObject:(DelegateTest*)obj WithCallback:(ReleaseCallbackTy)rc { |
| rc(obj); |
| } |
| |
| + (void) test: (Delegate *)d { |
| DelegateTest *obj1 = [[DelegateTest alloc] initWithDelegate: d]; // no-warning |
| DelegateTest *obj2 = [[DelegateTest alloc] init]; // no-warning |
| DelegateTest *obj3 = [[DelegateTest alloc] init]; // no-warning |
| updateObject(obj2, releaseObj); |
| [DelegateTest updateObject: obj3 |
| WithCallback: releaseObj]; |
| DelegateTest *obj4 = [[DelegateTest alloc] init]; // no-warning |
| [obj4 setDelegate: d]; |
| } |
| @end |
| |