1 // RUN: %clang_cc1 -triple powerpc64le-unknown-linux-gnu -emit-llvm -o - %s | FileCheck %s --check-prefix=PPC
2 // RUN: %clang_cc1 -mfloat-abi hard -triple armv7-unknown-linux-gnueabi -emit-llvm -o - %s | FileCheck %s --check-prefix=ARM32
3 // RUN: %clang_cc1 -mfloat-abi hard -triple aarch64-unknown-linux-gnu -emit-llvm -o - %s | FileCheck %s --check-prefix=ARM64
4 // RUN: %clang_cc1 -mfloat-abi hard -triple x86_64-unknown-windows-gnu -emit-llvm -o - %s | FileCheck %s --check-prefix=X64
5 // RUN: %clang_cc1 -mfloat-abi hard -triple aarch64-unknown-windows-msvc -emit-llvm -o - %s | FileCheck %s --check-prefix=WOA64
7 #if defined(__x86_64__)
8 #define CC __attribute__((vectorcall))
13 // Test that C++ classes are correctly classified as homogeneous aggregates.
24 struct D1
: Base1
{ // non-homogeneous aggregate
27 struct D2
: Base2
{ // homogeneous aggregate
30 struct D3
: Base1
, Base2
{ // non-homogeneous aggregate
33 struct D4
: Base2
, Base3
{ // homogeneous aggregate
40 struct D5
: I1
, I2
, I3
{}; // homogeneous aggregate
42 // PPC: define{{.*}} void @_Z7func_D12D1(%struct.D1* noalias sret(%struct.D1) align 8 %agg.result, [3 x i64] %x.coerce)
43 // ARM32: define{{.*}} arm_aapcs_vfpcc void @_Z7func_D12D1(%struct.D1* noalias sret(%struct.D1) align 8 %agg.result, [3 x i64] %x.coerce)
44 // ARM64: define{{.*}} void @_Z7func_D12D1(%struct.D1* noalias sret(%struct.D1) align 8 %agg.result, %struct.D1* noundef %x)
45 // X64: define dso_local x86_vectorcallcc void @"\01_Z7func_D12D1@@24"(%struct.D1* noalias sret(%struct.D1) align 8 %agg.result, %struct.D1* noundef %x)
46 D1 CC
func_D1(D1 x
) { return x
; }
48 // PPC: define{{.*}} [3 x double] @_Z7func_D22D2([3 x double] %x.coerce)
49 // ARM32: define{{.*}} arm_aapcs_vfpcc %struct.D2 @_Z7func_D22D2(%struct.D2 %x.coerce)
50 // ARM64: define{{.*}} %struct.D2 @_Z7func_D22D2([3 x double] %x.coerce)
51 // X64: define dso_local x86_vectorcallcc %struct.D2 @"\01_Z7func_D22D2@@24"(%struct.D2 inreg %x.coerce)
52 D2 CC
func_D2(D2 x
) { return x
; }
54 // PPC: define{{.*}} void @_Z7func_D32D3(%struct.D3* noalias sret(%struct.D3) align 8 %agg.result, [4 x i64] %x.coerce)
55 // ARM32: define{{.*}} arm_aapcs_vfpcc void @_Z7func_D32D3(%struct.D3* noalias sret(%struct.D3) align 8 %agg.result, [4 x i64] %x.coerce)
56 // ARM64: define{{.*}} void @_Z7func_D32D3(%struct.D3* noalias sret(%struct.D3) align 8 %agg.result, %struct.D3* noundef %x)
57 D3 CC
func_D3(D3 x
) { return x
; }
59 // PPC: define{{.*}} [4 x double] @_Z7func_D42D4([4 x double] %x.coerce)
60 // ARM32: define{{.*}} arm_aapcs_vfpcc %struct.D4 @_Z7func_D42D4(%struct.D4 %x.coerce)
61 // ARM64: define{{.*}} %struct.D4 @_Z7func_D42D4([4 x double] %x.coerce)
62 D4 CC
func_D4(D4 x
) { return x
; }
64 D5 CC
func_D5(D5 x
) { return x
; }
65 // PPC: define{{.*}} [3 x double] @_Z7func_D52D5([3 x double] %x.coerce)
66 // ARM32: define{{.*}} arm_aapcs_vfpcc %struct.D5 @_Z7func_D52D5(%struct.D5 %x.coerce)
68 // The C++ multiple inheritance expansion case is a little more complicated, so
69 // do some extra checking.
71 // ARM64-LABEL: define{{.*}} %struct.D5 @_Z7func_D52D5([3 x double] %x.coerce)
72 // ARM64: bitcast %struct.D5* %{{.*}} to [3 x double]*
73 // ARM64: store [3 x double] %x.coerce, [3 x double]*
79 // Check the call site.
81 // ARM64-LABEL: define{{.*}} void @_Z7call_D5P2D5(%struct.D5* noundef %p)
82 // ARM64: load [3 x double], [3 x double]*
83 // ARM64: call %struct.D5 @_Z7func_D52D5([3 x double] %{{.*}})
86 struct Float1
{ float x
; };
87 struct Float2
{ float y
; };
88 struct HVAWithEmptyBase
: Float1
, Empty
, Float2
{ float z
; };
90 // PPC: define{{.*}} void @_Z15with_empty_base16HVAWithEmptyBase([3 x float] %a.coerce)
91 // ARM64: define{{.*}} void @_Z15with_empty_base16HVAWithEmptyBase([3 x float] %a.coerce)
92 // ARM32: define{{.*}} arm_aapcs_vfpcc void @_Z15with_empty_base16HVAWithEmptyBase(%struct.HVAWithEmptyBase %a.coerce)
93 void CC
with_empty_base(HVAWithEmptyBase a
) {}
95 // FIXME: MSVC doesn't consider this an HVA because of the empty base.
96 // X64: define dso_local x86_vectorcallcc void @"\01_Z15with_empty_base16HVAWithEmptyBase@@16"(%struct.HVAWithEmptyBase inreg %a.coerce)
98 struct HVAWithEmptyBitField
: Float1
, Float2
{
99 int : 0; // Takes no space.
103 // PPC: define{{.*}} void @_Z19with_empty_bitfield20HVAWithEmptyBitField([3 x float] %a.coerce)
104 // ARM64: define{{.*}} void @_Z19with_empty_bitfield20HVAWithEmptyBitField([3 x float] %a.coerce)
105 // ARM32: define{{.*}} arm_aapcs_vfpcc void @_Z19with_empty_bitfield20HVAWithEmptyBitField(%struct.HVAWithEmptyBitField %a.coerce)
106 // X64: define dso_local x86_vectorcallcc void @"\01_Z19with_empty_bitfield20HVAWithEmptyBitField@@16"(%struct.HVAWithEmptyBitField inreg %a.coerce)
107 void CC
with_empty_bitfield(HVAWithEmptyBitField a
) {}
110 // MSVC on Arm includes "isCXX14Aggregate" as part of its definition of
111 // Homogeneous Floating-point Aggregate (HFA). Additionally, it has a different
112 // handling of C++14 aggregates, which can lead to confusion.
114 // Pod is a trivial HFA.
118 // Not an aggregate according to C++14 spec => not HFA according to MSVC.
119 struct NotCXX14Aggregate
{
123 // NotPod is a C++14 aggregate. But not HFA, because it contains
124 // NotCXX14Aggregate (which itself is not HFA because it's not a C++14
130 // A class with a base is returned using the sret calling convetion by MSVC.
131 struct HasEmptyBase
: public Empty
{
134 struct HasPodBase
: public Pod
{};
135 // WOA64-LABEL: define dso_local %"struct.pr47611::Pod" @"?copy@pr47611@@YA?AUPod@1@PEAU21@@Z"(%"struct.pr47611::Pod"* noundef %x)
136 Pod
copy(Pod
*x
) { return *x
; } // MSVC: ldp d0,d1,[x0], Clang: ldp d0,d1,[x0]
137 // WOA64-LABEL: define dso_local void @"?copy@pr47611@@YA?AUNotCXX14Aggregate@1@PEAU21@@Z"(%"struct.pr47611::NotCXX14Aggregate"* inreg noalias sret(%"struct.pr47611::NotCXX14Aggregate") align 8 %agg.result, %"struct.pr47611::NotCXX14Aggregate"* noundef %x)
138 NotCXX14Aggregate
copy(NotCXX14Aggregate
*x
) { return *x
; } // MSVC: stp x8,x9,[x0], Clang: str q0,[x0]
139 // WOA64-LABEL: define dso_local [2 x i64] @"?copy@pr47611@@YA?AUNotPod@1@PEAU21@@Z"(%"struct.pr47611::NotPod"* noundef %x)
140 NotPod
copy(NotPod
*x
) { return *x
; }
141 // WOA64-LABEL: define dso_local void @"?copy@pr47611@@YA?AUHasEmptyBase@1@PEAU21@@Z"(%"struct.pr47611::HasEmptyBase"* inreg noalias sret(%"struct.pr47611::HasEmptyBase") align 8 %agg.result, %"struct.pr47611::HasEmptyBase"* noundef %x)
142 HasEmptyBase
copy(HasEmptyBase
*x
) { return *x
; }
143 // WOA64-LABEL: define dso_local void @"?copy@pr47611@@YA?AUHasPodBase@1@PEAU21@@Z"(%"struct.pr47611::HasPodBase"* inreg noalias sret(%"struct.pr47611::HasPodBase") align 8 %agg.result, %"struct.pr47611::HasPodBase"* noundef %x)
144 HasPodBase
copy(HasPodBase
*x
) { return *x
; }
146 void call_copy_pod(Pod
*pod
) {
148 // WOA64-LABEL: define dso_local void @"?call_copy_pod@pr47611@@YAXPEAUPod@1@@Z"
149 // WOA64: %{{.*}} = call %"struct.pr47611::Pod" @"?copy@pr47611@@YA?AUPod@1@PEAU21@@Z"(%"struct.pr47611::Pod"* noundef %{{.*}})
152 void call_copy_notcxx14aggregate(NotCXX14Aggregate
*notcxx14aggregate
) {
153 *notcxx14aggregate
= copy(notcxx14aggregate
);
154 // WOA64-LABEL: define dso_local void @"?call_copy_notcxx14aggregate@pr47611@@YAXPEAUNotCXX14Aggregate@1@@Z"
155 // WOA64: call void @"?copy@pr47611@@YA?AUNotCXX14Aggregate@1@PEAU21@@Z"(%"struct.pr47611::NotCXX14Aggregate"* inreg sret(%"struct.pr47611::NotCXX14Aggregate") align 8 %{{.*}}, %"struct.pr47611::NotCXX14Aggregate"* noundef %{{.*}})
158 void call_copy_notpod(NotPod
*notPod
) {
159 *notPod
= copy(notPod
);
160 // WOA64-LABEL: define dso_local void @"?call_copy_notpod@pr47611@@YAXPEAUNotPod@1@@Z"
161 // WOA64: %{{.*}} = call [2 x i64] @"?copy@pr47611@@YA?AUNotPod@1@PEAU21@@Z"(%"struct.pr47611::NotPod"* noundef %{{.*}})
164 void call_copy_hasemptybase(HasEmptyBase
*hasEmptyBase
) {
165 *hasEmptyBase
= copy(hasEmptyBase
);
166 // WOA64-LABEL: define dso_local void @"?call_copy_hasemptybase@pr47611@@YAXPEAUHasEmptyBase@1@@Z"
167 // WOA64: call void @"?copy@pr47611@@YA?AUHasEmptyBase@1@PEAU21@@Z"(%"struct.pr47611::HasEmptyBase"* inreg sret(%"struct.pr47611::HasEmptyBase") align 8 %{{.*}}, %"struct.pr47611::HasEmptyBase"* noundef %{{.*}})
170 void call_copy_haspodbase(HasPodBase
*hasPodBase
) {
171 *hasPodBase
= copy(hasPodBase
);
172 // WOA64-LABEL: define dso_local void @"?call_copy_haspodbase@pr47611@@YAXPEAUHasPodBase@1@@Z"
173 // WOA64: call void @"?copy@pr47611@@YA?AUHasPodBase@1@PEAU21@@Z"(%"struct.pr47611::HasPodBase"* inreg sret(%"struct.pr47611::HasPodBase") align 8 %{{.*}}, %"struct.pr47611::HasPodBase"* noundef %{{.*}})
175 }; // namespace pr47611