xref: /aosp_15_r20/external/clang/test/CodeGenCXX/pointers-to-data-members.cpp (revision 67e74705e28f6214e480b399dd47ea732279e315)
1*67e74705SXin Li // RUN: %clang_cc1 %s -emit-llvm -o %t.ll -triple=x86_64-apple-darwin10
2*67e74705SXin Li // RUN: FileCheck %s < %t.ll
3*67e74705SXin Li // RUN: FileCheck -check-prefix=CHECK-GLOBAL %s < %t.ll
4*67e74705SXin Li 
5*67e74705SXin Li struct A { int a; int b; };
6*67e74705SXin Li struct B { int b; };
7*67e74705SXin Li struct C : B, A { };
8*67e74705SXin Li 
9*67e74705SXin Li // Zero init.
10*67e74705SXin Li namespace ZeroInit {
11*67e74705SXin Li   // CHECK-GLOBAL: @_ZN8ZeroInit1aE = global i64 -1
12*67e74705SXin Li   int A::* a;
13*67e74705SXin Li 
14*67e74705SXin Li   // CHECK-GLOBAL: @_ZN8ZeroInit2aaE = global [2 x i64] [i64 -1, i64 -1]
15*67e74705SXin Li   int A::* aa[2];
16*67e74705SXin Li 
17*67e74705SXin Li   // CHECK-GLOBAL: @_ZN8ZeroInit3aaaE = global [2 x [2 x i64]] {{\[}}[2 x i64] [i64 -1, i64 -1], [2 x i64] [i64 -1, i64 -1]]
18*67e74705SXin Li   int A::* aaa[2][2];
19*67e74705SXin Li 
20*67e74705SXin Li   // CHECK-GLOBAL: @_ZN8ZeroInit1bE = global i64 -1,
21*67e74705SXin Li   int A::* b = 0;
22*67e74705SXin Li 
23*67e74705SXin Li   // CHECK-GLOBAL: @_ZN8ZeroInit2saE = internal global %struct.anon { i64 -1 }
24*67e74705SXin Li   struct {
25*67e74705SXin Li     int A::*a;
26*67e74705SXin Li   } sa;
test_sa()27*67e74705SXin Li   void test_sa() { (void) sa; } // force emission
28*67e74705SXin Li 
29*67e74705SXin Li   // CHECK-GLOBAL: @_ZN8ZeroInit3ssaE = internal
30*67e74705SXin Li   // CHECK-GLOBAL: [2 x i64] [i64 -1, i64 -1]
31*67e74705SXin Li   struct {
32*67e74705SXin Li     int A::*aa[2];
33*67e74705SXin Li   } ssa[2];
test_ssa()34*67e74705SXin Li   void test_ssa() { (void) ssa; }
35*67e74705SXin Li 
36*67e74705SXin Li   // CHECK-GLOBAL: @_ZN8ZeroInit2ssE = internal global %struct.anon.1 { %struct.anon.2 { i64 -1 } }
37*67e74705SXin Li   struct {
38*67e74705SXin Li     struct {
39*67e74705SXin Li       int A::*pa;
40*67e74705SXin Li     } s;
41*67e74705SXin Li   } ss;
test_ss()42*67e74705SXin Li   void test_ss() { (void) ss; }
43*67e74705SXin Li 
44*67e74705SXin Li   struct A {
45*67e74705SXin Li     int A::*a;
46*67e74705SXin Li     int b;
47*67e74705SXin Li   };
48*67e74705SXin Li 
49*67e74705SXin Li   struct B {
50*67e74705SXin Li     A a[10];
51*67e74705SXin Li     char c;
52*67e74705SXin Li     int B::*b;
53*67e74705SXin Li   };
54*67e74705SXin Li 
55*67e74705SXin Li   struct C : A, B { int j; };
56*67e74705SXin Li   // CHECK-GLOBAL: @_ZN8ZeroInit1cE = global {{%.*}} <{ %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::B" { [10 x %"struct.ZeroInit::A"] [%"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }], i8 0, i64 -1 }, i32 0, [4 x i8] zeroinitializer }>, align 8
57*67e74705SXin Li   C c;
58*67e74705SXin Li }
59*67e74705SXin Li 
60*67e74705SXin Li // PR5674
61*67e74705SXin Li namespace PR5674 {
62*67e74705SXin Li   // CHECK-GLOBAL: @_ZN6PR56742pbE = global i64 4
63*67e74705SXin Li   int A::*pb = &A::b;
64*67e74705SXin Li }
65*67e74705SXin Li 
66*67e74705SXin Li // Casts.
67*67e74705SXin Li namespace Casts {
68*67e74705SXin Li 
69*67e74705SXin Li int A::*pa;
70*67e74705SXin Li int C::*pc;
71*67e74705SXin Li 
f()72*67e74705SXin Li void f() {
73*67e74705SXin Li   // CHECK:      store i64 -1, i64* @_ZN5Casts2paE
74*67e74705SXin Li   pa = 0;
75*67e74705SXin Li 
76*67e74705SXin Li   // CHECK-NEXT: [[TMP:%.*]] = load i64, i64* @_ZN5Casts2paE, align 8
77*67e74705SXin Li   // CHECK-NEXT: [[ADJ:%.*]] = add nsw i64 [[TMP]], 4
78*67e74705SXin Li   // CHECK-NEXT: [[ISNULL:%.*]] = icmp eq i64 [[TMP]], -1
79*67e74705SXin Li   // CHECK-NEXT: [[RES:%.*]] = select i1 [[ISNULL]], i64 [[TMP]], i64 [[ADJ]]
80*67e74705SXin Li   // CHECK-NEXT: store i64 [[RES]], i64* @_ZN5Casts2pcE
81*67e74705SXin Li   pc = pa;
82*67e74705SXin Li 
83*67e74705SXin Li   // CHECK-NEXT: [[TMP:%.*]] = load i64, i64* @_ZN5Casts2pcE, align 8
84*67e74705SXin Li   // CHECK-NEXT: [[ADJ:%.*]] = sub nsw i64 [[TMP]], 4
85*67e74705SXin Li   // CHECK-NEXT: [[ISNULL:%.*]] = icmp eq i64 [[TMP]], -1
86*67e74705SXin Li   // CHECK-NEXT: [[RES:%.*]] = select i1 [[ISNULL]], i64 [[TMP]], i64 [[ADJ]]
87*67e74705SXin Li   // CHECK-NEXT: store i64 [[RES]], i64* @_ZN5Casts2paE
88*67e74705SXin Li   pa = static_cast<int A::*>(pc);
89*67e74705SXin Li }
90*67e74705SXin Li 
91*67e74705SXin Li }
92*67e74705SXin Li 
93*67e74705SXin Li // Comparisons
94*67e74705SXin Li namespace Comparisons {
f()95*67e74705SXin Li   void f() {
96*67e74705SXin Li     int A::*a;
97*67e74705SXin Li 
98*67e74705SXin Li     // CHECK: icmp ne i64 {{.*}}, -1
99*67e74705SXin Li     if (a) { }
100*67e74705SXin Li 
101*67e74705SXin Li     // CHECK: icmp ne i64 {{.*}}, -1
102*67e74705SXin Li     if (a != 0) { }
103*67e74705SXin Li 
104*67e74705SXin Li     // CHECK: icmp ne i64 -1, {{.*}}
105*67e74705SXin Li     if (0 != a) { }
106*67e74705SXin Li 
107*67e74705SXin Li     // CHECK: icmp eq i64 {{.*}}, -1
108*67e74705SXin Li     if (a == 0) { }
109*67e74705SXin Li 
110*67e74705SXin Li     // CHECK: icmp eq i64 -1, {{.*}}
111*67e74705SXin Li     if (0 == a) { }
112*67e74705SXin Li   }
113*67e74705SXin Li }
114*67e74705SXin Li 
115*67e74705SXin Li namespace ValueInit {
116*67e74705SXin Li 
117*67e74705SXin Li struct A {
118*67e74705SXin Li   int A::*a;
119*67e74705SXin Li 
120*67e74705SXin Li   char c;
121*67e74705SXin Li 
122*67e74705SXin Li   A();
123*67e74705SXin Li };
124*67e74705SXin Li 
125*67e74705SXin Li // CHECK-LABEL: define void @_ZN9ValueInit1AC2Ev(%"struct.ValueInit::A"* %this) unnamed_addr
126*67e74705SXin Li // CHECK: store i64 -1, i64*
127*67e74705SXin Li // CHECK: ret void
A()128*67e74705SXin Li A::A() : a() {}
129*67e74705SXin Li 
130*67e74705SXin Li }
131*67e74705SXin Li 
132*67e74705SXin Li namespace VirtualBases {
133*67e74705SXin Li 
134*67e74705SXin Li struct A {
135*67e74705SXin Li   char c;
136*67e74705SXin Li   int A::*i;
137*67e74705SXin Li };
138*67e74705SXin Li 
139*67e74705SXin Li // CHECK-GLOBAL: @_ZN12VirtualBases1bE = global %"struct.VirtualBases::B" { i32 (...)** null, %"struct.VirtualBases::A" { i8 0, i64 -1 } }, align 8
140*67e74705SXin Li struct B : virtual A { };
141*67e74705SXin Li B b;
142*67e74705SXin Li 
143*67e74705SXin Li // CHECK-GLOBAL: @_ZN12VirtualBases1cE = global %"struct.VirtualBases::C" { i32 (...)** null, i64 -1, %"struct.VirtualBases::A" { i8 0, i64 -1 } }, align 8
144*67e74705SXin Li struct C : virtual A { int A::*i; };
145*67e74705SXin Li C c;
146*67e74705SXin Li 
147*67e74705SXin Li // CHECK-GLOBAL: @_ZN12VirtualBases1dE = global %"struct.VirtualBases::D" { %"struct.VirtualBases::C.base" { i32 (...)** null, i64 -1 }, i64 -1, %"struct.VirtualBases::A" { i8 0, i64 -1 } }, align 8
148*67e74705SXin Li struct D : C { int A::*i; };
149*67e74705SXin Li D d;
150*67e74705SXin Li 
151*67e74705SXin Li }
152*67e74705SXin Li 
153*67e74705SXin Li namespace Test1 {
154*67e74705SXin Li 
155*67e74705SXin Li // Don't crash when A contains a bit-field.
156*67e74705SXin Li struct A {
157*67e74705SXin Li   int A::* a;
158*67e74705SXin Li   int b : 10;
159*67e74705SXin Li };
160*67e74705SXin Li A a;
161*67e74705SXin Li 
162*67e74705SXin Li }
163*67e74705SXin Li 
164*67e74705SXin Li namespace BoolPtrToMember {
165*67e74705SXin Li   struct X {
166*67e74705SXin Li     bool member;
167*67e74705SXin Li   };
168*67e74705SXin Li 
169*67e74705SXin Li   // CHECK-LABEL: define dereferenceable({{[0-9]+}}) i8* @_ZN15BoolPtrToMember1fERNS_1XEMS0_b
f(X & x,bool X::* member)170*67e74705SXin Li   bool &f(X &x, bool X::*member) {
171*67e74705SXin Li     // CHECK: {{bitcast.* to i8\*}}
172*67e74705SXin Li     // CHECK-NEXT: getelementptr inbounds i8, i8*
173*67e74705SXin Li     // CHECK-NEXT: ret i8*
174*67e74705SXin Li     return x.*member;
175*67e74705SXin Li   }
176*67e74705SXin Li }
177*67e74705SXin Li 
178*67e74705SXin Li namespace PR8507 {
179*67e74705SXin Li 
180*67e74705SXin Li struct S;
f(S * p,double S::* pm)181*67e74705SXin Li void f(S* p, double S::*pm) {
182*67e74705SXin Li   if (0 < p->*pm) {
183*67e74705SXin Li   }
184*67e74705SXin Li }
185*67e74705SXin Li 
186*67e74705SXin Li }
187*67e74705SXin Li 
188*67e74705SXin Li namespace test4 {
189*67e74705SXin Li   struct A             { int A_i; };
190*67e74705SXin Li   struct B : virtual A { int A::*B_p; };
191*67e74705SXin Li   struct C : virtual B { int    *C_p; };
192*67e74705SXin Li   struct D :         C { int    *D_p; };
193*67e74705SXin Li 
194*67e74705SXin Li   // CHECK-GLOBAL: @_ZN5test41dE = global %"struct.test4::D" { %"struct.test4::C.base" zeroinitializer, i32* null, %"struct.test4::B.base" { i32 (...)** null, i64 -1 }, %"struct.test4::A" zeroinitializer }, align 8
195*67e74705SXin Li   D d;
196*67e74705SXin Li }
197*67e74705SXin Li 
198*67e74705SXin Li namespace PR11487 {
199*67e74705SXin Li   union U
200*67e74705SXin Li   {
201*67e74705SXin Li     int U::* mptr;
202*67e74705SXin Li     char x[16];
203*67e74705SXin Li   } x;
204*67e74705SXin Li   // CHECK-GLOBAL: @_ZN7PR114871xE = global %"union.PR11487::U" { i64 -1, [8 x i8] zeroinitializer }, align 8
205*67e74705SXin Li 
206*67e74705SXin Li }
207*67e74705SXin Li 
208*67e74705SXin Li namespace PR13097 {
209*67e74705SXin Li   struct X { int x; X(const X&); };
210*67e74705SXin Li   struct A {
211*67e74705SXin Li     int qq;
212*67e74705SXin Li       X x;
213*67e74705SXin Li   };
214*67e74705SXin Li   A f();
g()215*67e74705SXin Li   X g() { return f().*&A::x; }
216*67e74705SXin Li   // CHECK-LABEL: define void @_ZN7PR130971gEv
217*67e74705SXin Li   // CHECK: call void @_ZN7PR130971fEv
218*67e74705SXin Li   // CHECK-NOT: memcpy
219*67e74705SXin Li   // CHECK: call void @_ZN7PR130971XC1ERKS0_
220*67e74705SXin Li }
221*67e74705SXin Li 
222*67e74705SXin Li namespace PR21089 {
223*67e74705SXin Li struct A {
224*67e74705SXin Li   bool : 1;
225*67e74705SXin Li   int A::*x;
226*67e74705SXin Li   bool y;
227*67e74705SXin Li   A();
228*67e74705SXin Li };
229*67e74705SXin Li struct B : A {
230*67e74705SXin Li };
231*67e74705SXin Li B b;
232*67e74705SXin Li // CHECK-GLOBAL: @_ZN7PR210891bE = global %"struct.PR21089::B" { %"struct.PR21089::A.base" <{ i8 0, [7 x i8] zeroinitializer, i64 -1, i8 0 }>, [7 x i8] zeroinitializer }, align 8
233*67e74705SXin Li }
234*67e74705SXin Li 
235*67e74705SXin Li namespace PR21282 {
236*67e74705SXin Li union U {
237*67e74705SXin Li   int U::*x;
238*67e74705SXin Li   long y[2];
239*67e74705SXin Li };
240*67e74705SXin Li U u;
241*67e74705SXin Li // CHECK-GLOBAL: @_ZN7PR212821uE = global %"union.PR21282::U" { i64 -1, [8 x i8] zeroinitializer }, align 8
242*67e74705SXin Li }
243*67e74705SXin Li 
244*67e74705SXin Li namespace FlexibleArrayMember {
245*67e74705SXin Li struct S {
246*67e74705SXin Li   int S::*x[];
247*67e74705SXin Li };
248*67e74705SXin Li S s;
249*67e74705SXin Li // CHECK-GLOBAL: @_ZN19FlexibleArrayMember1sE = global %"struct.FlexibleArrayMember::S" zeroinitializer, align 8
250*67e74705SXin Li }
251*67e74705SXin Li 
252*67e74705SXin Li namespace IndirectPDM {
253*67e74705SXin Li union U {
254*67e74705SXin Li   union {
255*67e74705SXin Li     int U::*m;
256*67e74705SXin Li   };
257*67e74705SXin Li };
258*67e74705SXin Li U u;
259*67e74705SXin Li // CHECK-GLOBAL: @_ZN11IndirectPDM1uE = global %"union.IndirectPDM::U" { %union.anon { i64 -1 } }, align 8
260*67e74705SXin Li }
261