| // RUN: %clang_cc1 -fblocks -emit-llvm %s -o - -triple=i386-pc-win32 -std=c++98 | FileCheck %s |
| // RUN: %clang_cc1 -fblocks -emit-llvm %s -o - -triple=x86_64-pc-win32 -std=c++98| FileCheck -check-prefix X64 %s |
| |
| int a; |
| // CHECK-DAG: @"?a@@3HA" |
| |
| extern "C++" { |
| static int __attribute__((used)) ignore_transparent_context; |
| // CHECK-DAG: @ignore_transparent_context |
| } |
| |
| namespace N { |
| int b; |
| // CHECK-DAG: @"?b@N@@3HA" |
| |
| namespace { |
| int anonymous; |
| // CHECK-DAG: @"?anonymous@?A0x{{[^@]*}}@N@@3HA" |
| } |
| } |
| |
| static int c; |
| // CHECK-DAG: @c |
| |
| int _c(void) {return N::anonymous + c;} |
| // CHECK-DAG: @"?_c@@YAHXZ" |
| // X64-DAG: @"?_c@@YAHXZ" |
| |
| const int &NeedsReferenceTemporary = 2; |
| // CHECK-DAG: @"?NeedsReferenceTemporary@@3ABHB" = dso_local constant i32* @"?$RT1@NeedsReferenceTemporary@@3ABHB" |
| // X64-DAG: @"?NeedsReferenceTemporary@@3AEBHEB" = dso_local constant i32* @"?$RT1@NeedsReferenceTemporary@@3AEBHEB" |
| |
| class foo { |
| static const short d; |
| // CHECK-DAG: @"?d@foo@@0FB" |
| protected: |
| static volatile long e; |
| // CHECK-DAG: @"?e@foo@@1JC" |
| public: |
| static const volatile char f; |
| // CHECK-DAG: @"?f@foo@@2DD" |
| int operator+(int a); |
| foo(){} |
| // CHECK-DAG: @"??0foo@@QAE@XZ" |
| // X64-DAG: @"??0foo@@QEAA@XZ" |
| |
| ~foo(){} |
| // CHECK-DAG: @"??1foo@@QAE@XZ" |
| // X64-DAG: @"??1foo@@QEAA@XZ |
| |
| foo(int i){} |
| // CHECK-DAG: @"??0foo@@QAE@H@Z" |
| // X64-DAG: @"??0foo@@QEAA@H@Z" |
| |
| foo(char *q){} |
| // CHECK-DAG: @"??0foo@@QAE@PAD@Z" |
| // X64-DAG: @"??0foo@@QEAA@PEAD@Z" |
| |
| static foo* static_method() { return 0; } |
| |
| }f,s1(1),s2((char*)0); |
| |
| typedef foo (foo2); |
| |
| struct bar { |
| static int g; |
| }; |
| |
| union baz { |
| int a; |
| char b; |
| double c; |
| }; |
| |
| enum quux { |
| qone, |
| qtwo, |
| qthree |
| }; |
| |
| foo bar() { return foo(); } |
| // CHECK-DAG: @"?bar@@YA?AVfoo@@XZ" |
| // X64-DAG: @"?bar@@YA?AVfoo@@XZ" |
| |
| int foo::operator+(int a) { |
| // CHECK-DAG: @"??Hfoo@@QAEHH@Z" |
| // X64-DAG: @"??Hfoo@@QEAAHH@Z" |
| |
| foo::static_method(); |
| // CHECK-DAG: @"?static_method@foo@@SAPAV1@XZ" |
| // X64-DAG: @"?static_method@foo@@SAPEAV1@XZ" |
| bar(); |
| return a; |
| } |
| |
| const short foo::d = 0; |
| volatile long foo::e; |
| const volatile char foo::f = 'C'; |
| |
| int bar::g; |
| // CHECK-DAG: @"?g@bar@@2HA" |
| |
| extern int * const h1 = &a; |
| // CHECK-DAG: @"?h1@@3QAHA" |
| extern const int * const h2 = &a; |
| // CHECK-DAG: @"?h2@@3QBHB" |
| extern int * const __restrict h3 = &a; |
| // CHECK-DAG: @"?h3@@3QIAHIA" |
| // X64-DAG: @"?h3@@3QEIAHEIA" |
| |
| int i[10][20]; |
| // CHECK-DAG: @"?i@@3PAY0BE@HA" |
| |
| typedef int (*FunT)(int, int); |
| FunT FunArr[10][20]; |
| // CHECK-DAG: @"?FunArr@@3PAY0BE@P6AHHH@ZA" |
| // X64-DAG: @"?FunArr@@3PAY0BE@P6AHHH@ZA" |
| |
| int (__stdcall *j)(signed char, unsigned char); |
| // CHECK-DAG: @"?j@@3P6GHCE@ZA" |
| |
| const volatile char foo2::*k; |
| // CHECK-DAG: @"?k@@3PTfoo@@DT1@" |
| // X64-DAG: @"?k@@3PETfoo@@DET1@" |
| |
| int (foo2::*l)(int); |
| // CHECK-DAG: @"?l@@3P8foo@@AEHH@ZQ1@" |
| |
| // Ensure typedef CV qualifiers are mangled correctly |
| typedef const int cInt; |
| typedef volatile int vInt; |
| typedef const volatile int cvInt; |
| |
| extern cInt g_cInt = 1; |
| vInt g_vInt = 2; |
| cvInt g_cvInt = 3; |
| |
| // CHECK-DAG: @"?g_cInt@@3HB" |
| // CHECK-DAG: @"?g_vInt@@3HC" |
| // CHECK-DAG: @"?g_cvInt@@3HD" |
| |
| // Static functions are mangled, too. |
| // Also make sure calling conventions, arglists, and throw specs work. |
| static void __stdcall alpha(float a, double b) throw() {} |
| bool __fastcall beta(long long a, wchar_t b) throw(signed char, unsigned char) { |
| // CHECK-DAG: @"?beta@@YI_N_J_W@Z" |
| // X64-DAG: @"?beta@@YA_N_J_W@Z" |
| alpha(0.f, 0.0); |
| return false; |
| } |
| |
| // CHECK-DAG: @"?alpha@@YGXMN@Z" |
| // X64-DAG: @"?alpha@@YAXMN@Z" |
| |
| // Make sure tag-type mangling works. |
| void gamma(class foo, struct bar, union baz, enum quux) {} |
| // CHECK-DAG: @"?gamma@@YAXVfoo@@Ubar@@Tbaz@@W4quux@@@Z" |
| // X64-DAG: @"?gamma@@YAXVfoo@@Ubar@@Tbaz@@W4quux@@@Z" |
| |
| // Make sure pointer/reference-type mangling works. |
| void delta(int * const a, const long &) {} |
| // CHECK-DAG: @"?delta@@YAXQAHABJ@Z" |
| // X64-DAG: @"?delta@@YAXQEAHAEBJ@Z" |
| |
| // Array mangling. |
| void epsilon(int a[][10][20]) {} |
| // CHECK-DAG: @"?epsilon@@YAXQAY19BE@H@Z" |
| // X64-DAG: @"?epsilon@@YAXQEAY19BE@H@Z" |
| |
| void zeta(int (*)(int, int)) {} |
| // CHECK-DAG: @"?zeta@@YAXP6AHHH@Z@Z" |
| // X64-DAG: @"?zeta@@YAXP6AHHH@Z@Z" |
| |
| // Blocks mangling (Clang extension). A block should be mangled slightly |
| // differently from a similar function pointer. |
| void eta(int (^)(int, int)) {} |
| // CHECK-DAG: @"?eta@@YAXP_EAHHH@Z@Z" |
| |
| typedef int theta_arg(int,int); |
| void theta(theta_arg^ block) {} |
| // CHECK-DAG: @"?theta@@YAXP_EAHHH@Z@Z" |
| |
| void operator_new_delete() { |
| char *ptr = new char; |
| // CHECK-DAG: @"??2@YAPAXI@Z" |
| |
| delete ptr; |
| // CHECK-DAG: @"??3@YAXPAX@Z" |
| |
| char *array = new char[42]; |
| // CHECK-DAG: @"??_U@YAPAXI@Z" |
| |
| delete [] array; |
| // CHECK-DAG: @"??_V@YAXPAX@Z" |
| } |
| |
| // PR13022 |
| void (redundant_parens)(); |
| void redundant_parens_use() { redundant_parens(); } |
| // CHECK-DAG: @"?redundant_parens@@YAXXZ" |
| // X64-DAG: @"?redundant_parens@@YAXXZ" |
| |
| // PR13047 |
| typedef double RGB[3]; |
| RGB color1; |
| // CHECK-DAG: @"?color1@@3PANA" |
| extern const RGB color2 = {}; |
| // CHECK-DAG: @"?color2@@3QBNB" |
| extern RGB const color3[5] = {}; |
| // CHECK-DAG: @"?color3@@3QAY02$$CBNA" |
| extern RGB const ((color4)[5]) = {}; |
| // CHECK-DAG: @"?color4@@3QAY02$$CBNA" |
| |
| struct B; |
| volatile int B::* volatile memptr1; |
| // X64-DAG: @"?memptr1@@3RESB@@HES1@" |
| volatile int B::* memptr2; |
| // X64-DAG: @"?memptr2@@3PESB@@HES1@" |
| int B::* volatile memptr3; |
| // X64-DAG: @"?memptr3@@3REQB@@HEQ1@" |
| typedef int (*fun)(); |
| volatile fun B::* volatile funmemptr1; |
| // X64-DAG: @"?funmemptr1@@3RESB@@R6AHXZES1@" |
| volatile fun B::* funmemptr2; |
| // X64-DAG: @"?funmemptr2@@3PESB@@R6AHXZES1@" |
| fun B::* volatile funmemptr3; |
| // X64-DAG: @"?funmemptr3@@3REQB@@P6AHXZEQ1@" |
| void (B::* volatile memptrtofun1)(); |
| // X64-DAG: @"?memptrtofun1@@3R8B@@EAAXXZEQ1@" |
| const void (B::* memptrtofun2)(); |
| // X64-DAG: @"?memptrtofun2@@3P8B@@EAAXXZEQ1@" |
| volatile void (B::* memptrtofun3)(); |
| // X64-DAG: @"?memptrtofun3@@3P8B@@EAAXXZEQ1@" |
| int (B::* volatile memptrtofun4)(); |
| // X64-DAG: @"?memptrtofun4@@3R8B@@EAAHXZEQ1@" |
| volatile int (B::* memptrtofun5)(); |
| // X64-DAG: @"?memptrtofun5@@3P8B@@EAA?CHXZEQ1@" |
| const int (B::* memptrtofun6)(); |
| // X64-DAG: @"?memptrtofun6@@3P8B@@EAA?BHXZEQ1@" |
| fun (B::* volatile memptrtofun7)(); |
| // X64-DAG: @"?memptrtofun7@@3R8B@@EAAP6AHXZXZEQ1@" |
| volatile fun (B::* memptrtofun8)(); |
| // X64-DAG: @"?memptrtofun8@@3P8B@@EAAR6AHXZXZEQ1@" |
| const fun (B::* memptrtofun9)(); |
| // X64-DAG: @"?memptrtofun9@@3P8B@@EAAQ6AHXZXZEQ1@" |
| |
| // PR12603 |
| enum E {}; |
| // CHECK-DAG: "?fooE@@YA?AW4E@@XZ" |
| // X64-DAG: "?fooE@@YA?AW4E@@XZ" |
| E fooE() { return E(); } |
| |
| class X {}; |
| // CHECK-DAG: "?fooX@@YA?AVX@@XZ" |
| // X64-DAG: "?fooX@@YA?AVX@@XZ" |
| X fooX() { return X(); } |
| |
| namespace PR13182 { |
| extern char s0[]; |
| // CHECK-DAG: @"?s0@PR13182@@3PADA" |
| extern char s1[42]; |
| // CHECK-DAG: @"?s1@PR13182@@3PADA" |
| extern const char s2[]; |
| // CHECK-DAG: @"?s2@PR13182@@3QBDB" |
| extern const char s3[42]; |
| // CHECK-DAG: @"?s3@PR13182@@3QBDB" |
| extern volatile char s4[]; |
| // CHECK-DAG: @"?s4@PR13182@@3RCDC" |
| extern const volatile char s5[]; |
| // CHECK-DAG: @"?s5@PR13182@@3SDDD" |
| extern const char* const* s6; |
| // CHECK-DAG: @"?s6@PR13182@@3PBQBDB" |
| |
| char foo() { |
| return s0[0] + s1[0] + s2[0] + s3[0] + s4[0] + s5[0] + s6[0][0]; |
| } |
| } |
| |
| extern "C" inline void extern_c_func() { |
| static int local; |
| // CHECK-DAG: @"?local@?1??extern_c_func@@9@4HA" |
| // X64-DAG: @"?local@?1??extern_c_func@@9@4HA" |
| } |
| |
| void call_extern_c_func() { |
| extern_c_func(); |
| } |
| |
| int main() { return 0; } |
| // CHECK-DAG: @main |
| // X64-DAG: @main |
| |
| int wmain() { return 0; } |
| // CHECK-DAG: @wmain |
| // X64-DAG: @wmain |
| |
| int WinMain() { return 0; } |
| // CHECK-DAG: @WinMain |
| // X64-DAG: @WinMain |
| |
| int wWinMain() { return 0; } |
| // CHECK-DAG: @wWinMain |
| // X64-DAG: @wWinMain |
| |
| int DllMain() { return 0; } |
| // CHECK-DAG: @DllMain |
| // X64-DAG: @DllMain |
| |
| inline int inline_function_with_local_type() { |
| static struct { |
| int a_field; |
| } static_variable_in_inline_function = { 20 }, second_static = { 40 }; |
| // CHECK: @"?static_variable_in_inline_function@?1??inline_function_with_local_type@@YAHXZ@4U<unnamed-type-static_variable_in_inline_function>@?1??1@YAHXZ@A" |
| |
| return static_variable_in_inline_function.a_field + second_static.a_field; |
| } |
| |
| int call_inline_function_with_local_type() { |
| return inline_function_with_local_type(); |
| } |
| |
| template <typename T> |
| inline int templated_inline_function_with_local_type() { |
| static struct { |
| int a_field; |
| } static_variable_in_templated_inline_function = { 20 }, |
| second_static = { 40 }; |
| // CHECK: @"?static_variable_in_templated_inline_function@?1???$templated_inline_function_with_local_type@H@@YAHXZ@4U<unnamed-type-static_variable_in_templated_inline_function>@?1???$templated_inline_function_with_local_type@H@@YAHXZ@A" |
| |
| return static_variable_in_templated_inline_function.a_field + |
| second_static.a_field; |
| } |
| |
| int call_templated_inline_function_with_local_type() { |
| return templated_inline_function_with_local_type<int>(); |
| } |
| |
| // PR17371 |
| struct OverloadedNewDelete { |
| // __cdecl |
| void *operator new(__SIZE_TYPE__); |
| void *operator new[](__SIZE_TYPE__); |
| void operator delete(void *); |
| void operator delete[](void *); |
| // __thiscall |
| int operator+(int); |
| }; |
| |
| void *OverloadedNewDelete::operator new(__SIZE_TYPE__ s) { return 0; } |
| void *OverloadedNewDelete::operator new[](__SIZE_TYPE__ s) { return 0; } |
| void OverloadedNewDelete::operator delete(void *) { } |
| void OverloadedNewDelete::operator delete[](void *) { } |
| int OverloadedNewDelete::operator+(int x) { return x; }; |
| |
| // CHECK-DAG: ??2OverloadedNewDelete@@SAPAXI@Z |
| // CHECK-DAG: ??_UOverloadedNewDelete@@SAPAXI@Z |
| // CHECK-DAG: ??3OverloadedNewDelete@@SAXPAX@Z |
| // CHECK-DAG: ??_VOverloadedNewDelete@@SAXPAX@Z |
| // CHECK-DAG: ??HOverloadedNewDelete@@QAEHH@Z |
| |
| // X64-DAG: ??2OverloadedNewDelete@@SAPEAX_K@Z |
| // X64-DAG: ??_UOverloadedNewDelete@@SAPEAX_K@Z |
| // X64-DAG: ??3OverloadedNewDelete@@SAXPEAX@Z |
| // X64-DAG: ??_VOverloadedNewDelete@@SAXPEAX@Z |
| // X64-DAG: ??HOverloadedNewDelete@@QEAAHH@Z |
| |
| // Indirecting the function type through a typedef will require a calling |
| // convention adjustment before building the method decl. |
| |
| typedef void *__thiscall OperatorNewType(__SIZE_TYPE__); |
| typedef void __thiscall OperatorDeleteType(void *); |
| |
| struct TypedefNewDelete { |
| OperatorNewType operator new; |
| OperatorNewType operator new[]; |
| OperatorDeleteType operator delete; |
| OperatorDeleteType operator delete[]; |
| }; |
| |
| void *TypedefNewDelete::operator new(__SIZE_TYPE__ s) { return 0; } |
| void *TypedefNewDelete::operator new[](__SIZE_TYPE__ s) { return 0; } |
| void TypedefNewDelete::operator delete(void *) { } |
| void TypedefNewDelete::operator delete[](void *) { } |
| |
| // CHECK-DAG: ??2TypedefNewDelete@@SAPAXI@Z |
| // CHECK-DAG: ??_UTypedefNewDelete@@SAPAXI@Z |
| // CHECK-DAG: ??3TypedefNewDelete@@SAXPAX@Z |
| // CHECK-DAG: ??_VTypedefNewDelete@@SAXPAX@Z |
| |
| void __vectorcall vector_func() { } |
| // CHECK-DAG: @"?vector_func@@YQXXZ" |
| |
| template <void (*)(void)> |
| void fn_tmpl() {} |
| |
| template void fn_tmpl<extern_c_func>(); |
| // CHECK-DAG: @"??$fn_tmpl@$1?extern_c_func@@YAXXZ@@YAXXZ" |
| |
| extern "C" void __attribute__((overloadable)) overloaded_fn() {} |
| // CHECK-DAG: @"?overloaded_fn@@$$J0YAXXZ" |
| |
| extern "C" void overloaded_fn2() {} |
| // CHECK-DAG: @overloaded_fn2 |
| // |
| extern "C" void __attribute__((overloadable)) overloaded_fn3(); |
| extern "C" void overloaded_fn3() {} |
| // CHECK-DAG: @overloaded_fn3 |
| |
| namespace UnnamedType { |
| struct S { |
| typedef struct {} *T1[1]; |
| typedef struct {} T2; |
| typedef struct {} *T3, T4; |
| using T5 = struct {}; |
| using T6 = struct {} *; |
| }; |
| void f(S::T1) {} |
| void f(S::T2) {} |
| void f(S::T3) {} |
| void f(S::T4) {} |
| void f(S::T5) {} |
| void f(S::T6) {} |
| // CHECK-DAG: @"?f@UnnamedType@@YAXQAPAU<unnamed-type-T1>@S@1@@Z" |
| // CHECK-DAG: @"?f@UnnamedType@@YAXUT2@S@1@@Z" |
| // CHECK-DAG: @"?f@UnnamedType@@YAXPAUT4@S@1@@Z" |
| // CHECK-DAG: @"?f@UnnamedType@@YAXUT4@S@1@@Z" |
| // CHECK-DAG: @"?f@UnnamedType@@YAXUT5@S@1@@Z" |
| // CHECK-DAG: @"?f@UnnamedType@@YAXPAU<unnamed-type-T6>@S@1@@Z" |
| |
| // X64-DAG: @"?f@UnnamedType@@YAXQEAPEAU<unnamed-type-T1>@S@1@@Z" |
| // X64-DAG: @"?f@UnnamedType@@YAXUT2@S@1@@Z" |
| // X64-DAG: @"?f@UnnamedType@@YAXPEAUT4@S@1@@Z"(%"struct.UnnamedType::S::T4" |
| // X64-DAG: @"?f@UnnamedType@@YAXUT4@S@1@@Z" |
| // X64-DAG: @"?f@UnnamedType@@YAXUT5@S@1@@Z" |
| // X64-DAG: @"?f@UnnamedType@@YAXPEAU<unnamed-type-T6>@S@1@@Z" |
| } |
| |
| namespace PassObjectSize { |
| // NOTE: This mangling is subject to change. |
| // Reiterating from the comment in MicrosoftMangle, the scheme is pretend a |
| // parameter of type __clang::__pass_object_sizeN exists after each pass object |
| // size param P, where N is the Type of the pass_object_size attribute on P. |
| // |
| // e.g. we want to mangle: |
| // void foo(void *const __attribute__((pass_object_size(0)))); |
| // as if it were |
| // namespace __clang { enum __pass_object_size0 : size_t {}; } |
| // void foo(void *const, __clang::__pass_object_size0); |
| // where __clang is a top-level namespace. |
| |
| // CHECK-DAG: define dso_local i32 @"?foo@PassObjectSize@@YAHQAHW4__pass_object_size0@__clang@@@Z" |
| int foo(int *const i __attribute__((pass_object_size(0)))) { return 0; } |
| // CHECK-DAG: define dso_local i32 @"?bar@PassObjectSize@@YAHQAHW4__pass_object_size1@__clang@@@Z" |
| int bar(int *const i __attribute__((pass_object_size(1)))) { return 0; } |
| // CHECK-DAG: define dso_local i32 @"?qux@PassObjectSize@@YAHQAHW4__pass_object_size1@__clang@@0W4__pass_object_size0@3@@Z" |
| int qux(int *const i __attribute__((pass_object_size(1))), int *const j __attribute__((pass_object_size(0)))) { return 0; } |
| // CHECK-DAG: define dso_local i32 @"?zot@PassObjectSize@@YAHQAHW4__pass_object_size1@__clang@@01@Z" |
| int zot(int *const i __attribute__((pass_object_size(1))), int *const j __attribute__((pass_object_size(1)))) { return 0; } |
| } |
| |
| namespace Atomic { |
| // CHECK-DAG: define dso_local void @"?f@Atomic@@YAXU?$_Atomic@H@__clang@@@Z"( |
| void f(_Atomic(int)) {} |
| } |
| namespace Complex { |
| // CHECK-DAG: define dso_local void @"?f@Complex@@YAXU?$_Complex@H@__clang@@@Z"( |
| void f(_Complex int) {} |
| } |
| namespace Float16 { |
| // CHECK-DAG: define dso_local void @"?f@Float16@@YAXU_Float16@__clang@@@Z"( |
| void f(_Float16) {} |
| } |
| |
| namespace PR26029 { |
| template <class> |
| struct L { |
| L() {} |
| }; |
| template <class> |
| class H; |
| struct M : L<H<int *> > {}; |
| |
| template <class> |
| struct H {}; |
| |
| template <class GT> |
| void m_fn3() { |
| (H<GT *>()); |
| M(); |
| } |
| |
| void runOnFunction() { |
| L<H<int *> > b; |
| m_fn3<int>(); |
| } |
| // CHECK-DAG: call {{.*}} @"??0?$L@V?$H@PAH@PR26029@@@PR26029@@QAE@XZ" |
| } |