1 // RUN: %clang_cc1 %s -triple i686-linux -emit-llvm -o - -mconstructor-aliases | FileCheck --check-prefix=NOOPT %s
3 // RUN: %clang_cc1 %s -triple i686-linux -emit-llvm -o - -mconstructor-aliases -O1 -disable-llvm-passes > %t
4 // RUN: FileCheck --check-prefix=CHECK1 --input-file=%t %s
5 // RUN: FileCheck --check-prefix=CHECK2 --input-file=%t %s
6 // RUN: FileCheck --check-prefix=CHECK3 --input-file=%t %s
7 // RUN: FileCheck --check-prefix=CHECK4 --input-file=%t %s
8 // RUN: FileCheck --check-prefix=CHECK5 --input-file=%t %s
9 // RUN: FileCheck --check-prefix=CHECK6 --input-file=%t %s
11 // RUN: %clang_cc1 %s -triple i686-pc-windows-gnu -emit-llvm -o - -mconstructor-aliases -O1 -disable-llvm-passes | FileCheck --check-prefix=COFF %s
14 // Test that we produce the appropriate comdats when creating aliases to
15 // weak_odr constructors and destructors.
17 // CHECK1: @_ZN5test16foobarIvEC1Ev = weak_odr unnamed_addr alias void {{.*}} @_ZN5test16foobarIvEC2Ev
18 // CHECK1: @_ZN5test16foobarIvED1Ev = weak_odr unnamed_addr alias void (ptr), ptr @_ZN5test16foobarIvED2Ev
19 // CHECK1: define weak_odr void @_ZN5test16foobarIvEC2Ev({{.*}} comdat($_ZN5test16foobarIvEC5Ev)
20 // CHECK1: define weak_odr void @_ZN5test16foobarIvED2Ev({{.*}} comdat($_ZN5test16foobarIvED5Ev)
21 // CHECK1: define weak_odr void @_ZN5test16foobarIvED0Ev({{.*}} comdat($_ZN5test16foobarIvED5Ev)
24 // COFF doesn't support comdats with arbitrary names (C5/D5).
25 // COFF: define weak_odr {{.*}} void @_ZN5test16foobarIvEC2Ev({{.*}} comdat align
26 // COFF: define weak_odr {{.*}} void @_ZN5test16foobarIvEC1Ev({{.*}} comdat align
27 // COFF: define weak_odr {{.*}} void @_ZN5test16foobarIvED2Ev({{.*}} comdat align
28 // COFF: define weak_odr {{.*}} void @_ZN5test16foobarIvED0Ev({{.*}} comdat align
36 template struct foobar
<void>;
40 // test that when the destrucor is linkonce_odr we just replace every use of
43 // CHECK1: define internal void @__cxx_global_var_init()
44 // CHECK1: call void @_ZN5test26foobarIvEC2Ev
45 // CHECK1: define linkonce_odr void @_ZN5test26foobarIvEC2Ev({{.*}} comdat align
47 template <typename T
> struct foobar
{
54 // test that instead of an internal alias we just use the other destructor
57 // CHECK1: define internal void @__cxx_global_var_init.1()
58 // CHECK1: call i32 @__cxa_atexit{{.*}}_ZN5test312_GLOBAL__N_11AD2Ev
59 // CHECK1: define internal void @_ZN5test312_GLOBAL__N_11AD2Ev(
65 struct B
: public A
{};
72 // Test that we don't produce aliases from B to A. We cannot because we cannot
73 // guarantee that they will be present in every TU. Instead, we just call
74 // A's destructor directly.
76 // CHECK1: define internal void @__cxx_global_var_init.2()
77 // CHECK1: call i32 @__cxa_atexit{{.*}}_ZN5test41AD2Ev
78 // CHECK1: define linkonce_odr void @_ZN5test41AD2Ev({{.*}} comdat align
80 // test that we don't do this optimization at -O0 so that the debugger can
81 // see both destructors.
82 // NOOPT: define internal void @__cxx_global_var_init.2()
83 // NOOPT: call i32 @__cxa_atexit{{.*}}@_ZN5test41BD2Ev
84 // NOOPT: define linkonce_odr void @_ZN5test41BD2Ev({{.*}} comdat align
95 // similar to test4, but with an internal B.
97 // CHECK2: define internal void @__cxx_global_var_init.3()
98 // CHECK2: call i32 @__cxa_atexit{{.*}}_ZN5test51AD2Ev
99 // CHECK2: define linkonce_odr void @_ZN5test51AD2Ev({{.*}} comdat align
112 // Test that we use ~A directly, even when ~A is not defined. The symbol for
113 // ~B would have been internal and still contain a reference to ~A.
118 struct B
: public A
{
123 // CHECK3: define internal void @__cxx_global_var_init.4()
124 // CHECK3: call i32 @__cxa_atexit({{.*}}@_ZN5test61AD2Ev
128 // Test that we don't produce an alias from ~B to ~A<int> (or crash figuring
129 // out if we should).
131 // CHECK3: define{{.*}} void @_ZN5test71BD2Ev
132 template <typename
> struct A
{
138 template class A
<int>;
143 // Test that we replace ~zed with ~bar which is an alias to ~foo.
144 // CHECK4: @_ZN5test83barD2Ev ={{.*}} unnamed_addr alias {{.*}} @_ZN5test83fooD2Ev
145 // CHECK4: define internal void @__cxx_global_var_init.5()
146 // CHECK4: call i32 @__cxa_atexit({{.*}}@_ZN5test83barD2Ev
151 struct bar
: public foo
{
155 struct zed
: public bar
{};
161 __attribute__((stdcall)) ~foo() {
165 struct bar
: public foo
{};
168 // Test that we produce a call to bar's destructor. We used to call foo's, but
169 // it has a different calling conversion.
170 // CHECK4: call void @_ZN5test93barD2Ev
175 // CHECK5: @_ZTV1C = linkonce_odr unnamed_addr constant { [4 x ptr] } {{[^@]*}}@_ZTI1C, {{[^@]*}}@_ZN1CD2Ev, {{[^@]*}}@_ZN1CD0Ev]
176 // r194296 replaced C::~C with B::~B without emitting the later.
190 __attribute__((always_inline
)) ~B() {
194 extern template class B
<char>;
205 // Test that if a destructor is in a comdat, we don't try to emit is as an
206 // alias to a base class destructor.
212 } // closing the namespace causes ~bar to be sent to CodeGen
214 template <typename T
>
215 struct foo
: public bar
{
218 template <typename T
>
220 template class foo
<int>;
221 // CHECK5: define weak_odr void @_ZN6test103fooIiED2Ev({{.*}} comdat($_ZN6test103fooIiED5Ev)
225 // Test that when we don't have to worry about COMDATs we produce an alias
226 // from complate to base and from base to base class base.
231 struct foo
: public bar
{
235 // CHECK6: @_ZN6test113fooD2Ev ={{.*}} unnamed_addr alias {{.*}} @_ZN6test113barD2Ev
236 // CHECK6: @_ZN6test113fooD1Ev ={{.*}} unnamed_addr alias {{.*}} @_ZN6test113fooD2Ev
242 ~foo() { delete this; }
245 template class foo
<1>;
246 // CHECK6: @_ZN6test123fooILi1EED1Ev = weak_odr unnamed_addr alias {{.*}} @_ZN6test123fooILi1EED2Ev
247 // CHECK6: define weak_odr void @_ZN6test123fooILi1EED2Ev({{.*}}) {{.*}} comdat($_ZN6test123fooILi1EED5Ev)