llvm/clang/test/Analysis/inlining/retain-count-self-init.m

// RUN: %clang_analyze_cc1 -analyzer-checker=core,osx.cocoa.RetainCount,osx.cocoa.SelfInit -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;
@end

// We do not want to overhelm user with error messages in case they forgot to 
// assign to self and check that the result of [super init] is non-nil. So 
// stop tracking the receiver of init with respect to Retain Release checker.  
@interface ParentOfCell : NSObject
- (id)initWithInt: (int)inInt;
@end
@interface Cell : ParentOfCell{
  int x;
}
- (id)init;
+ (void)test;
@property int x;
@end
@implementation Cell
@synthesize x;
- (id) init {
  [super init];
  self.x = 3; // no-warning 
  return self; // expected-warning {{Returning 'self' while it is not set to the result of '[(super or self)}} 
}
- (id) initWithInt: (int)inInt {
  [super initWithInt: inInt];
  self.x = inInt; // no-warning 
  return self; // expected-warning {{Returning 'self' while it is not set to the result of '[(super or self)}} 
}
- (id) init2 {
  [self init]; // The call [self init] is inlined. We will warn inside the inlined body.
  self.x = 2; // no-warning 
  return self; 
}

- (id) initWithIntGood: (int)inInt {
    if (self = [super initWithInt: inInt]) {
      self.x = inInt; 
    }
    return self; 
}
+ (void) test {
  Cell *sharedCell1 = [[Cell alloc] init];
  [sharedCell1 release];
  Cell *sharedCell2 = [[Cell alloc] initWithInt: 3];
  [sharedCell2 release];
  Cell *sharedCell3 = [[Cell alloc] initWithIntGood: 3];
  [sharedCell3 release];
}

@end