1*67e74705SXin Li // RUN: %clang_cc1 %s -I%S -triple=x86_64-apple-darwin10 -fstrict-vtable-pointers -disable-llvm-optzns -O2 -emit-llvm -o %t.ll
2*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-CTORS %s < %t.ll
3*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-NEW %s < %t.ll
4*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-DTORS %s < %t.ll
5*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-LINK-REQ %s < %t.ll
6*67e74705SXin Li
7*67e74705SXin Li typedef __typeof__(sizeof(0)) size_t;
8*67e74705SXin Li void *operator new(size_t, void*) throw();
9*67e74705SXin Li
10*67e74705SXin Li struct NotTrivialDtor {
11*67e74705SXin Li ~NotTrivialDtor();
12*67e74705SXin Li };
13*67e74705SXin Li
14*67e74705SXin Li struct DynamicBase1 {
15*67e74705SXin Li NotTrivialDtor obj;
16*67e74705SXin Li virtual void foo();
17*67e74705SXin Li };
18*67e74705SXin Li
19*67e74705SXin Li struct DynamicDerived : DynamicBase1 {
20*67e74705SXin Li void foo();
21*67e74705SXin Li };
22*67e74705SXin Li
23*67e74705SXin Li struct DynamicBase2 {
24*67e74705SXin Li virtual void bar();
~DynamicBase2DynamicBase225*67e74705SXin Li ~DynamicBase2() {
26*67e74705SXin Li bar();
27*67e74705SXin Li }
28*67e74705SXin Li };
29*67e74705SXin Li
30*67e74705SXin Li struct DynamicDerivedMultiple : DynamicBase1, DynamicBase2 {
31*67e74705SXin Li virtual void foo();
32*67e74705SXin Li virtual void bar();
33*67e74705SXin Li };
34*67e74705SXin Li
35*67e74705SXin Li struct StaticBase {
36*67e74705SXin Li NotTrivialDtor obj;
37*67e74705SXin Li void bar();
38*67e74705SXin Li };
39*67e74705SXin Li
40*67e74705SXin Li struct DynamicFromStatic : StaticBase {
41*67e74705SXin Li virtual void bar();
42*67e74705SXin Li };
43*67e74705SXin Li
44*67e74705SXin Li struct DynamicFromVirtualStatic1 : virtual StaticBase {
45*67e74705SXin Li };
46*67e74705SXin Li
47*67e74705SXin Li struct DynamicFromVirtualStatic2 : virtual StaticBase {
48*67e74705SXin Li };
49*67e74705SXin Li
50*67e74705SXin Li struct DynamicFrom2Virtuals :
51*67e74705SXin Li DynamicFromVirtualStatic1,
52*67e74705SXin Li DynamicFromVirtualStatic2 {
53*67e74705SXin Li };
54*67e74705SXin Li
55*67e74705SXin Li // CHECK-NEW-LABEL: define void @_Z12LocalObjectsv()
56*67e74705SXin Li // CHECK-NEW-NOT: @llvm.invariant.group.barrier(
57*67e74705SXin Li // CHECK-NEW-LABEL: }
LocalObjects()58*67e74705SXin Li void LocalObjects() {
59*67e74705SXin Li DynamicBase1 DB;
60*67e74705SXin Li DB.foo();
61*67e74705SXin Li DynamicDerived DD;
62*67e74705SXin Li DD.foo();
63*67e74705SXin Li
64*67e74705SXin Li DynamicBase2 DB2;
65*67e74705SXin Li DB2.bar();
66*67e74705SXin Li
67*67e74705SXin Li StaticBase SB;
68*67e74705SXin Li SB.bar();
69*67e74705SXin Li
70*67e74705SXin Li DynamicDerivedMultiple DDM;
71*67e74705SXin Li DDM.foo();
72*67e74705SXin Li DDM.bar();
73*67e74705SXin Li
74*67e74705SXin Li DynamicFromStatic DFS;
75*67e74705SXin Li DFS.bar();
76*67e74705SXin Li DynamicFromVirtualStatic1 DFVS1;
77*67e74705SXin Li DFVS1.bar();
78*67e74705SXin Li DynamicFrom2Virtuals DF2V;
79*67e74705SXin Li DF2V.bar();
80*67e74705SXin Li }
81*67e74705SXin Li
82*67e74705SXin Li struct DynamicFromVirtualStatic1;
83*67e74705SXin Li // CHECK-CTORS-LABEL: define linkonce_odr void @_ZN25DynamicFromVirtualStatic1C1Ev
84*67e74705SXin Li // CHECK-CTORS-NOT: @llvm.invariant.group.barrier(
85*67e74705SXin Li // CHECK-CTORS-LABEL: }
86*67e74705SXin Li
87*67e74705SXin Li struct DynamicFrom2Virtuals;
88*67e74705SXin Li // CHECK-CTORS-LABEL: define linkonce_odr void @_ZN20DynamicFrom2VirtualsC1Ev
89*67e74705SXin Li // CHECK-CTORS: call i8* @llvm.invariant.group.barrier(
90*67e74705SXin Li // CHECK-CTORS-LABEL: }
91*67e74705SXin Li
92*67e74705SXin Li
93*67e74705SXin Li // CHECK-NEW-LABEL: define void @_Z9Pointers1v()
94*67e74705SXin Li // CHECK-NEW-NOT: @llvm.invariant.group.barrier(
95*67e74705SXin Li // CHECK-NEW-LABEL: call void @_ZN12DynamicBase1C1Ev(
96*67e74705SXin Li
97*67e74705SXin Li // CHECK-NEW: %[[THIS3:.*]] = call i8* @llvm.invariant.group.barrier(i8* %[[THIS2:.*]])
98*67e74705SXin Li // CHECK-NEW: %[[THIS4:.*]] = bitcast i8* %[[THIS3]] to %[[DynamicDerived:.*]]*
99*67e74705SXin Li // CHECK-NEW: call void @_ZN14DynamicDerivedC1Ev(%[[DynamicDerived:.*]]* %[[THIS4]])
100*67e74705SXin Li // CHECK-NEW-LABEL: }
Pointers1()101*67e74705SXin Li void Pointers1() {
102*67e74705SXin Li DynamicBase1 *DB = new DynamicBase1;
103*67e74705SXin Li DB->foo();
104*67e74705SXin Li
105*67e74705SXin Li DynamicDerived *DD = new (DB) DynamicDerived;
106*67e74705SXin Li DD->foo();
107*67e74705SXin Li DD->~DynamicDerived();
108*67e74705SXin Li }
109*67e74705SXin Li
110*67e74705SXin Li // CHECK-NEW-LABEL: define void @_Z14HackingObjectsv()
111*67e74705SXin Li // CHECK-NEW: call void @_ZN12DynamicBase1C1Ev
112*67e74705SXin Li // CHECK-NEW: call i8* @llvm.invariant.group.barrier(
113*67e74705SXin Li // CHECK-NEW: call void @_ZN14DynamicDerivedC1Ev(
114*67e74705SXin Li // CHECK-NEW: call i8* @llvm.invariant.group.barrier(
115*67e74705SXin Li // CHECK-NEW: call void @_ZN12DynamicBase1C1Ev(
116*67e74705SXin Li // CHECK-NEW-LABEL: }
HackingObjects()117*67e74705SXin Li void HackingObjects() {
118*67e74705SXin Li DynamicBase1 DB;
119*67e74705SXin Li DB.foo();
120*67e74705SXin Li
121*67e74705SXin Li DynamicDerived *DB2 = new (&DB) DynamicDerived;
122*67e74705SXin Li // Using DB now is prohibited.
123*67e74705SXin Li DB2->foo();
124*67e74705SXin Li DB2->~DynamicDerived();
125*67e74705SXin Li
126*67e74705SXin Li // We have to get back to the previous type to avoid calling wrong destructor
127*67e74705SXin Li new (&DB) DynamicBase1;
128*67e74705SXin Li DB.foo();
129*67e74705SXin Li }
130*67e74705SXin Li
131*67e74705SXin Li /*** Testing Constructors ***/
132*67e74705SXin Li struct DynamicBase1;
133*67e74705SXin Li // CHECK-CTORS-LABEL: define linkonce_odr void @_ZN12DynamicBase1C2Ev(
134*67e74705SXin Li // CHECK-CTORS-NOT: call i8* @llvm.invariant.group.barrier(
135*67e74705SXin Li // CHECK-CTORS-LABEL: }
136*67e74705SXin Li
137*67e74705SXin Li
138*67e74705SXin Li struct DynamicDerived;
139*67e74705SXin Li
140*67e74705SXin Li // CHECK-CTORS-LABEL: define linkonce_odr void @_ZN14DynamicDerivedC2Ev(
141*67e74705SXin Li // CHECK-CTORS: %[[THIS0:.*]] = load %[[DynamicDerived:.*]]*, %[[DynamicDerived]]** {{.*}}
142*67e74705SXin Li // CHECK-CTORS: %[[THIS1:.*]] = bitcast %[[DynamicDerived:.*]]* %[[THIS0]] to i8*
143*67e74705SXin Li // CHECK-CTORS: %[[THIS2:.*]] = call i8* @llvm.invariant.group.barrier(i8* %[[THIS1:.*]])
144*67e74705SXin Li // CHECK-CTORS: %[[THIS3:.*]] = bitcast i8* %[[THIS2]] to %[[DynamicDerived]]*
145*67e74705SXin Li // CHECK-CTORS: %[[THIS4:.*]] = bitcast %[[DynamicDerived]]* %[[THIS3]] to %[[DynamicBase:.*]]*
146*67e74705SXin Li // CHECK-CTORS: call void @_ZN12DynamicBase1C2Ev(%[[DynamicBase]]* %[[THIS4]])
147*67e74705SXin Li
148*67e74705SXin Li // CHECK-CTORS: %[[THIS5:.*]] = bitcast %struct.DynamicDerived* %[[THIS0]] to i32 (...)***
149*67e74705SXin Li // CHECK-CTORS: store {{.*}} %[[THIS5]]
150*67e74705SXin Li // CHECK-CTORS-LABEL: }
151*67e74705SXin Li
152*67e74705SXin Li struct DynamicDerivedMultiple;
153*67e74705SXin Li // CHECK-CTORS-LABEL: define linkonce_odr void @_ZN22DynamicDerivedMultipleC2Ev(
154*67e74705SXin Li
155*67e74705SXin Li // CHECK-CTORS: %[[THIS0:.*]] = load %[[CLASS:.*]]*, %[[CLASS]]** {{.*}}
156*67e74705SXin Li // CHECK-CTORS: %[[THIS1:.*]] = bitcast %[[CLASS:.*]]* %[[THIS0]] to i8*
157*67e74705SXin Li // CHECK-CTORS: %[[THIS2:.*]] = call i8* @llvm.invariant.group.barrier(i8* %[[THIS1]])
158*67e74705SXin Li // CHECK-CTORS: %[[THIS3:.*]] = bitcast i8* %[[THIS2]] to %[[CLASS]]*
159*67e74705SXin Li // CHECK-CTORS: %[[THIS4:.*]] = bitcast %[[CLASS]]* %[[THIS3]] to %[[BASE_CLASS:.*]]*
160*67e74705SXin Li // CHECK-CTORS: call void @_ZN12DynamicBase1C2Ev(%[[BASE_CLASS]]* %[[THIS4]])
161*67e74705SXin Li
162*67e74705SXin Li // CHECK-CTORS: call i8* @llvm.invariant.group.barrier(
163*67e74705SXin Li
164*67e74705SXin Li // CHECK-CTORS: call void @_ZN12DynamicBase2C2Ev(
165*67e74705SXin Li // CHECK-CTORS-NOT: @llvm.invariant.group.barrier
166*67e74705SXin Li
167*67e74705SXin Li
168*67e74705SXin Li // CHECK-CTORS: %[[THIS10:.*]] = bitcast %struct.DynamicDerivedMultiple* %[[THIS0]] to i32 (...)***
169*67e74705SXin Li // CHECK-CTORS: store {{.*}} @_ZTV22DynamicDerivedMultiple, i32 0, i32 2) {{.*}} %[[THIS10]]
170*67e74705SXin Li // CHECK-CTORS: %[[THIS11:.*]] = bitcast %struct.DynamicDerivedMultiple* %[[THIS0]] to i8*
171*67e74705SXin Li // CHECK-CTORS: %[[THIS_ADD:.*]] = getelementptr inbounds i8, i8* %[[THIS11]], i64 16
172*67e74705SXin Li // CHECK-CTORS: %[[THIS12:.*]] = bitcast i8* %[[THIS_ADD]] to i32 (...)***
173*67e74705SXin Li
174*67e74705SXin Li
175*67e74705SXin Li // CHECK-CTORS: store {{.*}} @_ZTV22DynamicDerivedMultiple, i32 0, i32 6) {{.*}} %[[THIS12]]
176*67e74705SXin Li // CHECK-CTORS-LABEL: }
177*67e74705SXin Li
178*67e74705SXin Li struct DynamicFromStatic;
179*67e74705SXin Li // CHECK-CTORS-LABEL: define linkonce_odr void @_ZN17DynamicFromStaticC2Ev(
180*67e74705SXin Li // CHECK-CTORS-NOT: @llvm.invariant.group.barrier(
181*67e74705SXin Li // CHECK-CTORS-LABEL: }
182*67e74705SXin Li
183*67e74705SXin Li
184*67e74705SXin Li /** DTORS **/
185*67e74705SXin Li // CHECK-DTORS-LABEL: define linkonce_odr void @_ZN10StaticBaseD2Ev(
186*67e74705SXin Li // CHECK-DTORS-NOT: call i8* @llvm.invariant.group.barrier(
187*67e74705SXin Li // CHECK-DTORS-LABEL: }
188*67e74705SXin Li
189*67e74705SXin Li
190*67e74705SXin Li // CHECK-DTORS-LABEL: define linkonce_odr void @_ZN25DynamicFromVirtualStatic2D2Ev(
191*67e74705SXin Li // CHECK-DTORS-NOT: invariant.barrier
192*67e74705SXin Li // CHECK-DTORS-LABEL: }
193*67e74705SXin Li
194*67e74705SXin Li // CHECK-DTORS-LABEL: define linkonce_odr void @_ZN17DynamicFromStaticD2Ev
195*67e74705SXin Li // CHECK-DTORS-NOT: call i8* @llvm.invariant.group.barrier(
196*67e74705SXin Li // CHECK-DTORS-LABEL: }
197*67e74705SXin Li
198*67e74705SXin Li
199*67e74705SXin Li // CHECK-DTORS-LABEL: define linkonce_odr void @_ZN22DynamicDerivedMultipleD2Ev(
200*67e74705SXin Li
201*67e74705SXin Li // CHECK-DTORS-LABEL: define linkonce_odr void @_ZN12DynamicBase2D2Ev(
202*67e74705SXin Li // CHECK-DTORS: call i8* @llvm.invariant.group.barrier(
203*67e74705SXin Li // CHECK-DTORS-LABEL: }
204*67e74705SXin Li
205*67e74705SXin Li // CHECK-DTORS-LABEL: define linkonce_odr void @_ZN12DynamicBase1D2Ev
206*67e74705SXin Li // CHECK-DTORS: call i8* @llvm.invariant.group.barrier(
207*67e74705SXin Li // CHECK-DTORS-LABEL: }
208*67e74705SXin Li
209*67e74705SXin Li // CHECK-DTORS-LABEL: define linkonce_odr void @_ZN14DynamicDerivedD2Ev
210*67e74705SXin Li // CHECK-DTORS-NOT: call i8* @llvm.invariant.group.barrier(
211*67e74705SXin Li // CHECK-DTORS-LABEL: }
212*67e74705SXin Li
213*67e74705SXin Li
214*67e74705SXin Li // CHECK-LINK-REQ: !llvm.module.flags = !{![[FIRST:.*]], ![[SEC:.*]]{{.*}}}
215*67e74705SXin Li
216*67e74705SXin Li // CHECK-LINK-REQ: ![[FIRST]] = !{i32 1, !"StrictVTablePointers", i32 1}
217*67e74705SXin Li // CHECK-LINK-REQ: ![[SEC]] = !{i32 3, !"StrictVTablePointersRequirement", ![[META:.*]]}
218*67e74705SXin Li // CHECK-LINK-REQ: ![[META]] = !{!"StrictVTablePointers", i32 1}
219*67e74705SXin Li
220