]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - test/CodeGenCXX/ctor-dtor-alias.cpp
Vendor import of clang trunk r238337:
[FreeBSD/FreeBSD.git] / test / CodeGenCXX / ctor-dtor-alias.cpp
1 // RUN: %clang_cc1 %s -triple i686-linux -emit-llvm -o - -mconstructor-aliases | FileCheck --check-prefix=NOOPT %s
2
3 // RUN: %clang_cc1 %s -triple i686-linux -emit-llvm -o - -mconstructor-aliases -O1 -disable-llvm-optzns > %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
10
11 // RUN: %clang_cc1 %s -triple i686-pc-windows-gnu -emit-llvm -o - -mconstructor-aliases -O1 -disable-llvm-optzns | FileCheck --check-prefix=COFF %s
12
13 namespace test1 {
14 // Test that we produce the apropriate comdats when creating aliases to
15 // weak_odr constructors and destructors.
16
17 // CHECK1: @_ZN5test16foobarIvEC1Ev = weak_odr alias void {{.*}} @_ZN5test16foobarIvEC2Ev
18 // CHECK1: @_ZN5test16foobarIvED1Ev = weak_odr alias void (%"struct.test1::foobar"*)* @_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)
22 // CHECK1-NOT: comdat
23
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
29
30 template <typename T>
31 struct foobar {
32   foobar() {}
33   virtual ~foobar() {}
34 };
35
36 template struct foobar<void>;
37 }
38
39 namespace test2 {
40 // test that when the destrucor is linkonce_odr we just replace every use of
41 // C1 with C2.
42
43 // CHECK1: define internal void @__cxx_global_var_init()
44 // CHECK1: call void @_ZN5test26foobarIvEC2Ev
45 // CHECK1: define linkonce_odr void @_ZN5test26foobarIvEC2Ev({{.*}} comdat align
46 void g();
47 template <typename T> struct foobar {
48   foobar() { g(); }
49 };
50 foobar<void> x;
51 }
52
53 namespace test3 {
54 // test that instead of an internal alias we just use the other destructor
55 // directly.
56
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(
60 namespace {
61 struct A {
62   ~A() {}
63 };
64
65 struct B : public A {};
66 }
67
68 B x;
69 }
70
71 namespace test4 {
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.
75
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
79
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
85   struct A {
86     virtual ~A() {}
87   };
88   struct B : public A{
89     ~B() {}
90   };
91   B X;
92 }
93
94 namespace test5 {
95   // similar to test4, but with an internal B.
96
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
100   struct A {
101     virtual ~A() {}
102   };
103   namespace {
104   struct B : public A{
105     ~B() {}
106   };
107   }
108   B X;
109 }
110
111 namespace test6 {
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.
114   struct A {
115     virtual ~A();
116   };
117   namespace {
118   struct B : public A {
119     ~B() {}
120   };
121   }
122   B X;
123   // CHECK3: define internal void @__cxx_global_var_init.4()
124   // CHECK3: call i32 @__cxa_atexit({{.*}}@_ZN5test61AD2Ev
125 }
126
127 namespace test7 {
128   // Test that we don't produce an alias from ~B to ~A<int> (or crash figuring
129   // out if we should).
130   // pr17875.
131   // CHECK3: define void @_ZN5test71BD2Ev
132   template <typename> struct A {
133     ~A() {}
134   };
135   class B : A<int> {
136     ~B();
137   };
138   template class A<int>;
139   B::~B() {}
140 }
141
142 namespace test8 {
143   // Test that we replace ~zed with ~bar which is an alias to ~foo.
144   // CHECK4: @_ZN5test83barD2Ev = alias {{.*}} @_ZN5test83fooD2Ev
145   // CHECK4: define internal void @__cxx_global_var_init.5()
146   // CHECK4: call i32 @__cxa_atexit({{.*}}@_ZN5test83barD2Ev
147   struct foo {
148     ~foo();
149   };
150   foo::~foo() {}
151   struct bar : public foo {
152     ~bar();
153   };
154   bar::~bar() {}
155   struct zed : public bar {};
156   zed foo;
157 }
158
159 namespace test9 {
160 struct foo {
161   __attribute__((stdcall)) ~foo() {
162   }
163 };
164
165 struct bar : public foo {};
166
167 void zed() {
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
171   bar ptr;
172 }
173 }
174
175 // CHECK5: @_ZTV1C = linkonce_odr unnamed_addr constant [4 x i8*] [{{[^@]*}}@_ZTI1C {{[^@]*}}@_ZN1CD2Ev {{[^@]*}}@_ZN1CD0Ev {{[^@]*}}]
176 // r194296 replaced C::~C with B::~B without emitting the later.
177
178 class A {
179 public:
180   A(int);
181   virtual ~A();
182 };
183
184 template <class>
185 class B : A {
186 public:
187   B()
188       : A(0) {
189   }
190   __attribute__((always_inline)) ~B() {
191   }
192 };
193
194 extern template class B<char>;
195
196 class C : B<char> {
197 };
198
199 void
200 fn1() {
201   new C;
202 }
203
204 namespace test10 {
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.
207 struct bar {
208   ~bar();
209 };
210 bar::~bar() {
211 }
212 } // closing the namespace causes ~bar to be sent to CodeGen
213 namespace test10 {
214 template <typename T>
215 struct foo : public bar {
216   ~foo();
217 };
218 template <typename T>
219 foo<T>::~foo() {}
220 template class foo<int>;
221 // CHECK5: define weak_odr void @_ZN6test103fooIiED2Ev({{.*}} comdat($_ZN6test103fooIiED5Ev)
222 }
223
224 namespace test11 {
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.
227 struct bar {
228   ~bar();
229 };
230 bar::~bar() {}
231 struct foo : public bar {
232   ~foo();
233 };
234 foo::~foo() {}
235 // CHECK6: @_ZN6test113fooD2Ev = alias {{.*}} @_ZN6test113barD2Ev
236 // CHECK6: @_ZN6test113fooD1Ev = alias {{.*}} @_ZN6test113fooD2Ev
237 }
238
239 namespace test12 {
240 template <int>
241 struct foo {
242   ~foo() { delete this; }
243 };
244
245 template class foo<1>;
246 // CHECK6: @_ZN6test123fooILi1EED1Ev = weak_odr alias {{.*}} @_ZN6test123fooILi1EED2Ev
247 // CHECK6: define weak_odr void @_ZN6test123fooILi1EED2Ev({{.*}}) {{.*}} comdat($_ZN6test123fooILi1EED5Ev)
248 }