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*);
}
|