Run DCE after a LoopFlatten test to reduce spurious output [nfc]
[llvm-project.git] / clang / test / CodeGenCXX / devirtualize-virtual-function-calls-final.cpp
blob931b60db69d6c355f2489b1f893fa4ff6670383c
1 // RUN: %clang_cc1 -triple i386-unknown-unknown -std=c++11 %s -emit-llvm -o - | FileCheck %s
3 namespace Test1 {
4 struct A {
5 virtual int f() final;
6 };
8 // CHECK-LABEL: define{{.*}} i32 @_ZN5Test11fEPNS_1AE
9 int f(A *a) {
10 // CHECK: call noundef i32 @_ZN5Test11A1fEv
11 return a->f();
15 namespace Test2 {
16 struct A final {
17 virtual int f();
20 // CHECK-LABEL: define{{.*}} i32 @_ZN5Test21fEPNS_1AE
21 int f(A *a) {
22 // CHECK: call noundef i32 @_ZN5Test21A1fEv
23 return a->f();
27 namespace Test2a {
28 struct A {
29 virtual ~A() final {}
30 virtual int f();
33 // CHECK-LABEL: define{{.*}} i32 @_ZN6Test2a1fEPNS_1AE
34 int f(A *a) {
35 // CHECK: call noundef i32 @_ZN6Test2a1A1fEv
36 return a->f();
41 namespace Test3 {
42 struct A {
43 virtual int f(); };
45 struct B final : A { };
47 // CHECK-LABEL: define{{.*}} i32 @_ZN5Test31fEPNS_1BE
48 int f(B *b) {
49 // CHECK: call noundef i32 @_ZN5Test31A1fEv
50 return b->f();
53 // CHECK-LABEL: define{{.*}} i32 @_ZN5Test31fERNS_1BE
54 int f(B &b) {
55 // CHECK: call noundef i32 @_ZN5Test31A1fEv
56 return b.f();
59 // CHECK-LABEL: define{{.*}} i32 @_ZN5Test31fEPv
60 int f(void *v) {
61 // CHECK: call noundef i32 @_ZN5Test31A1fEv
62 return static_cast<B*>(v)->f();
66 namespace Test4 {
67 struct A {
68 virtual void f();
69 virtual int operator-();
72 struct B final : A {
73 virtual void f();
74 virtual int operator-();
77 // CHECK-LABEL: define{{.*}} void @_ZN5Test41fEPNS_1BE
78 void f(B* d) {
79 // CHECK: call void @_ZN5Test41B1fEv
80 static_cast<A*>(d)->f();
81 // CHECK: call noundef i32 @_ZN5Test41BngEv
82 -static_cast<A&>(*d);
86 namespace Test5 {
87 struct A {
88 virtual void f();
89 virtual int operator-();
92 struct B : A {
93 virtual void f();
94 virtual int operator-();
97 struct C final : B {
100 // CHECK-LABEL: define{{.*}} void @_ZN5Test51fEPNS_1CE
101 void f(C* d) {
102 // FIXME: It should be possible to devirtualize this case, but that is
103 // not implemented yet.
104 // CHECK: getelementptr
105 // CHECK-NEXT: %[[FUNC:.*]] = load
106 // CHECK-NEXT: call void %[[FUNC]]
107 static_cast<A*>(d)->f();
109 // CHECK-LABEL: define{{.*}} void @_ZN5Test53fopEPNS_1CE
110 void fop(C* d) {
111 // FIXME: It should be possible to devirtualize this case, but that is
112 // not implemented yet.
113 // CHECK: getelementptr
114 // CHECK-NEXT: %[[FUNC:.*]] = load
115 // CHECK-NEXT: call noundef i32 %[[FUNC]]
116 -static_cast<A&>(*d);
120 namespace Test6 {
121 struct A {
122 virtual ~A();
125 struct B : public A {
126 virtual ~B();
129 struct C {
130 virtual ~C();
133 struct D final : public C, public B {
136 // CHECK-LABEL: define{{.*}} void @_ZN5Test61fEPNS_1DE
137 void f(D* d) {
138 // CHECK: call void @_ZN5Test61DD1Ev
139 static_cast<A*>(d)->~A();
143 namespace Test7 {
144 struct foo {
145 virtual void g() {}
148 struct bar {
149 virtual int f() { return 0; }
152 struct zed final : public foo, public bar {
153 int z;
154 virtual int f() {return z;}
157 // CHECK-LABEL: define{{.*}} i32 @_ZN5Test71fEPNS_3zedE
158 int f(zed *z) {
159 // CHECK: alloca
160 // CHECK-NEXT: store
161 // CHECK-NEXT: load
162 // CHECK-NEXT: call noundef i32 @_ZN5Test73zed1fEv
163 // CHECK-NEXT: ret
164 return static_cast<bar*>(z)->f();
168 namespace Test8 {
169 struct A { virtual ~A() {} };
170 struct B {
171 int b;
172 virtual int foo() { return b; }
174 struct C final : A, B { };
175 // CHECK-LABEL: define{{.*}} i32 @_ZN5Test84testEPNS_1CE
176 int test(C *c) {
177 // CHECK: %[[THIS:.*]] = phi
178 // CHECK-NEXT: call noundef i32 @_ZN5Test81B3fooEv(ptr {{[^,]*}} %[[THIS]])
179 return static_cast<B*>(c)->foo();
183 namespace Test9 {
184 struct A {
185 int a;
187 struct B {
188 int b;
190 struct C : public B, public A {
192 struct RA {
193 virtual A *f() {
194 return 0;
196 virtual A *operator-() {
197 return 0;
200 struct RC final : public RA {
201 virtual C *f() {
202 C *x = new C();
203 x->a = 1;
204 x->b = 2;
205 return x;
207 virtual C *operator-() {
208 C *x = new C();
209 x->a = 1;
210 x->b = 2;
211 return x;
214 // CHECK: define {{.*}} @_ZN5Test91fEPNS_2RCE
215 A *f(RC *x) {
216 // FIXME: It should be possible to devirtualize this case, but that is
217 // not implemented yet.
218 // CHECK: load
219 // CHECK: [[VTABLE:%.+]] = load {{.+}}
220 // CHECK: [[VFN:%.+]] = getelementptr inbounds {{.+}} [[VTABLE]], i{{[0-9]+}} 0
221 // CHECK-NEXT: %[[FUNC:.*]] = load {{.+}} [[VFN]]
222 // CHECK-NEXT: = call {{.*}} %[[FUNC]]
223 return static_cast<RA*>(x)->f();
225 // CHECK: define {{.*}} @_ZN5Test93fopEPNS_2RCE
226 A *fop(RC *x) {
227 // FIXME: It should be possible to devirtualize this case, but that is
228 // not implemented yet.
229 // CHECK: load
230 // CHECK: [[VTABLE:%.+]] = load {{.+}}
231 // CHECK: [[VFN:%.+]] = getelementptr inbounds {{.+}} [[VTABLE]], i{{[0-9]+}} 1
232 // CHECK-NEXT: %[[FUNC:.*]] = load {{.+}} [[VFN]]
233 // CHECK-NEXT: = call {{.*}} %[[FUNC]]
234 return -static_cast<RA&>(*x);
238 namespace Test10 {
239 struct A {
240 virtual int f();
243 struct B : A {
244 int f() final;
247 // CHECK-LABEL: define{{.*}} i32 @_ZN6Test101fEPNS_1BE
248 int f(B *b) {
249 // CHECK: call noundef i32 @_ZN6Test101B1fEv
250 return static_cast<A *>(b)->f();
254 namespace TestVBase {
255 struct A { virtual void f(); };
256 struct B : virtual A {};
257 struct C : virtual A { void f() override; };
259 extern struct BC final : B, C {} &bc;
260 extern struct BCusingA final : B, C { using A::f; } &bc_using_a;
261 extern struct BCusingB final : B, C { using B::f; } &bc_using_b;
262 extern struct BCusingC final : B, C { using C::f; } &bc_using_c;
264 extern struct CB final : C, B {} &cb;
265 extern struct CBusingA final : C, B { using A::f; } &cb_using_a;
266 extern struct CBusingB final : C, B { using B::f; } &cb_using_b;
267 extern struct CBusingC final : C, B { using C::f; } &cb_using_c;
269 // CHECK-LABEL: @_ZN9TestVBase4testEv(
270 void test() {
271 // FIXME: The 'using A' case can be devirtualized to call A's virtual
272 // adjustment thunk for C::f.
273 // FIXME: The 'using B' case can be devirtualized, but requires us to emit
274 // a derived-to-base or base-to-derived conversion as part of
275 // devirtualization.
277 // CHECK: call void @_ZN9TestVBase1C1fEv(
278 bc.f();
279 // CHECK: call void %
280 bc_using_a.f();
281 // CHECK: call void %
282 bc_using_b.f();
283 // CHECK: call void @_ZN9TestVBase1C1fEv(
284 bc_using_c.f();
286 // CHECK: call void @_ZN9TestVBase1C1fEv(
287 cb.f();
288 // CHECK: call void %
289 cb_using_a.f();
290 // CHECK: call void %
291 cb_using_b.f();
292 // CHECK: call void @_ZN9TestVBase1C1fEv(
293 cb_using_c.f();
297 namespace Test11 {
298 // Check that the definitions of Derived's operators are emitted.
300 // CHECK-LABEL: define linkonce_odr void @_ZN6Test111SIiE4foo1Ev(
301 // CHECK: call void @_ZN6Test111SIiE7DerivedclEv(
302 // CHECK: call noundef zeroext i1 @_ZN6Test111SIiE7DerivedeqERKNS_4BaseE(
303 // CHECK: call noundef zeroext i1 @_ZN6Test111SIiE7DerivedntEv(
304 // CHECK: call noundef nonnull align 4 dereferenceable(4) ptr @_ZN6Test111SIiE7DerivedixEi(
305 // CHECK: define linkonce_odr void @_ZN6Test111SIiE7DerivedclEv(
306 // CHECK: define linkonce_odr noundef zeroext i1 @_ZN6Test111SIiE7DerivedeqERKNS_4BaseE(
307 // CHECK: define linkonce_odr noundef zeroext i1 @_ZN6Test111SIiE7DerivedntEv(
308 // CHECK: define linkonce_odr noundef nonnull align 4 dereferenceable(4) ptr @_ZN6Test111SIiE7DerivedixEi(
309 class Base {
310 public:
311 virtual void operator()() {}
312 virtual bool operator==(const Base &other) { return false; }
313 virtual bool operator!() { return false; }
314 virtual Base &operator[](int i) { return *this; }
317 template<class T>
318 struct S {
319 class Derived final : public Base {
320 public:
321 void operator()() override {}
322 bool operator==(const Base &other) override { return true; }
323 bool operator!() override { return true; }
324 Base &operator[](int i) override { return *this; }
327 Derived *ptr = nullptr, *ptr2 = nullptr;
329 void foo1() {
330 if (ptr && ptr2) {
331 // These calls get devirtualized. Linkage fails if the definitions of
332 // the called functions are not emitted.
333 (*ptr)();
334 (void)(*ptr == *ptr2);
335 (void)(!(*ptr));
336 (void)((*ptr)[1]);
341 void foo2() {
342 S<int> *s = new S<int>;
343 s->foo1();