forked from OSchip/llvm-project
Allow objc @() syntax for enum types.
Previously we would reject it as illegal using a value of enum type and on ObjC++ it was illegal to use an enumerator as well. rdar://11454917 llvm-svn: 156843
This commit is contained in:
parent
da01ba37e0
commit
8e6951d81b
|
@ -1540,6 +1540,8 @@ def err_undeclared_nsstring : Error<
|
||||||
"cannot box a string value because NSString has not been declared">;
|
"cannot box a string value because NSString has not been declared">;
|
||||||
def err_objc_illegal_boxed_expression_type : Error<
|
def err_objc_illegal_boxed_expression_type : Error<
|
||||||
"illegal type %0 used in a boxed expression">;
|
"illegal type %0 used in a boxed expression">;
|
||||||
|
def err_objc_incomplete_boxed_expression_type : Error<
|
||||||
|
"incomplete type %0 used in a boxed expression">;
|
||||||
def err_undeclared_nsarray : Error<
|
def err_undeclared_nsarray : Error<
|
||||||
"NSArray must be available to use Objective-C array literals">;
|
"NSArray must be available to use Objective-C array literals">;
|
||||||
def err_undeclared_nsdictionary : Error<
|
def err_undeclared_nsdictionary : Error<
|
||||||
|
|
|
@ -539,6 +539,17 @@ ExprResult Sema::BuildObjCBoxedExpr(SourceRange SR, Expr *ValueExpr) {
|
||||||
// Look for the appropriate method within NSNumber.
|
// Look for the appropriate method within NSNumber.
|
||||||
BoxingMethod = getNSNumberFactoryMethod(*this, SR.getBegin(), ValueType);
|
BoxingMethod = getNSNumberFactoryMethod(*this, SR.getBegin(), ValueType);
|
||||||
BoxedType = NSNumberPointer;
|
BoxedType = NSNumberPointer;
|
||||||
|
|
||||||
|
} else if (const EnumType *ET = ValueType->getAs<EnumType>()) {
|
||||||
|
if (!ET->getDecl()->isComplete()) {
|
||||||
|
Diag(SR.getBegin(), diag::err_objc_incomplete_boxed_expression_type)
|
||||||
|
<< ValueType << ValueExpr->getSourceRange();
|
||||||
|
return ExprError();
|
||||||
|
}
|
||||||
|
|
||||||
|
BoxingMethod = getNSNumberFactoryMethod(*this, SR.getBegin(),
|
||||||
|
ET->getDecl()->getIntegerType());
|
||||||
|
BoxedType = NSNumberPointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!BoxingMethod) {
|
if (!BoxingMethod) {
|
||||||
|
|
|
@ -1,5 +1,29 @@
|
||||||
// RUN: %clang_cc1 -fsyntax-only -verify -Wattributes %s
|
// RUN: %clang_cc1 -fsyntax-only -verify -Wattributes %s
|
||||||
|
|
||||||
|
typedef long NSInteger;
|
||||||
|
typedef unsigned long NSUInteger;
|
||||||
|
typedef signed char BOOL;
|
||||||
|
|
||||||
|
@interface NSNumber
|
||||||
|
@end
|
||||||
|
@interface NSNumber (NSNumberCreation)
|
||||||
|
+ (NSNumber *)numberWithChar:(char)value;
|
||||||
|
+ (NSNumber *)numberWithUnsignedChar:(unsigned char)value;
|
||||||
|
+ (NSNumber *)numberWithShort:(short)value;
|
||||||
|
+ (NSNumber *)numberWithUnsignedShort:(unsigned short)value;
|
||||||
|
+ (NSNumber *)numberWithInt:(int)value;
|
||||||
|
+ (NSNumber *)numberWithUnsignedInt:(unsigned int)value;
|
||||||
|
+ (NSNumber *)numberWithLong:(long)value;
|
||||||
|
+ (NSNumber *)numberWithUnsignedLong:(unsigned long)value;
|
||||||
|
+ (NSNumber *)numberWithLongLong:(long long)value;
|
||||||
|
+ (NSNumber *)numberWithUnsignedLongLong:(unsigned long long)value;
|
||||||
|
+ (NSNumber *)numberWithFloat:(float)value;
|
||||||
|
+ (NSNumber *)numberWithDouble:(double)value;
|
||||||
|
+ (NSNumber *)numberWithBool:(BOOL)value;
|
||||||
|
+ (NSNumber *)numberWithInteger:(NSInteger)value;
|
||||||
|
+ (NSNumber *)numberWithUnsignedInteger:(NSUInteger)value;
|
||||||
|
@end
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
int x, y, z;
|
int x, y, z;
|
||||||
} point;
|
} point;
|
||||||
|
@ -19,3 +43,16 @@ void testPointers() {
|
||||||
void testInvalid() {
|
void testInvalid() {
|
||||||
@(not_defined); // expected-error {{use of undeclared identifier 'not_defined'}}
|
@(not_defined); // expected-error {{use of undeclared identifier 'not_defined'}}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum MyEnum {
|
||||||
|
ME_foo
|
||||||
|
};
|
||||||
|
|
||||||
|
enum ForwE;
|
||||||
|
|
||||||
|
void testEnum(void *p) {
|
||||||
|
enum MyEnum myen;
|
||||||
|
id box = @(myen);
|
||||||
|
box = @(ME_foo);
|
||||||
|
box = @(*(enum ForwE*)p); // expected-error {{incomplete type 'enum ForwE' used in a boxed expression}}
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,58 @@
|
||||||
|
// RUN: %clang_cc1 -fsyntax-only -verify -Wattributes %s
|
||||||
|
|
||||||
|
typedef long NSInteger;
|
||||||
|
typedef unsigned long NSUInteger;
|
||||||
|
typedef signed char BOOL;
|
||||||
|
|
||||||
|
@interface NSNumber
|
||||||
|
@end
|
||||||
|
@interface NSNumber (NSNumberCreation)
|
||||||
|
+ (NSNumber *)numberWithChar:(char)value;
|
||||||
|
+ (NSNumber *)numberWithUnsignedChar:(unsigned char)value;
|
||||||
|
+ (NSNumber *)numberWithShort:(short)value;
|
||||||
|
+ (NSNumber *)numberWithUnsignedShort:(unsigned short)value;
|
||||||
|
+ (NSNumber *)numberWithInt:(int)value;
|
||||||
|
+ (NSNumber *)numberWithUnsignedInt:(unsigned int)value;
|
||||||
|
+ (NSNumber *)numberWithLong:(long)value;
|
||||||
|
+ (NSNumber *)numberWithUnsignedLong:(unsigned long)value;
|
||||||
|
+ (NSNumber *)numberWithLongLong:(long long)value;
|
||||||
|
+ (NSNumber *)numberWithUnsignedLongLong:(unsigned long long)value;
|
||||||
|
+ (NSNumber *)numberWithFloat:(float)value;
|
||||||
|
+ (NSNumber *)numberWithDouble:(double)value;
|
||||||
|
+ (NSNumber *)numberWithBool:(BOOL)value;
|
||||||
|
+ (NSNumber *)numberWithInteger:(NSInteger)value;
|
||||||
|
+ (NSNumber *)numberWithUnsignedInteger:(NSUInteger)value;
|
||||||
|
@end
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int x, y, z;
|
||||||
|
} point;
|
||||||
|
|
||||||
|
void testStruct() {
|
||||||
|
point p = { 0, 0, 0 };
|
||||||
|
id boxed = @(p); // expected-error {{illegal type 'point' used in a boxed expression}}
|
||||||
|
}
|
||||||
|
|
||||||
|
void testPointers() {
|
||||||
|
void *null = 0;
|
||||||
|
id boxed_null = @(null); // expected-error {{illegal type 'void *' used in a boxed expression}}
|
||||||
|
int numbers[] = { 0, 1, 2 };
|
||||||
|
id boxed_numbers = @(numbers); // expected-error {{illegal type 'int *' used in a boxed expression}}
|
||||||
|
}
|
||||||
|
|
||||||
|
void testInvalid() {
|
||||||
|
@(not_defined); // expected-error {{use of undeclared identifier 'not_defined'}}
|
||||||
|
}
|
||||||
|
|
||||||
|
enum MyEnum {
|
||||||
|
ME_foo
|
||||||
|
};
|
||||||
|
|
||||||
|
enum ForwE; // expected-error {{ISO C++ forbids forward references to 'enum' types}}
|
||||||
|
|
||||||
|
void testEnum(void *p) {
|
||||||
|
enum MyEnum myen;
|
||||||
|
id box = @(myen);
|
||||||
|
box = @(ME_foo);
|
||||||
|
box = @(*(enum ForwE*)p); // expected-error {{incomplete type 'enum ForwE' used in a boxed expression}}
|
||||||
|
}
|
Loading…
Reference in New Issue