1 /*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include "code_generator_arm_vixl.h"
18 #include "mirror/array-inl.h"
19
20 namespace vixl32 = vixl::aarch32;
21 using namespace vixl32; // NOLINT(build/namespaces)
22
23 namespace art HIDDEN {
24 namespace arm {
25
26 using helpers::DRegisterFrom;
27 using helpers::Int64ConstantFrom;
28 using helpers::InputDRegisterAt;
29 using helpers::InputRegisterAt;
30 using helpers::OutputDRegister;
31 using helpers::OutputRegister;
32 using helpers::RegisterFrom;
33
34 #define __ GetVIXLAssembler()->
35
VisitVecReplicateScalar(HVecReplicateScalar * instruction)36 void LocationsBuilderARMVIXL::VisitVecReplicateScalar(HVecReplicateScalar* instruction) {
37 LocationSummary* locations = new (GetGraph()->GetAllocator()) LocationSummary(instruction);
38 switch (instruction->GetPackedType()) {
39 case DataType::Type::kBool:
40 case DataType::Type::kUint8:
41 case DataType::Type::kInt8:
42 case DataType::Type::kUint16:
43 case DataType::Type::kInt16:
44 case DataType::Type::kInt32:
45 locations->SetInAt(0, Location::RequiresRegister());
46 locations->SetOut(Location::RequiresFpuRegister());
47 break;
48 default:
49 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
50 UNREACHABLE();
51 }
52 }
53
VisitVecReplicateScalar(HVecReplicateScalar * instruction)54 void InstructionCodeGeneratorARMVIXL::VisitVecReplicateScalar(HVecReplicateScalar* instruction) {
55 LocationSummary* locations = instruction->GetLocations();
56 vixl32::DRegister dst = DRegisterFrom(locations->Out());
57 switch (instruction->GetPackedType()) {
58 case DataType::Type::kBool:
59 case DataType::Type::kUint8:
60 case DataType::Type::kInt8:
61 DCHECK_EQ(8u, instruction->GetVectorLength());
62 __ Vdup(Untyped8, dst, InputRegisterAt(instruction, 0));
63 break;
64 case DataType::Type::kUint16:
65 case DataType::Type::kInt16:
66 DCHECK_EQ(4u, instruction->GetVectorLength());
67 __ Vdup(Untyped16, dst, InputRegisterAt(instruction, 0));
68 break;
69 case DataType::Type::kInt32:
70 DCHECK_EQ(2u, instruction->GetVectorLength());
71 __ Vdup(Untyped32, dst, InputRegisterAt(instruction, 0));
72 break;
73 default:
74 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
75 UNREACHABLE();
76 }
77 }
78
VisitVecExtractScalar(HVecExtractScalar * instruction)79 void LocationsBuilderARMVIXL::VisitVecExtractScalar(HVecExtractScalar* instruction) {
80 LocationSummary* locations = new (GetGraph()->GetAllocator()) LocationSummary(instruction);
81 switch (instruction->GetPackedType()) {
82 case DataType::Type::kInt32:
83 locations->SetInAt(0, Location::RequiresFpuRegister());
84 locations->SetOut(Location::RequiresRegister());
85 break;
86 default:
87 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
88 UNREACHABLE();
89 }
90 }
91
VisitVecExtractScalar(HVecExtractScalar * instruction)92 void InstructionCodeGeneratorARMVIXL::VisitVecExtractScalar(HVecExtractScalar* instruction) {
93 LocationSummary* locations = instruction->GetLocations();
94 vixl32::DRegister src = DRegisterFrom(locations->InAt(0));
95 switch (instruction->GetPackedType()) {
96 case DataType::Type::kInt32:
97 DCHECK_EQ(2u, instruction->GetVectorLength());
98 __ Vmov(OutputRegister(instruction), DRegisterLane(src, 0));
99 break;
100 default:
101 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
102 UNREACHABLE();
103 }
104 }
105
106 // Helper to set up locations for vector unary operations.
CreateVecUnOpLocations(ArenaAllocator * allocator,HVecUnaryOperation * instruction)107 static void CreateVecUnOpLocations(ArenaAllocator* allocator, HVecUnaryOperation* instruction) {
108 LocationSummary* locations = new (allocator) LocationSummary(instruction);
109 switch (instruction->GetPackedType()) {
110 case DataType::Type::kBool:
111 locations->SetInAt(0, Location::RequiresFpuRegister());
112 locations->SetOut(Location::RequiresFpuRegister(),
113 instruction->IsVecNot() ? Location::kOutputOverlap
114 : Location::kNoOutputOverlap);
115 break;
116 case DataType::Type::kUint8:
117 case DataType::Type::kInt8:
118 case DataType::Type::kUint16:
119 case DataType::Type::kInt16:
120 case DataType::Type::kInt32:
121 locations->SetInAt(0, Location::RequiresFpuRegister());
122 locations->SetOut(Location::RequiresFpuRegister(), Location::kNoOutputOverlap);
123 break;
124 default:
125 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
126 UNREACHABLE();
127 }
128 }
129
VisitVecReduce(HVecReduce * instruction)130 void LocationsBuilderARMVIXL::VisitVecReduce(HVecReduce* instruction) {
131 CreateVecUnOpLocations(GetGraph()->GetAllocator(), instruction);
132 }
133
VisitVecReduce(HVecReduce * instruction)134 void InstructionCodeGeneratorARMVIXL::VisitVecReduce(HVecReduce* instruction) {
135 LocationSummary* locations = instruction->GetLocations();
136 vixl32::DRegister src = DRegisterFrom(locations->InAt(0));
137 vixl32::DRegister dst = DRegisterFrom(locations->Out());
138 switch (instruction->GetPackedType()) {
139 case DataType::Type::kInt32:
140 DCHECK_EQ(2u, instruction->GetVectorLength());
141 switch (instruction->GetReductionKind()) {
142 case HVecReduce::kSum:
143 __ Vpadd(DataTypeValue::I32, dst, src, src);
144 break;
145 case HVecReduce::kMin:
146 __ Vpmin(DataTypeValue::S32, dst, src, src);
147 break;
148 case HVecReduce::kMax:
149 __ Vpmax(DataTypeValue::S32, dst, src, src);
150 break;
151 }
152 break;
153 default:
154 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
155 UNREACHABLE();
156 }
157 }
158
VisitVecCnv(HVecCnv * instruction)159 void LocationsBuilderARMVIXL::VisitVecCnv(HVecCnv* instruction) {
160 CreateVecUnOpLocations(GetGraph()->GetAllocator(), instruction);
161 }
162
VisitVecCnv(HVecCnv * instruction)163 void InstructionCodeGeneratorARMVIXL::VisitVecCnv(HVecCnv* instruction) {
164 LOG(FATAL) << "No SIMD for " << instruction->GetId();
165 }
166
VisitVecNeg(HVecNeg * instruction)167 void LocationsBuilderARMVIXL::VisitVecNeg(HVecNeg* instruction) {
168 CreateVecUnOpLocations(GetGraph()->GetAllocator(), instruction);
169 }
170
VisitVecNeg(HVecNeg * instruction)171 void InstructionCodeGeneratorARMVIXL::VisitVecNeg(HVecNeg* instruction) {
172 LocationSummary* locations = instruction->GetLocations();
173 vixl32::DRegister src = DRegisterFrom(locations->InAt(0));
174 vixl32::DRegister dst = DRegisterFrom(locations->Out());
175 switch (instruction->GetPackedType()) {
176 case DataType::Type::kUint8:
177 case DataType::Type::kInt8:
178 DCHECK_EQ(8u, instruction->GetVectorLength());
179 __ Vneg(DataTypeValue::S8, dst, src);
180 break;
181 case DataType::Type::kUint16:
182 case DataType::Type::kInt16:
183 DCHECK_EQ(4u, instruction->GetVectorLength());
184 __ Vneg(DataTypeValue::S16, dst, src);
185 break;
186 case DataType::Type::kInt32:
187 DCHECK_EQ(2u, instruction->GetVectorLength());
188 __ Vneg(DataTypeValue::S32, dst, src);
189 break;
190 default:
191 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
192 UNREACHABLE();
193 }
194 }
195
VisitVecAbs(HVecAbs * instruction)196 void LocationsBuilderARMVIXL::VisitVecAbs(HVecAbs* instruction) {
197 CreateVecUnOpLocations(GetGraph()->GetAllocator(), instruction);
198 }
199
VisitVecAbs(HVecAbs * instruction)200 void InstructionCodeGeneratorARMVIXL::VisitVecAbs(HVecAbs* instruction) {
201 LocationSummary* locations = instruction->GetLocations();
202 vixl32::DRegister src = DRegisterFrom(locations->InAt(0));
203 vixl32::DRegister dst = DRegisterFrom(locations->Out());
204 switch (instruction->GetPackedType()) {
205 case DataType::Type::kInt8:
206 DCHECK_EQ(8u, instruction->GetVectorLength());
207 __ Vabs(DataTypeValue::S8, dst, src);
208 break;
209 case DataType::Type::kInt16:
210 DCHECK_EQ(4u, instruction->GetVectorLength());
211 __ Vabs(DataTypeValue::S16, dst, src);
212 break;
213 case DataType::Type::kInt32:
214 DCHECK_EQ(2u, instruction->GetVectorLength());
215 __ Vabs(DataTypeValue::S32, dst, src);
216 break;
217 default:
218 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
219 UNREACHABLE();
220 }
221 }
222
VisitVecNot(HVecNot * instruction)223 void LocationsBuilderARMVIXL::VisitVecNot(HVecNot* instruction) {
224 CreateVecUnOpLocations(GetGraph()->GetAllocator(), instruction);
225 }
226
VisitVecNot(HVecNot * instruction)227 void InstructionCodeGeneratorARMVIXL::VisitVecNot(HVecNot* instruction) {
228 LocationSummary* locations = instruction->GetLocations();
229 vixl32::DRegister src = DRegisterFrom(locations->InAt(0));
230 vixl32::DRegister dst = DRegisterFrom(locations->Out());
231 switch (instruction->GetPackedType()) {
232 case DataType::Type::kBool: // special case boolean-not
233 DCHECK_EQ(8u, instruction->GetVectorLength());
234 __ Vmov(I8, dst, 1);
235 __ Veor(dst, dst, src);
236 break;
237 case DataType::Type::kUint8:
238 case DataType::Type::kInt8:
239 case DataType::Type::kUint16:
240 case DataType::Type::kInt16:
241 case DataType::Type::kInt32:
242 __ Vmvn(I8, dst, src); // lanes do not matter
243 break;
244 default:
245 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
246 UNREACHABLE();
247 }
248 }
249
250 // Helper to set up locations for vector binary operations.
CreateVecBinOpLocations(ArenaAllocator * allocator,HVecBinaryOperation * instruction)251 static void CreateVecBinOpLocations(ArenaAllocator* allocator, HVecBinaryOperation* instruction) {
252 LocationSummary* locations = new (allocator) LocationSummary(instruction);
253 switch (instruction->GetPackedType()) {
254 case DataType::Type::kBool:
255 case DataType::Type::kUint8:
256 case DataType::Type::kInt8:
257 case DataType::Type::kUint16:
258 case DataType::Type::kInt16:
259 case DataType::Type::kInt32:
260 locations->SetInAt(0, Location::RequiresFpuRegister());
261 locations->SetInAt(1, Location::RequiresFpuRegister());
262 locations->SetOut(Location::RequiresFpuRegister(), Location::kNoOutputOverlap);
263 break;
264 default:
265 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
266 UNREACHABLE();
267 }
268 }
269
VisitVecAdd(HVecAdd * instruction)270 void LocationsBuilderARMVIXL::VisitVecAdd(HVecAdd* instruction) {
271 CreateVecBinOpLocations(GetGraph()->GetAllocator(), instruction);
272 }
273
VisitVecAdd(HVecAdd * instruction)274 void InstructionCodeGeneratorARMVIXL::VisitVecAdd(HVecAdd* instruction) {
275 LocationSummary* locations = instruction->GetLocations();
276 vixl32::DRegister lhs = DRegisterFrom(locations->InAt(0));
277 vixl32::DRegister rhs = DRegisterFrom(locations->InAt(1));
278 vixl32::DRegister dst = DRegisterFrom(locations->Out());
279 switch (instruction->GetPackedType()) {
280 case DataType::Type::kUint8:
281 case DataType::Type::kInt8:
282 DCHECK_EQ(8u, instruction->GetVectorLength());
283 __ Vadd(I8, dst, lhs, rhs);
284 break;
285 case DataType::Type::kUint16:
286 case DataType::Type::kInt16:
287 DCHECK_EQ(4u, instruction->GetVectorLength());
288 __ Vadd(I16, dst, lhs, rhs);
289 break;
290 case DataType::Type::kInt32:
291 DCHECK_EQ(2u, instruction->GetVectorLength());
292 __ Vadd(I32, dst, lhs, rhs);
293 break;
294 default:
295 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
296 UNREACHABLE();
297 }
298 }
299
VisitVecSaturationAdd(HVecSaturationAdd * instruction)300 void LocationsBuilderARMVIXL::VisitVecSaturationAdd(HVecSaturationAdd* instruction) {
301 CreateVecBinOpLocations(GetGraph()->GetAllocator(), instruction);
302 }
303
VisitVecSaturationAdd(HVecSaturationAdd * instruction)304 void InstructionCodeGeneratorARMVIXL::VisitVecSaturationAdd(HVecSaturationAdd* instruction) {
305 LocationSummary* locations = instruction->GetLocations();
306 vixl32::DRegister lhs = DRegisterFrom(locations->InAt(0));
307 vixl32::DRegister rhs = DRegisterFrom(locations->InAt(1));
308 vixl32::DRegister dst = DRegisterFrom(locations->Out());
309 switch (instruction->GetPackedType()) {
310 case DataType::Type::kUint8:
311 DCHECK_EQ(8u, instruction->GetVectorLength());
312 __ Vqadd(DataTypeValue::U8, dst, lhs, rhs);
313 break;
314 case DataType::Type::kInt8:
315 DCHECK_EQ(8u, instruction->GetVectorLength());
316 __ Vqadd(DataTypeValue::S8, dst, lhs, rhs);
317 break;
318 case DataType::Type::kUint16:
319 DCHECK_EQ(4u, instruction->GetVectorLength());
320 __ Vqadd(DataTypeValue::U16, dst, lhs, rhs);
321 break;
322 case DataType::Type::kInt16:
323 DCHECK_EQ(4u, instruction->GetVectorLength());
324 __ Vqadd(DataTypeValue::S16, dst, lhs, rhs);
325 break;
326 default:
327 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
328 UNREACHABLE();
329 }
330 }
331
VisitVecHalvingAdd(HVecHalvingAdd * instruction)332 void LocationsBuilderARMVIXL::VisitVecHalvingAdd(HVecHalvingAdd* instruction) {
333 CreateVecBinOpLocations(GetGraph()->GetAllocator(), instruction);
334 }
335
VisitVecHalvingAdd(HVecHalvingAdd * instruction)336 void InstructionCodeGeneratorARMVIXL::VisitVecHalvingAdd(HVecHalvingAdd* instruction) {
337 LocationSummary* locations = instruction->GetLocations();
338 vixl32::DRegister lhs = DRegisterFrom(locations->InAt(0));
339 vixl32::DRegister rhs = DRegisterFrom(locations->InAt(1));
340 vixl32::DRegister dst = DRegisterFrom(locations->Out());
341 switch (instruction->GetPackedType()) {
342 case DataType::Type::kUint8:
343 DCHECK_EQ(8u, instruction->GetVectorLength());
344 instruction->IsRounded()
345 ? __ Vrhadd(DataTypeValue::U8, dst, lhs, rhs)
346 : __ Vhadd(DataTypeValue::U8, dst, lhs, rhs);
347 break;
348 case DataType::Type::kInt8:
349 DCHECK_EQ(8u, instruction->GetVectorLength());
350 instruction->IsRounded()
351 ? __ Vrhadd(DataTypeValue::S8, dst, lhs, rhs)
352 : __ Vhadd(DataTypeValue::S8, dst, lhs, rhs);
353 break;
354 case DataType::Type::kUint16:
355 DCHECK_EQ(4u, instruction->GetVectorLength());
356 instruction->IsRounded()
357 ? __ Vrhadd(DataTypeValue::U16, dst, lhs, rhs)
358 : __ Vhadd(DataTypeValue::U16, dst, lhs, rhs);
359 break;
360 case DataType::Type::kInt16:
361 DCHECK_EQ(4u, instruction->GetVectorLength());
362 instruction->IsRounded()
363 ? __ Vrhadd(DataTypeValue::S16, dst, lhs, rhs)
364 : __ Vhadd(DataTypeValue::S16, dst, lhs, rhs);
365 break;
366 default:
367 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
368 UNREACHABLE();
369 }
370 }
371
VisitVecSub(HVecSub * instruction)372 void LocationsBuilderARMVIXL::VisitVecSub(HVecSub* instruction) {
373 CreateVecBinOpLocations(GetGraph()->GetAllocator(), instruction);
374 }
375
VisitVecSub(HVecSub * instruction)376 void InstructionCodeGeneratorARMVIXL::VisitVecSub(HVecSub* instruction) {
377 LocationSummary* locations = instruction->GetLocations();
378 vixl32::DRegister lhs = DRegisterFrom(locations->InAt(0));
379 vixl32::DRegister rhs = DRegisterFrom(locations->InAt(1));
380 vixl32::DRegister dst = DRegisterFrom(locations->Out());
381 switch (instruction->GetPackedType()) {
382 case DataType::Type::kUint8:
383 case DataType::Type::kInt8:
384 DCHECK_EQ(8u, instruction->GetVectorLength());
385 __ Vsub(I8, dst, lhs, rhs);
386 break;
387 case DataType::Type::kUint16:
388 case DataType::Type::kInt16:
389 DCHECK_EQ(4u, instruction->GetVectorLength());
390 __ Vsub(I16, dst, lhs, rhs);
391 break;
392 case DataType::Type::kInt32:
393 DCHECK_EQ(2u, instruction->GetVectorLength());
394 __ Vsub(I32, dst, lhs, rhs);
395 break;
396 default:
397 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
398 UNREACHABLE();
399 }
400 }
401
VisitVecSaturationSub(HVecSaturationSub * instruction)402 void LocationsBuilderARMVIXL::VisitVecSaturationSub(HVecSaturationSub* instruction) {
403 CreateVecBinOpLocations(GetGraph()->GetAllocator(), instruction);
404 }
405
VisitVecSaturationSub(HVecSaturationSub * instruction)406 void InstructionCodeGeneratorARMVIXL::VisitVecSaturationSub(HVecSaturationSub* instruction) {
407 LocationSummary* locations = instruction->GetLocations();
408 vixl32::DRegister lhs = DRegisterFrom(locations->InAt(0));
409 vixl32::DRegister rhs = DRegisterFrom(locations->InAt(1));
410 vixl32::DRegister dst = DRegisterFrom(locations->Out());
411 switch (instruction->GetPackedType()) {
412 case DataType::Type::kUint8:
413 DCHECK_EQ(8u, instruction->GetVectorLength());
414 __ Vqsub(DataTypeValue::U8, dst, lhs, rhs);
415 break;
416 case DataType::Type::kInt8:
417 DCHECK_EQ(8u, instruction->GetVectorLength());
418 __ Vqsub(DataTypeValue::S8, dst, lhs, rhs);
419 break;
420 case DataType::Type::kUint16:
421 DCHECK_EQ(4u, instruction->GetVectorLength());
422 __ Vqsub(DataTypeValue::U16, dst, lhs, rhs);
423 break;
424 case DataType::Type::kInt16:
425 DCHECK_EQ(4u, instruction->GetVectorLength());
426 __ Vqsub(DataTypeValue::S16, dst, lhs, rhs);
427 break;
428 default:
429 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
430 UNREACHABLE();
431 }
432 }
433
VisitVecMul(HVecMul * instruction)434 void LocationsBuilderARMVIXL::VisitVecMul(HVecMul* instruction) {
435 CreateVecBinOpLocations(GetGraph()->GetAllocator(), instruction);
436 }
437
VisitVecMul(HVecMul * instruction)438 void InstructionCodeGeneratorARMVIXL::VisitVecMul(HVecMul* instruction) {
439 LocationSummary* locations = instruction->GetLocations();
440 vixl32::DRegister lhs = DRegisterFrom(locations->InAt(0));
441 vixl32::DRegister rhs = DRegisterFrom(locations->InAt(1));
442 vixl32::DRegister dst = DRegisterFrom(locations->Out());
443 switch (instruction->GetPackedType()) {
444 case DataType::Type::kUint8:
445 case DataType::Type::kInt8:
446 DCHECK_EQ(8u, instruction->GetVectorLength());
447 __ Vmul(I8, dst, lhs, rhs);
448 break;
449 case DataType::Type::kUint16:
450 case DataType::Type::kInt16:
451 DCHECK_EQ(4u, instruction->GetVectorLength());
452 __ Vmul(I16, dst, lhs, rhs);
453 break;
454 case DataType::Type::kInt32:
455 DCHECK_EQ(2u, instruction->GetVectorLength());
456 __ Vmul(I32, dst, lhs, rhs);
457 break;
458 default:
459 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
460 UNREACHABLE();
461 }
462 }
463
VisitVecDiv(HVecDiv * instruction)464 void LocationsBuilderARMVIXL::VisitVecDiv(HVecDiv* instruction) {
465 CreateVecBinOpLocations(GetGraph()->GetAllocator(), instruction);
466 }
467
VisitVecDiv(HVecDiv * instruction)468 void InstructionCodeGeneratorARMVIXL::VisitVecDiv(HVecDiv* instruction) {
469 LOG(FATAL) << "No SIMD for " << instruction->GetId();
470 }
471
VisitVecMin(HVecMin * instruction)472 void LocationsBuilderARMVIXL::VisitVecMin(HVecMin* instruction) {
473 CreateVecBinOpLocations(GetGraph()->GetAllocator(), instruction);
474 }
475
VisitVecMin(HVecMin * instruction)476 void InstructionCodeGeneratorARMVIXL::VisitVecMin(HVecMin* instruction) {
477 LocationSummary* locations = instruction->GetLocations();
478 vixl32::DRegister lhs = DRegisterFrom(locations->InAt(0));
479 vixl32::DRegister rhs = DRegisterFrom(locations->InAt(1));
480 vixl32::DRegister dst = DRegisterFrom(locations->Out());
481 switch (instruction->GetPackedType()) {
482 case DataType::Type::kUint8:
483 DCHECK_EQ(8u, instruction->GetVectorLength());
484 __ Vmin(DataTypeValue::U8, dst, lhs, rhs);
485 break;
486 case DataType::Type::kInt8:
487 DCHECK_EQ(8u, instruction->GetVectorLength());
488 __ Vmin(DataTypeValue::S8, dst, lhs, rhs);
489 break;
490 case DataType::Type::kUint16:
491 DCHECK_EQ(4u, instruction->GetVectorLength());
492 __ Vmin(DataTypeValue::U16, dst, lhs, rhs);
493 break;
494 case DataType::Type::kInt16:
495 DCHECK_EQ(4u, instruction->GetVectorLength());
496 __ Vmin(DataTypeValue::S16, dst, lhs, rhs);
497 break;
498 case DataType::Type::kUint32:
499 DCHECK_EQ(2u, instruction->GetVectorLength());
500 __ Vmin(DataTypeValue::U32, dst, lhs, rhs);
501 break;
502 case DataType::Type::kInt32:
503 DCHECK_EQ(2u, instruction->GetVectorLength());
504 __ Vmin(DataTypeValue::S32, dst, lhs, rhs);
505 break;
506 default:
507 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
508 UNREACHABLE();
509 }
510 }
511
VisitVecMax(HVecMax * instruction)512 void LocationsBuilderARMVIXL::VisitVecMax(HVecMax* instruction) {
513 CreateVecBinOpLocations(GetGraph()->GetAllocator(), instruction);
514 }
515
VisitVecMax(HVecMax * instruction)516 void InstructionCodeGeneratorARMVIXL::VisitVecMax(HVecMax* instruction) {
517 LocationSummary* locations = instruction->GetLocations();
518 vixl32::DRegister lhs = DRegisterFrom(locations->InAt(0));
519 vixl32::DRegister rhs = DRegisterFrom(locations->InAt(1));
520 vixl32::DRegister dst = DRegisterFrom(locations->Out());
521 switch (instruction->GetPackedType()) {
522 case DataType::Type::kUint8:
523 DCHECK_EQ(8u, instruction->GetVectorLength());
524 __ Vmax(DataTypeValue::U8, dst, lhs, rhs);
525 break;
526 case DataType::Type::kInt8:
527 DCHECK_EQ(8u, instruction->GetVectorLength());
528 __ Vmax(DataTypeValue::S8, dst, lhs, rhs);
529 break;
530 case DataType::Type::kUint16:
531 DCHECK_EQ(4u, instruction->GetVectorLength());
532 __ Vmax(DataTypeValue::U16, dst, lhs, rhs);
533 break;
534 case DataType::Type::kInt16:
535 DCHECK_EQ(4u, instruction->GetVectorLength());
536 __ Vmax(DataTypeValue::S16, dst, lhs, rhs);
537 break;
538 case DataType::Type::kUint32:
539 DCHECK_EQ(2u, instruction->GetVectorLength());
540 __ Vmax(DataTypeValue::U32, dst, lhs, rhs);
541 break;
542 case DataType::Type::kInt32:
543 DCHECK_EQ(2u, instruction->GetVectorLength());
544 __ Vmax(DataTypeValue::S32, dst, lhs, rhs);
545 break;
546 default:
547 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
548 UNREACHABLE();
549 }
550 }
551
VisitVecAnd(HVecAnd * instruction)552 void LocationsBuilderARMVIXL::VisitVecAnd(HVecAnd* instruction) {
553 // TODO: Allow constants supported by VAND (immediate).
554 CreateVecBinOpLocations(GetGraph()->GetAllocator(), instruction);
555 }
556
VisitVecAnd(HVecAnd * instruction)557 void InstructionCodeGeneratorARMVIXL::VisitVecAnd(HVecAnd* instruction) {
558 LocationSummary* locations = instruction->GetLocations();
559 vixl32::DRegister lhs = DRegisterFrom(locations->InAt(0));
560 vixl32::DRegister rhs = DRegisterFrom(locations->InAt(1));
561 vixl32::DRegister dst = DRegisterFrom(locations->Out());
562 switch (instruction->GetPackedType()) {
563 case DataType::Type::kBool:
564 case DataType::Type::kUint8:
565 case DataType::Type::kInt8:
566 case DataType::Type::kUint16:
567 case DataType::Type::kInt16:
568 case DataType::Type::kInt32:
569 __ Vand(I8, dst, lhs, rhs);
570 break;
571 default:
572 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
573 UNREACHABLE();
574 }
575 }
576
VisitVecAndNot(HVecAndNot * instruction)577 void LocationsBuilderARMVIXL::VisitVecAndNot(HVecAndNot* instruction) {
578 CreateVecBinOpLocations(GetGraph()->GetAllocator(), instruction);
579 }
580
VisitVecAndNot(HVecAndNot * instruction)581 void InstructionCodeGeneratorARMVIXL::VisitVecAndNot(HVecAndNot* instruction) {
582 LOG(FATAL) << "No SIMD for " << instruction->GetId();
583 }
584
VisitVecOr(HVecOr * instruction)585 void LocationsBuilderARMVIXL::VisitVecOr(HVecOr* instruction) {
586 CreateVecBinOpLocations(GetGraph()->GetAllocator(), instruction);
587 }
588
VisitVecOr(HVecOr * instruction)589 void InstructionCodeGeneratorARMVIXL::VisitVecOr(HVecOr* instruction) {
590 LocationSummary* locations = instruction->GetLocations();
591 vixl32::DRegister lhs = DRegisterFrom(locations->InAt(0));
592 vixl32::DRegister rhs = DRegisterFrom(locations->InAt(1));
593 vixl32::DRegister dst = DRegisterFrom(locations->Out());
594 switch (instruction->GetPackedType()) {
595 case DataType::Type::kBool:
596 case DataType::Type::kUint8:
597 case DataType::Type::kInt8:
598 case DataType::Type::kUint16:
599 case DataType::Type::kInt16:
600 case DataType::Type::kInt32:
601 __ Vorr(I8, dst, lhs, rhs);
602 break;
603 default:
604 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
605 UNREACHABLE();
606 }
607 }
608
VisitVecXor(HVecXor * instruction)609 void LocationsBuilderARMVIXL::VisitVecXor(HVecXor* instruction) {
610 CreateVecBinOpLocations(GetGraph()->GetAllocator(), instruction);
611 }
612
VisitVecXor(HVecXor * instruction)613 void InstructionCodeGeneratorARMVIXL::VisitVecXor(HVecXor* instruction) {
614 LocationSummary* locations = instruction->GetLocations();
615 vixl32::DRegister lhs = DRegisterFrom(locations->InAt(0));
616 vixl32::DRegister rhs = DRegisterFrom(locations->InAt(1));
617 vixl32::DRegister dst = DRegisterFrom(locations->Out());
618 switch (instruction->GetPackedType()) {
619 case DataType::Type::kBool:
620 case DataType::Type::kUint8:
621 case DataType::Type::kInt8:
622 case DataType::Type::kUint16:
623 case DataType::Type::kInt16:
624 case DataType::Type::kInt32:
625 __ Veor(I8, dst, lhs, rhs);
626 break;
627 default:
628 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
629 UNREACHABLE();
630 }
631 }
632
633 // Helper to set up locations for vector shift operations.
CreateVecShiftLocations(ArenaAllocator * allocator,HVecBinaryOperation * instruction)634 static void CreateVecShiftLocations(ArenaAllocator* allocator, HVecBinaryOperation* instruction) {
635 LocationSummary* locations = new (allocator) LocationSummary(instruction);
636 switch (instruction->GetPackedType()) {
637 case DataType::Type::kUint8:
638 case DataType::Type::kInt8:
639 case DataType::Type::kUint16:
640 case DataType::Type::kInt16:
641 case DataType::Type::kInt32:
642 locations->SetInAt(0, Location::RequiresFpuRegister());
643 locations->SetInAt(1, Location::ConstantLocation(instruction->InputAt(1)));
644 locations->SetOut(Location::RequiresFpuRegister(), Location::kNoOutputOverlap);
645 break;
646 default:
647 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
648 UNREACHABLE();
649 }
650 }
651
VisitVecShl(HVecShl * instruction)652 void LocationsBuilderARMVIXL::VisitVecShl(HVecShl* instruction) {
653 CreateVecShiftLocations(GetGraph()->GetAllocator(), instruction);
654 }
655
VisitVecShl(HVecShl * instruction)656 void InstructionCodeGeneratorARMVIXL::VisitVecShl(HVecShl* instruction) {
657 LocationSummary* locations = instruction->GetLocations();
658 vixl32::DRegister lhs = DRegisterFrom(locations->InAt(0));
659 vixl32::DRegister dst = DRegisterFrom(locations->Out());
660 int32_t value = locations->InAt(1).GetConstant()->AsIntConstant()->GetValue();
661 switch (instruction->GetPackedType()) {
662 case DataType::Type::kUint8:
663 case DataType::Type::kInt8:
664 DCHECK_EQ(8u, instruction->GetVectorLength());
665 __ Vshl(I8, dst, lhs, value);
666 break;
667 case DataType::Type::kUint16:
668 case DataType::Type::kInt16:
669 DCHECK_EQ(4u, instruction->GetVectorLength());
670 __ Vshl(I16, dst, lhs, value);
671 break;
672 case DataType::Type::kInt32:
673 DCHECK_EQ(2u, instruction->GetVectorLength());
674 __ Vshl(I32, dst, lhs, value);
675 break;
676 default:
677 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
678 UNREACHABLE();
679 }
680 }
681
VisitVecShr(HVecShr * instruction)682 void LocationsBuilderARMVIXL::VisitVecShr(HVecShr* instruction) {
683 CreateVecShiftLocations(GetGraph()->GetAllocator(), instruction);
684 }
685
VisitVecShr(HVecShr * instruction)686 void InstructionCodeGeneratorARMVIXL::VisitVecShr(HVecShr* instruction) {
687 LocationSummary* locations = instruction->GetLocations();
688 vixl32::DRegister lhs = DRegisterFrom(locations->InAt(0));
689 vixl32::DRegister dst = DRegisterFrom(locations->Out());
690 int32_t value = locations->InAt(1).GetConstant()->AsIntConstant()->GetValue();
691 switch (instruction->GetPackedType()) {
692 case DataType::Type::kUint8:
693 case DataType::Type::kInt8:
694 DCHECK_EQ(8u, instruction->GetVectorLength());
695 __ Vshr(DataTypeValue::S8, dst, lhs, value);
696 break;
697 case DataType::Type::kUint16:
698 case DataType::Type::kInt16:
699 DCHECK_EQ(4u, instruction->GetVectorLength());
700 __ Vshr(DataTypeValue::S16, dst, lhs, value);
701 break;
702 case DataType::Type::kInt32:
703 DCHECK_EQ(2u, instruction->GetVectorLength());
704 __ Vshr(DataTypeValue::S32, dst, lhs, value);
705 break;
706 default:
707 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
708 UNREACHABLE();
709 }
710 }
711
VisitVecUShr(HVecUShr * instruction)712 void LocationsBuilderARMVIXL::VisitVecUShr(HVecUShr* instruction) {
713 CreateVecShiftLocations(GetGraph()->GetAllocator(), instruction);
714 }
715
VisitVecUShr(HVecUShr * instruction)716 void InstructionCodeGeneratorARMVIXL::VisitVecUShr(HVecUShr* instruction) {
717 LocationSummary* locations = instruction->GetLocations();
718 vixl32::DRegister lhs = DRegisterFrom(locations->InAt(0));
719 vixl32::DRegister dst = DRegisterFrom(locations->Out());
720 int32_t value = locations->InAt(1).GetConstant()->AsIntConstant()->GetValue();
721 switch (instruction->GetPackedType()) {
722 case DataType::Type::kUint8:
723 case DataType::Type::kInt8:
724 DCHECK_EQ(8u, instruction->GetVectorLength());
725 __ Vshr(DataTypeValue::U8, dst, lhs, value);
726 break;
727 case DataType::Type::kUint16:
728 case DataType::Type::kInt16:
729 DCHECK_EQ(4u, instruction->GetVectorLength());
730 __ Vshr(DataTypeValue::U16, dst, lhs, value);
731 break;
732 case DataType::Type::kInt32:
733 DCHECK_EQ(2u, instruction->GetVectorLength());
734 __ Vshr(DataTypeValue::U32, dst, lhs, value);
735 break;
736 default:
737 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
738 UNREACHABLE();
739 }
740 }
741
VisitVecSetScalars(HVecSetScalars * instruction)742 void LocationsBuilderARMVIXL::VisitVecSetScalars(HVecSetScalars* instruction) {
743 LocationSummary* locations = new (GetGraph()->GetAllocator()) LocationSummary(instruction);
744
745 DCHECK_EQ(1u, instruction->InputCount()); // only one input currently implemented
746
747 HInstruction* input = instruction->InputAt(0);
748 bool is_zero = IsZeroBitPattern(input);
749
750 switch (instruction->GetPackedType()) {
751 case DataType::Type::kInt32:
752 locations->SetInAt(0, is_zero ? Location::ConstantLocation(input)
753 : Location::RequiresRegister());
754 locations->SetOut(Location::RequiresFpuRegister());
755 break;
756 default:
757 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
758 UNREACHABLE();
759 }
760 }
761
VisitVecSetScalars(HVecSetScalars * instruction)762 void InstructionCodeGeneratorARMVIXL::VisitVecSetScalars(HVecSetScalars* instruction) {
763 LocationSummary* locations = instruction->GetLocations();
764 vixl32::DRegister dst = DRegisterFrom(locations->Out());
765
766 DCHECK_EQ(1u, instruction->InputCount()); // only one input currently implemented
767
768 // Zero out all other elements first.
769 __ Vmov(I32, dst, 0);
770
771 // Shorthand for any type of zero.
772 if (IsZeroBitPattern(instruction->InputAt(0))) {
773 return;
774 }
775
776 // Set required elements.
777 switch (instruction->GetPackedType()) {
778 case DataType::Type::kInt32:
779 DCHECK_EQ(2u, instruction->GetVectorLength());
780 __ Vmov(Untyped32, DRegisterLane(dst, 0), InputRegisterAt(instruction, 0));
781 break;
782 default:
783 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
784 UNREACHABLE();
785 }
786 }
787
788 // Helper to set up locations for vector accumulations.
CreateVecAccumLocations(ArenaAllocator * allocator,HVecOperation * instruction)789 static void CreateVecAccumLocations(ArenaAllocator* allocator, HVecOperation* instruction) {
790 LocationSummary* locations = new (allocator) LocationSummary(instruction);
791 switch (instruction->GetPackedType()) {
792 case DataType::Type::kUint8:
793 case DataType::Type::kInt8:
794 case DataType::Type::kUint16:
795 case DataType::Type::kInt16:
796 case DataType::Type::kInt32:
797 case DataType::Type::kInt64:
798 locations->SetInAt(0, Location::RequiresFpuRegister());
799 locations->SetInAt(1, Location::RequiresFpuRegister());
800 locations->SetInAt(2, Location::RequiresFpuRegister());
801 locations->SetOut(Location::SameAsFirstInput());
802 break;
803 default:
804 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
805 UNREACHABLE();
806 }
807 }
808
VisitVecMultiplyAccumulate(HVecMultiplyAccumulate * instruction)809 void LocationsBuilderARMVIXL::VisitVecMultiplyAccumulate(HVecMultiplyAccumulate* instruction) {
810 CreateVecAccumLocations(GetGraph()->GetAllocator(), instruction);
811 }
812
VisitVecMultiplyAccumulate(HVecMultiplyAccumulate * instruction)813 void InstructionCodeGeneratorARMVIXL::VisitVecMultiplyAccumulate(HVecMultiplyAccumulate* instruction) {
814 LOG(FATAL) << "No SIMD for " << instruction->GetId();
815 }
816
VisitVecSADAccumulate(HVecSADAccumulate * instruction)817 void LocationsBuilderARMVIXL::VisitVecSADAccumulate(HVecSADAccumulate* instruction) {
818 CreateVecAccumLocations(GetGraph()->GetAllocator(), instruction);
819 }
820
VisitVecSADAccumulate(HVecSADAccumulate * instruction)821 void InstructionCodeGeneratorARMVIXL::VisitVecSADAccumulate(HVecSADAccumulate* instruction) {
822 LocationSummary* locations = instruction->GetLocations();
823 vixl32::DRegister acc = DRegisterFrom(locations->InAt(0));
824 vixl32::DRegister left = DRegisterFrom(locations->InAt(1));
825 vixl32::DRegister right = DRegisterFrom(locations->InAt(2));
826
827 DCHECK(locations->InAt(0).Equals(locations->Out()));
828
829 // Handle all feasible acc_T += sad(a_S, b_S) type combinations (T x S).
830 HVecOperation* a = instruction->InputAt(1)->AsVecOperation();
831 HVecOperation* b = instruction->InputAt(2)->AsVecOperation();
832 DCHECK_EQ(a->GetPackedType(), b->GetPackedType());
833 switch (a->GetPackedType()) {
834 case DataType::Type::kInt32:
835 DCHECK_EQ(2u, a->GetVectorLength());
836 switch (instruction->GetPackedType()) {
837 case DataType::Type::kInt32: {
838 DCHECK_EQ(2u, instruction->GetVectorLength());
839 UseScratchRegisterScope temps(GetVIXLAssembler());
840 vixl32::DRegister tmp = temps.AcquireD();
841 __ Vsub(DataTypeValue::I32, tmp, left, right);
842 __ Vabs(DataTypeValue::S32, tmp, tmp);
843 __ Vadd(DataTypeValue::I32, acc, acc, tmp);
844 break;
845 }
846 default:
847 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
848 UNREACHABLE();
849 }
850 break;
851 default:
852 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
853 UNREACHABLE();
854 }
855 }
856
VisitVecDotProd(HVecDotProd * instruction)857 void LocationsBuilderARMVIXL::VisitVecDotProd(HVecDotProd* instruction) {
858 LOG(FATAL) << "No SIMD for " << instruction->GetId();
859 }
860
VisitVecDotProd(HVecDotProd * instruction)861 void InstructionCodeGeneratorARMVIXL::VisitVecDotProd(HVecDotProd* instruction) {
862 LOG(FATAL) << "No SIMD for " << instruction->GetId();
863 }
864
865 // Return whether the vector memory access operation is guaranteed to be word-aligned (ARM word
866 // size equals to 4).
IsWordAligned(HVecMemoryOperation * instruction)867 static bool IsWordAligned(HVecMemoryOperation* instruction) {
868 return instruction->GetAlignment().IsAlignedAt(4u);
869 }
870
871 // Helper to set up locations for vector memory operations.
CreateVecMemLocations(ArenaAllocator * allocator,HVecMemoryOperation * instruction,bool is_load)872 static void CreateVecMemLocations(ArenaAllocator* allocator,
873 HVecMemoryOperation* instruction,
874 bool is_load) {
875 LocationSummary* locations = new (allocator) LocationSummary(instruction);
876 switch (instruction->GetPackedType()) {
877 case DataType::Type::kBool:
878 case DataType::Type::kUint8:
879 case DataType::Type::kInt8:
880 case DataType::Type::kUint16:
881 case DataType::Type::kInt16:
882 case DataType::Type::kInt32:
883 locations->SetInAt(0, Location::RequiresRegister());
884 locations->SetInAt(1, Location::RegisterOrConstant(instruction->InputAt(1)));
885 if (is_load) {
886 locations->SetOut(Location::RequiresFpuRegister());
887 } else {
888 locations->SetInAt(2, Location::RequiresFpuRegister());
889 }
890 break;
891 default:
892 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
893 UNREACHABLE();
894 }
895 }
896
897 // Helper to set up locations for vector memory operations. Returns the memory operand and,
898 // if used, sets the output parameter scratch to a temporary register used in this operand,
899 // so that the client can release it right after the memory operand use.
VecAddress(HVecMemoryOperation * instruction,UseScratchRegisterScope * temps_scope,vixl32::Register * scratch)900 MemOperand InstructionCodeGeneratorARMVIXL::VecAddress(
901 HVecMemoryOperation* instruction,
902 UseScratchRegisterScope* temps_scope,
903 /*out*/ vixl32::Register* scratch) {
904 LocationSummary* locations = instruction->GetLocations();
905 vixl32::Register base = InputRegisterAt(instruction, 0);
906
907 Location index = locations->InAt(1);
908 size_t size = DataType::Size(instruction->GetPackedType());
909 uint32_t offset = mirror::Array::DataOffset(size).Uint32Value();
910 size_t shift = ComponentSizeShiftWidth(size);
911
912 // HIntermediateAddress optimization is only applied for scalar ArrayGet and ArraySet.
913 DCHECK(!instruction->InputAt(0)->IsIntermediateAddress());
914
915 if (index.IsConstant()) {
916 offset += Int64ConstantFrom(index) << shift;
917 return MemOperand(base, offset);
918 } else {
919 *scratch = temps_scope->Acquire();
920 __ Add(*scratch, base, Operand(RegisterFrom(index), ShiftType::LSL, shift));
921
922 return MemOperand(*scratch, offset);
923 }
924 }
925
VecAddressUnaligned(HVecMemoryOperation * instruction,UseScratchRegisterScope * temps_scope,vixl32::Register * scratch)926 AlignedMemOperand InstructionCodeGeneratorARMVIXL::VecAddressUnaligned(
927 HVecMemoryOperation* instruction,
928 UseScratchRegisterScope* temps_scope,
929 /*out*/ vixl32::Register* scratch) {
930 LocationSummary* locations = instruction->GetLocations();
931 vixl32::Register base = InputRegisterAt(instruction, 0);
932
933 Location index = locations->InAt(1);
934 size_t size = DataType::Size(instruction->GetPackedType());
935 uint32_t offset = mirror::Array::DataOffset(size).Uint32Value();
936 size_t shift = ComponentSizeShiftWidth(size);
937
938 // HIntermediateAddress optimization is only applied for scalar ArrayGet and ArraySet.
939 DCHECK(!instruction->InputAt(0)->IsIntermediateAddress());
940
941 if (index.IsConstant()) {
942 offset += Int64ConstantFrom(index) << shift;
943 __ Add(*scratch, base, offset);
944 } else {
945 *scratch = temps_scope->Acquire();
946 __ Add(*scratch, base, offset);
947 __ Add(*scratch, *scratch, Operand(RegisterFrom(index), ShiftType::LSL, shift));
948 }
949 return AlignedMemOperand(*scratch, kNoAlignment);
950 }
951
VisitVecLoad(HVecLoad * instruction)952 void LocationsBuilderARMVIXL::VisitVecLoad(HVecLoad* instruction) {
953 CreateVecMemLocations(GetGraph()->GetAllocator(), instruction, /*is_load*/ true);
954 }
955
VisitVecLoad(HVecLoad * instruction)956 void InstructionCodeGeneratorARMVIXL::VisitVecLoad(HVecLoad* instruction) {
957 vixl32::DRegister reg = OutputDRegister(instruction);
958 UseScratchRegisterScope temps(GetVIXLAssembler());
959 vixl32::Register scratch;
960
961 DCHECK_IMPLIES(instruction->GetPackedType() == DataType::Type::kUint16,
962 !instruction->IsStringCharAt());
963
964 switch (instruction->GetPackedType()) {
965 case DataType::Type::kBool:
966 case DataType::Type::kUint8:
967 case DataType::Type::kInt8:
968 DCHECK_EQ(8u, instruction->GetVectorLength());
969 if (IsWordAligned(instruction)) {
970 __ Vldr(reg, VecAddress(instruction, &temps, &scratch));
971 } else {
972 __ Vld1(Untyped8,
973 NeonRegisterList(reg, kMultipleLanes),
974 VecAddressUnaligned(instruction, &temps, &scratch));
975 }
976 break;
977 case DataType::Type::kUint16:
978 case DataType::Type::kInt16:
979 DCHECK_EQ(4u, instruction->GetVectorLength());
980 if (IsWordAligned(instruction)) {
981 __ Vldr(reg, VecAddress(instruction, &temps, &scratch));
982 } else {
983 __ Vld1(Untyped16,
984 NeonRegisterList(reg, kMultipleLanes),
985 VecAddressUnaligned(instruction, &temps, &scratch));
986 }
987 break;
988 case DataType::Type::kInt32:
989 DCHECK_EQ(2u, instruction->GetVectorLength());
990 if (IsWordAligned(instruction)) {
991 __ Vldr(reg, VecAddress(instruction, &temps, &scratch));
992 } else {
993 __ Vld1(Untyped32,
994 NeonRegisterList(reg, kMultipleLanes),
995 VecAddressUnaligned(instruction, &temps, &scratch));
996 }
997 break;
998 default:
999 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
1000 UNREACHABLE();
1001 }
1002 }
1003
VisitVecStore(HVecStore * instruction)1004 void LocationsBuilderARMVIXL::VisitVecStore(HVecStore* instruction) {
1005 CreateVecMemLocations(GetGraph()->GetAllocator(), instruction, /*is_load*/ false);
1006 }
1007
VisitVecStore(HVecStore * instruction)1008 void InstructionCodeGeneratorARMVIXL::VisitVecStore(HVecStore* instruction) {
1009 vixl32::DRegister reg = InputDRegisterAt(instruction, 2);
1010 UseScratchRegisterScope temps(GetVIXLAssembler());
1011 vixl32::Register scratch;
1012 switch (instruction->GetPackedType()) {
1013 case DataType::Type::kBool:
1014 case DataType::Type::kUint8:
1015 case DataType::Type::kInt8:
1016 DCHECK_EQ(8u, instruction->GetVectorLength());
1017 if (IsWordAligned(instruction)) {
1018 __ Vstr(reg, VecAddress(instruction, &temps, &scratch));
1019 } else {
1020 __ Vst1(Untyped8,
1021 NeonRegisterList(reg, kMultipleLanes),
1022 VecAddressUnaligned(instruction, &temps, &scratch));
1023 }
1024 break;
1025 case DataType::Type::kUint16:
1026 case DataType::Type::kInt16:
1027 DCHECK_EQ(4u, instruction->GetVectorLength());
1028 if (IsWordAligned(instruction)) {
1029 __ Vstr(reg, VecAddress(instruction, &temps, &scratch));
1030 } else {
1031 __ Vst1(Untyped16,
1032 NeonRegisterList(reg, kMultipleLanes),
1033 VecAddressUnaligned(instruction, &temps, &scratch));
1034 }
1035 break;
1036 case DataType::Type::kInt32:
1037 DCHECK_EQ(2u, instruction->GetVectorLength());
1038 if (IsWordAligned(instruction)) {
1039 __ Vstr(reg, VecAddress(instruction, &temps, &scratch));
1040 } else {
1041 __ Vst1(Untyped32,
1042 NeonRegisterList(reg, kMultipleLanes),
1043 VecAddressUnaligned(instruction, &temps, &scratch));
1044 }
1045 break;
1046 default:
1047 LOG(FATAL) << "Unsupported SIMD type: " << instruction->GetPackedType();
1048 UNREACHABLE();
1049 }
1050 }
1051
VisitVecPredSetAll(HVecPredSetAll * instruction)1052 void LocationsBuilderARMVIXL::VisitVecPredSetAll(HVecPredSetAll* instruction) {
1053 LOG(FATAL) << "No SIMD for " << instruction->GetId();
1054 UNREACHABLE();
1055 }
1056
VisitVecPredSetAll(HVecPredSetAll * instruction)1057 void InstructionCodeGeneratorARMVIXL::VisitVecPredSetAll(HVecPredSetAll* instruction) {
1058 LOG(FATAL) << "No SIMD for " << instruction->GetId();
1059 UNREACHABLE();
1060 }
1061
VisitVecPredWhile(HVecPredWhile * instruction)1062 void LocationsBuilderARMVIXL::VisitVecPredWhile(HVecPredWhile* instruction) {
1063 LOG(FATAL) << "No SIMD for " << instruction->GetId();
1064 UNREACHABLE();
1065 }
1066
VisitVecPredWhile(HVecPredWhile * instruction)1067 void InstructionCodeGeneratorARMVIXL::VisitVecPredWhile(HVecPredWhile* instruction) {
1068 LOG(FATAL) << "No SIMD for " << instruction->GetId();
1069 UNREACHABLE();
1070 }
1071
VisitVecPredToBoolean(HVecPredToBoolean * instruction)1072 void LocationsBuilderARMVIXL::VisitVecPredToBoolean(HVecPredToBoolean* instruction) {
1073 LOG(FATAL) << "No SIMD for " << instruction->GetId();
1074 UNREACHABLE();
1075 }
1076
VisitVecPredToBoolean(HVecPredToBoolean * instruction)1077 void InstructionCodeGeneratorARMVIXL::VisitVecPredToBoolean(HVecPredToBoolean* instruction) {
1078 LOG(FATAL) << "No SIMD for " << instruction->GetId();
1079 UNREACHABLE();
1080 }
1081
VisitVecCondition(HVecCondition * instruction)1082 void LocationsBuilderARMVIXL::VisitVecCondition(HVecCondition* instruction) {
1083 LOG(FATAL) << "No SIMD for " << instruction->GetId();
1084 UNREACHABLE();
1085 }
1086
VisitVecCondition(HVecCondition * instruction)1087 void InstructionCodeGeneratorARMVIXL::VisitVecCondition(HVecCondition* instruction) {
1088 LOG(FATAL) << "No SIMD for " << instruction->GetId();
1089 UNREACHABLE();
1090 }
1091
VisitVecPredNot(HVecPredNot * instruction)1092 void LocationsBuilderARMVIXL::VisitVecPredNot(HVecPredNot* instruction) {
1093 LOG(FATAL) << "No SIMD for " << instruction->GetId();
1094 UNREACHABLE();
1095 }
1096
VisitVecPredNot(HVecPredNot * instruction)1097 void InstructionCodeGeneratorARMVIXL::VisitVecPredNot(HVecPredNot* instruction) {
1098 LOG(FATAL) << "No SIMD for " << instruction->GetId();
1099 UNREACHABLE();
1100 }
1101
1102 #undef __
1103
1104 } // namespace arm
1105 } // namespace art
1106