[ORC] Add std::tuple support to SimplePackedSerialization.
[llvm-project.git] / llvm / test / CodeGen / AArch64 / sve-masked-gather-legalize.ll
blobf97fa34392a25291826a24125f3725065b2307fb
1 ; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py
2 ; RUN: llc -mtriple=aarch64--linux-gnu -mattr=+sve -aarch64-enable-mgather-combine=0 < %s | FileCheck %s
3 ; RUN: llc -mtriple=aarch64--linux-gnu -mattr=+sve -aarch64-enable-mgather-combine=1 < %s | FileCheck %s
5 ; Test for multiple uses of the mgather where the s/zext should not be combined
7 define <vscale x 2 x i64> @masked_sgather_sext(i8* %base, <vscale x 2 x i64> %offsets, <vscale x 2 x i1> %mask, <vscale x 2 x i8> %vals) {
8 ; CHECK-LABEL: masked_sgather_sext:
9 ; CHECK:       // %bb.0:
10 ; CHECK-NEXT:    ld1sb { z0.d }, p0/z, [x0, z0.d]
11 ; CHECK-NEXT:    ptrue p0.d
12 ; CHECK-NEXT:    movprfx z2, z0
13 ; CHECK-NEXT:    sxtb z2.d, p0/m, z0.d
14 ; CHECK-NEXT:    add z0.d, z0.d, z1.d
15 ; CHECK-NEXT:    sxtb z0.d, p0/m, z0.d
16 ; CHECK-NEXT:    mul z0.d, p0/m, z0.d, z2.d
17 ; CHECK-NEXT:    ret
18   %ptrs = getelementptr i8, i8* %base, <vscale x 2 x i64> %offsets
19   %data = call <vscale x 2 x i8> @llvm.masked.gather.nxv2i8(<vscale x 2 x i8*> %ptrs, i32 1, <vscale x 2 x i1> %mask, <vscale x 2 x i8> undef)
20   %data.sext = sext <vscale x 2 x i8> %data to <vscale x 2 x i64>
21   %add = add <vscale x 2 x i8> %data, %vals
22   %add.sext = sext <vscale x 2 x i8> %add to <vscale x 2 x i64>
23   %mul = mul <vscale x 2 x i64> %data.sext, %add.sext
24   ret <vscale x 2 x i64> %mul
27 define <vscale x 2 x i64> @masked_sgather_zext(i8* %base, <vscale x 2 x i64> %offsets, <vscale x 2 x i1> %mask, <vscale x 2 x i8> %vals) {
28 ; CHECK-LABEL: masked_sgather_zext:
29 ; CHECK:       // %bb.0:
30 ; CHECK-NEXT: ld1sb { z0.d }, p0/z, [x0, z0.d]
31 ; CHECK-NEXT: ptrue p0.d
32 ; CHECK-NEXT: add z1.d, z0.d, z1.d
33 ; CHECK-NEXT: and z0.d, z0.d, #0xff
34 ; CHECK-NEXT: and z1.d, z1.d, #0xff
35 ; CHECK-NEXT: mul z0.d, p0/m, z0.d, z1.d
36 ; CHECK-NEXT: ret
37   %ptrs = getelementptr i8, i8* %base, <vscale x 2 x i64> %offsets
38   %data = call <vscale x 2 x i8> @llvm.masked.gather.nxv2i8(<vscale x 2 x i8*> %ptrs, i32 1, <vscale x 2 x i1> %mask, <vscale x 2 x i8> undef)
39   %data.zext = zext <vscale x 2 x i8> %data to <vscale x 2 x i64>
40   %add = add <vscale x 2 x i8> %data, %vals
41   %add.zext = zext <vscale x 2 x i8> %add to <vscale x 2 x i64>
42   %mul = mul <vscale x 2 x i64> %data.zext, %add.zext
43   ret <vscale x 2 x i64> %mul
46 ; Tests that exercise various type legalisation scenarios for ISD::MGATHER.
48 ; Code generate load of an illegal datatype via promotion.
49 define <vscale x 2 x i8> @masked_gather_nxv2i8(<vscale x 2 x i8*> %ptrs, <vscale x 2 x i1> %mask) {
50 ; CHECK-LABEL: masked_gather_nxv2i8:
51 ; CHECK: ld1sb { z0.d }, p0/z, [z0.d]
52 ; CHECK: ret
53   %data = call <vscale x 2 x i8> @llvm.masked.gather.nxv2i8(<vscale x 2 x i8*> %ptrs, i32 1, <vscale x 2 x i1> %mask, <vscale x 2 x i8> undef)
54   ret <vscale x 2 x i8> %data
57 ; Code generate load of an illegal datatype via promotion.
58 define <vscale x 2 x i16> @masked_gather_nxv2i16(<vscale x 2 x i16*> %ptrs, <vscale x 2 x i1> %mask) {
59 ; CHECK-LABEL: masked_gather_nxv2i16:
60 ; CHECK: ld1sh { z0.d }, p0/z, [z0.d]
61 ; CHECK: ret
62   %data = call <vscale x 2 x i16> @llvm.masked.gather.nxv2i16(<vscale x 2 x i16*> %ptrs, i32 2, <vscale x 2 x i1> %mask, <vscale x 2 x i16> undef)
63   ret <vscale x 2 x i16> %data
66 ; Code generate load of an illegal datatype via promotion.
67 define <vscale x 2 x i32> @masked_gather_nxv2i32(<vscale x 2 x i32*> %ptrs, <vscale x 2 x i1> %mask) {
68 ; CHECK-LABEL: masked_gather_nxv2i32:
69 ; CHECK: ld1sw { z0.d }, p0/z, [z0.d]
70 ; CHECK: ret
71   %data = call <vscale x 2 x i32> @llvm.masked.gather.nxv2i32(<vscale x 2 x i32*> %ptrs, i32 4, <vscale x 2 x i1> %mask, <vscale x 2 x i32> undef)
72   ret <vscale x 2 x i32> %data
75 define <vscale x 4 x half> @masked_gather_nxv4f16(<vscale x 4 x half*> %ptrs, <vscale x 4 x i1> %mask) {
76 ; CHECK-LABEL: masked_gather_nxv4f16:
77 ; CHECK:       // %bb.0:
78 ; CHECK-NEXT:    pfalse p1.b
79 ; CHECK-NEXT:    zip2 p2.s, p0.s, p1.s
80 ; CHECK-NEXT:    zip1 p0.s, p0.s, p1.s
81 ; CHECK-NEXT:    ld1h { z1.d }, p2/z, [z1.d]
82 ; CHECK-NEXT:    ld1h { z0.d }, p0/z, [z0.d]
83 ; CHECK-NEXT:    uzp1 z0.s, z0.s, z1.s
84 ; CHECK-NEXT:    ret
85   %data = call <vscale x 4 x half> @llvm.masked.gather.nxv4f16(<vscale x 4 x half*> %ptrs, i32 0, <vscale x 4 x i1> %mask, <vscale x 4 x half> undef)
86   ret <vscale x 4 x half> %data
89 define <vscale x 2 x float> @masked_gather_nxv2f32(float* %base, <vscale x 2 x i16> %indices, <vscale x 2 x i1> %mask) {
90 ; CHECK-LABEL: masked_gather_nxv2f32:
91 ; CHECK:       // %bb.0:
92 ; CHECK-NEXT:    ptrue p1.d
93 ; CHECK-NEXT:    sxth z0.d, p1/m, z0.d
94 ; CHECK-NEXT:    ld1w { z0.d }, p0/z, [x0, z0.d, sxtw #2]
95 ; CHECK-NEXT:    ret
96   %ptrs = getelementptr float, float* %base, <vscale x 2 x i16> %indices
97   %data = call <vscale x 2 x float> @llvm.masked.gather.nxv2f32(<vscale x 2 x float*> %ptrs, i32 1, <vscale x 2 x i1> %mask, <vscale x 2 x float> undef)
98   ret <vscale x 2 x float> %data
101 define <vscale x 8 x half> @masked_gather_nxv8f16(<vscale x 8 x half*> %ptrs, <vscale x 8 x i1> %mask) {
102 ; CHECK-LABEL: masked_gather_nxv8f16:
103 ; CHECK:       // %bb.0:
104 ; CHECK-NEXT:    pfalse p1.b
105 ; CHECK-NEXT:    zip2 p2.h, p0.h, p1.h
106 ; CHECK-NEXT:    zip1 p0.h, p0.h, p1.h
107 ; CHECK-NEXT:    zip2 p3.s, p2.s, p1.s
108 ; CHECK-NEXT:    zip1 p2.s, p2.s, p1.s
109 ; CHECK-NEXT:    ld1h { z3.d }, p3/z, [z3.d]
110 ; CHECK-NEXT:    ld1h { z2.d }, p2/z, [z2.d]
111 ; CHECK-NEXT:    zip2 p2.s, p0.s, p1.s
112 ; CHECK-NEXT:    zip1 p0.s, p0.s, p1.s
113 ; CHECK-NEXT:    ld1h { z1.d }, p2/z, [z1.d]
114 ; CHECK-NEXT:    ld1h { z0.d }, p0/z, [z0.d]
115 ; CHECK-NEXT:    uzp1 z2.s, z2.s, z3.s
116 ; CHECK-NEXT:    uzp1 z0.s, z0.s, z1.s
117 ; CHECK-NEXT:    uzp1 z0.h, z0.h, z2.h
118 ; CHECK-NEXT:    ret
119   %data = call <vscale x 8 x half> @llvm.masked.gather.nxv8f16(<vscale x 8 x half*> %ptrs, i32 2, <vscale x 8 x i1> %mask, <vscale x 8 x half> undef)
120   ret <vscale x 8 x half> %data
123 define <vscale x 8 x bfloat> @masked_gather_nxv8bf16(bfloat* %base, <vscale x 8 x i16> %indices, <vscale x 8 x i1> %mask) #0 {
124 ; CHECK-LABEL: masked_gather_nxv8bf16:
125 ; CHECK:       // %bb.0:
126 ; CHECK-NEXT:    pfalse p1.b
127 ; CHECK-NEXT:    sunpkhi z1.s, z0.h
128 ; CHECK-NEXT:    sunpklo z0.s, z0.h
129 ; CHECK-NEXT:    zip2 p2.h, p0.h, p1.h
130 ; CHECK-NEXT:    zip1 p0.h, p0.h, p1.h
131 ; CHECK-NEXT:    ld1h { z1.s }, p2/z, [x0, z1.s, sxtw #1]
132 ; CHECK-NEXT:    ld1h { z0.s }, p0/z, [x0, z0.s, sxtw #1]
133 ; CHECK-NEXT:    uzp1 z0.h, z0.h, z1.h
134 ; CHECK-NEXT:    ret
135   %ptrs = getelementptr bfloat, bfloat* %base, <vscale x 8 x i16> %indices
136   %data = call <vscale x 8 x bfloat> @llvm.masked.gather.nxv8bf16(<vscale x 8 x bfloat*> %ptrs, i32 1, <vscale x 8 x i1> %mask, <vscale x 8 x bfloat> undef)
137   ret <vscale x 8 x bfloat> %data
140 define <vscale x 4 x double> @masked_gather_nxv4f64(double* %base, <vscale x 4 x i16> %indices, <vscale x 4 x i1> %mask) {;
141 ; CHECK-LABEL: masked_gather_nxv4f64:
142 ; CHECK:       // %bb.0:
143 ; CHECK-NEXT:    ptrue p1.s
144 ; CHECK-NEXT:    pfalse p2.b
145 ; CHECK-NEXT:    sxth z0.s, p1/m, z0.s
146 ; CHECK-NEXT:    zip1 p1.s, p0.s, p2.s
147 ; CHECK-NEXT:    zip2 p0.s, p0.s, p2.s
148 ; CHECK-NEXT:    sunpklo z1.d, z0.s
149 ; CHECK-NEXT:    sunpkhi z2.d, z0.s
150 ; CHECK-NEXT:    ld1d { z0.d }, p1/z, [x0, z1.d, lsl #3]
151 ; CHECK-NEXT:    ld1d { z1.d }, p0/z, [x0, z2.d, lsl #3]
152 ; CHECK-NEXT:    ret
153   %ptrs = getelementptr double, double* %base, <vscale x 4 x i16> %indices
154   %data = call <vscale x 4 x double> @llvm.masked.gather.nxv4f64(<vscale x 4 x double*> %ptrs, i32 1, <vscale x 4 x i1> %mask, <vscale x 4 x double> undef)
155   ret <vscale x 4 x double> %data
158 define <vscale x 8 x float> @masked_gather_nxv8f32(float* %base, <vscale x 8 x i32> %offsets, <vscale x 8 x i1> %mask) {
159 ; CHECK-LABEL: masked_gather_nxv8f32:
160 ; CHECK:       // %bb.0:
161 ; CHECK-NEXT:    pfalse p1.b
162 ; CHECK-NEXT:    zip1 p2.h, p0.h, p1.h
163 ; CHECK-NEXT:    zip2 p0.h, p0.h, p1.h
164 ; CHECK-NEXT:    ld1w { z0.s }, p2/z, [x0, z0.s, uxtw #2]
165 ; CHECK-NEXT:    ld1w { z1.s }, p0/z, [x0, z1.s, uxtw #2]
166 ; CHECK-NEXT:    ret
167   %offsets.zext = zext <vscale x 8 x i32> %offsets to <vscale x 8 x i64>
168   %ptrs = getelementptr float, float* %base, <vscale x 8 x i64> %offsets.zext
169   %vals = call <vscale x 8 x float> @llvm.masked.gather.nxv8f32(<vscale x 8 x float*> %ptrs, i32 4, <vscale x 8 x i1> %mask, <vscale x 8 x float> undef)
170   ret <vscale x 8 x float> %vals
173 ; Code generate the worst case scenario when all vector types are legal.
174 define <vscale x 16 x i8> @masked_gather_nxv16i8(i8* %base, <vscale x 16 x i8> %indices, <vscale x 16 x i1> %mask) {
175 ; CHECK-LABEL: masked_gather_nxv16i8:
176 ; CHECK-DAG: ld1sb { {{z[0-9]+}}.s }, {{p[0-9]+}}/z, [x0, {{z[0-9]+}}.s, sxtw]
177 ; CHECK-DAG: ld1sb { {{z[0-9]+}}.s }, {{p[0-9]+}}/z, [x0, {{z[0-9]+}}.s, sxtw]
178 ; CHECK-DAG: ld1sb { {{z[0-9]+}}.s }, {{p[0-9]+}}/z, [x0, {{z[0-9]+}}.s, sxtw]
179 ; CHECK-DAG: ld1sb { {{z[0-9]+}}.s }, {{p[0-9]+}}/z, [x0, {{z[0-9]+}}.s, sxtw]
180 ; CHECK: ret
181   %ptrs = getelementptr i8, i8* %base, <vscale x 16 x i8> %indices
182   %data = call <vscale x 16 x i8> @llvm.masked.gather.nxv16i8(<vscale x 16 x i8*> %ptrs, i32 1, <vscale x 16 x i1> %mask, <vscale x 16 x i8> undef)
183   ret <vscale x 16 x i8> %data
186 ; Code generate the worst case scenario when all vector types are illegal.
187 define <vscale x 32 x i32> @masked_gather_nxv32i32(i32* %base, <vscale x 32 x i32> %indices, <vscale x 32 x i1> %mask) {
188 ; CHECK-LABEL: masked_gather_nxv32i32:
189 ; CHECK-NOT: unpkhi
190 ; CHECK-DAG: ld1w { {{z[0-9]+}}.s }, {{p[0-9]+}}/z, [x0, z0.s, sxtw #2]
191 ; CHECK-DAG: ld1w { {{z[0-9]+}}.s }, {{p[0-9]+}}/z, [x0, z1.s, sxtw #2]
192 ; CHECK-DAG: ld1w { {{z[0-9]+}}.s }, {{p[0-9]+}}/z, [x0, z2.s, sxtw #2]
193 ; CHECK-DAG: ld1w { {{z[0-9]+}}.s }, {{p[0-9]+}}/z, [x0, z3.s, sxtw #2]
194 ; CHECK-DAG: ld1w { {{z[0-9]+}}.s }, {{p[0-9]+}}/z, [x0, z4.s, sxtw #2]
195 ; CHECK-DAG: ld1w { {{z[0-9]+}}.s }, {{p[0-9]+}}/z, [x0, z5.s, sxtw #2]
196 ; CHECK-DAG: ld1w { {{z[0-9]+}}.s }, {{p[0-9]+}}/z, [x0, z6.s, sxtw #2]
197 ; CHECK-DAG: ld1w { {{z[0-9]+}}.s }, {{p[0-9]+}}/z, [x0, z7.s, sxtw #2]
198 ; CHECK: ret
199   %ptrs = getelementptr i32, i32* %base, <vscale x 32 x i32> %indices
200   %data = call <vscale x 32 x i32> @llvm.masked.gather.nxv32i32(<vscale x 32 x i32*> %ptrs, i32 4, <vscale x 32 x i1> %mask, <vscale x 32 x i32> undef)
201   ret <vscale x 32 x i32> %data
204 ; TODO: Currently, the sign extend gets applied to the values after a 'uzp1' of two
205 ; registers, so it doesn't get folded away. Same for any other vector-of-pointers
206 ; style gathers which don't fit in an <vscale x 2 x type*> single register. Better folding
207 ; is required before we can check those off.
208 define <vscale x 4 x i32> @masked_sgather_nxv4i8(<vscale x 4 x i8*> %ptrs, <vscale x 4 x i1> %mask) {
209 ; CHECK-LABEL: masked_sgather_nxv4i8:
210 ; CHECK:         pfalse p1.b
211 ; CHECK-NEXT:    zip2 p2.s, p0.s, p1.s
212 ; CHECK-NEXT:    zip1 p0.s, p0.s, p1.s
213 ; CHECK-NEXT:    ld1sb { z1.d }, p2/z, [z1.d]
214 ; CHECK-NEXT:    ld1sb { z0.d }, p0/z, [z0.d]
215 ; CHECK-NEXT:    ptrue p0.s
216 ; CHECK-NEXT:    uzp1 z0.s, z0.s, z1.s
217 ; CHECK-NEXT:    sxtb z0.s, p0/m, z0.s
218 ; CHECK-NEXT:    ret
219   %vals = call <vscale x 4 x i8> @llvm.masked.gather.nxv4i8(<vscale x 4 x i8*> %ptrs, i32 1, <vscale x 4 x i1> %mask, <vscale x 4 x i8> undef)
220   %svals = sext <vscale x 4 x i8> %vals to <vscale x 4 x i32>
221   ret <vscale x 4 x i32> %svals
224 declare <vscale x 2 x i8> @llvm.masked.gather.nxv2i8(<vscale x 2 x i8*>, i32, <vscale x 2 x i1>, <vscale x 2 x i8>)
225 declare <vscale x 2 x i16> @llvm.masked.gather.nxv2i16(<vscale x 2 x i16*>, i32, <vscale x 2 x i1>, <vscale x 2 x i16>)
226 declare <vscale x 2 x i32> @llvm.masked.gather.nxv2i32(<vscale x 2 x i32*>, i32, <vscale x 2 x i1>, <vscale x 2 x i32>)
227 declare <vscale x 4 x i8> @llvm.masked.gather.nxv4i8(<vscale x 4 x i8*>, i32, <vscale x 4 x i1>, <vscale x 4 x i8>)
228 declare <vscale x 16 x i8> @llvm.masked.gather.nxv16i8(<vscale x 16 x i8*>, i32, <vscale x 16 x i1>, <vscale x 16 x i8>)
229 declare <vscale x 32 x i32> @llvm.masked.gather.nxv32i32(<vscale x 32 x i32*>, i32, <vscale x 32 x i1>, <vscale x 32 x i32>)
231 declare <vscale x 4 x half> @llvm.masked.gather.nxv4f16(<vscale x 4 x half*>, i32, <vscale x 4 x i1>, <vscale x 4 x half>)
232 declare <vscale x 8 x half> @llvm.masked.gather.nxv8f16(<vscale x 8 x half*>, i32, <vscale x 8 x i1>, <vscale x 8 x half>)
233 declare <vscale x 8 x bfloat> @llvm.masked.gather.nxv8bf16(<vscale x 8 x bfloat*>, i32, <vscale x 8 x i1>, <vscale x 8 x bfloat>)
234 declare <vscale x 2 x float> @llvm.masked.gather.nxv2f32(<vscale x 2 x float*>, i32, <vscale x 2 x i1>, <vscale x 2 x float>)
235 declare <vscale x 8 x float> @llvm.masked.gather.nxv8f32(<vscale x 8 x float*>, i32, <vscale x 8 x i1>, <vscale x 8 x float>)
236 declare <vscale x 4 x double> @llvm.masked.gather.nxv4f64(<vscale x 4 x double*>, i32, <vscale x 4 x i1>, <vscale x 4 x double>)
237 attributes #0 = { "target-features"="+sve,+bf16" }