Implement __attribute__((objc_direct)), __attribute__((objc_direct_members))
__attribute__((objc_direct)) is an attribute on methods declaration, and
__attribute__((objc_direct_members)) on implementation, categories or
extensions.
A `direct` property specifier is added (@property(direct) type name)
These attributes / specifiers cause the method to have no associated
Objective-C metadata (for the property or the method itself), and the
calling convention to be a direct C function call.
The symbol for the method has enforced hidden visibility and such direct
calls are hence unreachable cross image. An explicit C function must be
made if so desired to wrap them.
The implicit `self` and `_cmd` arguments are preserved, however to
maintain compatibility with the usual `objc_msgSend` semantics,
3 fundamental precautions are taken:
1) for instance methods, `self` is nil-checked. On arm64 backends this
typically adds a single instruction (cbz x0, <closest-ret>) to the
codegen, for the vast majority of the cases when the return type is a
scalar.
2) for class methods, because the class may not be realized/initialized
yet, a call to `[self self]` is emitted. When the proper deployment
target is used, this is optimized to `objc_opt_self(self)`.
However, long term we might want to emit something better that the
optimizer can reason about. When inlining kicks in, these calls
aren't optimized away as the optimizer has no idea that a single call
is really necessary.
3) the calling convention for the `_cmd` argument is changed: the caller
leaves the second argument to the call undefined, and the selector is
loaded inside the body when it's referenced only.
As far as error reporting goes, the compiler refuses:
- making any overloads direct,
- making an overload of a direct method,
- implementations marked as direct when the declaration in the
interface isn't (the other way around is allowed, as the direct
attribute is inherited from the declaration),
- marking methods required for protocol conformance as direct,
- messaging an unqualified `id` with a direct method,
- forming any @selector() expression with only direct selectors.
As warnings:
- any inconsistency of direct-related calling convention when
@selector() or messaging is used,
- forming any @selector() expression with a possibly direct selector.
Lastly an `objc_direct_members` attribute is added that can decorate
`@implementation` blocks and causes methods only declared there (and in
no `@interface`) to be automatically direct. When decorating an
`@interface` then all methods and properties declared in this block are
marked direct.
Radar-ID: rdar://problem/2684889
Differential Revision: https://reviews.llvm.org/D69991
Reviewed-By: John McCall
2019-11-08 15:14:58 +08:00
|
|
|
// RUN: %clang_cc1 -fsyntax-only -verify -Wselector-type-mismatch %s
|
|
|
|
|
|
|
|
@protocol Proto
|
|
|
|
- (void)protoMethod; // expected-note {{previous declaration is here}}
|
|
|
|
+ (void)classProtoMethod; // expected-note {{previous declaration is here}}
|
|
|
|
@end
|
|
|
|
|
|
|
|
@protocol ProtoDirectFail
|
|
|
|
- (void)protoMethod __attribute__((objc_direct)); // expected-error {{'objc_direct' attribute cannot be applied to methods declared in an Objective-C protocol}}
|
|
|
|
+ (void)classProtoMethod __attribute__((objc_direct)); // expected-error {{'objc_direct' attribute cannot be applied to methods declared in an Objective-C protocol}}
|
|
|
|
@end
|
|
|
|
|
|
|
|
__attribute__((objc_root_class))
|
|
|
|
@interface Root
|
|
|
|
- (void)rootRegular; // expected-note {{previous declaration is here}}
|
|
|
|
+ (void)classRootRegular; // expected-note {{previous declaration is here}}
|
|
|
|
- (void)rootDirect __attribute__((objc_direct)); // expected-note {{previous declaration is here}};
|
|
|
|
+ (void)classRootDirect __attribute__((objc_direct)); // expected-note {{previous declaration is here}};
|
|
|
|
- (void)otherRootDirect __attribute__((objc_direct)); // expected-note {{direct method 'otherRootDirect' declared here}}
|
|
|
|
+ (void)otherClassRootDirect __attribute__((objc_direct)); // expected-note {{direct method 'otherClassRootDirect' declared here}}
|
2020-01-31 08:48:11 +08:00
|
|
|
+ (void)otherOtherClassRootDirect __attribute__((objc_direct)); // expected-note {{direct method 'otherOtherClassRootDirect' declared here}}
|
Implement __attribute__((objc_direct)), __attribute__((objc_direct_members))
__attribute__((objc_direct)) is an attribute on methods declaration, and
__attribute__((objc_direct_members)) on implementation, categories or
extensions.
A `direct` property specifier is added (@property(direct) type name)
These attributes / specifiers cause the method to have no associated
Objective-C metadata (for the property or the method itself), and the
calling convention to be a direct C function call.
The symbol for the method has enforced hidden visibility and such direct
calls are hence unreachable cross image. An explicit C function must be
made if so desired to wrap them.
The implicit `self` and `_cmd` arguments are preserved, however to
maintain compatibility with the usual `objc_msgSend` semantics,
3 fundamental precautions are taken:
1) for instance methods, `self` is nil-checked. On arm64 backends this
typically adds a single instruction (cbz x0, <closest-ret>) to the
codegen, for the vast majority of the cases when the return type is a
scalar.
2) for class methods, because the class may not be realized/initialized
yet, a call to `[self self]` is emitted. When the proper deployment
target is used, this is optimized to `objc_opt_self(self)`.
However, long term we might want to emit something better that the
optimizer can reason about. When inlining kicks in, these calls
aren't optimized away as the optimizer has no idea that a single call
is really necessary.
3) the calling convention for the `_cmd` argument is changed: the caller
leaves the second argument to the call undefined, and the selector is
loaded inside the body when it's referenced only.
As far as error reporting goes, the compiler refuses:
- making any overloads direct,
- making an overload of a direct method,
- implementations marked as direct when the declaration in the
interface isn't (the other way around is allowed, as the direct
attribute is inherited from the declaration),
- marking methods required for protocol conformance as direct,
- messaging an unqualified `id` with a direct method,
- forming any @selector() expression with only direct selectors.
As warnings:
- any inconsistency of direct-related calling convention when
@selector() or messaging is used,
- forming any @selector() expression with a possibly direct selector.
Lastly an `objc_direct_members` attribute is added that can decorate
`@implementation` blocks and causes methods only declared there (and in
no `@interface`) to be automatically direct. When decorating an
`@interface` then all methods and properties declared in this block are
marked direct.
Radar-ID: rdar://problem/2684889
Differential Revision: https://reviews.llvm.org/D69991
Reviewed-By: John McCall
2019-11-08 15:14:58 +08:00
|
|
|
- (void)notDirectInIface; // expected-note {{previous declaration is here}}
|
|
|
|
+ (void)classNotDirectInIface; // expected-note {{previous declaration is here}}
|
|
|
|
@end
|
|
|
|
|
|
|
|
__attribute__((objc_direct_members))
|
|
|
|
@interface Root ()
|
|
|
|
- (void)rootExtensionDirect; // expected-note {{previous declaration is here}}
|
|
|
|
+ (void)classRootExtensionDirect; // expected-note {{previous declaration is here}}
|
|
|
|
@end
|
|
|
|
|
|
|
|
__attribute__((objc_direct_members))
|
|
|
|
@interface Root(Direct)
|
|
|
|
- (void)rootCategoryDirect; // expected-note {{previous declaration is here}}
|
|
|
|
+ (void)classRootCategoryDirect; // expected-note {{previous declaration is here}}
|
|
|
|
@end
|
|
|
|
|
|
|
|
@interface Root ()
|
|
|
|
- (void)rootExtensionRegular; // expected-note {{previous declaration is here}}
|
|
|
|
+ (void)classRootExtensionRegular; // expected-note {{previous declaration is here}}
|
|
|
|
- (void)rootExtensionDirect2 __attribute__((objc_direct)); // expected-note {{previous declaration is here}}
|
|
|
|
+ (void)classRootExtensionDirect2 __attribute__((objc_direct)); // expected-note {{previous declaration is here}}
|
|
|
|
@end
|
|
|
|
|
|
|
|
@interface Root (Direct2)
|
|
|
|
- (void)rootCategoryRegular; // expected-note {{previous declaration is here}}
|
|
|
|
+ (void)classRootCategoryRegular; // expected-note {{previous declaration is here}}
|
|
|
|
- (void)rootCategoryDirect2 __attribute__((objc_direct)); // expected-note {{previous declaration is here}}
|
|
|
|
+ (void)classRootCategoryDirect2 __attribute__((objc_direct)); // expected-note {{previous declaration is here}}
|
|
|
|
@end
|
|
|
|
|
2020-01-31 08:48:11 +08:00
|
|
|
__attribute__((objc_direct_members))
|
|
|
|
@interface SubDirectMembers : Root
|
|
|
|
@property int foo; // expected-note {{previous declaration is here}}
|
2020-01-31 10:21:25 +08:00
|
|
|
- (instancetype)init;
|
|
|
|
@end
|
|
|
|
|
Implement __attribute__((objc_direct)), __attribute__((objc_direct_members))
__attribute__((objc_direct)) is an attribute on methods declaration, and
__attribute__((objc_direct_members)) on implementation, categories or
extensions.
A `direct` property specifier is added (@property(direct) type name)
These attributes / specifiers cause the method to have no associated
Objective-C metadata (for the property or the method itself), and the
calling convention to be a direct C function call.
The symbol for the method has enforced hidden visibility and such direct
calls are hence unreachable cross image. An explicit C function must be
made if so desired to wrap them.
The implicit `self` and `_cmd` arguments are preserved, however to
maintain compatibility with the usual `objc_msgSend` semantics,
3 fundamental precautions are taken:
1) for instance methods, `self` is nil-checked. On arm64 backends this
typically adds a single instruction (cbz x0, <closest-ret>) to the
codegen, for the vast majority of the cases when the return type is a
scalar.
2) for class methods, because the class may not be realized/initialized
yet, a call to `[self self]` is emitted. When the proper deployment
target is used, this is optimized to `objc_opt_self(self)`.
However, long term we might want to emit something better that the
optimizer can reason about. When inlining kicks in, these calls
aren't optimized away as the optimizer has no idea that a single call
is really necessary.
3) the calling convention for the `_cmd` argument is changed: the caller
leaves the second argument to the call undefined, and the selector is
loaded inside the body when it's referenced only.
As far as error reporting goes, the compiler refuses:
- making any overloads direct,
- making an overload of a direct method,
- implementations marked as direct when the declaration in the
interface isn't (the other way around is allowed, as the direct
attribute is inherited from the declaration),
- marking methods required for protocol conformance as direct,
- messaging an unqualified `id` with a direct method,
- forming any @selector() expression with only direct selectors.
As warnings:
- any inconsistency of direct-related calling convention when
@selector() or messaging is used,
- forming any @selector() expression with a possibly direct selector.
Lastly an `objc_direct_members` attribute is added that can decorate
`@implementation` blocks and causes methods only declared there (and in
no `@interface`) to be automatically direct. When decorating an
`@interface` then all methods and properties declared in this block are
marked direct.
Radar-ID: rdar://problem/2684889
Differential Revision: https://reviews.llvm.org/D69991
Reviewed-By: John McCall
2019-11-08 15:14:58 +08:00
|
|
|
@interface Sub : Root <Proto>
|
|
|
|
/* invalid overrides with directs */
|
|
|
|
- (void)rootRegular __attribute__((objc_direct)); // expected-error {{methods that override superclass methods cannot be direct}}
|
|
|
|
+ (void)classRootRegular __attribute__((objc_direct)); // expected-error {{methods that override superclass methods cannot be direct}}
|
|
|
|
- (void)protoMethod __attribute__((objc_direct)); // expected-error {{methods that implement protocol requirements cannot be direct}}
|
|
|
|
+ (void)classProtoMethod __attribute__((objc_direct)); // expected-error {{methods that implement protocol requirements cannot be direct}}
|
|
|
|
- (void)rootExtensionRegular __attribute__((objc_direct)); // expected-error {{methods that override superclass methods cannot be direct}}
|
|
|
|
+ (void)classRootExtensionRegular __attribute__((objc_direct)); // expected-error {{methods that override superclass methods cannot be direct}}
|
|
|
|
- (void)rootCategoryRegular __attribute__((objc_direct)); // expected-error {{methods that override superclass methods cannot be direct}}
|
|
|
|
+ (void)classRootCategoryRegular __attribute__((objc_direct)); // expected-error {{methods that override superclass methods cannot be direct}}
|
|
|
|
|
|
|
|
/* invalid overrides of directs */
|
|
|
|
- (void)rootDirect; // expected-error {{cannot override a method that is declared direct by a superclass}}
|
|
|
|
+ (void)classRootDirect; // expected-error {{cannot override a method that is declared direct by a superclass}}
|
|
|
|
- (void)rootExtensionDirect; // expected-error {{cannot override a method that is declared direct by a superclass}}
|
|
|
|
+ (void)classRootExtensionDirect; // expected-error {{cannot override a method that is declared direct by a superclass}}
|
|
|
|
- (void)rootExtensionDirect2; // expected-error {{cannot override a method that is declared direct by a superclass}}
|
|
|
|
+ (void)classRootExtensionDirect2; // expected-error {{cannot override a method that is declared direct by a superclass}}
|
|
|
|
- (void)rootCategoryDirect; // expected-error {{cannot override a method that is declared direct by a superclass}}
|
|
|
|
+ (void)classRootCategoryDirect; // expected-error {{cannot override a method that is declared direct by a superclass}}
|
|
|
|
- (void)rootCategoryDirect2; // expected-error {{cannot override a method that is declared direct by a superclass}}
|
|
|
|
+ (void)classRootCategoryDirect2; // expected-error {{cannot override a method that is declared direct by a superclass}}
|
|
|
|
@end
|
|
|
|
|
|
|
|
__attribute__((objc_direct_members))
|
|
|
|
@implementation Root
|
|
|
|
- (void)rootRegular {
|
|
|
|
}
|
|
|
|
+ (void)classRootRegular {
|
|
|
|
}
|
|
|
|
- (void)rootDirect {
|
|
|
|
}
|
|
|
|
+ (void)classRootDirect {
|
|
|
|
}
|
|
|
|
- (void)otherRootDirect {
|
|
|
|
}
|
2020-01-15 11:51:08 +08:00
|
|
|
+ (void)someRootDirectMethod { // expected-note {{direct method 'someRootDirectMethod' declared here}}
|
|
|
|
}
|
Implement __attribute__((objc_direct)), __attribute__((objc_direct_members))
__attribute__((objc_direct)) is an attribute on methods declaration, and
__attribute__((objc_direct_members)) on implementation, categories or
extensions.
A `direct` property specifier is added (@property(direct) type name)
These attributes / specifiers cause the method to have no associated
Objective-C metadata (for the property or the method itself), and the
calling convention to be a direct C function call.
The symbol for the method has enforced hidden visibility and such direct
calls are hence unreachable cross image. An explicit C function must be
made if so desired to wrap them.
The implicit `self` and `_cmd` arguments are preserved, however to
maintain compatibility with the usual `objc_msgSend` semantics,
3 fundamental precautions are taken:
1) for instance methods, `self` is nil-checked. On arm64 backends this
typically adds a single instruction (cbz x0, <closest-ret>) to the
codegen, for the vast majority of the cases when the return type is a
scalar.
2) for class methods, because the class may not be realized/initialized
yet, a call to `[self self]` is emitted. When the proper deployment
target is used, this is optimized to `objc_opt_self(self)`.
However, long term we might want to emit something better that the
optimizer can reason about. When inlining kicks in, these calls
aren't optimized away as the optimizer has no idea that a single call
is really necessary.
3) the calling convention for the `_cmd` argument is changed: the caller
leaves the second argument to the call undefined, and the selector is
loaded inside the body when it's referenced only.
As far as error reporting goes, the compiler refuses:
- making any overloads direct,
- making an overload of a direct method,
- implementations marked as direct when the declaration in the
interface isn't (the other way around is allowed, as the direct
attribute is inherited from the declaration),
- marking methods required for protocol conformance as direct,
- messaging an unqualified `id` with a direct method,
- forming any @selector() expression with only direct selectors.
As warnings:
- any inconsistency of direct-related calling convention when
@selector() or messaging is used,
- forming any @selector() expression with a possibly direct selector.
Lastly an `objc_direct_members` attribute is added that can decorate
`@implementation` blocks and causes methods only declared there (and in
no `@interface`) to be automatically direct. When decorating an
`@interface` then all methods and properties declared in this block are
marked direct.
Radar-ID: rdar://problem/2684889
Differential Revision: https://reviews.llvm.org/D69991
Reviewed-By: John McCall
2019-11-08 15:14:58 +08:00
|
|
|
+ (void)otherClassRootDirect {
|
2020-01-15 11:51:08 +08:00
|
|
|
[self someRootDirectMethod]; // expected-error {{messaging a Class with a method that is possibly direct}}
|
Implement __attribute__((objc_direct)), __attribute__((objc_direct_members))
__attribute__((objc_direct)) is an attribute on methods declaration, and
__attribute__((objc_direct_members)) on implementation, categories or
extensions.
A `direct` property specifier is added (@property(direct) type name)
These attributes / specifiers cause the method to have no associated
Objective-C metadata (for the property or the method itself), and the
calling convention to be a direct C function call.
The symbol for the method has enforced hidden visibility and such direct
calls are hence unreachable cross image. An explicit C function must be
made if so desired to wrap them.
The implicit `self` and `_cmd` arguments are preserved, however to
maintain compatibility with the usual `objc_msgSend` semantics,
3 fundamental precautions are taken:
1) for instance methods, `self` is nil-checked. On arm64 backends this
typically adds a single instruction (cbz x0, <closest-ret>) to the
codegen, for the vast majority of the cases when the return type is a
scalar.
2) for class methods, because the class may not be realized/initialized
yet, a call to `[self self]` is emitted. When the proper deployment
target is used, this is optimized to `objc_opt_self(self)`.
However, long term we might want to emit something better that the
optimizer can reason about. When inlining kicks in, these calls
aren't optimized away as the optimizer has no idea that a single call
is really necessary.
3) the calling convention for the `_cmd` argument is changed: the caller
leaves the second argument to the call undefined, and the selector is
loaded inside the body when it's referenced only.
As far as error reporting goes, the compiler refuses:
- making any overloads direct,
- making an overload of a direct method,
- implementations marked as direct when the declaration in the
interface isn't (the other way around is allowed, as the direct
attribute is inherited from the declaration),
- marking methods required for protocol conformance as direct,
- messaging an unqualified `id` with a direct method,
- forming any @selector() expression with only direct selectors.
As warnings:
- any inconsistency of direct-related calling convention when
@selector() or messaging is used,
- forming any @selector() expression with a possibly direct selector.
Lastly an `objc_direct_members` attribute is added that can decorate
`@implementation` blocks and causes methods only declared there (and in
no `@interface`) to be automatically direct. When decorating an
`@interface` then all methods and properties declared in this block are
marked direct.
Radar-ID: rdar://problem/2684889
Differential Revision: https://reviews.llvm.org/D69991
Reviewed-By: John McCall
2019-11-08 15:14:58 +08:00
|
|
|
}
|
2020-01-31 08:48:11 +08:00
|
|
|
+ (void)otherOtherClassRootDirect {
|
|
|
|
}
|
Implement __attribute__((objc_direct)), __attribute__((objc_direct_members))
__attribute__((objc_direct)) is an attribute on methods declaration, and
__attribute__((objc_direct_members)) on implementation, categories or
extensions.
A `direct` property specifier is added (@property(direct) type name)
These attributes / specifiers cause the method to have no associated
Objective-C metadata (for the property or the method itself), and the
calling convention to be a direct C function call.
The symbol for the method has enforced hidden visibility and such direct
calls are hence unreachable cross image. An explicit C function must be
made if so desired to wrap them.
The implicit `self` and `_cmd` arguments are preserved, however to
maintain compatibility with the usual `objc_msgSend` semantics,
3 fundamental precautions are taken:
1) for instance methods, `self` is nil-checked. On arm64 backends this
typically adds a single instruction (cbz x0, <closest-ret>) to the
codegen, for the vast majority of the cases when the return type is a
scalar.
2) for class methods, because the class may not be realized/initialized
yet, a call to `[self self]` is emitted. When the proper deployment
target is used, this is optimized to `objc_opt_self(self)`.
However, long term we might want to emit something better that the
optimizer can reason about. When inlining kicks in, these calls
aren't optimized away as the optimizer has no idea that a single call
is really necessary.
3) the calling convention for the `_cmd` argument is changed: the caller
leaves the second argument to the call undefined, and the selector is
loaded inside the body when it's referenced only.
As far as error reporting goes, the compiler refuses:
- making any overloads direct,
- making an overload of a direct method,
- implementations marked as direct when the declaration in the
interface isn't (the other way around is allowed, as the direct
attribute is inherited from the declaration),
- marking methods required for protocol conformance as direct,
- messaging an unqualified `id` with a direct method,
- forming any @selector() expression with only direct selectors.
As warnings:
- any inconsistency of direct-related calling convention when
@selector() or messaging is used,
- forming any @selector() expression with a possibly direct selector.
Lastly an `objc_direct_members` attribute is added that can decorate
`@implementation` blocks and causes methods only declared there (and in
no `@interface`) to be automatically direct. When decorating an
`@interface` then all methods and properties declared in this block are
marked direct.
Radar-ID: rdar://problem/2684889
Differential Revision: https://reviews.llvm.org/D69991
Reviewed-By: John McCall
2019-11-08 15:14:58 +08:00
|
|
|
- (void)rootExtensionDirect {
|
|
|
|
}
|
|
|
|
+ (void)classRootExtensionDirect {
|
|
|
|
}
|
|
|
|
- (void)rootExtensionRegular {
|
|
|
|
}
|
|
|
|
+ (void)classRootExtensionRegular {
|
|
|
|
}
|
|
|
|
- (void)rootExtensionDirect2 {
|
|
|
|
}
|
|
|
|
+ (void)classRootExtensionDirect2 {
|
|
|
|
}
|
|
|
|
- (void)notDirectInIface __attribute__((objc_direct)) // expected-error {{direct method implementation was previously declared not direct}}
|
|
|
|
{
|
|
|
|
}
|
|
|
|
+ (void)classNotDirectInIface __attribute__((objc_direct)) // expected-error {{direct method implementation was previously declared not direct}}
|
|
|
|
{
|
|
|
|
}
|
|
|
|
- (void)direct1 { // expected-note {{direct method 'direct1' declared here}}
|
|
|
|
}
|
|
|
|
- (void)direct2 { // expected-note {{direct method 'direct2' declared here}}
|
|
|
|
}
|
|
|
|
@end
|
|
|
|
|
|
|
|
@interface Foo : Root
|
|
|
|
- (id)directMismatch1; // expected-note {{using}}
|
|
|
|
- (id)directMismatch2; // expected-note {{method 'directMismatch2' declared here}}
|
|
|
|
@end
|
|
|
|
|
|
|
|
@interface Bar : Root
|
|
|
|
- (void)directMismatch1 __attribute__((objc_direct)); // expected-note {{also found}}
|
|
|
|
- (void)directMismatch2 __attribute__((objc_direct)); // expected-note {{method 'directMismatch2' declared here}}
|
|
|
|
@end
|
|
|
|
|
|
|
|
@interface ValidSub : Root
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation ValidSub
|
|
|
|
- (void)someValidSubMethod {
|
|
|
|
[super otherRootDirect]; // expected-error {{messaging super with a direct method}}
|
|
|
|
}
|
2020-01-31 08:48:11 +08:00
|
|
|
+ (void)someValidSubMethod {
|
|
|
|
[super otherOtherClassRootDirect]; // expected-error {{messaging super with a direct method}}
|
|
|
|
}
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation SubDirectMembers
|
|
|
|
@dynamic foo; // expected-error {{direct property cannot be @dynamic}}
|
|
|
|
- (instancetype)init {
|
|
|
|
return self;
|
|
|
|
}
|
Implement __attribute__((objc_direct)), __attribute__((objc_direct_members))
__attribute__((objc_direct)) is an attribute on methods declaration, and
__attribute__((objc_direct_members)) on implementation, categories or
extensions.
A `direct` property specifier is added (@property(direct) type name)
These attributes / specifiers cause the method to have no associated
Objective-C metadata (for the property or the method itself), and the
calling convention to be a direct C function call.
The symbol for the method has enforced hidden visibility and such direct
calls are hence unreachable cross image. An explicit C function must be
made if so desired to wrap them.
The implicit `self` and `_cmd` arguments are preserved, however to
maintain compatibility with the usual `objc_msgSend` semantics,
3 fundamental precautions are taken:
1) for instance methods, `self` is nil-checked. On arm64 backends this
typically adds a single instruction (cbz x0, <closest-ret>) to the
codegen, for the vast majority of the cases when the return type is a
scalar.
2) for class methods, because the class may not be realized/initialized
yet, a call to `[self self]` is emitted. When the proper deployment
target is used, this is optimized to `objc_opt_self(self)`.
However, long term we might want to emit something better that the
optimizer can reason about. When inlining kicks in, these calls
aren't optimized away as the optimizer has no idea that a single call
is really necessary.
3) the calling convention for the `_cmd` argument is changed: the caller
leaves the second argument to the call undefined, and the selector is
loaded inside the body when it's referenced only.
As far as error reporting goes, the compiler refuses:
- making any overloads direct,
- making an overload of a direct method,
- implementations marked as direct when the declaration in the
interface isn't (the other way around is allowed, as the direct
attribute is inherited from the declaration),
- marking methods required for protocol conformance as direct,
- messaging an unqualified `id` with a direct method,
- forming any @selector() expression with only direct selectors.
As warnings:
- any inconsistency of direct-related calling convention when
@selector() or messaging is used,
- forming any @selector() expression with a possibly direct selector.
Lastly an `objc_direct_members` attribute is added that can decorate
`@implementation` blocks and causes methods only declared there (and in
no `@interface`) to be automatically direct. When decorating an
`@interface` then all methods and properties declared in this block are
marked direct.
Radar-ID: rdar://problem/2684889
Differential Revision: https://reviews.llvm.org/D69991
Reviewed-By: John McCall
2019-11-08 15:14:58 +08:00
|
|
|
@end
|
|
|
|
|
|
|
|
extern void callMethod(id obj, Class cls);
|
|
|
|
extern void useSel(SEL sel);
|
|
|
|
|
|
|
|
void callMethod(id obj, Class cls) {
|
|
|
|
[Root otherClassRootDirect];
|
|
|
|
[cls otherClassRootDirect]; // expected-error {{messaging a Class with a method that is possibly direct}}
|
|
|
|
[obj direct1]; // expected-error {{messaging unqualified id with a method that is possibly direct}}
|
|
|
|
[(Root *)obj direct1];
|
|
|
|
[obj directMismatch1]; // expected-warning {{multiple methods named 'directMismatch1' found}}
|
|
|
|
useSel(@selector(direct2)); // expected-error {{@selector expression formed with direct selector 'direct2'}}
|
|
|
|
useSel(@selector(directMismatch2)); // expected-warning {{several methods with selector 'directMismatch2' of mismatched types are found for the @selector expression}}
|
|
|
|
}
|