| Ted Kremenek | 28eace6 | 2013-11-23 01:01:34 +0000 | [diff] [blame] | 1 | // RUN: %clang_cc1 -triple x86_64-apple-darwin11 -fsyntax-only -verify %s -Wno-objc-root-class |
| 2 | |
| Ted Kremenek | f41cf7f1 | 2013-12-10 19:43:48 +0000 | [diff] [blame] | 3 | // Mark this protocol as requiring all of its methods and properties |
| 4 | // to be explicitly implemented in the adopting class. |
| 5 | __attribute__((objc_protocol_requires_explicit_implementation)) |
| Ted Kremenek | 28eace6 | 2013-11-23 01:01:34 +0000 | [diff] [blame] | 6 | @protocol Protocol |
| 7 | - (void) theBestOfTimes; // expected-note {{method 'theBestOfTimes' declared here}} |
| Ted Kremenek | f41cf7f1 | 2013-12-10 19:43:48 +0000 | [diff] [blame] | 8 | @property (readonly) id theWorstOfTimes; |
| Ted Kremenek | 28eace6 | 2013-11-23 01:01:34 +0000 | [diff] [blame] | 9 | @end |
| 10 | |
| Ted Kremenek | f41cf7f1 | 2013-12-10 19:43:48 +0000 | [diff] [blame] | 11 | // In this example, ClassA adopts the protocol. We won't |
| 12 | // provide the implementation here, but this protocol will |
| 13 | // be adopted later by a subclass. |
| Ted Kremenek | 28eace6 | 2013-11-23 01:01:34 +0000 | [diff] [blame] | 14 | @interface ClassA <Protocol> |
| 15 | - (void) theBestOfTimes; |
| Ted Kremenek | 28eace6 | 2013-11-23 01:01:34 +0000 | [diff] [blame] | 16 | @property (readonly) id theWorstOfTimes; |
| 17 | @end |
| 18 | |
| Ted Kremenek | f41cf7f1 | 2013-12-10 19:43:48 +0000 | [diff] [blame] | 19 | // This class subclasses ClassA (which adopts 'Protocol'), |
| 20 | // but does not provide the needed implementation. |
| Ted Kremenek | 2ccf19e | 2013-12-13 05:58:51 +0000 | [diff] [blame] | 21 | @interface ClassB : ClassA <Protocol> |
| Ted Kremenek | 28eace6 | 2013-11-23 01:01:34 +0000 | [diff] [blame] | 22 | @end |
| 23 | |
| Ted Kremenek | 2ccf19e | 2013-12-13 05:58:51 +0000 | [diff] [blame] | 24 | @implementation ClassB // expected-warning {{method 'theBestOfTimes' in protocol 'Protocol' not implemented}} |
| Ted Kremenek | 28eace6 | 2013-11-23 01:01:34 +0000 | [diff] [blame] | 25 | @end |
| 26 | |
| Ted Kremenek | c152c52 | 2013-12-12 06:20:42 +0000 | [diff] [blame] | 27 | // Test that inherited protocols do not get the explicit conformance requirement. |
| 28 | @protocol Inherited |
| 29 | - (void) fairIsFoul; |
| 30 | @end |
| 31 | |
| 32 | __attribute__((objc_protocol_requires_explicit_implementation)) |
| 33 | @protocol Derived <Inherited> |
| 34 | - (void) foulIsFair; // expected-note {{method 'foulIsFair' declared here}} |
| 35 | @end |
| 36 | |
| 37 | @interface ClassC <Inherited> |
| 38 | @end |
| 39 | |
| Ted Kremenek | 2ccf19e | 2013-12-13 05:58:51 +0000 | [diff] [blame] | 40 | @interface ClassD : ClassC <Derived> |
| Ted Kremenek | c152c52 | 2013-12-12 06:20:42 +0000 | [diff] [blame] | 41 | @end |
| 42 | |
| Ted Kremenek | 2ccf19e | 2013-12-13 05:58:51 +0000 | [diff] [blame] | 43 | @implementation ClassD // expected-warning {{method 'foulIsFair' in protocol 'Derived' not implemented}} |
| Ted Kremenek | c152c52 | 2013-12-12 06:20:42 +0000 | [diff] [blame] | 44 | @end |
| 45 | |
| Ted Kremenek | f41cf7f1 | 2013-12-10 19:43:48 +0000 | [diff] [blame] | 46 | // Test that the attribute is used correctly. |
| 47 | __attribute__((objc_protocol_requires_explicit_implementation(1+2))) // expected-error {{attribute takes no arguments}} |
| 48 | @protocol AnotherProtocol @end |
| Ted Kremenek | 28eace6 | 2013-11-23 01:01:34 +0000 | [diff] [blame] | 49 | |
| Ted Kremenek | f41cf7f1 | 2013-12-10 19:43:48 +0000 | [diff] [blame] | 50 | // Cannot put the attribute on classes or other non-protocol declarations. |
| 51 | __attribute__((objc_protocol_requires_explicit_implementation)) // expected-error {{attribute only applies to Objective-C protocols}} |
| 52 | @interface AnotherClass @end |
| Ted Kremenek | 28eace6 | 2013-11-23 01:01:34 +0000 | [diff] [blame] | 53 | |
| Ted Kremenek | f41cf7f1 | 2013-12-10 19:43:48 +0000 | [diff] [blame] | 54 | __attribute__((objc_protocol_requires_explicit_implementation)) // expected-error {{attribute only applies to Objective-C protocols}} |
| 55 | int x; |
| Ted Kremenek | 28eace6 | 2013-11-23 01:01:34 +0000 | [diff] [blame] | 56 | |
| Ted Kremenek | 33e430f | 2013-12-13 06:26:14 +0000 | [diff] [blame] | 57 | // Test that inherited protocols with the attribute |
| 58 | // are treated properly. |
| 59 | __attribute__((objc_protocol_requires_explicit_implementation)) |
| 60 | @protocol ProtocolA |
| 61 | @required |
| 62 | - (void)rlyeh; |
| 63 | - (void)innsmouth; |
| 64 | @end |
| 65 | |
| 66 | @protocol ProtocolB <ProtocolA> |
| 67 | @required |
| 68 | - (void)dunwich; |
| 69 | - (id)innsmouth; |
| 70 | @end |
| 71 | |
| 72 | @protocol ProtocolC |
| 73 | @required |
| 74 | - (void)rlyeh; |
| 75 | - (void)innsmouth; |
| 76 | - (void)dunwich; |
| 77 | @end |
| 78 | |
| 79 | @interface MyObject <ProtocolC> |
| 80 | @end |
| 81 | |
| 82 | @interface MyLovecraft <ProtocolA> |
| 83 | @end |
| 84 | |
| 85 | @interface MyShoggoth : MyLovecraft <ProtocolB> |
| 86 | @end |
| 87 | |
| 88 | @implementation MyObject |
| 89 | - (void)innsmouth {} |
| 90 | - (void)rlyeh {} |
| 91 | - (void)dunwich {} |
| 92 | @end |
| 93 | |
| 94 | @implementation MyLovecraft |
| 95 | - (void)innsmouth {} |
| 96 | - (void)rlyeh {} |
| 97 | @end |
| 98 | |
| 99 | @implementation MyShoggoth |
| 100 | - (void)dunwich {} |
| 101 | @end |
| 102 | |