Home | History | Annotate | Download | only in SemaObjCXX
      1 // RUN: %clang_cc1 -fsyntax-only -verify %s
      2 
      3 #if !__has_feature(objc_instancetype)
      4 # error Missing 'instancetype' feature macro.
      5 #endif
      6 
      7 @interface Root
      8 + (instancetype)alloc;
      9 - (instancetype)init; // expected-note{{overridden method is part of the 'init' method family}}
     10 - (instancetype)self; // expected-note {{explicitly declared 'instancetype'}}
     11 - (Class)class;
     12 
     13 @property (assign) Root *selfProp;
     14 - (instancetype)selfProp;
     15 @end
     16 
     17 @protocol Proto1
     18 @optional
     19 - (instancetype)methodInProto1;
     20 @end
     21 
     22 @protocol Proto2
     23 @optional
     24 - (instancetype)methodInProto2; // expected-note{{overridden method returns an instance of its class type}}
     25 - (instancetype)otherMethodInProto2; // expected-note{{overridden method returns an instance of its class type}}
     26 @end
     27 
     28 @interface Subclass1 : Root // expected-note 4 {{receiver is instance of class declared here}}
     29 - (instancetype)initSubclass1;
     30 - (void)methodOnSubclass1;
     31 + (instancetype)allocSubclass1;
     32 @end
     33 
     34 @interface Subclass2 : Root
     35 - (instancetype)initSubclass2;
     36 - (void)methodOnSubclass2;
     37 @end
     38 
     39 // Sanity check: the basic initialization pattern.
     40 void test_instancetype_alloc_init_simple() {
     41   Root *r1 = [[Root alloc] init];
     42   Subclass1 *sc1 = [[Subclass1 alloc] init];
     43 }
     44 
     45 // Test that message sends to instancetype methods have the right type.
     46 void test_instancetype_narrow_method_search() {
     47   // instancetype on class methods
     48   Subclass1 *sc1 = [[Subclass1 alloc] initSubclass2]; // expected-warning{{'Subclass1' may not respond to 'initSubclass2'}}
     49   Subclass2 *sc2 = [[Subclass2 alloc] initSubclass2]; // okay
     50 
     51   // instancetype on instance methods
     52   [[[Subclass1 alloc] init] methodOnSubclass2]; // expected-warning{{'Subclass1' may not respond to 'methodOnSubclass2'}}
     53   [[[Subclass2 alloc] init] methodOnSubclass2];
     54   
     55   // instancetype on class methods using protocols
     56   typedef Subclass1<Proto1> SC1Proto1;
     57   typedef Subclass1<Proto2> SC1Proto2;
     58   [[SC1Proto1 alloc] methodInProto2]; // expected-warning{{method '-methodInProto2' not found (return type defaults to 'id')}}
     59   [[SC1Proto2 alloc] methodInProto2];
     60 
     61   // instancetype on instance methods
     62   Subclass1<Proto1> *sc1proto1 = 0;
     63   [[sc1proto1 self] methodInProto2]; // expected-warning{{method '-methodInProto2' not found (return type defaults to 'id')}}
     64   Subclass1<Proto2> *sc1proto2 = 0;
     65   [[sc1proto2 self] methodInProto2];
     66 
     67   // Exact type checks
     68   typeof([[Subclass1 alloc] init]) *ptr1 = (Subclass1 **)0;
     69   typeof([[Subclass2 alloc] init]) *ptr2 = (Subclass2 **)0;
     70 
     71   // Message sends to Class.
     72   Subclass1<Proto1> *sc1proto1_2 = [[[sc1proto1 class] alloc] init];
     73 
     74   // Property access
     75   [sc1proto1.self methodInProto2]; // expected-warning{{method '-methodInProto2' not found (return type defaults to 'id')}}
     76   [sc1proto2.self methodInProto2];
     77   [Subclass1.alloc initSubclass2]; // expected-warning{{'Subclass1' may not respond to 'initSubclass2'}}
     78   [Subclass2.alloc initSubclass2];
     79 
     80   [sc1proto1.selfProp methodInProto2]; // expected-warning{{method '-methodInProto2' not found (return type defaults to 'id')}}
     81   [sc1proto2.selfProp methodInProto2];
     82 }
     83 
     84 // Test that message sends to super methods have the right type.
     85 @interface Subsubclass1 : Subclass1
     86 - (instancetype)initSubclass1;
     87 + (instancetype)allocSubclass1;
     88 
     89 - (void)onlyInSubsubclass1;
     90 @end
     91 
     92 @implementation Subsubclass1
     93 - (instancetype)initSubclass1 {
     94   // Check based on method search.
     95   [[super initSubclass1] methodOnSubclass2]; // expected-warning{{'Subsubclass1' may not respond to 'methodOnSubclass2'}}
     96   [super.initSubclass1 methodOnSubclass2]; // expected-warning{{'Subsubclass1' may not respond to 'methodOnSubclass2'}}
     97 
     98   self = [super init]; // common pattern
     99 
    100   // Exact type check.
    101   typeof([super initSubclass1]) *ptr1 = (Subsubclass1**)0;
    102 
    103   return self;
    104 }
    105 
    106 + (instancetype)allocSubclass1 {
    107   // Check based on method search.
    108   [[super allocSubclass1] methodOnSubclass2]; // expected-warning{{'Subsubclass1' may not respond to 'methodOnSubclass2'}}
    109 
    110   // The ASTs don't model super property accesses well enough to get this right
    111   [super.allocSubclass1 methodOnSubclass2]; // expected-warning{{'Subsubclass1' may not respond to 'methodOnSubclass2'}}
    112 
    113   // Exact type check.
    114   typeof([super allocSubclass1]) *ptr1 = (Subsubclass1**)0;
    115   
    116   return [super allocSubclass1];
    117 }
    118 
    119 - (void)onlyInSubsubclass1 {}
    120 @end
    121 
    122 // Check compatibility rules for inheritance of related return types.
    123 @class Subclass4;
    124 
    125 @interface Subclass3 <Proto1, Proto2>
    126 - (Subclass3 *)methodInProto1;
    127 - (Subclass4 *)methodInProto2; // expected-warning{{method is expected to return an instance of its class type 'Subclass3', but is declared to return 'Subclass4 *'}}
    128 @end
    129 
    130 @interface Subclass4 : Root
    131 + (Subclass4 *)alloc; // okay
    132 - (Subclass3 *)init; // expected-warning{{method is expected to return an instance of its class type 'Subclass4', but is declared to return 'Subclass3 *'}}
    133 - (id)self; // expected-note{{overridden method is part of the 'self' method family}}
    134 - (instancetype)initOther;
    135 @end
    136 
    137 @protocol Proto3 <Proto1, Proto2>
    138 @optional
    139 - (id)methodInProto1;
    140 - (Subclass1 *)methodInProto2;
    141 - (int)otherMethodInProto2; // expected-warning{{protocol method is expected to return an instance of the implementing class, but is declared to return 'int'}}
    142 @end
    143 
    144 @implementation Subclass4
    145 + (id)alloc {
    146   return self; // FIXME: we accept this in ObjC++ but not ObjC?
    147 }
    148 
    149 - (Subclass3 *)init { return 0; } // don't complain: we lost the related return type
    150 
    151 - (Subclass3 *)self { return 0; } // expected-warning{{method is expected to return an instance of its class type 'Subclass4', but is declared to return 'Subclass3 *'}}
    152 
    153 - (Subclass4 *)initOther { return 0; }
    154 
    155 @end
    156 
    157 // Check that inherited related return types influence the types of
    158 // message sends.
    159 void test_instancetype_inherited() {
    160   [[Subclass4 alloc] initSubclass1]; // expected-warning{{'Subclass4' may not respond to 'initSubclass1'}}
    161   [[Subclass4 alloc] initOther];
    162 }
    163 
    164 // Check that related return types tighten up the semantics of
    165 // Objective-C method implementations.
    166 @implementation Subclass2
    167 - (instancetype)initSubclass2 { // expected-note {{explicitly declared 'instancetype'}}
    168   Subclass1 *sc1 = [[Subclass1 alloc] init];
    169   return sc1; // expected-error{{cannot initialize return object of type 'Subclass2 *' with an lvalue of type 'Subclass1 *'}}
    170 }
    171 - (void)methodOnSubclass2 {}
    172 - (id)self {
    173   Subclass1 *sc1 = [[Subclass1 alloc] init];
    174   return sc1; // expected-error{{cannot initialize return object of type 'Subclass2 *' with an lvalue of type 'Subclass1 *'}}
    175 }
    176 @end
    177 
    178 @interface MyClass : Root
    179 + (int)myClassMethod;
    180 @end
    181 
    182 @implementation MyClass
    183 + (int)myClassMethod { return 0; }
    184 
    185 - (void)blah {
    186   int i = [[MyClass self] myClassMethod];
    187 }
    188 
    189 @end
    190 
    191 // rdar://12493140
    192 @protocol P4
    193 - (instancetype) foo; // expected-note {{current method is explicitly declared 'instancetype' and is expected to return an instance of its class type}}
    194 @end
    195 @interface A4 : Root <P4>
    196 - (instancetype) bar; // expected-note {{current method is explicitly declared 'instancetype' and is expected to return an instance of its class type}}
    197 - (instancetype) baz; // expected-note {{overridden method returns an instance of its class type}} expected-note {{previous definition is here}}
    198 @end
    199 @interface B4 : Root @end
    200 
    201 @implementation A4 {
    202   B4 *_b;
    203 }
    204 - (id) foo {
    205   return _b; // expected-error {{cannot initialize return object of type 'A4 *' with an lvalue of type 'B4 *'}}
    206 }
    207 - (id) bar {
    208   return _b; // expected-error {{cannot initialize return object of type 'A4 *' with an lvalue of type 'B4 *'}}
    209 }
    210 
    211 // This is really just to ensure that we don't crash.
    212 // FIXME: only one diagnostic, please
    213 - (float) baz { // expected-warning {{method is expected to return an instance of its class type 'A4', but is declared to return 'float'}} expected-warning {{conflicting return type in implementation}}
    214   return 0;
    215 }
    216 @end
    217