Index: lib/AST/ExprConstant.cpp =================================================================== --- lib/AST/ExprConstant.cpp +++ lib/AST/ExprConstant.cpp @@ -6185,7 +6185,8 @@ /// EvaluateBuiltinClassifyType - Evaluate __builtin_classify_type the same way /// as GCC. -static int EvaluateBuiltinClassifyType(const CallExpr *E) { +static int EvaluateBuiltinClassifyType(const CallExpr *E, + const LangOptions &LangOpts) { // The following enum mimics the values returned by GCC. // FIXME: Does GCC differ between lvalue and rvalue references here? enum gcc_type_class { @@ -6209,11 +6210,11 @@ if (ArgTy->isVoidType()) return void_type_class; else if (ArgTy->isEnumeralType()) - return enumeral_type_class; + return LangOpts.CPlusPlus ? enumeral_type_class : integer_type_class; else if (ArgTy->isBooleanType()) return boolean_type_class; else if (ArgTy->isCharType()) - return string_type_class; // gcc doesn't appear to use char_type_class + return integer_type_class; // gcc doesn't appear to use char_type_class else if (ArgTy->isIntegerType()) return integer_type_class; else if (ArgTy->isPointerType()) @@ -6225,16 +6226,18 @@ else if (ArgTy->isComplexType()) return complex_type_class; else if (ArgTy->isFunctionType()) - return function_type_class; + return LangOpts.CPlusPlus ? function_type_class : pointer_type_class; + else if (ArgTy->isMemberFunctionPointerType()) + return method_type_class; else if (ArgTy->isStructureOrClassType()) return record_type_class; else if (ArgTy->isUnionType()) return union_type_class; else if (ArgTy->isArrayType()) - return array_type_class; + return LangOpts.CPlusPlus ? array_type_class : pointer_type_class; else if (ArgTy->isUnionType()) return union_type_class; - else // FIXME: offset_type_class, method_type_class, & lang_type_class? + else // FIXME: offset_type_class & lang_type_class? llvm_unreachable("CallExpr::isBuiltinClassifyType(): unimplemented type"); } @@ -6607,7 +6610,7 @@ } case Builtin::BI__builtin_classify_type: - return Success(EvaluateBuiltinClassifyType(E), E); + return Success(EvaluateBuiltinClassifyType(E, Info.getLangOpts()), E); // FIXME: BI__builtin_clrsb // FIXME: BI__builtin_clrsbl Index: test/Sema/builtin-classify-type.c =================================================================== --- test/Sema/builtin-classify-type.c +++ test/Sema/builtin-classify-type.c @@ -0,0 +1,42 @@ +// RUN: %clang_cc1 -fsyntax-only -verify %s + +// expected-no-diagnostics + +enum gcc_type_class { + no_type_class = -1, + void_type_class, integer_type_class, char_type_class, + enumeral_type_class, boolean_type_class, + pointer_type_class, reference_type_class, offset_type_class, + real_type_class, complex_type_class, + function_type_class, method_type_class, + record_type_class, union_type_class, + array_type_class, string_type_class, + lang_type_class +}; + +void foo() { + int i; + char c; + enum { red, green, blue } enum_obj; + int *p; + double d; + _Complex double cc; + extern void f(); + struct { int a; float b; } s_obj; + union { int a; float b; } u_obj; + int arr[10]; + + int a1[__builtin_classify_type(f()) == void_type_class ? 1 : -1]; + int a2[__builtin_classify_type(i) == integer_type_class ? 1 : -1]; + int a3[__builtin_classify_type(c) == integer_type_class ? 1 : -1]; + int a4[__builtin_classify_type(enum_obj) == integer_type_class ? 1 : -1]; + int a5[__builtin_classify_type(p) == pointer_type_class ? 1 : -1]; + int a6[__builtin_classify_type(d) == real_type_class ? 1 : -1]; + int a7[__builtin_classify_type(cc) == complex_type_class ? 1 : -1]; + int a8[__builtin_classify_type(f) == pointer_type_class ? 1 : -1]; + int a0[__builtin_classify_type(s_obj) == record_type_class ? 1 : -1]; + int a10[__builtin_classify_type(u_obj) == union_type_class ? 1 : -1]; + int a11[__builtin_classify_type(arr) == pointer_type_class ? 1 : -1]; + int a12[__builtin_classify_type("abc") == pointer_type_class ? 1 : -1]; +} + Index: test/SemaCXX/builtin-classify-type.cpp =================================================================== --- test/SemaCXX/builtin-classify-type.cpp +++ test/SemaCXX/builtin-classify-type.cpp @@ -0,0 +1,52 @@ +// RUN: %clang_cc1 -fsyntax-only -verify %s + +// expected-no-diagnostics + +enum gcc_type_class { + no_type_class = -1, + void_type_class, integer_type_class, char_type_class, + enumeral_type_class, boolean_type_class, + pointer_type_class, reference_type_class, offset_type_class, + real_type_class, complex_type_class, + function_type_class, method_type_class, + record_type_class, union_type_class, + array_type_class, string_type_class, + lang_type_class +}; + +class cl { +public: + void bar() {} +}; + +int builtin_result; + +void foo() { + int i; + char c; + enum { red, green, blue} enum_obj; + bool b; + int *p; + int &r = i; + double d; + extern void f(); + cl cl_obj; + union { int a; float b; } u_obj; + int arr[10]; + + int a1[__builtin_classify_type(f()) == void_type_class ? 1 : -1]; + int a2[__builtin_classify_type(i) == integer_type_class ? 1 : -1]; + int a3[__builtin_classify_type(c) == integer_type_class ? 1 : -1]; + int a4[__builtin_classify_type(enum_obj) == enumeral_type_class ? 1 : -1]; + int a5[__builtin_classify_type(b) == boolean_type_class ? 1 : -1]; + int a6[__builtin_classify_type(p) == pointer_type_class ? 1 : -1]; + int a7[__builtin_classify_type(r) == integer_type_class ? 1 : -1]; + int a8[__builtin_classify_type(d) == real_type_class ? 1 : -1]; + int a9[__builtin_classify_type(f) == function_type_class ? 1 : -1]; + int a10[__builtin_classify_type(&cl::bar) == method_type_class ? 1 : -1]; + int a11[__builtin_classify_type(cl_obj) == record_type_class ? 1 : -1]; + int a12[__builtin_classify_type(u_obj) == union_type_class ? 1 : -1]; + int a13[__builtin_classify_type(arr) == array_type_class ? 1 : -1]; + int a14[__builtin_classify_type("abc") == array_type_class ? 1 : -1]; +} +