// RUN: %clang_cc1 -fsyntax-only -triple x86_64-linux-pc %s -verify -DBAD_CONVERSION // RUN: %clang_cc1 -fsyntax-only -triple i386-windows-pc %s -verify -DBAD_CONVERSION -DWIN32 // RUN: %clang_cc1 -fsyntax-only -triple x86_64-linux-pc %s -ast-dump | FileCheck %s --check-prefixes=CHECK,LIN64,NODEF // RUN: %clang_cc1 -fsyntax-only -triple i386-windows-pc %s -ast-dump -DWIN32 | FileCheck %s --check-prefixes=CHECK,WIN32,NODEF // RUN: %clang_cc1 -fsyntax-only -triple x86_64-linux-pc -fdefault-calling-conv=vectorcall %s -verify -DBAD_VEC_CONVERS // RUN: %clang_cc1 -fsyntax-only -triple x86_64-linux-pc -fdefault-calling-conv=vectorcall %s -ast-dump | FileCheck %s --check-prefixes=CHECK,VECTDEF void useage() { auto normal = [](int, float, double) {}; // #1 auto vectorcall = [](int, float, double) __attribute__((vectorcall)){}; // #2 #ifdef WIN32 auto thiscall = [](int, float, double) __attribute__((thiscall)){}; // #3 #endif // WIN32 auto cdecl = [](int, float, double) __attribute__((cdecl)){}; auto genericlambda = [](auto a) {}; // #4 auto genericvectorcalllambda = [](auto a) __attribute__((vectorcall)){}; // #5 // None of these should be ambiguous. (void)+normal; (void)+vectorcall; #ifdef WIN32 (void)+thiscall; #endif // WIN32 (void)+cdecl; #ifdef BAD_CONVERSION // expected-error-re@+1 {{invalid argument type {{.*}} to unary expression}} (void)+genericlambda; // expected-error-re@+1 {{invalid argument type {{.*}} to unary expression}} (void)+genericvectorcalllambda; #endif // BAD_CONVERSION // CHECK: VarDecl {{.*}} normal ' // CHECK: LambdaExpr // WIN32: CXXMethodDecl {{.*}} operator() 'void (int, float, double) __attribute__((thiscall)) const' // LIN64: CXXMethodDecl {{.*}} operator() 'void (int, float, double) const' // VECTDEF: CXXMethodDecl {{.*}} operator() 'void (int, float, double) const' // NODEF: CXXConversionDecl {{.*}} operator void (*)(int, float, double) 'void // NODEF: CXXMethodDecl {{.*}} __invoke 'void (int, float, double)' static inline // VECTDEF: CXXConversionDecl {{.*}} operator void (*)(int, float, double) __attribute__((vectorcall)) 'void // VECTDEF: CXXMethodDecl {{.*}} __invoke 'void (int, float, double) __attribute__((vectorcall))' static inline // CHECK: VarDecl {{.*}} vectorcall ' // CHECK: LambdaExpr // CHECK: CXXMethodDecl {{.*}} operator() 'void (int, float, double) __attribute__((vectorcall)) const' // CHECK: CXXConversionDecl {{.*}} operator void (*)(int, float, double) __attribute__((vectorcall)) 'void // CHECK: CXXMethodDecl {{.*}} __invoke 'void (int, float, double) __attribute__((vectorcall))' static inline // WIN32: VarDecl {{.*}} thiscall ' // WIN32: LambdaExpr // WIN32: CXXMethodDecl {{.*}} operator() 'void (int, float, double) __attribute__((thiscall)) const' // WIN32: CXXConversionDecl {{.*}} operator void (*)(int, float, double) 'void // WIN32: CXXMethodDecl {{.*}} __invoke 'void (int, float, double)' static inline // CHECK: VarDecl {{.*}} cdecl ' // CHECK: LambdaExpr // CHECK: CXXMethodDecl {{.*}} operator() 'void (int, float, double) const' // NODEF: CXXConversionDecl {{.*}} operator void (*)(int, float, double) 'void // NODEF: CXXMethodDecl {{.*}} __invoke 'void (int, float, double)' static inline // VECTDEF: CXXConversionDecl {{.*}} operator void (*)(int, float, double) __attribute__((vectorcall)) 'void // VECTDEF: CXXMethodDecl {{.*}} __invoke 'void (int, float, double) __attribute__((vectorcall))' static inline // CHECK: VarDecl {{.*}} genericlambda ' // CHECK: LambdaExpr // // CHECK: FunctionTemplateDecl {{.*}} operator() // LIN64: CXXMethodDecl {{.*}} operator() 'auto (auto) const' inline // LIN64: CXXMethodDecl {{.*}} operator() 'void (char) const' inline // LIN64: CXXMethodDecl {{.*}} operator() 'void (int) const' inline // WIN32: CXXMethodDecl {{.*}} operator() 'auto (auto) __attribute__((thiscall)) const' inline // WIN32: CXXMethodDecl {{.*}} operator() 'void (char) __attribute__((thiscall)) const' inline // WIN32: CXXMethodDecl {{.*}} operator() 'void (int) __attribute__((thiscall)) const' inline // // NODEF: FunctionTemplateDecl {{.*}} operator auto (*)(type-parameter-0-0) // VECDEF: FunctionTemplateDecl {{.*}} operator auto (*)(type-parameter-0-0) __attribute__((vectorcall)) // LIN64: CXXConversionDecl {{.*}} operator auto (*)(type-parameter-0-0) 'auto (*() const noexcept)(auto)' // LIN64: CXXConversionDecl {{.*}} operator auto (*)(char) 'void (*() const noexcept)(char)' // LIN64: CXXConversionDecl {{.*}} operator auto (*)(int) 'void (*() const noexcept)(int)' // WIN32: CXXConversionDecl {{.*}} operator auto (*)(type-parameter-0-0) 'auto (*() __attribute__((thiscall)) const noexcept)(auto)' // WIN32: CXXConversionDecl {{.*}} operator auto (*)(char) 'void (*() __attribute__((thiscall)) const noexcept)(char)' // WIN32: CXXConversionDecl {{.*}} operator auto (*)(int) 'void (*() __attribute__((thiscall)) const noexcept)(int)' // VECDEF: CXXConversionDecl {{.*}} operator auto (*)(type-parameter-0-0) __attribute__((vectorcall)) 'auto (*() const noexcept)(auto)' __attribute__((vectorcall)) // VECDEF: CXXConversionDecl {{.*}} operator auto (*)(char) __attribute__((vectorcall)) 'void (*() const noexcept)(char)' __attribute__((vectorcall)) // VECDEF: CXXConversionDecl {{.*}} operator auto (*)(int) __attribute__((vectorcall)) 'void (*() const noexcept)(int)' __attribute__((vectorcall)) // // CHECK: FunctionTemplateDecl {{.*}} __invoke // NODEF: CXXMethodDecl {{.*}} __invoke 'auto (auto)' // NODEF: CXXMethodDecl {{.*}} __invoke 'void (char)' // NODEF: CXXMethodDecl {{.*}} __invoke 'void (int)' // VECDEF: CXXMethodDecl {{.*}} __invoke 'auto (auto) __attribute__((vectorcall))' // VECDEF: CXXMethodDecl {{.*}} __invoke 'void (char) __attribute__((vectorcall))' // VECDEF: CXXMethodDecl {{.*}} __invoke 'void (int) __attribute__((vectorcall))' // // ONLY WIN32 has the duplicate here. // WIN32: FunctionTemplateDecl {{.*}} operator auto (*)(type-parameter-0-0) __attribute__((thiscall)) // WIN32: CXXConversionDecl {{.*}} operator auto (*)(type-parameter-0-0) __attribute__((thiscall)) 'auto (*() __attribute__((thiscall)) const noexcept)(auto) __attribute__((thiscall))' // WIN32: CXXConversionDecl {{.*}} operator auto (*)(char) __attribute__((thiscall)) 'void (*() __attribute__((thiscall)) const noexcept)(char) __attribute__((thiscall))' // WIN32: CXXConversionDecl {{.*}} operator auto (*)(int) __attribute__((thiscall)) 'void (*() __attribute__((thiscall)) const noexcept)(int) __attribute__((thiscall))' // // WIN32: FunctionTemplateDecl {{.*}} __invoke // WIN32: CXXMethodDecl {{.*}} __invoke 'auto (auto) __attribute__((thiscall))' // WIN32: CXXMethodDecl {{.*}} __invoke 'void (char) __attribute__((thiscall))' // WIN32: CXXMethodDecl {{.*}} __invoke 'void (int) __attribute__((thiscall))' // CHECK: VarDecl {{.*}} genericvectorcalllambda ' // CHECK: LambdaExpr // CHECK: FunctionTemplateDecl {{.*}} operator() // CHECK: CXXMethodDecl {{.*}} operator() 'auto (auto) __attribute__((vectorcall)) const' inline // CHECK: CXXMethodDecl {{.*}} operator() 'void (char) __attribute__((vectorcall)) const' inline // CHECK: CXXMethodDecl {{.*}} operator() 'void (int) __attribute__((vectorcall)) const' inline // CHECK: FunctionTemplateDecl {{.*}} operator auto (*)(type-parameter-0-0) __attribute__((vectorcall)) // LIN64: CXXConversionDecl {{.*}} operator auto (*)(type-parameter-0-0) __attribute__((vectorcall)) 'auto (*() const noexcept)(auto) __attribute__((vectorcall))' // LIN64: CXXConversionDecl {{.*}} operator auto (*)(char) __attribute__((vectorcall)) 'void (*() const noexcept)(char) __attribute__((vectorcall))' // LIN64: CXXConversionDecl {{.*}} operator auto (*)(int) __attribute__((vectorcall)) 'void (*() const noexcept)(int) __attribute__((vectorcall))' // WIN32: CXXConversionDecl {{.*}} operator auto (*)(type-parameter-0-0) __attribute__((vectorcall)) 'auto (*() __attribute__((thiscall)) const noexcept)(auto) __attribute__((vectorcall))' // WIN32: CXXConversionDecl {{.*}} operator auto (*)(char) __attribute__((vectorcall)) 'void (*() __attribute__((thiscall)) const noexcept)(char) __attribute__((vectorcall))' // WIN32: CXXConversionDecl {{.*}} operator auto (*)(int) __attribute__((vectorcall)) 'void (*() __attribute__((thiscall)) const noexcept)(int) __attribute__((vectorcall))' // CHECK: FunctionTemplateDecl {{.*}} __invoke // CHECK: CXXMethodDecl {{.*}} __invoke 'auto (auto) __attribute__((vectorcall))' // CHECK: CXXMethodDecl {{.*}} __invoke 'void (char) __attribute__((vectorcall))' // CHECK: CXXMethodDecl {{.*}} __invoke 'void (int) __attribute__((vectorcall))' // NODEF: UnaryOperator {{.*}} 'void (*)(int, float, double)' prefix '+' // NODEF-NEXT: ImplicitCastExpr {{.*}} 'void (*)(int, float, double)' // NODEF-NEXT: CXXMemberCallExpr {{.*}}'void (*)(int, float, double)' // VECTDEF: UnaryOperator {{.*}} 'void (*)(int, float, double) __attribute__((vectorcall))' prefix '+' // VECTDEF-NEXT: ImplicitCastExpr {{.*}} 'void (*)(int, float, double) __attribute__((vectorcall))' // VECTDEF-NEXT: CXXMemberCallExpr {{.*}}'void (*)(int, float, double) __attribute__((vectorcall))' // CHECK: UnaryOperator {{.*}} 'void (*)(int, float, double) __attribute__((vectorcall))' prefix '+' // CHECK-NEXT: ImplicitCastExpr {{.*}} 'void (*)(int, float, double) __attribute__((vectorcall))' // CHECK-NEXT: CXXMemberCallExpr {{.*}}'void (*)(int, float, double) __attribute__((vectorcall))' // WIN32: UnaryOperator {{.*}} 'void (*)(int, float, double)' prefix '+' // WIN32-NEXT: ImplicitCastExpr {{.*}} 'void (*)(int, float, double)' // WIN32-NEXT: CXXMemberCallExpr {{.*}}'void (*)(int, float, double)' // NODEF: UnaryOperator {{.*}} 'void (*)(int, float, double)' prefix '+' // NODEF-NEXT: ImplicitCastExpr {{.*}} 'void (*)(int, float, double)' // NODEF-NEXT: CXXMemberCallExpr {{.*}}'void (*)(int, float, double)' // VECTDEF: UnaryOperator {{.*}} 'void (*)(int, float, double) __attribute__((vectorcall))' prefix '+' // VECTDEF-NEXT: ImplicitCastExpr {{.*}} 'void (*)(int, float, double) __attribute__((vectorcall))' // VECTDEF-NEXT: CXXMemberCallExpr {{.*}}'void (*)(int, float, double) __attribute__((vectorcall))' #ifdef BAD_CONVERSION // expected-error-re@+2 {{no viable conversion from {{.*}} to 'void (*)(int, float, double) __attribute__((vectorcall))}} // expected-note@#1 {{candidate function}} void (*__attribute__((vectorcall)) normal_ptr2)(int, float, double) = normal; // expected-error-re@+2 {{no viable conversion from {{.*}} to 'void (*)(int, float, double)}} // expected-note@#2 {{candidate function}} void (*vectorcall_ptr2)(int, float, double) = vectorcall; #ifdef WIN32 void (*__attribute__((thiscall)) thiscall_ptr2)(int, float, double) = thiscall; #endif // WIN32 // expected-error-re@+2 {{no viable conversion from {{.*}} to 'void (*)(char) __attribute__((vectorcall))'}} // expected-note@#4 {{candidate function}} void(__vectorcall * generic_ptr)(char) = genericlambda; // expected-error-re@+2 {{no viable conversion from {{.*}} to 'void (*)(char)}} // expected-note@#5 {{candidate function}} void (*generic_ptr2)(char) = genericvectorcalllambda; #endif // BAD_CONVERSION #ifdef BAD_VEC_CONVERS void (*__attribute__((vectorcall)) normal_ptr2)(int, float, double) = normal; void (*normal_ptr3)(int, float, double) = normal; // expected-error-re@+2 {{no viable conversion from {{.*}} to 'void (*)(int, float, double) __attribute__((regcall))}} // expected-note@#1 {{candidate function}} void (*__attribute__((regcall)) normalptr4)(int, float, double) = normal; void (*__attribute__((vectorcall)) vectorcall_ptr2)(int, float, double) = vectorcall; void (*vectorcall_ptr3)(int, float, double) = vectorcall; #endif // BAD_VEC_CONVERS // Required to force emission of the invoker. void (*normal_ptr)(int, float, double) = normal; void (*__attribute__((vectorcall)) vectorcall_ptr)(int, float, double) = vectorcall; #ifdef WIN32 void (*thiscall_ptr)(int, float, double) = thiscall; #endif // WIN32 void (*cdecl_ptr)(int, float, double) = cdecl; void (*generic_ptr3)(char) = genericlambda; void (*generic_ptr4)(int) = genericlambda; #ifdef WIN32 void (*__attribute__((thiscall)) generic_ptr3b)(char) = genericlambda; void (*__attribute__((thiscall)) generic_ptr4b)(int) = genericlambda; #endif void (*__attribute__((vectorcall)) generic_ptr5)(char) = genericvectorcalllambda; void (*__attribute__((vectorcall)) generic_ptr6)(int) = genericvectorcalllambda; }