1 // REQUIRES: aarch64-registered-target
2 // RUN: %clang_cc1 -triple arm64-none-linux-gnu -target-feature +neon \
3 // RUN: -ffp-contract=fast -emit-llvm -o - %s | opt -S -mem2reg \
4 // RUN: | FileCheck %s
5
6 // Test new aarch64 intrinsics with poly128
7 // FIXME: Currently, poly128_t equals to uint128, which will be spilt into
8 // two 64-bit GPR(eg X0, X1). Now moving data from X0, X1 to FPR128 will
9 // introduce 2 store and 1 load instructions(store X0, X1 to memory and
10 // then load back to Q0). If target has NEON, this is better replaced by
11 // FMOV or INS.
12
13 #include <arm_neon.h>
14
15 // CHECK-LABEL: define void @test_vstrq_p128(i128* %ptr, i128 %val) #0 {
16 // CHECK: [[TMP0:%.*]] = bitcast i128* %ptr to i8*
17 // CHECK: [[TMP1:%.*]] = bitcast i8* [[TMP0]] to i128*
18 // CHECK: store i128 %val, i128* [[TMP1]]
19 // CHECK: ret void
test_vstrq_p128(poly128_t * ptr,poly128_t val)20 void test_vstrq_p128(poly128_t * ptr, poly128_t val) {
21 vstrq_p128(ptr, val);
22
23 }
24
25 // CHECK-LABEL: define i128 @test_vldrq_p128(i128* %ptr) #0 {
26 // CHECK: [[TMP0:%.*]] = bitcast i128* %ptr to i8*
27 // CHECK: [[TMP1:%.*]] = bitcast i8* [[TMP0]] to i128*
28 // CHECK: [[TMP2:%.*]] = load i128, i128* [[TMP1]]
29 // CHECK: ret i128 [[TMP2]]
test_vldrq_p128(poly128_t * ptr)30 poly128_t test_vldrq_p128(poly128_t * ptr) {
31 return vldrq_p128(ptr);
32
33 }
34
35 // CHECK-LABEL: define void @test_ld_st_p128(i128* %ptr) #0 {
36 // CHECK: [[TMP0:%.*]] = bitcast i128* %ptr to i8*
37 // CHECK: [[TMP1:%.*]] = bitcast i8* [[TMP0]] to i128*
38 // CHECK: [[TMP2:%.*]] = load i128, i128* [[TMP1]]
39 // CHECK: [[ADD_PTR:%.*]] = getelementptr inbounds i128, i128* %ptr, i64 1
40 // CHECK: [[TMP3:%.*]] = bitcast i128* [[ADD_PTR]] to i8*
41 // CHECK: [[TMP4:%.*]] = bitcast i8* [[TMP3]] to i128*
42 // CHECK: store i128 [[TMP2]], i128* [[TMP4]]
43 // CHECK: ret void
test_ld_st_p128(poly128_t * ptr)44 void test_ld_st_p128(poly128_t * ptr) {
45 vstrq_p128(ptr+1, vldrq_p128(ptr));
46
47 }
48
49 // CHECK-LABEL: define i128 @test_vmull_p64(i64 %a, i64 %b) #0 {
50 // CHECK: [[VMULL_P64_I:%.*]] = call <16 x i8> @llvm.aarch64.neon.pmull64(i64 %a, i64 %b) #2
51 // CHECK: [[VMULL_P641_I:%.*]] = bitcast <16 x i8> [[VMULL_P64_I]] to i128
52 // CHECK: ret i128 [[VMULL_P641_I]]
test_vmull_p64(poly64_t a,poly64_t b)53 poly128_t test_vmull_p64(poly64_t a, poly64_t b) {
54 return vmull_p64(a, b);
55 }
56
57 // CHECK-LABEL: define i128 @test_vmull_high_p64(<2 x i64> %a, <2 x i64> %b) #0 {
58 // CHECK: [[SHUFFLE_I_I:%.*]] = shufflevector <2 x i64> %a, <2 x i64> %a, <1 x i32> <i32 1>
59 // CHECK: [[TMP0:%.*]] = bitcast <1 x i64> [[SHUFFLE_I_I]] to i64
60 // CHECK: [[SHUFFLE_I7_I:%.*]] = shufflevector <2 x i64> %b, <2 x i64> %b, <1 x i32> <i32 1>
61 // CHECK: [[TMP1:%.*]] = bitcast <1 x i64> [[SHUFFLE_I7_I]] to i64
62 // CHECK: [[VMULL_P64_I_I:%.*]] = call <16 x i8> @llvm.aarch64.neon.pmull64(i64 [[TMP0]], i64 [[TMP1]]) #2
63 // CHECK: [[VMULL_P641_I_I:%.*]] = bitcast <16 x i8> [[VMULL_P64_I_I]] to i128
64 // CHECK: ret i128 [[VMULL_P641_I_I]]
test_vmull_high_p64(poly64x2_t a,poly64x2_t b)65 poly128_t test_vmull_high_p64(poly64x2_t a, poly64x2_t b) {
66 return vmull_high_p64(a, b);
67 }
68
69 // CHECK-LABEL: define i128 @test_vreinterpretq_p128_s8(<16 x i8> %a) #0 {
70 // CHECK: [[TMP0:%.*]] = bitcast <16 x i8> %a to i128
71 // CHECK: ret i128 [[TMP0]]
test_vreinterpretq_p128_s8(int8x16_t a)72 poly128_t test_vreinterpretq_p128_s8(int8x16_t a) {
73 return vreinterpretq_p128_s8(a);
74 }
75
76 // CHECK-LABEL: define i128 @test_vreinterpretq_p128_s16(<8 x i16> %a) #0 {
77 // CHECK: [[TMP0:%.*]] = bitcast <8 x i16> %a to i128
78 // CHECK: ret i128 [[TMP0]]
test_vreinterpretq_p128_s16(int16x8_t a)79 poly128_t test_vreinterpretq_p128_s16(int16x8_t a) {
80 return vreinterpretq_p128_s16(a);
81 }
82
83 // CHECK-LABEL: define i128 @test_vreinterpretq_p128_s32(<4 x i32> %a) #0 {
84 // CHECK: [[TMP0:%.*]] = bitcast <4 x i32> %a to i128
85 // CHECK: ret i128 [[TMP0]]
test_vreinterpretq_p128_s32(int32x4_t a)86 poly128_t test_vreinterpretq_p128_s32(int32x4_t a) {
87 return vreinterpretq_p128_s32(a);
88 }
89
90 // CHECK-LABEL: define i128 @test_vreinterpretq_p128_s64(<2 x i64> %a) #0 {
91 // CHECK: [[TMP0:%.*]] = bitcast <2 x i64> %a to i128
92 // CHECK: ret i128 [[TMP0]]
test_vreinterpretq_p128_s64(int64x2_t a)93 poly128_t test_vreinterpretq_p128_s64(int64x2_t a) {
94 return vreinterpretq_p128_s64(a);
95 }
96
97 // CHECK-LABEL: define i128 @test_vreinterpretq_p128_u8(<16 x i8> %a) #0 {
98 // CHECK: [[TMP0:%.*]] = bitcast <16 x i8> %a to i128
99 // CHECK: ret i128 [[TMP0]]
test_vreinterpretq_p128_u8(uint8x16_t a)100 poly128_t test_vreinterpretq_p128_u8(uint8x16_t a) {
101 return vreinterpretq_p128_u8(a);
102 }
103
104 // CHECK-LABEL: define i128 @test_vreinterpretq_p128_u16(<8 x i16> %a) #0 {
105 // CHECK: [[TMP0:%.*]] = bitcast <8 x i16> %a to i128
106 // CHECK: ret i128 [[TMP0]]
test_vreinterpretq_p128_u16(uint16x8_t a)107 poly128_t test_vreinterpretq_p128_u16(uint16x8_t a) {
108 return vreinterpretq_p128_u16(a);
109 }
110
111 // CHECK-LABEL: define i128 @test_vreinterpretq_p128_u32(<4 x i32> %a) #0 {
112 // CHECK: [[TMP0:%.*]] = bitcast <4 x i32> %a to i128
113 // CHECK: ret i128 [[TMP0]]
test_vreinterpretq_p128_u32(uint32x4_t a)114 poly128_t test_vreinterpretq_p128_u32(uint32x4_t a) {
115 return vreinterpretq_p128_u32(a);
116 }
117
118 // CHECK-LABEL: define i128 @test_vreinterpretq_p128_u64(<2 x i64> %a) #0 {
119 // CHECK: [[TMP0:%.*]] = bitcast <2 x i64> %a to i128
120 // CHECK: ret i128 [[TMP0]]
test_vreinterpretq_p128_u64(uint64x2_t a)121 poly128_t test_vreinterpretq_p128_u64(uint64x2_t a) {
122 return vreinterpretq_p128_u64(a);
123 }
124
125 // CHECK-LABEL: define i128 @test_vreinterpretq_p128_f32(<4 x float> %a) #0 {
126 // CHECK: [[TMP0:%.*]] = bitcast <4 x float> %a to i128
127 // CHECK: ret i128 [[TMP0]]
test_vreinterpretq_p128_f32(float32x4_t a)128 poly128_t test_vreinterpretq_p128_f32(float32x4_t a) {
129 return vreinterpretq_p128_f32(a);
130 }
131
132 // CHECK-LABEL: define i128 @test_vreinterpretq_p128_f64(<2 x double> %a) #0 {
133 // CHECK: [[TMP0:%.*]] = bitcast <2 x double> %a to i128
134 // CHECK: ret i128 [[TMP0]]
test_vreinterpretq_p128_f64(float64x2_t a)135 poly128_t test_vreinterpretq_p128_f64(float64x2_t a) {
136 return vreinterpretq_p128_f64(a);
137 }
138
139 // CHECK-LABEL: define i128 @test_vreinterpretq_p128_p8(<16 x i8> %a) #0 {
140 // CHECK: [[TMP0:%.*]] = bitcast <16 x i8> %a to i128
141 // CHECK: ret i128 [[TMP0]]
test_vreinterpretq_p128_p8(poly8x16_t a)142 poly128_t test_vreinterpretq_p128_p8(poly8x16_t a) {
143 return vreinterpretq_p128_p8(a);
144 }
145
146 // CHECK-LABEL: define i128 @test_vreinterpretq_p128_p16(<8 x i16> %a) #0 {
147 // CHECK: [[TMP0:%.*]] = bitcast <8 x i16> %a to i128
148 // CHECK: ret i128 [[TMP0]]
test_vreinterpretq_p128_p16(poly16x8_t a)149 poly128_t test_vreinterpretq_p128_p16(poly16x8_t a) {
150 return vreinterpretq_p128_p16(a);
151 }
152
153 // CHECK-LABEL: define i128 @test_vreinterpretq_p128_p64(<2 x i64> %a) #0 {
154 // CHECK: [[TMP0:%.*]] = bitcast <2 x i64> %a to i128
155 // CHECK: ret i128 [[TMP0]]
test_vreinterpretq_p128_p64(poly64x2_t a)156 poly128_t test_vreinterpretq_p128_p64(poly64x2_t a) {
157 return vreinterpretq_p128_p64(a);
158 }
159
160 // CHECK-LABEL: define <16 x i8> @test_vreinterpretq_s8_p128(i128 %a) #0 {
161 // CHECK: [[TMP0:%.*]] = bitcast i128 %a to <16 x i8>
162 // CHECK: ret <16 x i8> [[TMP0]]
test_vreinterpretq_s8_p128(poly128_t a)163 int8x16_t test_vreinterpretq_s8_p128(poly128_t a) {
164 return vreinterpretq_s8_p128(a);
165 }
166
167 // CHECK-LABEL: define <8 x i16> @test_vreinterpretq_s16_p128(i128 %a) #0 {
168 // CHECK: [[TMP0:%.*]] = bitcast i128 %a to <8 x i16>
169 // CHECK: ret <8 x i16> [[TMP0]]
test_vreinterpretq_s16_p128(poly128_t a)170 int16x8_t test_vreinterpretq_s16_p128(poly128_t a) {
171 return vreinterpretq_s16_p128(a);
172 }
173
174 // CHECK-LABEL: define <4 x i32> @test_vreinterpretq_s32_p128(i128 %a) #0 {
175 // CHECK: [[TMP0:%.*]] = bitcast i128 %a to <4 x i32>
176 // CHECK: ret <4 x i32> [[TMP0]]
test_vreinterpretq_s32_p128(poly128_t a)177 int32x4_t test_vreinterpretq_s32_p128(poly128_t a) {
178 return vreinterpretq_s32_p128(a);
179 }
180
181 // CHECK-LABEL: define <2 x i64> @test_vreinterpretq_s64_p128(i128 %a) #0 {
182 // CHECK: [[TMP0:%.*]] = bitcast i128 %a to <2 x i64>
183 // CHECK: ret <2 x i64> [[TMP0]]
test_vreinterpretq_s64_p128(poly128_t a)184 int64x2_t test_vreinterpretq_s64_p128(poly128_t a) {
185 return vreinterpretq_s64_p128(a);
186 }
187
188 // CHECK-LABEL: define <16 x i8> @test_vreinterpretq_u8_p128(i128 %a) #0 {
189 // CHECK: [[TMP0:%.*]] = bitcast i128 %a to <16 x i8>
190 // CHECK: ret <16 x i8> [[TMP0]]
test_vreinterpretq_u8_p128(poly128_t a)191 uint8x16_t test_vreinterpretq_u8_p128(poly128_t a) {
192 return vreinterpretq_u8_p128(a);
193 }
194
195 // CHECK-LABEL: define <8 x i16> @test_vreinterpretq_u16_p128(i128 %a) #0 {
196 // CHECK: [[TMP0:%.*]] = bitcast i128 %a to <8 x i16>
197 // CHECK: ret <8 x i16> [[TMP0]]
test_vreinterpretq_u16_p128(poly128_t a)198 uint16x8_t test_vreinterpretq_u16_p128(poly128_t a) {
199 return vreinterpretq_u16_p128(a);
200 }
201
202 // CHECK-LABEL: define <4 x i32> @test_vreinterpretq_u32_p128(i128 %a) #0 {
203 // CHECK: [[TMP0:%.*]] = bitcast i128 %a to <4 x i32>
204 // CHECK: ret <4 x i32> [[TMP0]]
test_vreinterpretq_u32_p128(poly128_t a)205 uint32x4_t test_vreinterpretq_u32_p128(poly128_t a) {
206 return vreinterpretq_u32_p128(a);
207 }
208
209 // CHECK-LABEL: define <2 x i64> @test_vreinterpretq_u64_p128(i128 %a) #0 {
210 // CHECK: [[TMP0:%.*]] = bitcast i128 %a to <2 x i64>
211 // CHECK: ret <2 x i64> [[TMP0]]
test_vreinterpretq_u64_p128(poly128_t a)212 uint64x2_t test_vreinterpretq_u64_p128(poly128_t a) {
213 return vreinterpretq_u64_p128(a);
214 }
215
216 // CHECK-LABEL: define <4 x float> @test_vreinterpretq_f32_p128(i128 %a) #0 {
217 // CHECK: [[TMP0:%.*]] = bitcast i128 %a to <4 x float>
218 // CHECK: ret <4 x float> [[TMP0]]
test_vreinterpretq_f32_p128(poly128_t a)219 float32x4_t test_vreinterpretq_f32_p128(poly128_t a) {
220 return vreinterpretq_f32_p128(a);
221 }
222
223 // CHECK-LABEL: define <2 x double> @test_vreinterpretq_f64_p128(i128 %a) #0 {
224 // CHECK: [[TMP0:%.*]] = bitcast i128 %a to <2 x double>
225 // CHECK: ret <2 x double> [[TMP0]]
test_vreinterpretq_f64_p128(poly128_t a)226 float64x2_t test_vreinterpretq_f64_p128(poly128_t a) {
227 return vreinterpretq_f64_p128(a);
228 }
229
230 // CHECK-LABEL: define <16 x i8> @test_vreinterpretq_p8_p128(i128 %a) #0 {
231 // CHECK: [[TMP0:%.*]] = bitcast i128 %a to <16 x i8>
232 // CHECK: ret <16 x i8> [[TMP0]]
test_vreinterpretq_p8_p128(poly128_t a)233 poly8x16_t test_vreinterpretq_p8_p128(poly128_t a) {
234 return vreinterpretq_p8_p128(a);
235 }
236
237 // CHECK-LABEL: define <8 x i16> @test_vreinterpretq_p16_p128(i128 %a) #0 {
238 // CHECK: [[TMP0:%.*]] = bitcast i128 %a to <8 x i16>
239 // CHECK: ret <8 x i16> [[TMP0]]
test_vreinterpretq_p16_p128(poly128_t a)240 poly16x8_t test_vreinterpretq_p16_p128(poly128_t a) {
241 return vreinterpretq_p16_p128(a);
242 }
243
244 // CHECK-LABEL: define <2 x i64> @test_vreinterpretq_p64_p128(i128 %a) #0 {
245 // CHECK: [[TMP0:%.*]] = bitcast i128 %a to <2 x i64>
246 // CHECK: ret <2 x i64> [[TMP0]]
test_vreinterpretq_p64_p128(poly128_t a)247 poly64x2_t test_vreinterpretq_p64_p128(poly128_t a) {
248 return vreinterpretq_p64_p128(a);
249 }
250
251
252