1*67e74705SXin Li // RUN: %clang_cc1 -triple x86_64-unknown-unknown -emit-llvm -o - %s | FileCheck %s
2*67e74705SXin Li
3*67e74705SXin Li // Basic base class test.
4*67e74705SXin Li struct f0_s0 { unsigned a; };
5*67e74705SXin Li struct f0_s1 : public f0_s0 { void *b; };
6*67e74705SXin Li // CHECK-LABEL: define void @_Z2f05f0_s1(i32 %a0.coerce0, i8* %a0.coerce1)
f0(f0_s1 a0)7*67e74705SXin Li void f0(f0_s1 a0) { }
8*67e74705SXin Li
9*67e74705SXin Li // Check with two eight-bytes in base class.
10*67e74705SXin Li struct f1_s0 { unsigned a; unsigned b; float c; };
11*67e74705SXin Li struct f1_s1 : public f1_s0 { float d;};
12*67e74705SXin Li // CHECK-LABEL: define void @_Z2f15f1_s1(i64 %a0.coerce0, <2 x float> %a0.coerce1)
f1(f1_s1 a0)13*67e74705SXin Li void f1(f1_s1 a0) { }
14*67e74705SXin Li
15*67e74705SXin Li // Check with two eight-bytes in base class and merge.
16*67e74705SXin Li struct f2_s0 { unsigned a; unsigned b; float c; };
17*67e74705SXin Li struct f2_s1 : public f2_s0 { char d;};
18*67e74705SXin Li // CHECK-LABEL: define void @_Z2f25f2_s1(i64 %a0.coerce0, i64 %a0.coerce1)
f2(f2_s1 a0)19*67e74705SXin Li void f2(f2_s1 a0) { }
20*67e74705SXin Li
21*67e74705SXin Li // PR5831
22*67e74705SXin Li // CHECK-LABEL: define void @_Z2f34s3_1(i64 %x.coerce)
23*67e74705SXin Li struct s3_0 {};
24*67e74705SXin Li struct s3_1 { struct s3_0 a; long b; };
f3(struct s3_1 x)25*67e74705SXin Li void f3(struct s3_1 x) {}
26*67e74705SXin Li
27*67e74705SXin Li // CHECK-LABEL: define i64 @_Z4f4_0M2s4i(i64 %a)
28*67e74705SXin Li // CHECK: define {{.*}} @_Z4f4_1M2s4FivE(i64 %a.coerce0, i64 %a.coerce1)
29*67e74705SXin Li struct s4 {};
30*67e74705SXin Li typedef int s4::* s4_mdp;
31*67e74705SXin Li typedef int (s4::*s4_mfp)();
f4_0(s4_mdp a)32*67e74705SXin Li s4_mdp f4_0(s4_mdp a) { return a; }
f4_1(s4_mfp a)33*67e74705SXin Li s4_mfp f4_1(s4_mfp a) { return a; }
34*67e74705SXin Li
35*67e74705SXin Li // A struct with <= one eightbyte before a member data pointer should still
36*67e74705SXin Li // be allowed in registers.
37*67e74705SXin Li // CHECK-LABEL: define void @{{.*}}f_struct_with_mdp{{.*}}(i8* %a.coerce0, i64 %a.coerce1)
38*67e74705SXin Li struct struct_with_mdp { char *a; s4_mdp b; };
f_struct_with_mdp(struct_with_mdp a)39*67e74705SXin Li void f_struct_with_mdp(struct_with_mdp a) { (void)a; }
40*67e74705SXin Li
41*67e74705SXin Li // A struct with anything before a member function will be too big and
42*67e74705SXin Li // goes in memory.
43*67e74705SXin Li // CHECK-LABEL: define void @{{.*}}f_struct_with_mfp_0{{.*}}(%struct{{.*}} byval align 8 %a)
44*67e74705SXin Li struct struct_with_mfp_0 { char a; s4_mfp b; };
f_struct_with_mfp_0(struct_with_mfp_0 a)45*67e74705SXin Li void f_struct_with_mfp_0(struct_with_mfp_0 a) { (void)a; }
46*67e74705SXin Li
47*67e74705SXin Li // CHECK-LABEL: define void @{{.*}}f_struct_with_mfp_1{{.*}}(%struct{{.*}} byval align 8 %a)
48*67e74705SXin Li struct struct_with_mfp_1 { void *a; s4_mfp b; };
f_struct_with_mfp_1(struct_with_mfp_1 a)49*67e74705SXin Li void f_struct_with_mfp_1(struct_with_mfp_1 a) { (void)a; }
50*67e74705SXin Li
51*67e74705SXin Li namespace PR7523 {
52*67e74705SXin Li struct StringRef {
53*67e74705SXin Li char *a;
54*67e74705SXin Li };
55*67e74705SXin Li
56*67e74705SXin Li void AddKeyword(StringRef, int x);
57*67e74705SXin Li
foo()58*67e74705SXin Li void foo() {
59*67e74705SXin Li // CHECK-LABEL: define void @_ZN6PR75233fooEv()
60*67e74705SXin Li // CHECK: call void @_ZN6PR752310AddKeywordENS_9StringRefEi(i8* {{.*}}, i32 4)
61*67e74705SXin Li AddKeyword(StringRef(), 4);
62*67e74705SXin Li }
63*67e74705SXin Li }
64*67e74705SXin Li
65*67e74705SXin Li namespace PR7742 { // Also rdar://8250764
66*67e74705SXin Li struct s2 {
67*67e74705SXin Li float a[2];
68*67e74705SXin Li };
69*67e74705SXin Li
70*67e74705SXin Li struct c2 : public s2 {};
71*67e74705SXin Li
72*67e74705SXin Li // CHECK-LABEL: define <2 x float> @_ZN6PR77423fooEPNS_2c2E(%"struct.PR7742::c2"* %P)
foo(c2 * P)73*67e74705SXin Li c2 foo(c2 *P) {
74*67e74705SXin Li return c2();
75*67e74705SXin Li }
76*67e74705SXin Li
77*67e74705SXin Li }
78*67e74705SXin Li
79*67e74705SXin Li namespace PR5179 {
80*67e74705SXin Li struct B {};
81*67e74705SXin Li
82*67e74705SXin Li struct B1 : B {
83*67e74705SXin Li int* pa;
84*67e74705SXin Li };
85*67e74705SXin Li
86*67e74705SXin Li struct B2 : B {
87*67e74705SXin Li B1 b1;
88*67e74705SXin Li };
89*67e74705SXin Li
90*67e74705SXin Li // CHECK-LABEL: define i8* @_ZN6PR51793barENS_2B2E(i32* %b2.coerce)
bar(B2 b2)91*67e74705SXin Li const void *bar(B2 b2) {
92*67e74705SXin Li return b2.b1.pa;
93*67e74705SXin Li }
94*67e74705SXin Li }
95*67e74705SXin Li
96*67e74705SXin Li namespace test5 {
97*67e74705SXin Li struct Xbase { };
98*67e74705SXin Li struct Empty { };
99*67e74705SXin Li struct Y;
100*67e74705SXin Li struct X : public Xbase {
101*67e74705SXin Li Empty empty;
102*67e74705SXin Li Y f();
103*67e74705SXin Li };
104*67e74705SXin Li struct Y : public X {
105*67e74705SXin Li Empty empty;
106*67e74705SXin Li };
107*67e74705SXin Li X getX();
108*67e74705SXin Li int takeY(const Y&, int y);
g()109*67e74705SXin Li void g() {
110*67e74705SXin Li // rdar://8340348 - The temporary for the X object needs to have a defined
111*67e74705SXin Li // address when passed into X::f as 'this'.
112*67e74705SXin Li takeY(getX().f(), 42);
113*67e74705SXin Li }
114*67e74705SXin Li // CHECK: void @_ZN5test51gEv()
115*67e74705SXin Li // CHECK: alloca %"struct.test5::Y"
116*67e74705SXin Li // CHECK: alloca %"struct.test5::X"
117*67e74705SXin Li // CHECK: alloca %"struct.test5::Y"
118*67e74705SXin Li }
119*67e74705SXin Li
120*67e74705SXin Li
121*67e74705SXin Li // rdar://8360877
122*67e74705SXin Li namespace test6 {
123*67e74705SXin Li struct outer {
124*67e74705SXin Li int x;
125*67e74705SXin Li struct epsilon_matcher {} e;
126*67e74705SXin Li int f;
127*67e74705SXin Li };
128*67e74705SXin Li
test(outer x)129*67e74705SXin Li int test(outer x) {
130*67e74705SXin Li return x.x + x.f;
131*67e74705SXin Li }
132*67e74705SXin Li // CHECK-LABEL: define i32 @_ZN5test64testENS_5outerE(i64 %x.coerce0, i32 %x.coerce1)
133*67e74705SXin Li }
134*67e74705SXin Li
135*67e74705SXin Li namespace test7 {
136*67e74705SXin Li struct StringRef {char* ptr; long len; };
137*67e74705SXin Li class A { public: ~A(); };
x(A,A,long,long,StringRef)138*67e74705SXin Li A x(A, A, long, long, StringRef) { return A(); }
139*67e74705SXin Li // Check that the StringRef is passed byval instead of expanded
140*67e74705SXin Li // (which would split it between registers and memory).
141*67e74705SXin Li // rdar://problem/9686430
142*67e74705SXin Li // CHECK: define void @_ZN5test71xENS_1AES0_llNS_9StringRefE({{.*}} byval align 8)
143*67e74705SXin Li
144*67e74705SXin Li // And a couple extra related tests:
y(A,long double,long,long,StringRef)145*67e74705SXin Li A y(A, long double, long, long, StringRef) { return A(); }
146*67e74705SXin Li // CHECK: define void @_ZN5test71yENS_1AEellNS_9StringRefE({{.*}} i8*
147*67e74705SXin Li struct StringDouble {char * ptr; double d;};
z(A,A,A,A,A,StringDouble)148*67e74705SXin Li A z(A, A, A, A, A, StringDouble) { return A(); }
zz(A,A,A,A,StringDouble)149*67e74705SXin Li A zz(A, A, A, A, StringDouble) { return A(); }
150*67e74705SXin Li // CHECK: define void @_ZN5test71zENS_1AES0_S0_S0_S0_NS_12StringDoubleE({{.*}} byval align 8)
151*67e74705SXin Li // CHECK: define void @_ZN5test72zzENS_1AES0_S0_S0_NS_12StringDoubleE({{.*}} i8*
152*67e74705SXin Li }
153*67e74705SXin Li
154*67e74705SXin Li namespace test8 {
155*67e74705SXin Li // CHECK: declare void @_ZN5test83fooENS_1BE(%"class.test8::B"* byval align 8)
156*67e74705SXin Li class A {
157*67e74705SXin Li char big[17];
158*67e74705SXin Li };
159*67e74705SXin Li
160*67e74705SXin Li class B : public A {};
161*67e74705SXin Li
162*67e74705SXin Li void foo(B b);
bar()163*67e74705SXin Li void bar() {
164*67e74705SXin Li B b;
165*67e74705SXin Li foo(b);
166*67e74705SXin Li }
167*67e74705SXin Li }
168*67e74705SXin Li
169*67e74705SXin Li // PR4242
170*67e74705SXin Li namespace test9 {
171*67e74705SXin Li // Large enough to be passed indirectly.
172*67e74705SXin Li struct S { void *data[3]; };
173*67e74705SXin Li
174*67e74705SXin Li struct T { void *data[2]; };
175*67e74705SXin Li
176*67e74705SXin Li // CHECK: define void @_ZN5test93fooEPNS_1SEPNS_1TE([[S:%.*]]*, [[T:%.*]]*)
foo(S *,T *)177*67e74705SXin Li void foo(S*, T*) {}
178*67e74705SXin Li
179*67e74705SXin Li // CHECK: define void @_ZN5test91aEiiiiNS_1TEPv([[S]]* noalias sret {{%.*}}, i32, i32, i32, i32, [[T]]* byval align 8, i8*)
a(int,int,int,int,T,void *)180*67e74705SXin Li S a(int, int, int, int, T, void*) {
181*67e74705SXin Li return S();
182*67e74705SXin Li }
183*67e74705SXin Li
184*67e74705SXin Li // CHECK: define [[S]]* @_ZN5test91bEPNS_1SEiiiiNS_1TEPv([[S]]* {{%.*}}, i32, i32, i32, i32, [[T:%.*]]* byval align 8, i8*)
b(S * sret,int,int,int,int,T,void *)185*67e74705SXin Li S* b(S* sret, int, int, int, int, T, void*) {
186*67e74705SXin Li return sret;
187*67e74705SXin Li }
188*67e74705SXin Li
189*67e74705SXin Li // CHECK: define void @_ZN5test91cEiiiNS_1TEPv([[S]]* noalias sret {{%.*}}, i32, i32, i32, i8* {{%.*}}, i8* {{%.*}}, i8*)
c(int,int,int,T,void *)190*67e74705SXin Li S c(int, int, int, T, void*) {
191*67e74705SXin Li return S();
192*67e74705SXin Li }
193*67e74705SXin Li
194*67e74705SXin Li // CHECK: define [[S]]* @_ZN5test91dEPNS_1SEiiiNS_1TEPv([[S]]* {{%.*}}, i32, i32, i32, i8* {{%.*}}, i8* {{%.*}}, i8*)
d(S * sret,int,int,int,T,void *)195*67e74705SXin Li S* d(S* sret, int, int, int, T, void*) {
196*67e74705SXin Li return sret;
197*67e74705SXin Li }
198*67e74705SXin Li }
199*67e74705SXin Li
200*67e74705SXin Li namespace test10 {
201*67e74705SXin Li #pragma pack(1)
202*67e74705SXin Li struct BasePacked {
203*67e74705SXin Li char one;
204*67e74705SXin Li short two;
205*67e74705SXin Li };
206*67e74705SXin Li #pragma pack()
207*67e74705SXin Li struct DerivedPacked : public BasePacked {
208*67e74705SXin Li int three;
209*67e74705SXin Li };
210*67e74705SXin Li // CHECK-LABEL: define i32 @_ZN6test1020FuncForDerivedPackedENS_13DerivedPackedE({{.*}}* byval align 8
FuncForDerivedPacked(DerivedPacked d)211*67e74705SXin Li int FuncForDerivedPacked(DerivedPacked d) {
212*67e74705SXin Li return d.three;
213*67e74705SXin Li }
214*67e74705SXin Li }
215*67e74705SXin Li
216*67e74705SXin Li namespace test11 {
217*67e74705SXin Li union U {
218*67e74705SXin Li float f1;
219*67e74705SXin Li char __attribute__((__vector_size__(1))) f2;
220*67e74705SXin Li };
f(union U u)221*67e74705SXin Li int f(union U u) { return u.f2[1]; }
222*67e74705SXin Li // CHECK-LABEL: define i32 @_ZN6test111fENS_1UE(i32
223*67e74705SXin Li }
224