aboutsummaryrefslogtreecommitdiff
path: root/clang/test/CIR/CodeGen/virtual-destructor-calls.cpp
blob: 08a6b21ca91d31358b5fba63e3c16612c237e723 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
// RUN: %clang_cc1 -triple aarch64-none-linux-android21 -std=c++20 -mconstructor-aliases -O0 -fclangir -emit-cir %s -o %t.cir
// RUN: FileCheck --check-prefix=CIR --input-file=%t.cir %s
// RUN: %clang_cc1 -triple aarch64-none-linux-android21 -std=c++20 -mconstructor-aliases -O0 -fclangir -emit-llvm %s -o %t-cir.ll
// RUN: FileCheck --check-prefix=LLVM --input-file=%t-cir.ll %s
// RUN: %clang_cc1 -triple aarch64-none-linux-android21 -std=c++20 -mconstructor-aliases -O0 -emit-llvm %s -o %t.ll
// RUN: FileCheck --check-prefix=OGCG --input-file=%t.ll %s

// TODO(cir): Try to emit base destructor as an alias at O1 or higher.

// FIXME: LLVM IR dialect does not yet support function ptr globals, which precludes
// a lot of the proper semantics for properly representing alias functions in LLVM
// (see the note on LLVM_O1 below).

struct Member {
  ~Member();
};

struct A {
  virtual ~A();
};

struct B : A {
  Member m;
  virtual ~B();
};

B::~B() { }

// Aliases are inserted before the function definitions in LLVM IR
// FIXME: These should have unnamed_addr set.
// LLVM: @_ZN1BD1Ev = alias void (ptr), ptr @_ZN1BD2Ev
// LLVM: @_ZN1CD1Ev = alias void (ptr), ptr @_ZN1CD2Ev

// OGCG: @_ZN1BD1Ev = unnamed_addr alias void (ptr), ptr @_ZN1BD2Ev
// OGCG: @_ZN1CD1Ev = unnamed_addr alias void (ptr), ptr @_ZN1CD2Ev


// Base (D2) dtor for B: calls A's base dtor.

// CIR: cir.func{{.*}} @_ZN1BD2Ev
// CIR:   cir.call @_ZN6MemberD1Ev
// CIR:   cir.call @_ZN1AD2Ev

// LLVM: define{{.*}} void @_ZN1BD2Ev
// LLVM:   call void @_ZN6MemberD1Ev
// LLVM:   call void @_ZN1AD2Ev

// OGCG: define{{.*}} @_ZN1BD2Ev
// OGCG:   call void @_ZN6MemberD1Ev
// OGCG:   call void @_ZN1AD2Ev

// Complete (D1) dtor for B: just an alias because there are no virtual bases.

// CIR: cir.func{{.*}} @_ZN1BD1Ev(!cir.ptr<!rec_B>) alias(@_ZN1BD2Ev)
// This is defined above for LLVM and OGCG.

// Deleting (D0) dtor for B: defers to the complete dtor but also calls operator delete.

// CIR: cir.func{{.*}} @_ZN1BD0Ev
// CIR:   cir.call @_ZN1BD1Ev(%[[THIS:.*]]) nothrow : (!cir.ptr<!rec_B>) -> ()
// CIR:   %[[THIS_VOID:.*]] = cir.cast bitcast %[[THIS]] : !cir.ptr<!rec_B> -> !cir.ptr<!void>
// CIR:   %[[SIZE:.*]] = cir.const #cir.int<16>
// CIR:   cir.call @_ZdlPvm(%[[THIS_VOID]], %[[SIZE]])

// LLVM: define{{.*}} void @_ZN1BD0Ev
// LLVM:   call void @_ZN1BD1Ev(ptr %[[THIS:.*]])
// LLVM:   call void @_ZdlPvm(ptr %[[THIS]], i64 16)

// OGCG: define{{.*}} @_ZN1BD0Ev
// OGCG:   call void @_ZN1BD1Ev(ptr{{.*}} %[[THIS:.*]])
// OGCG:   call void @_ZdlPvm(ptr{{.*}} %[[THIS]], i64{{.*}} 16)

struct C : B {
  ~C();
};

C::~C() { }

// Base (D2) dtor for C: calls B's base dtor.

// CIR: cir.func{{.*}} @_ZN1CD2Ev
// CIR:   %[[B:.*]] = cir.base_class_addr %[[THIS:.*]] : !cir.ptr<!rec_C> nonnull [0] -> !cir.ptr<!rec_B>
// CIR:   cir.call @_ZN1BD2Ev(%[[B]])

// LLVM: define{{.*}} void @_ZN1CD2Ev
// LLVM:   call void @_ZN1BD2Ev

// OGCG: define{{.*}} @_ZN1CD2Ev
// OGCG:   call void @_ZN1BD2Ev

// Complete (D1) dtor for C: just an alias because there are no virtual bases.

// CIR: cir.func{{.*}} @_ZN1CD1Ev(!cir.ptr<!rec_C>) alias(@_ZN1CD2Ev)
// This is defined above for LLVM and OGCG.


// Deleting (D0) dtor for C: defers to the complete dtor but also calls operator delete.

// CIR: cir.func{{.*}} @_ZN1CD0Ev
// CIR:   cir.call @_ZN1CD1Ev(%[[THIS:.*]]) nothrow : (!cir.ptr<!rec_C>) -> ()
// CIR:   %[[THIS_VOID:.*]] = cir.cast bitcast %[[THIS]] : !cir.ptr<!rec_C> -> !cir.ptr<!void>
// CIR:   %[[SIZE:.*]] = cir.const #cir.int<16>
// CIR:   cir.call @_ZdlPvm(%[[THIS_VOID]], %[[SIZE]])

// LLVM: define{{.*}} void @_ZN1CD0Ev
// LLVM:   call void @_ZN1CD1Ev(ptr %[[THIS:.*]])
// LLVM:   call void @_ZdlPvm(ptr %[[THIS]], i64 16)

// OGCG: define{{.*}} @_ZN1CD0Ev
// OGCG:   call void @_ZN1CD1Ev(ptr{{.*}} %[[THIS:.*]])
// OGCG:   call void @_ZdlPvm(ptr{{.*}} %[[THIS]], i64{{.*}} 16)

namespace PR12798 {
  // A qualified call to a base class destructor should not undergo virtual
  // dispatch. Template instantiation used to lose the qualifier.
  struct A { virtual ~A(); };
  template<typename T> void f(T *p) { p->A::~A(); }

  // CIR: cir.func{{.*}} @_ZN7PR127981fINS_1AEEEvPT_
  // CIR:   cir.call @_ZN7PR127981AD1Ev

  // LLVM: define{{.*}} @_ZN7PR127981fINS_1AEEEvPT_
  // LLVM:   call void @_ZN7PR127981AD1Ev

  // OGCG: define{{.*}} @_ZN7PR127981fINS_1AEEEvPT_
  // OGCG:   call void @_ZN7PR127981AD1Ev

  template void f(A*);
}