1 ; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py
2 ; RUN: llc -mtriple=thumbv8.1m.main-none-none-eabi -mattr=+mve.fp %s -o - | FileCheck %s
4 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16(i16* %base, <8 x i16>* %offptr) {
5 ; CHECK-LABEL: scaled_v8i16_i16:
6 ; CHECK: @ %bb.0: @ %entry
7 ; CHECK-NEXT: vldrh.u16 q1, [r1]
8 ; CHECK-NEXT: vldrh.u16 q0, [r0, q1, uxtw #1]
11 %offs = load <8 x i16>, <8 x i16>* %offptr, align 2
12 %offs.zext = zext <8 x i16> %offs to <8 x i32>
13 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> %offs.zext
14 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> undef)
18 define arm_aapcs_vfpcc <8 x half> @scaled_v8f16_i16(i16* %base, <8 x i16>* %offptr) {
19 ; CHECK-LABEL: scaled_v8f16_i16:
20 ; CHECK: @ %bb.0: @ %entry
21 ; CHECK-NEXT: vldrh.u16 q1, [r1]
22 ; CHECK-NEXT: vldrh.u16 q0, [r0, q1, uxtw #1]
25 %offs = load <8 x i16>, <8 x i16>* %offptr, align 2
26 %offs.zext = zext <8 x i16> %offs to <8 x i32>
27 %i16_ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> %offs.zext
28 %ptrs = bitcast <8 x i16*> %i16_ptrs to <8 x half*>
29 %gather = call <8 x half> @llvm.masked.gather.v8f16.v8p0f16(<8 x half*> %ptrs, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x half> undef)
30 ret <8 x half> %gather
33 define arm_aapcs_vfpcc <8 x half> @scaled_v8f16_half(half* %base, <8 x i16>* %offptr) {
34 ; CHECK-LABEL: scaled_v8f16_half:
35 ; CHECK: @ %bb.0: @ %entry
36 ; CHECK-NEXT: vldrh.u16 q1, [r1]
37 ; CHECK-NEXT: vldrh.u16 q0, [r0, q1, uxtw #1]
40 %offs = load <8 x i16>, <8 x i16>* %offptr, align 2
41 %offs.zext = zext <8 x i16> %offs to <8 x i32>
42 %ptrs = getelementptr inbounds half, half* %base, <8 x i32> %offs.zext
43 %gather = call <8 x half> @llvm.masked.gather.v8f16.v8p0f16(<8 x half*> %ptrs, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x half> undef)
44 ret <8 x half> %gather
47 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_sext(i16* %base, <8 x i16>* %offptr) {
48 ; CHECK-LABEL: scaled_v8i16_sext:
49 ; CHECK: @ %bb.0: @ %entry
50 ; CHECK-NEXT: .save {r4, r5, r7, lr}
51 ; CHECK-NEXT: push {r4, r5, r7, lr}
52 ; CHECK-NEXT: vldrh.s32 q0, [r1, #8]
53 ; CHECK-NEXT: vshl.i32 q0, q0, #1
54 ; CHECK-NEXT: vadd.i32 q0, q0, r0
55 ; CHECK-NEXT: vmov r2, r12, d0
56 ; CHECK-NEXT: vmov r3, lr, d1
57 ; CHECK-NEXT: vldrh.s32 q0, [r1]
58 ; CHECK-NEXT: vshl.i32 q0, q0, #1
59 ; CHECK-NEXT: vadd.i32 q0, q0, r0
60 ; CHECK-NEXT: vmov r4, r5, d0
61 ; CHECK-NEXT: vmov r0, r1, d1
62 ; CHECK-NEXT: ldrh r2, [r2]
63 ; CHECK-NEXT: ldrh.w r12, [r12]
64 ; CHECK-NEXT: ldrh r3, [r3]
65 ; CHECK-NEXT: ldrh.w lr, [lr]
66 ; CHECK-NEXT: ldrh r4, [r4]
67 ; CHECK-NEXT: ldrh r5, [r5]
68 ; CHECK-NEXT: vmov.16 q0[0], r4
69 ; CHECK-NEXT: ldrh r0, [r0]
70 ; CHECK-NEXT: vmov.16 q0[1], r5
71 ; CHECK-NEXT: ldrh r1, [r1]
72 ; CHECK-NEXT: vmov.16 q0[2], r0
73 ; CHECK-NEXT: vmov.16 q0[3], r1
74 ; CHECK-NEXT: vmov.16 q0[4], r2
75 ; CHECK-NEXT: vmov.16 q0[5], r12
76 ; CHECK-NEXT: vmov.16 q0[6], r3
77 ; CHECK-NEXT: vmov.16 q0[7], lr
78 ; CHECK-NEXT: pop {r4, r5, r7, pc}
80 %offs = load <8 x i16>, <8 x i16>* %offptr, align 2
81 %offs.sext = sext <8 x i16> %offs to <8 x i32>
82 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> %offs.sext
83 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> undef)
87 define arm_aapcs_vfpcc <8 x half> @scaled_v8f16_sext(i16* %base, <8 x i16>* %offptr) {
88 ; CHECK-LABEL: scaled_v8f16_sext:
89 ; CHECK: @ %bb.0: @ %entry
90 ; CHECK-NEXT: vldrh.s32 q0, [r1]
91 ; CHECK-NEXT: vshl.i32 q0, q0, #1
92 ; CHECK-NEXT: vadd.i32 q0, q0, r0
93 ; CHECK-NEXT: vmov r2, r3, d0
94 ; CHECK-NEXT: vldr.16 s4, [r3]
95 ; CHECK-NEXT: vldr.16 s0, [r2]
96 ; CHECK-NEXT: vmov r2, r3, d1
97 ; CHECK-NEXT: vins.f16 s0, s4
98 ; CHECK-NEXT: vldrh.s32 q1, [r1, #8]
99 ; CHECK-NEXT: vldr.16 s2, [r3]
100 ; CHECK-NEXT: vldr.16 s1, [r2]
101 ; CHECK-NEXT: vshl.i32 q1, q1, #1
102 ; CHECK-NEXT: vadd.i32 q1, q1, r0
103 ; CHECK-NEXT: vins.f16 s1, s2
104 ; CHECK-NEXT: vmov r0, r1, d2
105 ; CHECK-NEXT: vldr.16 s4, [r1]
106 ; CHECK-NEXT: vldr.16 s2, [r0]
107 ; CHECK-NEXT: vmov r0, r1, d3
108 ; CHECK-NEXT: vins.f16 s2, s4
109 ; CHECK-NEXT: vldr.16 s4, [r1]
110 ; CHECK-NEXT: vldr.16 s3, [r0]
111 ; CHECK-NEXT: vins.f16 s3, s4
114 %offs = load <8 x i16>, <8 x i16>* %offptr, align 2
115 %offs.sext = sext <8 x i16> %offs to <8 x i32>
116 %i16_ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> %offs.sext
117 %ptrs = bitcast <8 x i16*> %i16_ptrs to <8 x half*>
118 %gather = call <8 x half> @llvm.masked.gather.v8f16.v8p0f16(<8 x half*> %ptrs, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x half> undef)
119 ret <8 x half> %gather
122 define arm_aapcs_vfpcc <8 x i16> @unsigned_scaled_v8i16_i8(i16* %base, <8 x i8>* %offptr) {
123 ; CHECK-LABEL: unsigned_scaled_v8i16_i8:
124 ; CHECK: @ %bb.0: @ %entry
125 ; CHECK-NEXT: vldrb.u16 q1, [r1]
126 ; CHECK-NEXT: vldrh.u16 q0, [r0, q1, uxtw #1]
129 %offs = load <8 x i8>, <8 x i8>* %offptr, align 1
130 %offs.zext = zext <8 x i8> %offs to <8 x i32>
131 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> %offs.zext
132 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> undef)
133 ret <8 x i16> %gather
136 define arm_aapcs_vfpcc <8 x half> @unsigned_scaled_v8f16_i8(i16* %base, <8 x i8>* %offptr) {
137 ; CHECK-LABEL: unsigned_scaled_v8f16_i8:
138 ; CHECK: @ %bb.0: @ %entry
139 ; CHECK-NEXT: vldrb.u16 q1, [r1]
140 ; CHECK-NEXT: vldrh.u16 q0, [r0, q1, uxtw #1]
143 %offs = load <8 x i8>, <8 x i8>* %offptr, align 1
144 %offs.zext = zext <8 x i8> %offs to <8 x i32>
145 %i16_ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> %offs.zext
146 %ptrs = bitcast <8 x i16*> %i16_ptrs to <8 x half*>
147 %gather = call <8 x half> @llvm.masked.gather.v8f16.v8p0f16(<8 x half*> %ptrs, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x half> undef)
148 ret <8 x half> %gather
151 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_passthru0t(i16* %base, <8 x i16>* %offptr) {
152 ; CHECK-LABEL: scaled_v8i16_i16_passthru0t:
153 ; CHECK: @ %bb.0: @ %entry
154 ; CHECK-NEXT: vldrh.u16 q1, [r1]
155 ; CHECK-NEXT: vldrh.u16 q0, [r0, q1, uxtw #1]
158 %offs = load <8 x i16>, <8 x i16>* %offptr, align 2
159 %offs.zext = zext <8 x i16> %offs to <8 x i32>
160 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> %offs.zext
161 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> zeroinitializer)
162 ret <8 x i16> %gather
165 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_passthru1t(i16* %base, <8 x i16>* %offptr) {
166 ; CHECK-LABEL: scaled_v8i16_i16_passthru1t:
167 ; CHECK: @ %bb.0: @ %entry
168 ; CHECK-NEXT: vldrh.u16 q1, [r1]
169 ; CHECK-NEXT: vldrh.u16 q0, [r0, q1, uxtw #1]
172 %offs = load <8 x i16>, <8 x i16>* %offptr, align 2
173 %offs.zext = zext <8 x i16> %offs to <8 x i32>
174 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> %offs.zext
175 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> <i16 1, i16 1, i16 1, i16 1, i16 1, i16 1, i16 1, i16 1>)
176 ret <8 x i16> %gather
179 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_passthru1f(i16* %base, <8 x i16>* %offptr) {
180 ; CHECK-LABEL: scaled_v8i16_i16_passthru1f:
181 ; CHECK: @ %bb.0: @ %entry
182 ; CHECK-NEXT: movw r2, #65487
183 ; CHECK-NEXT: vmov.i16 q0, #0x1
184 ; CHECK-NEXT: vmsr p0, r2
185 ; CHECK-NEXT: vldrh.u16 q1, [r1]
187 ; CHECK-NEXT: vldrht.u16 q2, [r0, q1, uxtw #1]
188 ; CHECK-NEXT: vpsel q0, q2, q0
191 %offs = load <8 x i16>, <8 x i16>* %offptr, align 2
192 %offs.zext = zext <8 x i16> %offs to <8 x i32>
193 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> %offs.zext
194 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs, i32 2, <8 x i1> <i1 true, i1 true, i1 false, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> <i16 1, i16 1, i16 1, i16 1, i16 1, i16 1, i16 1, i16 1>)
195 ret <8 x i16> %gather
198 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_passthru0f(i16* %base, <8 x i16>* %offptr) {
199 ; CHECK-LABEL: scaled_v8i16_i16_passthru0f:
200 ; CHECK: @ %bb.0: @ %entry
201 ; CHECK-NEXT: movw r2, #65523
202 ; CHECK-NEXT: vmsr p0, r2
203 ; CHECK-NEXT: vldrh.u16 q1, [r1]
205 ; CHECK-NEXT: vldrht.u16 q0, [r0, q1, uxtw #1]
208 %offs = load <8 x i16>, <8 x i16>* %offptr, align 2
209 %offs.zext = zext <8 x i16> %offs to <8 x i32>
210 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> %offs.zext
211 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs, i32 2, <8 x i1> <i1 true, i1 false, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> <i16 0, i16 0, i16 0, i16 0, i16 0, i16 0, i16 0, i16 0>)
212 ret <8 x i16> %gather
215 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_passthru_icmp0(i16* %base, <8 x i16>* %offptr) {
216 ; CHECK-LABEL: scaled_v8i16_i16_passthru_icmp0:
217 ; CHECK: @ %bb.0: @ %entry
218 ; CHECK-NEXT: vldrh.u16 q1, [r1]
219 ; CHECK-NEXT: vpt.s16 gt, q1, zr
220 ; CHECK-NEXT: vldrht.u16 q0, [r0, q1, uxtw #1]
223 %offs = load <8 x i16>, <8 x i16>* %offptr, align 2
224 %offs.zext = zext <8 x i16> %offs to <8 x i32>
225 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> %offs.zext
226 %mask = icmp sgt <8 x i16> %offs, zeroinitializer
227 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs, i32 2, <8 x i1> %mask, <8 x i16> <i16 0, i16 0, i16 0, i16 0, i16 0, i16 0, i16 0, i16 0>)
228 ret <8 x i16> %gather
231 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_passthru_icmp1(i16* %base, <8 x i16>* %offptr) {
232 ; CHECK-LABEL: scaled_v8i16_i16_passthru_icmp1:
233 ; CHECK: @ %bb.0: @ %entry
234 ; CHECK-NEXT: vmov.i16 q0, #0x1
235 ; CHECK-NEXT: vldrh.u16 q1, [r1]
236 ; CHECK-NEXT: vpt.s16 gt, q1, zr
237 ; CHECK-NEXT: vldrht.u16 q2, [r0, q1, uxtw #1]
238 ; CHECK-NEXT: vpsel q0, q2, q0
241 %offs = load <8 x i16>, <8 x i16>* %offptr, align 2
242 %offs.zext = zext <8 x i16> %offs to <8 x i32>
243 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> %offs.zext
244 %mask = icmp sgt <8 x i16> %offs, zeroinitializer
245 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs, i32 2, <8 x i1> %mask, <8 x i16> <i16 1, i16 1, i16 1, i16 1, i16 1, i16 1, i16 1, i16 1>)
246 ret <8 x i16> %gather
249 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_2gep(i16* %base, <8 x i16>* %offptr) {
250 ; CHECK-LABEL: scaled_v8i16_i16_2gep:
251 ; CHECK: @ %bb.0: @ %entry
252 ; CHECK-NEXT: .save {r4, r5, r7, lr}
253 ; CHECK-NEXT: push {r4, r5, r7, lr}
254 ; CHECK-NEXT: vldrh.s32 q1, [r1, #8]
255 ; CHECK-NEXT: vmov.i32 q0, #0x28
256 ; CHECK-NEXT: vshl.i32 q1, q1, #1
257 ; CHECK-NEXT: vadd.i32 q1, q1, r0
258 ; CHECK-NEXT: vadd.i32 q1, q1, q0
259 ; CHECK-NEXT: vmov r2, r12, d2
260 ; CHECK-NEXT: vmov r3, lr, d3
261 ; CHECK-NEXT: vldrh.s32 q1, [r1]
262 ; CHECK-NEXT: vshl.i32 q1, q1, #1
263 ; CHECK-NEXT: vadd.i32 q1, q1, r0
264 ; CHECK-NEXT: vadd.i32 q0, q1, q0
265 ; CHECK-NEXT: vmov r4, r5, d0
266 ; CHECK-NEXT: vmov r0, r1, d1
267 ; CHECK-NEXT: ldrh r2, [r2]
268 ; CHECK-NEXT: ldrh.w r12, [r12]
269 ; CHECK-NEXT: ldrh r3, [r3]
270 ; CHECK-NEXT: ldrh.w lr, [lr]
271 ; CHECK-NEXT: ldrh r4, [r4]
272 ; CHECK-NEXT: ldrh r5, [r5]
273 ; CHECK-NEXT: vmov.16 q0[0], r4
274 ; CHECK-NEXT: ldrh r0, [r0]
275 ; CHECK-NEXT: vmov.16 q0[1], r5
276 ; CHECK-NEXT: ldrh r1, [r1]
277 ; CHECK-NEXT: vmov.16 q0[2], r0
278 ; CHECK-NEXT: vmov.16 q0[3], r1
279 ; CHECK-NEXT: vmov.16 q0[4], r2
280 ; CHECK-NEXT: vmov.16 q0[5], r12
281 ; CHECK-NEXT: vmov.16 q0[6], r3
282 ; CHECK-NEXT: vmov.16 q0[7], lr
283 ; CHECK-NEXT: pop {r4, r5, r7, pc}
285 %offs = load <8 x i16>, <8 x i16>* %offptr, align 2
286 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i16> %offs
287 %ptrs2 = getelementptr inbounds i16, <8 x i16*> %ptrs, i16 20
288 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs2, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> undef)
289 ret <8 x i16> %gather
292 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_2gep2(i16* %base, <8 x i16>* %offptr) {
293 ; CHECK-LABEL: scaled_v8i16_i16_2gep2:
294 ; CHECK: @ %bb.0: @ %entry
295 ; CHECK-NEXT: adr r1, .LCPI14_0
296 ; CHECK-NEXT: vldrw.u32 q1, [r1]
297 ; CHECK-NEXT: vldrh.u16 q0, [r0, q1, uxtw #1]
299 ; CHECK-NEXT: .p2align 4
300 ; CHECK-NEXT: @ %bb.1:
301 ; CHECK-NEXT: .LCPI14_0:
302 ; CHECK-NEXT: .short 20 @ 0x14
303 ; CHECK-NEXT: .short 23 @ 0x17
304 ; CHECK-NEXT: .short 26 @ 0x1a
305 ; CHECK-NEXT: .short 29 @ 0x1d
306 ; CHECK-NEXT: .short 32 @ 0x20
307 ; CHECK-NEXT: .short 35 @ 0x23
308 ; CHECK-NEXT: .short 38 @ 0x26
309 ; CHECK-NEXT: .short 41 @ 0x29
311 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i16> <i16 0, i16 3, i16 6, i16 9, i16 12, i16 15, i16 18, i16 21>
312 %ptrs2 = getelementptr inbounds i16,<8 x i16*> %ptrs, i16 20
313 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs2, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> undef)
314 ret <8 x i16> %gather
317 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_biggep(i16* %base) {
318 ; CHECK-LABEL: scaled_v8i16_i16_biggep:
319 ; CHECK: @ %bb.0: @ %entry
320 ; CHECK-NEXT: adr r1, .LCPI15_0
321 ; CHECK-NEXT: vldrw.u32 q1, [r1]
322 ; CHECK-NEXT: vldrh.u16 q0, [r0, q1, uxtw #1]
324 ; CHECK-NEXT: .p2align 4
325 ; CHECK-NEXT: @ %bb.1:
326 ; CHECK-NEXT: .LCPI15_0:
327 ; CHECK-NEXT: .short 20 @ 0x14
328 ; CHECK-NEXT: .short 23 @ 0x17
329 ; CHECK-NEXT: .short 26 @ 0x1a
330 ; CHECK-NEXT: .short 29 @ 0x1d
331 ; CHECK-NEXT: .short 32 @ 0x20
332 ; CHECK-NEXT: .short 35 @ 0x23
333 ; CHECK-NEXT: .short 38 @ 0x26
334 ; CHECK-NEXT: .short 41 @ 0x29
336 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> <i32 0, i32 3, i32 6, i32 9, i32 12, i32 15, i32 18, i32 21>
337 %ptrs2 = getelementptr inbounds i16,<8 x i16*> %ptrs, i32 20
338 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs2, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> undef)
339 ret <8 x i16> %gather
342 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_biggep2(i16* %base) {
343 ; CHECK-LABEL: scaled_v8i16_i16_biggep2:
344 ; CHECK: @ %bb.0: @ %entry
345 ; CHECK-NEXT: adr r1, .LCPI16_0
346 ; CHECK-NEXT: vldrw.u32 q1, [r1]
347 ; CHECK-NEXT: vldrh.u16 q0, [r0, q1, uxtw #1]
349 ; CHECK-NEXT: .p2align 4
350 ; CHECK-NEXT: @ %bb.1:
351 ; CHECK-NEXT: .LCPI16_0:
352 ; CHECK-NEXT: .short 0 @ 0x0
353 ; CHECK-NEXT: .short 3 @ 0x3
354 ; CHECK-NEXT: .short 6 @ 0x6
355 ; CHECK-NEXT: .short 9 @ 0x9
356 ; CHECK-NEXT: .short 12 @ 0xc
357 ; CHECK-NEXT: .short 15 @ 0xf
358 ; CHECK-NEXT: .short 18 @ 0x12
359 ; CHECK-NEXT: .short 21 @ 0x15
361 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> <i32 0, i32 3, i32 6, i32 9, i32 12, i32 15, i32 18, i32 21>
362 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> undef)
363 ret <8 x i16> %gather
366 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_biggep3(i16* %base) {
367 ; CHECK-LABEL: scaled_v8i16_i16_biggep3:
368 ; CHECK: @ %bb.0: @ %entry
369 ; CHECK-NEXT: .save {r4, r5, r7, lr}
370 ; CHECK-NEXT: push {r4, r5, r7, lr}
371 ; CHECK-NEXT: adr r1, .LCPI17_0
372 ; CHECK-NEXT: adr r2, .LCPI17_1
373 ; CHECK-NEXT: vldrw.u32 q0, [r1]
374 ; CHECK-NEXT: vadd.i32 q0, q0, r0
375 ; CHECK-NEXT: vmov r1, lr, d0
376 ; CHECK-NEXT: vmov r3, r12, d1
377 ; CHECK-NEXT: vldrw.u32 q0, [r2]
378 ; CHECK-NEXT: vadd.i32 q0, q0, r0
379 ; CHECK-NEXT: vmov r4, r5, d0
380 ; CHECK-NEXT: vmov r0, r2, d1
381 ; CHECK-NEXT: ldrh r1, [r1]
382 ; CHECK-NEXT: ldrh.w lr, [lr]
383 ; CHECK-NEXT: ldrh r3, [r3]
384 ; CHECK-NEXT: ldrh.w r12, [r12]
385 ; CHECK-NEXT: ldrh r4, [r4]
386 ; CHECK-NEXT: ldrh r5, [r5]
387 ; CHECK-NEXT: vmov.16 q0[0], r4
388 ; CHECK-NEXT: ldrh r0, [r0]
389 ; CHECK-NEXT: vmov.16 q0[1], r5
390 ; CHECK-NEXT: ldrh r2, [r2]
391 ; CHECK-NEXT: vmov.16 q0[2], r0
392 ; CHECK-NEXT: vmov.16 q0[3], r2
393 ; CHECK-NEXT: vmov.16 q0[4], r1
394 ; CHECK-NEXT: vmov.16 q0[5], lr
395 ; CHECK-NEXT: vmov.16 q0[6], r3
396 ; CHECK-NEXT: vmov.16 q0[7], r12
397 ; CHECK-NEXT: pop {r4, r5, r7, pc}
398 ; CHECK-NEXT: .p2align 4
399 ; CHECK-NEXT: @ %bb.1:
400 ; CHECK-NEXT: .LCPI17_0:
401 ; CHECK-NEXT: .long 131096 @ 0x20018
402 ; CHECK-NEXT: .long 131102 @ 0x2001e
403 ; CHECK-NEXT: .long 131108 @ 0x20024
404 ; CHECK-NEXT: .long 131114 @ 0x2002a
405 ; CHECK-NEXT: .LCPI17_1:
406 ; CHECK-NEXT: .long 131072 @ 0x20000
407 ; CHECK-NEXT: .long 131078 @ 0x20006
408 ; CHECK-NEXT: .long 131084 @ 0x2000c
409 ; CHECK-NEXT: .long 131090 @ 0x20012
411 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> <i32 0, i32 3, i32 6, i32 9, i32 12, i32 15, i32 18, i32 21>
412 %ptrs2 = getelementptr inbounds i16,<8 x i16*> %ptrs, i32 65536
413 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs2, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> undef)
414 ret <8 x i16> %gather
417 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_biggep4(i16* %base) {
418 ; CHECK-LABEL: scaled_v8i16_i16_biggep4:
419 ; CHECK: @ %bb.0: @ %entry
420 ; CHECK-NEXT: .save {r4, r5, r7, lr}
421 ; CHECK-NEXT: push {r4, r5, r7, lr}
422 ; CHECK-NEXT: adr r1, .LCPI18_0
423 ; CHECK-NEXT: adr r2, .LCPI18_1
424 ; CHECK-NEXT: vldrw.u32 q0, [r1]
425 ; CHECK-NEXT: vadd.i32 q0, q0, r0
426 ; CHECK-NEXT: vmov r1, lr, d0
427 ; CHECK-NEXT: vmov r3, r12, d1
428 ; CHECK-NEXT: vldrw.u32 q0, [r2]
429 ; CHECK-NEXT: vadd.i32 q0, q0, r0
430 ; CHECK-NEXT: vmov r4, r5, d0
431 ; CHECK-NEXT: vmov r0, r2, d1
432 ; CHECK-NEXT: ldrh r1, [r1]
433 ; CHECK-NEXT: ldrh.w lr, [lr]
434 ; CHECK-NEXT: ldrh r3, [r3]
435 ; CHECK-NEXT: ldrh.w r12, [r12]
436 ; CHECK-NEXT: ldrh r4, [r4]
437 ; CHECK-NEXT: ldrh r5, [r5]
438 ; CHECK-NEXT: vmov.16 q0[0], r4
439 ; CHECK-NEXT: ldrh r0, [r0]
440 ; CHECK-NEXT: vmov.16 q0[1], r5
441 ; CHECK-NEXT: ldrh r2, [r2]
442 ; CHECK-NEXT: vmov.16 q0[2], r0
443 ; CHECK-NEXT: vmov.16 q0[3], r2
444 ; CHECK-NEXT: vmov.16 q0[4], r1
445 ; CHECK-NEXT: vmov.16 q0[5], lr
446 ; CHECK-NEXT: vmov.16 q0[6], r3
447 ; CHECK-NEXT: vmov.16 q0[7], r12
448 ; CHECK-NEXT: pop {r4, r5, r7, pc}
449 ; CHECK-NEXT: .p2align 4
450 ; CHECK-NEXT: @ %bb.1:
451 ; CHECK-NEXT: .LCPI18_0:
452 ; CHECK-NEXT: .long 24 @ 0x18
453 ; CHECK-NEXT: .long 131072 @ 0x20000
454 ; CHECK-NEXT: .long 36 @ 0x24
455 ; CHECK-NEXT: .long 42 @ 0x2a
456 ; CHECK-NEXT: .LCPI18_1:
457 ; CHECK-NEXT: .long 0 @ 0x0
458 ; CHECK-NEXT: .long 6 @ 0x6
459 ; CHECK-NEXT: .long 12 @ 0xc
460 ; CHECK-NEXT: .long 18 @ 0x12
462 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> <i32 0, i32 3, i32 6, i32 9, i32 12, i32 65536, i32 18, i32 21>
463 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> undef)
464 ret <8 x i16> %gather
467 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_biggep5(<8 x i16*> %base) {
468 ; CHECK-LABEL: scaled_v8i16_i16_biggep5:
469 ; CHECK: @ %bb.0: @ %entry
470 ; CHECK-NEXT: .save {r4, r5, r6, lr}
471 ; CHECK-NEXT: push {r4, r5, r6, lr}
472 ; CHECK-NEXT: vmov.i32 q2, #0x20000
473 ; CHECK-NEXT: vadd.i32 q0, q0, q2
474 ; CHECK-NEXT: vadd.i32 q1, q1, q2
475 ; CHECK-NEXT: vmov r4, r5, d0
476 ; CHECK-NEXT: vmov r1, lr, d1
477 ; CHECK-NEXT: vmov r2, r3, d3
478 ; CHECK-NEXT: vmov r0, r12, d2
479 ; CHECK-NEXT: ldrh r4, [r4]
480 ; CHECK-NEXT: ldrh r5, [r5]
481 ; CHECK-NEXT: vmov.16 q0[0], r4
482 ; CHECK-NEXT: ldrh r1, [r1]
483 ; CHECK-NEXT: vmov.16 q0[1], r5
484 ; CHECK-NEXT: ldrh r6, [r3]
485 ; CHECK-NEXT: ldrh.w r3, [lr]
486 ; CHECK-NEXT: vmov.16 q0[2], r1
487 ; CHECK-NEXT: ldrh r0, [r0]
488 ; CHECK-NEXT: vmov.16 q0[3], r3
489 ; CHECK-NEXT: ldrh.w r12, [r12]
490 ; CHECK-NEXT: vmov.16 q0[4], r0
491 ; CHECK-NEXT: ldrh r2, [r2]
492 ; CHECK-NEXT: vmov.16 q0[5], r12
493 ; CHECK-NEXT: vmov.16 q0[6], r2
494 ; CHECK-NEXT: vmov.16 q0[7], r6
495 ; CHECK-NEXT: pop {r4, r5, r6, pc}
497 %ptrs2 = getelementptr inbounds i16,<8 x i16*> %base, i32 65536
498 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs2, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> undef)
499 ret <8 x i16> %gather
502 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_biggep6(i16* %base) {
503 ; CHECK-LABEL: scaled_v8i16_i16_biggep6:
504 ; CHECK: @ %bb.0: @ %entry
505 ; CHECK-NEXT: .save {r4, r5, r7, lr}
506 ; CHECK-NEXT: push {r4, r5, r7, lr}
507 ; CHECK-NEXT: adr r1, .LCPI20_0
508 ; CHECK-NEXT: adr r2, .LCPI20_1
509 ; CHECK-NEXT: vldrw.u32 q0, [r1]
510 ; CHECK-NEXT: vadd.i32 q0, q0, r0
511 ; CHECK-NEXT: vmov r1, lr, d0
512 ; CHECK-NEXT: vmov r3, r12, d1
513 ; CHECK-NEXT: vldrw.u32 q0, [r2]
514 ; CHECK-NEXT: vadd.i32 q0, q0, r0
515 ; CHECK-NEXT: vmov r4, r5, d0
516 ; CHECK-NEXT: vmov r0, r2, d1
517 ; CHECK-NEXT: ldrh r1, [r1]
518 ; CHECK-NEXT: ldrh.w lr, [lr]
519 ; CHECK-NEXT: ldrh r3, [r3]
520 ; CHECK-NEXT: ldrh.w r12, [r12]
521 ; CHECK-NEXT: ldrh r4, [r4]
522 ; CHECK-NEXT: ldrh r5, [r5]
523 ; CHECK-NEXT: vmov.16 q0[0], r4
524 ; CHECK-NEXT: ldrh r0, [r0]
525 ; CHECK-NEXT: vmov.16 q0[1], r5
526 ; CHECK-NEXT: ldrh r2, [r2]
527 ; CHECK-NEXT: vmov.16 q0[2], r0
528 ; CHECK-NEXT: vmov.16 q0[3], r2
529 ; CHECK-NEXT: vmov.16 q0[4], r1
530 ; CHECK-NEXT: vmov.16 q0[5], lr
531 ; CHECK-NEXT: vmov.16 q0[6], r3
532 ; CHECK-NEXT: vmov.16 q0[7], r12
533 ; CHECK-NEXT: pop {r4, r5, r7, pc}
534 ; CHECK-NEXT: .p2align 4
535 ; CHECK-NEXT: @ %bb.1:
536 ; CHECK-NEXT: .LCPI20_0:
537 ; CHECK-NEXT: .long 131074 @ 0x20002
538 ; CHECK-NEXT: .long 32 @ 0x20
539 ; CHECK-NEXT: .long 38 @ 0x26
540 ; CHECK-NEXT: .long 44 @ 0x2c
541 ; CHECK-NEXT: .LCPI20_1:
542 ; CHECK-NEXT: .long 2 @ 0x2
543 ; CHECK-NEXT: .long 8 @ 0x8
544 ; CHECK-NEXT: .long 14 @ 0xe
545 ; CHECK-NEXT: .long 20 @ 0x14
547 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i32> <i32 0, i32 3, i32 6, i32 9, i32 65536, i32 15, i32 18, i32 21>
548 %ptrs2 = getelementptr inbounds i16,<8 x i16*> %ptrs, i32 1
549 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs2, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> undef)
550 ret <8 x i16> %gather
553 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_biggep7(i16* %base, <8 x i16>* %offptr) {
554 ; CHECK-LABEL: scaled_v8i16_i16_biggep7:
555 ; CHECK: @ %bb.0: @ %entry
556 ; CHECK-NEXT: .save {r4, r5, r7, lr}
557 ; CHECK-NEXT: push {r4, r5, r7, lr}
558 ; CHECK-NEXT: adr r1, .LCPI21_0
559 ; CHECK-NEXT: adr r2, .LCPI21_1
560 ; CHECK-NEXT: vldrw.u32 q0, [r1]
561 ; CHECK-NEXT: vadd.i32 q0, q0, r0
562 ; CHECK-NEXT: vmov r1, lr, d0
563 ; CHECK-NEXT: vmov r3, r12, d1
564 ; CHECK-NEXT: vldrw.u32 q0, [r2]
565 ; CHECK-NEXT: vadd.i32 q0, q0, r0
566 ; CHECK-NEXT: vmov r4, r5, d0
567 ; CHECK-NEXT: vmov r0, r2, d1
568 ; CHECK-NEXT: ldrh r1, [r1]
569 ; CHECK-NEXT: ldrh.w lr, [lr]
570 ; CHECK-NEXT: ldrh r3, [r3]
571 ; CHECK-NEXT: ldrh.w r12, [r12]
572 ; CHECK-NEXT: ldrh r4, [r4]
573 ; CHECK-NEXT: ldrh r5, [r5]
574 ; CHECK-NEXT: vmov.16 q0[0], r4
575 ; CHECK-NEXT: ldrh r0, [r0]
576 ; CHECK-NEXT: vmov.16 q0[1], r5
577 ; CHECK-NEXT: ldrh r2, [r2]
578 ; CHECK-NEXT: vmov.16 q0[2], r0
579 ; CHECK-NEXT: vmov.16 q0[3], r2
580 ; CHECK-NEXT: vmov.16 q0[4], r1
581 ; CHECK-NEXT: vmov.16 q0[5], lr
582 ; CHECK-NEXT: vmov.16 q0[6], r3
583 ; CHECK-NEXT: vmov.16 q0[7], r12
584 ; CHECK-NEXT: pop {r4, r5, r7, pc}
585 ; CHECK-NEXT: .p2align 4
586 ; CHECK-NEXT: @ %bb.1:
587 ; CHECK-NEXT: .LCPI21_0:
588 ; CHECK-NEXT: .long 1224 @ 0x4c8
589 ; CHECK-NEXT: .long 1230 @ 0x4ce
590 ; CHECK-NEXT: .long 1236 @ 0x4d4
591 ; CHECK-NEXT: .long 1242 @ 0x4da
592 ; CHECK-NEXT: .LCPI21_1:
593 ; CHECK-NEXT: .long 128 @ 0x80
594 ; CHECK-NEXT: .long 1206 @ 0x4b6
595 ; CHECK-NEXT: .long 1212 @ 0x4bc
596 ; CHECK-NEXT: .long 1218 @ 0x4c2
598 %ptrs = getelementptr inbounds i16, i16* %base, <8 x i16> <i16 65000, i16 3, i16 6, i16 9, i16 12, i16 15, i16 18, i16 21>
599 %ptrs2 = getelementptr inbounds i16,<8 x i16*> %ptrs, i16 600
600 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs2, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> undef)
601 ret <8 x i16> %gather
604 define arm_aapcs_vfpcc <8 x i16> @scaled_v8i16_i16_basei32(i32* %base, <8 x i16>* %offptr) {
605 ; CHECK-LABEL: scaled_v8i16_i16_basei32:
606 ; CHECK: @ %bb.0: @ %entry
607 ; CHECK-NEXT: .save {r4, r5, r7, lr}
608 ; CHECK-NEXT: push {r4, r5, r7, lr}
609 ; CHECK-NEXT: vldrh.u32 q0, [r1, #8]
610 ; CHECK-NEXT: vshl.i32 q0, q0, #2
611 ; CHECK-NEXT: vadd.i32 q0, q0, r0
612 ; CHECK-NEXT: vmov r2, r12, d0
613 ; CHECK-NEXT: vmov r3, lr, d1
614 ; CHECK-NEXT: vldrh.u32 q0, [r1]
615 ; CHECK-NEXT: vshl.i32 q0, q0, #2
616 ; CHECK-NEXT: vadd.i32 q0, q0, r0
617 ; CHECK-NEXT: vmov r4, r5, d0
618 ; CHECK-NEXT: vmov r0, r1, d1
619 ; CHECK-NEXT: ldrh r2, [r2]
620 ; CHECK-NEXT: ldrh.w r12, [r12]
621 ; CHECK-NEXT: ldrh r3, [r3]
622 ; CHECK-NEXT: ldrh.w lr, [lr]
623 ; CHECK-NEXT: ldrh r4, [r4]
624 ; CHECK-NEXT: ldrh r5, [r5]
625 ; CHECK-NEXT: vmov.16 q0[0], r4
626 ; CHECK-NEXT: ldrh r0, [r0]
627 ; CHECK-NEXT: vmov.16 q0[1], r5
628 ; CHECK-NEXT: ldrh r1, [r1]
629 ; CHECK-NEXT: vmov.16 q0[2], r0
630 ; CHECK-NEXT: vmov.16 q0[3], r1
631 ; CHECK-NEXT: vmov.16 q0[4], r2
632 ; CHECK-NEXT: vmov.16 q0[5], r12
633 ; CHECK-NEXT: vmov.16 q0[6], r3
634 ; CHECK-NEXT: vmov.16 q0[7], lr
635 ; CHECK-NEXT: pop {r4, r5, r7, pc}
637 %offs = load <8 x i16>, <8 x i16>* %offptr, align 2
638 %offs.zext = zext <8 x i16> %offs to <8 x i32>
639 %ptrs = getelementptr inbounds i32, i32* %base, <8 x i32> %offs.zext
640 %ptrs.cast = bitcast <8 x i32*> %ptrs to <8 x i16*>
641 %gather = call <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*> %ptrs.cast, i32 2, <8 x i1> <i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true, i1 true>, <8 x i16> undef)
642 ret <8 x i16> %gather
645 declare <8 x i8> @llvm.masked.gather.v8i8.v8p0i8(<8 x i8*>, i32, <8 x i1>, <8 x i8>) #1
646 declare <8 x i16> @llvm.masked.gather.v8i16.v8p0i16(<8 x i16*>, i32, <8 x i1>, <8 x i16>) #1
647 declare <8 x half> @llvm.masked.gather.v8f16.v8p0f16(<8 x half*>, i32, <8 x i1>, <8 x half>) #1