2012-04-06 22:12:22 +04:00
// RUN: %clang_cc1 -fsyntax-only -fobjc-fragile-abi -verify -Wno-objc-root-class %s
2010-02-19 18:18:45 +03:00
@interface I1
2010-04-22 02:36:40 +04:00
- (int*)method;
2010-02-19 18:18:45 +03:00
@end
@implementation I1
2010-04-22 02:36:40 +04:00
- (int*)method {
2010-02-19 18:18:45 +03:00
struct x { };
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@102021 91177308-0d34-0410-b5e6-96231b3b80d8
2010-04-21 23:57:20 +04:00
[x method]; // expected-error{{receiver type 'x' is not an Objective-C class}}
2010-04-22 02:36:40 +04:00
return 0;
2010-02-19 18:18:45 +03:00
}
@end
2010-02-19 19:08:35 +03:00
typedef struct { int x; } ivar;
@interface I2 {
id ivar;
}
2010-04-22 02:36:40 +04:00
- (int*)method;
2010-02-19 19:08:35 +03:00
+ (void)method;
@end
2010-04-22 02:36:40 +04:00
struct I2_holder {
I2_holder();
I2 *get();
};
I2 *operator+(I2_holder, int);
2010-02-19 19:08:35 +03:00
@implementation I2
2010-04-22 02:36:40 +04:00
- (int*)method {
2010-02-19 19:08:35 +03:00
[ivar method];
2010-04-22 02:36:40 +04:00
// Test instance messages that start with a simple-type-specifier.
[I2_holder().get() method];
[I2_holder().get() + 17 method];
return 0;
2010-02-19 19:08:35 +03:00
}
+ (void)method {
2010-08-28 14:40:52 +04:00
[ivar method]; // expected-error{{receiver type 'ivar' is not an Objective-C class}}
2010-02-19 19:08:35 +03:00
}
@end
2010-04-22 02:36:40 +04:00
// Class message sends
@interface I3
+ (int*)method;
@end
@interface I4 : I3
+ (int*)otherMethod;
@end
template<typename T>
struct identity {
typedef T type;
};
@implementation I4
+ (int *)otherMethod {
// Test class messages that use non-trivial simple-type-specifiers
// or typename-specifiers.
if (false) {
if (true)
2010-06-17 02:31:08 +04:00
return [typename identity<I3>::type method]; // expected-warning{{occurs outside of a template}}
2010-04-22 02:36:40 +04:00
return [::I3 method];
}
int* ip1 = {[super method]};
int* ip2 = {[::I3 method]};
2010-06-17 02:31:08 +04:00
int* ip3 = {[typename identity<I3>::type method]}; // expected-warning{{occurs outside of a template}}
int* ip4 = {[typename identity<I2_holder>::type().get() method]}; // expected-warning{{occurs outside of a template}}
2010-04-22 02:36:40 +04:00
int array[5] = {[3] = 2};
return [super method];
}
@end
2010-04-22 03:24:10 +04:00
struct String {
String(const char *);
};
struct MutableString : public String { };
// C++-specific parameter types
@interface I5
2010-04-22 04:20:18 +04:00
- method:(const String&)str1
other:(String&)str2; // expected-note{{passing argument to parameter 'str2' here}}
2010-04-22 03:24:10 +04:00
@end
void test_I5(I5 *i5, String s) {
[i5 method:"hello" other:s];
2010-09-05 04:04:01 +04:00
[i5 method:s other:"world"]; // expected-error{{non-const lvalue reference to type 'String' cannot bind to a value of unrelated type 'const char [6]'}}
2010-04-22 03:24:10 +04:00
}
2010-09-28 21:48:56 +04:00
// <rdar://problem/8483253>
@interface A
struct X { };
+ (A *)create:(void (*)(void *x, X r, void *data))callback
callbackData:(void *)callback_data;
@end
void foo(void)
{
void *fun;
void *ptr;
X r;
A *im = [A create:(void (*)(void *cgl_ctx, X r, void *data)) fun
callbackData:ptr];
}
2011-01-11 06:23:19 +03:00
// <rdar://problem/8807070>
template<typename T> struct X1; // expected-note{{template is declared here}}
@interface B
+ (X1<int>)blah;
+ (X1<float>&)blarg;
@end
void f() {
[B blah]; // expected-error{{implicit instantiation of undefined template 'X1<int>'}}
[B blarg];
}