xref: /openbmc/qemu/target/arm/tcg/translate-vfp.c (revision bf616ce4)
1 /*
2  *  ARM translation: AArch32 VFP instructions
3  *
4  *  Copyright (c) 2003 Fabrice Bellard
5  *  Copyright (c) 2005-2007 CodeSourcery
6  *  Copyright (c) 2007 OpenedHand, Ltd.
7  *  Copyright (c) 2019 Linaro, Ltd.
8  *
9  * This library is free software; you can redistribute it and/or
10  * modify it under the terms of the GNU Lesser General Public
11  * License as published by the Free Software Foundation; either
12  * version 2.1 of the License, or (at your option) any later version.
13  *
14  * This library is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
17  * Lesser General Public License for more details.
18  *
19  * You should have received a copy of the GNU Lesser General Public
20  * License along with this library; if not, see <http://www.gnu.org/licenses/>.
21  */
22 
23 #include "qemu/osdep.h"
24 #include "translate.h"
25 #include "translate-a32.h"
26 
27 /* Include the generated VFP decoder */
28 #include "decode-vfp.c.inc"
29 #include "decode-vfp-uncond.c.inc"
30 
31 static inline void vfp_load_reg64(TCGv_i64 var, int reg)
32 {
33     tcg_gen_ld_i64(var, tcg_env, vfp_reg_offset(true, reg));
34 }
35 
36 static inline void vfp_store_reg64(TCGv_i64 var, int reg)
37 {
38     tcg_gen_st_i64(var, tcg_env, vfp_reg_offset(true, reg));
39 }
40 
41 static inline void vfp_load_reg32(TCGv_i32 var, int reg)
42 {
43     tcg_gen_ld_i32(var, tcg_env, vfp_reg_offset(false, reg));
44 }
45 
46 static inline void vfp_store_reg32(TCGv_i32 var, int reg)
47 {
48     tcg_gen_st_i32(var, tcg_env, vfp_reg_offset(false, reg));
49 }
50 
51 /*
52  * The imm8 encodes the sign bit, enough bits to represent an exponent in
53  * the range 01....1xx to 10....0xx, and the most significant 4 bits of
54  * the mantissa; see VFPExpandImm() in the v8 ARM ARM.
55  */
56 uint64_t vfp_expand_imm(int size, uint8_t imm8)
57 {
58     uint64_t imm;
59 
60     switch (size) {
61     case MO_64:
62         imm = (extract32(imm8, 7, 1) ? 0x8000 : 0) |
63             (extract32(imm8, 6, 1) ? 0x3fc0 : 0x4000) |
64             extract32(imm8, 0, 6);
65         imm <<= 48;
66         break;
67     case MO_32:
68         imm = (extract32(imm8, 7, 1) ? 0x8000 : 0) |
69             (extract32(imm8, 6, 1) ? 0x3e00 : 0x4000) |
70             (extract32(imm8, 0, 6) << 3);
71         imm <<= 16;
72         break;
73     case MO_16:
74         imm = (extract32(imm8, 7, 1) ? 0x8000 : 0) |
75             (extract32(imm8, 6, 1) ? 0x3000 : 0x4000) |
76             (extract32(imm8, 0, 6) << 6);
77         break;
78     default:
79         g_assert_not_reached();
80     }
81     return imm;
82 }
83 
84 /*
85  * Return the offset of a 16-bit half of the specified VFP single-precision
86  * register. If top is true, returns the top 16 bits; otherwise the bottom
87  * 16 bits.
88  */
89 static inline long vfp_f16_offset(unsigned reg, bool top)
90 {
91     long offs = vfp_reg_offset(false, reg);
92 #if HOST_BIG_ENDIAN
93     if (!top) {
94         offs += 2;
95     }
96 #else
97     if (top) {
98         offs += 2;
99     }
100 #endif
101     return offs;
102 }
103 
104 /*
105  * Generate code for M-profile lazy FP state preservation if needed;
106  * this corresponds to the pseudocode PreserveFPState() function.
107  */
108 static void gen_preserve_fp_state(DisasContext *s, bool skip_context_update)
109 {
110     if (s->v7m_lspact) {
111         /*
112          * Lazy state saving affects external memory and also the NVIC,
113          * so we must mark it as an IO operation for icount (and cause
114          * this to be the last insn in the TB).
115          */
116         if (translator_io_start(&s->base)) {
117             s->base.is_jmp = DISAS_UPDATE_EXIT;
118         }
119         gen_helper_v7m_preserve_fp_state(tcg_env);
120         /*
121          * If the preserve_fp_state helper doesn't throw an exception
122          * then it will clear LSPACT; we don't need to repeat this for
123          * any further FP insns in this TB.
124          */
125         s->v7m_lspact = false;
126         /*
127          * The helper might have zeroed VPR, so we do not know the
128          * correct value for the MVE_NO_PRED TB flag any more.
129          * If we're about to create a new fp context then that
130          * will precisely determine the MVE_NO_PRED value (see
131          * gen_update_fp_context()). Otherwise, we must:
132          *  - set s->mve_no_pred to false, so this instruction
133          *    is generated to use helper functions
134          *  - end the TB now, without chaining to the next TB
135          */
136         if (skip_context_update || !s->v7m_new_fp_ctxt_needed) {
137             s->mve_no_pred = false;
138             s->base.is_jmp = DISAS_UPDATE_NOCHAIN;
139         }
140     }
141 }
142 
143 /*
144  * Generate code for M-profile FP context handling: update the
145  * ownership of the FP context, and create a new context if
146  * necessary. This corresponds to the parts of the pseudocode
147  * ExecuteFPCheck() after the initial PreserveFPState() call.
148  */
149 static void gen_update_fp_context(DisasContext *s)
150 {
151     /* Update ownership of FP context: set FPCCR.S to match current state */
152     if (s->v8m_fpccr_s_wrong) {
153         TCGv_i32 tmp;
154 
155         tmp = load_cpu_field(v7m.fpccr[M_REG_S]);
156         if (s->v8m_secure) {
157             tcg_gen_ori_i32(tmp, tmp, R_V7M_FPCCR_S_MASK);
158         } else {
159             tcg_gen_andi_i32(tmp, tmp, ~R_V7M_FPCCR_S_MASK);
160         }
161         store_cpu_field(tmp, v7m.fpccr[M_REG_S]);
162         /* Don't need to do this for any further FP insns in this TB */
163         s->v8m_fpccr_s_wrong = false;
164     }
165 
166     if (s->v7m_new_fp_ctxt_needed) {
167         /*
168          * Create new FP context by updating CONTROL.FPCA, CONTROL.SFPA,
169          * the FPSCR, and VPR.
170          */
171         TCGv_i32 control, fpscr;
172         uint32_t bits = R_V7M_CONTROL_FPCA_MASK;
173 
174         fpscr = load_cpu_field(v7m.fpdscr[s->v8m_secure]);
175         gen_helper_vfp_set_fpscr(tcg_env, fpscr);
176         if (dc_isar_feature(aa32_mve, s)) {
177             store_cpu_field(tcg_constant_i32(0), v7m.vpr);
178         }
179         /*
180          * We just updated the FPSCR and VPR. Some of this state is cached
181          * in the MVE_NO_PRED TB flag. We want to avoid having to end the
182          * TB here, which means we need the new value of the MVE_NO_PRED
183          * flag to be exactly known here and the same for all executions.
184          * Luckily FPDSCR.LTPSIZE is always constant 4 and the VPR is
185          * always set to 0, so the new MVE_NO_PRED flag is always 1
186          * if and only if we have MVE.
187          *
188          * (The other FPSCR state cached in TB flags is VECLEN and VECSTRIDE,
189          * but those do not exist for M-profile, so are not relevant here.)
190          */
191         s->mve_no_pred = dc_isar_feature(aa32_mve, s);
192 
193         if (s->v8m_secure) {
194             bits |= R_V7M_CONTROL_SFPA_MASK;
195         }
196         control = load_cpu_field(v7m.control[M_REG_S]);
197         tcg_gen_ori_i32(control, control, bits);
198         store_cpu_field(control, v7m.control[M_REG_S]);
199         /* Don't need to do this for any further FP insns in this TB */
200         s->v7m_new_fp_ctxt_needed = false;
201     }
202 }
203 
204 /*
205  * Check that VFP access is enabled, A-profile specific version.
206  *
207  * If VFP is enabled, return true. If not, emit code to generate an
208  * appropriate exception and return false.
209  * The ignore_vfp_enabled argument specifies that we should ignore
210  * whether VFP is enabled via FPEXC.EN: this should be true for FMXR/FMRX
211  * accesses to FPSID, FPEXC, MVFR0, MVFR1, MVFR2, and false for all other insns.
212  */
213 static bool vfp_access_check_a(DisasContext *s, bool ignore_vfp_enabled)
214 {
215     if (s->fp_excp_el) {
216         /*
217          * The full syndrome is only used for HSR when HCPTR traps:
218          * For v8, when TA==0, coproc is RES0.
219          * For v7, any use of a Floating-point instruction or access
220          * to a Floating-point Extension register that is trapped to
221          * Hyp mode because of a trap configured in the HCPTR sets
222          * this field to 0xA.
223          */
224         int coproc = arm_dc_feature(s, ARM_FEATURE_V8) ? 0 : 0xa;
225         uint32_t syn = syn_fp_access_trap(1, 0xe, false, coproc);
226 
227         gen_exception_insn_el(s, 0, EXCP_UDEF, syn, s->fp_excp_el);
228         return false;
229     }
230 
231     /*
232      * Note that rebuild_hflags_a32 has already accounted for being in EL0
233      * and the higher EL in A64 mode, etc.  Unlike A64 mode, there do not
234      * appear to be any insns which touch VFP which are allowed.
235      */
236     if (s->sme_trap_nonstreaming) {
237         gen_exception_insn(s, 0, EXCP_UDEF,
238                            syn_smetrap(SME_ET_Streaming,
239                                        curr_insn_len(s) == 2));
240         return false;
241     }
242 
243     if (!s->vfp_enabled && !ignore_vfp_enabled) {
244         assert(!arm_dc_feature(s, ARM_FEATURE_M));
245         unallocated_encoding(s);
246         return false;
247     }
248     return true;
249 }
250 
251 /*
252  * Check that VFP access is enabled, M-profile specific version.
253  *
254  * If VFP is enabled, do the necessary M-profile lazy-FP handling and then
255  * return true. If not, emit code to generate an appropriate exception and
256  * return false.
257  * skip_context_update is true to skip the "update FP context" part of this.
258  */
259 bool vfp_access_check_m(DisasContext *s, bool skip_context_update)
260 {
261     if (s->fp_excp_el) {
262         /*
263          * M-profile mostly catches the "FPU disabled" case early, in
264          * disas_m_nocp(), but a few insns (eg LCTP, WLSTP, DLSTP)
265          * which do coprocessor-checks are outside the large ranges of
266          * the encoding space handled by the patterns in m-nocp.decode,
267          * and for them we may need to raise NOCP here.
268          */
269         gen_exception_insn_el(s, 0, EXCP_NOCP,
270                               syn_uncategorized(), s->fp_excp_el);
271         return false;
272     }
273 
274     /* Handle M-profile lazy FP state mechanics */
275 
276     /* Trigger lazy-state preservation if necessary */
277     gen_preserve_fp_state(s, skip_context_update);
278 
279     if (!skip_context_update) {
280         /* Update ownership of FP context and create new FP context if needed */
281         gen_update_fp_context(s);
282     }
283 
284     return true;
285 }
286 
287 /*
288  * The most usual kind of VFP access check, for everything except
289  * FMXR/FMRX to the always-available special registers.
290  */
291 bool vfp_access_check(DisasContext *s)
292 {
293     if (arm_dc_feature(s, ARM_FEATURE_M)) {
294         return vfp_access_check_m(s, false);
295     } else {
296         return vfp_access_check_a(s, false);
297     }
298 }
299 
300 static bool trans_VSEL(DisasContext *s, arg_VSEL *a)
301 {
302     uint32_t rd, rn, rm;
303     int sz = a->sz;
304 
305     if (!dc_isar_feature(aa32_vsel, s)) {
306         return false;
307     }
308 
309     if (sz == 3 && !dc_isar_feature(aa32_fpdp_v2, s)) {
310         return false;
311     }
312 
313     if (sz == 1 && !dc_isar_feature(aa32_fp16_arith, s)) {
314         return false;
315     }
316 
317     /* UNDEF accesses to D16-D31 if they don't exist */
318     if (sz == 3 && !dc_isar_feature(aa32_simd_r32, s) &&
319         ((a->vm | a->vn | a->vd) & 0x10)) {
320         return false;
321     }
322 
323     rd = a->vd;
324     rn = a->vn;
325     rm = a->vm;
326 
327     if (!vfp_access_check(s)) {
328         return true;
329     }
330 
331     if (sz == 3) {
332         TCGv_i64 frn, frm, dest;
333         TCGv_i64 tmp, zero, zf, nf, vf;
334 
335         zero = tcg_constant_i64(0);
336 
337         frn = tcg_temp_new_i64();
338         frm = tcg_temp_new_i64();
339         dest = tcg_temp_new_i64();
340 
341         zf = tcg_temp_new_i64();
342         nf = tcg_temp_new_i64();
343         vf = tcg_temp_new_i64();
344 
345         tcg_gen_extu_i32_i64(zf, cpu_ZF);
346         tcg_gen_ext_i32_i64(nf, cpu_NF);
347         tcg_gen_ext_i32_i64(vf, cpu_VF);
348 
349         vfp_load_reg64(frn, rn);
350         vfp_load_reg64(frm, rm);
351         switch (a->cc) {
352         case 0: /* eq: Z */
353             tcg_gen_movcond_i64(TCG_COND_EQ, dest, zf, zero, frn, frm);
354             break;
355         case 1: /* vs: V */
356             tcg_gen_movcond_i64(TCG_COND_LT, dest, vf, zero, frn, frm);
357             break;
358         case 2: /* ge: N == V -> N ^ V == 0 */
359             tmp = tcg_temp_new_i64();
360             tcg_gen_xor_i64(tmp, vf, nf);
361             tcg_gen_movcond_i64(TCG_COND_GE, dest, tmp, zero, frn, frm);
362             break;
363         case 3: /* gt: !Z && N == V */
364             tcg_gen_movcond_i64(TCG_COND_NE, dest, zf, zero, frn, frm);
365             tmp = tcg_temp_new_i64();
366             tcg_gen_xor_i64(tmp, vf, nf);
367             tcg_gen_movcond_i64(TCG_COND_GE, dest, tmp, zero, dest, frm);
368             break;
369         }
370         vfp_store_reg64(dest, rd);
371     } else {
372         TCGv_i32 frn, frm, dest;
373         TCGv_i32 tmp, zero;
374 
375         zero = tcg_constant_i32(0);
376 
377         frn = tcg_temp_new_i32();
378         frm = tcg_temp_new_i32();
379         dest = tcg_temp_new_i32();
380         vfp_load_reg32(frn, rn);
381         vfp_load_reg32(frm, rm);
382         switch (a->cc) {
383         case 0: /* eq: Z */
384             tcg_gen_movcond_i32(TCG_COND_EQ, dest, cpu_ZF, zero, frn, frm);
385             break;
386         case 1: /* vs: V */
387             tcg_gen_movcond_i32(TCG_COND_LT, dest, cpu_VF, zero, frn, frm);
388             break;
389         case 2: /* ge: N == V -> N ^ V == 0 */
390             tmp = tcg_temp_new_i32();
391             tcg_gen_xor_i32(tmp, cpu_VF, cpu_NF);
392             tcg_gen_movcond_i32(TCG_COND_GE, dest, tmp, zero, frn, frm);
393             break;
394         case 3: /* gt: !Z && N == V */
395             tcg_gen_movcond_i32(TCG_COND_NE, dest, cpu_ZF, zero, frn, frm);
396             tmp = tcg_temp_new_i32();
397             tcg_gen_xor_i32(tmp, cpu_VF, cpu_NF);
398             tcg_gen_movcond_i32(TCG_COND_GE, dest, tmp, zero, dest, frm);
399             break;
400         }
401         /* For fp16 the top half is always zeroes */
402         if (sz == 1) {
403             tcg_gen_andi_i32(dest, dest, 0xffff);
404         }
405         vfp_store_reg32(dest, rd);
406     }
407 
408     return true;
409 }
410 
411 /*
412  * Table for converting the most common AArch32 encoding of
413  * rounding mode to arm_fprounding order (which matches the
414  * common AArch64 order); see ARM ARM pseudocode FPDecodeRM().
415  */
416 static const uint8_t fp_decode_rm[] = {
417     FPROUNDING_TIEAWAY,
418     FPROUNDING_TIEEVEN,
419     FPROUNDING_POSINF,
420     FPROUNDING_NEGINF,
421 };
422 
423 static bool trans_VRINT(DisasContext *s, arg_VRINT *a)
424 {
425     uint32_t rd, rm;
426     int sz = a->sz;
427     TCGv_ptr fpst;
428     TCGv_i32 tcg_rmode;
429     int rounding = fp_decode_rm[a->rm];
430 
431     if (!dc_isar_feature(aa32_vrint, s)) {
432         return false;
433     }
434 
435     if (sz == 3 && !dc_isar_feature(aa32_fpdp_v2, s)) {
436         return false;
437     }
438 
439     if (sz == 1 && !dc_isar_feature(aa32_fp16_arith, s)) {
440         return false;
441     }
442 
443     /* UNDEF accesses to D16-D31 if they don't exist */
444     if (sz == 3 && !dc_isar_feature(aa32_simd_r32, s) &&
445         ((a->vm | a->vd) & 0x10)) {
446         return false;
447     }
448 
449     rd = a->vd;
450     rm = a->vm;
451 
452     if (!vfp_access_check(s)) {
453         return true;
454     }
455 
456     if (sz == 1) {
457         fpst = fpstatus_ptr(FPST_FPCR_F16);
458     } else {
459         fpst = fpstatus_ptr(FPST_FPCR);
460     }
461 
462     tcg_rmode = gen_set_rmode(rounding, fpst);
463 
464     if (sz == 3) {
465         TCGv_i64 tcg_op;
466         TCGv_i64 tcg_res;
467         tcg_op = tcg_temp_new_i64();
468         tcg_res = tcg_temp_new_i64();
469         vfp_load_reg64(tcg_op, rm);
470         gen_helper_rintd(tcg_res, tcg_op, fpst);
471         vfp_store_reg64(tcg_res, rd);
472     } else {
473         TCGv_i32 tcg_op;
474         TCGv_i32 tcg_res;
475         tcg_op = tcg_temp_new_i32();
476         tcg_res = tcg_temp_new_i32();
477         vfp_load_reg32(tcg_op, rm);
478         if (sz == 1) {
479             gen_helper_rinth(tcg_res, tcg_op, fpst);
480         } else {
481             gen_helper_rints(tcg_res, tcg_op, fpst);
482         }
483         vfp_store_reg32(tcg_res, rd);
484     }
485 
486     gen_restore_rmode(tcg_rmode, fpst);
487     return true;
488 }
489 
490 static bool trans_VCVT(DisasContext *s, arg_VCVT *a)
491 {
492     uint32_t rd, rm;
493     int sz = a->sz;
494     TCGv_ptr fpst;
495     TCGv_i32 tcg_rmode, tcg_shift;
496     int rounding = fp_decode_rm[a->rm];
497     bool is_signed = a->op;
498 
499     if (!dc_isar_feature(aa32_vcvt_dr, s)) {
500         return false;
501     }
502 
503     if (sz == 3 && !dc_isar_feature(aa32_fpdp_v2, s)) {
504         return false;
505     }
506 
507     if (sz == 1 && !dc_isar_feature(aa32_fp16_arith, s)) {
508         return false;
509     }
510 
511     /* UNDEF accesses to D16-D31 if they don't exist */
512     if (sz == 3 && !dc_isar_feature(aa32_simd_r32, s) && (a->vm & 0x10)) {
513         return false;
514     }
515 
516     rd = a->vd;
517     rm = a->vm;
518 
519     if (!vfp_access_check(s)) {
520         return true;
521     }
522 
523     if (sz == 1) {
524         fpst = fpstatus_ptr(FPST_FPCR_F16);
525     } else {
526         fpst = fpstatus_ptr(FPST_FPCR);
527     }
528 
529     tcg_shift = tcg_constant_i32(0);
530     tcg_rmode = gen_set_rmode(rounding, fpst);
531 
532     if (sz == 3) {
533         TCGv_i64 tcg_double, tcg_res;
534         TCGv_i32 tcg_tmp;
535         tcg_double = tcg_temp_new_i64();
536         tcg_res = tcg_temp_new_i64();
537         tcg_tmp = tcg_temp_new_i32();
538         vfp_load_reg64(tcg_double, rm);
539         if (is_signed) {
540             gen_helper_vfp_tosld(tcg_res, tcg_double, tcg_shift, fpst);
541         } else {
542             gen_helper_vfp_tould(tcg_res, tcg_double, tcg_shift, fpst);
543         }
544         tcg_gen_extrl_i64_i32(tcg_tmp, tcg_res);
545         vfp_store_reg32(tcg_tmp, rd);
546     } else {
547         TCGv_i32 tcg_single, tcg_res;
548         tcg_single = tcg_temp_new_i32();
549         tcg_res = tcg_temp_new_i32();
550         vfp_load_reg32(tcg_single, rm);
551         if (sz == 1) {
552             if (is_signed) {
553                 gen_helper_vfp_toslh(tcg_res, tcg_single, tcg_shift, fpst);
554             } else {
555                 gen_helper_vfp_toulh(tcg_res, tcg_single, tcg_shift, fpst);
556             }
557         } else {
558             if (is_signed) {
559                 gen_helper_vfp_tosls(tcg_res, tcg_single, tcg_shift, fpst);
560             } else {
561                 gen_helper_vfp_touls(tcg_res, tcg_single, tcg_shift, fpst);
562             }
563         }
564         vfp_store_reg32(tcg_res, rd);
565     }
566 
567     gen_restore_rmode(tcg_rmode, fpst);
568     return true;
569 }
570 
571 bool mve_skip_vmov(DisasContext *s, int vn, int index, int size)
572 {
573     /*
574      * In a CPU with MVE, the VMOV (vector lane to general-purpose register)
575      * and VMOV (general-purpose register to vector lane) insns are not
576      * predicated, but they are subject to beatwise execution if they are
577      * not in an IT block.
578      *
579      * Since our implementation always executes all 4 beats in one tick,
580      * this means only that if PSR.ECI says we should not be executing
581      * the beat corresponding to the lane of the vector register being
582      * accessed then we should skip performing the move, and that we need
583      * to do the usual check for bad ECI state and advance of ECI state.
584      *
585      * Note that if PSR.ECI is non-zero then we cannot be in an IT block.
586      *
587      * Return true if this VMOV scalar <-> gpreg should be skipped because
588      * the MVE PSR.ECI state says we skip the beat where the store happens.
589      */
590 
591     /* Calculate the byte offset into Qn which we're going to access */
592     int ofs = (index << size) + ((vn & 1) * 8);
593 
594     if (!dc_isar_feature(aa32_mve, s)) {
595         return false;
596     }
597 
598     switch (s->eci) {
599     case ECI_NONE:
600         return false;
601     case ECI_A0:
602         return ofs < 4;
603     case ECI_A0A1:
604         return ofs < 8;
605     case ECI_A0A1A2:
606     case ECI_A0A1A2B0:
607         return ofs < 12;
608     default:
609         g_assert_not_reached();
610     }
611 }
612 
613 static bool trans_VMOV_to_gp(DisasContext *s, arg_VMOV_to_gp *a)
614 {
615     /* VMOV scalar to general purpose register */
616     TCGv_i32 tmp;
617 
618     /*
619      * SIZE == MO_32 is a VFP instruction; otherwise NEON. MVE has
620      * all sizes, whether the CPU has fp or not.
621      */
622     if (!dc_isar_feature(aa32_mve, s)) {
623         if (a->size == MO_32
624             ? !dc_isar_feature(aa32_fpsp_v2, s)
625             : !arm_dc_feature(s, ARM_FEATURE_NEON)) {
626             return false;
627         }
628     }
629 
630     /* UNDEF accesses to D16-D31 if they don't exist */
631     if (!dc_isar_feature(aa32_simd_r32, s) && (a->vn & 0x10)) {
632         return false;
633     }
634 
635     if (dc_isar_feature(aa32_mve, s)) {
636         if (!mve_eci_check(s)) {
637             return true;
638         }
639     }
640 
641     if (!vfp_access_check(s)) {
642         return true;
643     }
644 
645     if (!mve_skip_vmov(s, a->vn, a->index, a->size)) {
646         tmp = tcg_temp_new_i32();
647         read_neon_element32(tmp, a->vn, a->index,
648                             a->size | (a->u ? 0 : MO_SIGN));
649         store_reg(s, a->rt, tmp);
650     }
651 
652     if (dc_isar_feature(aa32_mve, s)) {
653         mve_update_and_store_eci(s);
654     }
655     return true;
656 }
657 
658 static bool trans_VMOV_from_gp(DisasContext *s, arg_VMOV_from_gp *a)
659 {
660     /* VMOV general purpose register to scalar */
661     TCGv_i32 tmp;
662 
663     /*
664      * SIZE == MO_32 is a VFP instruction; otherwise NEON. MVE has
665      * all sizes, whether the CPU has fp or not.
666      */
667     if (!dc_isar_feature(aa32_mve, s)) {
668         if (a->size == MO_32
669             ? !dc_isar_feature(aa32_fpsp_v2, s)
670             : !arm_dc_feature(s, ARM_FEATURE_NEON)) {
671             return false;
672         }
673     }
674 
675     /* UNDEF accesses to D16-D31 if they don't exist */
676     if (!dc_isar_feature(aa32_simd_r32, s) && (a->vn & 0x10)) {
677         return false;
678     }
679 
680     if (dc_isar_feature(aa32_mve, s)) {
681         if (!mve_eci_check(s)) {
682             return true;
683         }
684     }
685 
686     if (!vfp_access_check(s)) {
687         return true;
688     }
689 
690     if (!mve_skip_vmov(s, a->vn, a->index, a->size)) {
691         tmp = load_reg(s, a->rt);
692         write_neon_element32(tmp, a->vn, a->index, a->size);
693     }
694 
695     if (dc_isar_feature(aa32_mve, s)) {
696         mve_update_and_store_eci(s);
697     }
698     return true;
699 }
700 
701 static bool trans_VDUP(DisasContext *s, arg_VDUP *a)
702 {
703     /* VDUP (general purpose register) */
704     TCGv_i32 tmp;
705     int size, vec_size;
706 
707     if (!arm_dc_feature(s, ARM_FEATURE_NEON)) {
708         return false;
709     }
710 
711     /* UNDEF accesses to D16-D31 if they don't exist */
712     if (!dc_isar_feature(aa32_simd_r32, s) && (a->vn & 0x10)) {
713         return false;
714     }
715 
716     if (a->b && a->e) {
717         return false;
718     }
719 
720     if (a->q && (a->vn & 1)) {
721         return false;
722     }
723 
724     vec_size = a->q ? 16 : 8;
725     if (a->b) {
726         size = 0;
727     } else if (a->e) {
728         size = 1;
729     } else {
730         size = 2;
731     }
732 
733     if (!vfp_access_check(s)) {
734         return true;
735     }
736 
737     tmp = load_reg(s, a->rt);
738     tcg_gen_gvec_dup_i32(size, neon_full_reg_offset(a->vn),
739                          vec_size, vec_size, tmp);
740     return true;
741 }
742 
743 static bool trans_VMSR_VMRS(DisasContext *s, arg_VMSR_VMRS *a)
744 {
745     TCGv_i32 tmp;
746     bool ignore_vfp_enabled = false;
747 
748     if (arm_dc_feature(s, ARM_FEATURE_M)) {
749         /* M profile version was already handled in m-nocp.decode */
750         return false;
751     }
752 
753     if (!dc_isar_feature(aa32_fpsp_v2, s)) {
754         return false;
755     }
756 
757     switch (a->reg) {
758     case ARM_VFP_FPSID:
759         /*
760          * VFPv2 allows access to FPSID from userspace; VFPv3 restricts
761          * all ID registers to privileged access only.
762          */
763         if (IS_USER(s) && dc_isar_feature(aa32_fpsp_v3, s)) {
764             return false;
765         }
766         ignore_vfp_enabled = true;
767         break;
768     case ARM_VFP_MVFR0:
769     case ARM_VFP_MVFR1:
770         if (IS_USER(s) || !arm_dc_feature(s, ARM_FEATURE_MVFR)) {
771             return false;
772         }
773         ignore_vfp_enabled = true;
774         break;
775     case ARM_VFP_MVFR2:
776         if (IS_USER(s) || !arm_dc_feature(s, ARM_FEATURE_V8)) {
777             return false;
778         }
779         ignore_vfp_enabled = true;
780         break;
781     case ARM_VFP_FPSCR:
782         break;
783     case ARM_VFP_FPEXC:
784         if (IS_USER(s)) {
785             return false;
786         }
787         ignore_vfp_enabled = true;
788         break;
789     case ARM_VFP_FPINST:
790     case ARM_VFP_FPINST2:
791         /* Not present in VFPv3 */
792         if (IS_USER(s) || dc_isar_feature(aa32_fpsp_v3, s)) {
793             return false;
794         }
795         break;
796     default:
797         return false;
798     }
799 
800     /*
801      * Call vfp_access_check_a() directly, because we need to tell
802      * it to ignore FPEXC.EN for some register accesses.
803      */
804     if (!vfp_access_check_a(s, ignore_vfp_enabled)) {
805         return true;
806     }
807 
808     if (a->l) {
809         /* VMRS, move VFP special register to gp register */
810         switch (a->reg) {
811         case ARM_VFP_MVFR0:
812         case ARM_VFP_MVFR1:
813         case ARM_VFP_MVFR2:
814         case ARM_VFP_FPSID:
815             if (s->current_el == 1) {
816                 gen_set_condexec(s);
817                 gen_update_pc(s, 0);
818                 gen_helper_check_hcr_el2_trap(tcg_env,
819                                               tcg_constant_i32(a->rt),
820                                               tcg_constant_i32(a->reg));
821             }
822             /* fall through */
823         case ARM_VFP_FPEXC:
824         case ARM_VFP_FPINST:
825         case ARM_VFP_FPINST2:
826             tmp = load_cpu_field(vfp.xregs[a->reg]);
827             break;
828         case ARM_VFP_FPSCR:
829             if (a->rt == 15) {
830                 tmp = load_cpu_field(vfp.xregs[ARM_VFP_FPSCR]);
831                 tcg_gen_andi_i32(tmp, tmp, FPCR_NZCV_MASK);
832             } else {
833                 tmp = tcg_temp_new_i32();
834                 gen_helper_vfp_get_fpscr(tmp, tcg_env);
835             }
836             break;
837         default:
838             g_assert_not_reached();
839         }
840 
841         if (a->rt == 15) {
842             /* Set the 4 flag bits in the CPSR.  */
843             gen_set_nzcv(tmp);
844         } else {
845             store_reg(s, a->rt, tmp);
846         }
847     } else {
848         /* VMSR, move gp register to VFP special register */
849         switch (a->reg) {
850         case ARM_VFP_FPSID:
851         case ARM_VFP_MVFR0:
852         case ARM_VFP_MVFR1:
853         case ARM_VFP_MVFR2:
854             /* Writes are ignored.  */
855             break;
856         case ARM_VFP_FPSCR:
857             tmp = load_reg(s, a->rt);
858             gen_helper_vfp_set_fpscr(tcg_env, tmp);
859             gen_lookup_tb(s);
860             break;
861         case ARM_VFP_FPEXC:
862             /*
863              * TODO: VFP subarchitecture support.
864              * For now, keep the EN bit only
865              */
866             tmp = load_reg(s, a->rt);
867             tcg_gen_andi_i32(tmp, tmp, 1 << 30);
868             store_cpu_field(tmp, vfp.xregs[a->reg]);
869             gen_lookup_tb(s);
870             break;
871         case ARM_VFP_FPINST:
872         case ARM_VFP_FPINST2:
873             tmp = load_reg(s, a->rt);
874             store_cpu_field(tmp, vfp.xregs[a->reg]);
875             break;
876         default:
877             g_assert_not_reached();
878         }
879     }
880 
881     return true;
882 }
883 
884 
885 static bool trans_VMOV_half(DisasContext *s, arg_VMOV_single *a)
886 {
887     TCGv_i32 tmp;
888 
889     if (!dc_isar_feature(aa32_fp16_arith, s)) {
890         return false;
891     }
892 
893     if (a->rt == 15) {
894         /* UNPREDICTABLE; we choose to UNDEF */
895         return false;
896     }
897 
898     if (!vfp_access_check(s)) {
899         return true;
900     }
901 
902     if (a->l) {
903         /* VFP to general purpose register */
904         tmp = tcg_temp_new_i32();
905         vfp_load_reg32(tmp, a->vn);
906         tcg_gen_andi_i32(tmp, tmp, 0xffff);
907         store_reg(s, a->rt, tmp);
908     } else {
909         /* general purpose register to VFP */
910         tmp = load_reg(s, a->rt);
911         tcg_gen_andi_i32(tmp, tmp, 0xffff);
912         vfp_store_reg32(tmp, a->vn);
913     }
914 
915     return true;
916 }
917 
918 static bool trans_VMOV_single(DisasContext *s, arg_VMOV_single *a)
919 {
920     TCGv_i32 tmp;
921 
922     if (!dc_isar_feature(aa32_fpsp_v2, s) && !dc_isar_feature(aa32_mve, s)) {
923         return false;
924     }
925 
926     if (!vfp_access_check(s)) {
927         return true;
928     }
929 
930     if (a->l) {
931         /* VFP to general purpose register */
932         tmp = tcg_temp_new_i32();
933         vfp_load_reg32(tmp, a->vn);
934         if (a->rt == 15) {
935             /* Set the 4 flag bits in the CPSR.  */
936             gen_set_nzcv(tmp);
937         } else {
938             store_reg(s, a->rt, tmp);
939         }
940     } else {
941         /* general purpose register to VFP */
942         tmp = load_reg(s, a->rt);
943         vfp_store_reg32(tmp, a->vn);
944     }
945 
946     return true;
947 }
948 
949 static bool trans_VMOV_64_sp(DisasContext *s, arg_VMOV_64_sp *a)
950 {
951     TCGv_i32 tmp;
952 
953     if (!dc_isar_feature(aa32_fpsp_v2, s) && !dc_isar_feature(aa32_mve, s)) {
954         return false;
955     }
956 
957     /*
958      * VMOV between two general-purpose registers and two single precision
959      * floating point registers
960      */
961     if (!vfp_access_check(s)) {
962         return true;
963     }
964 
965     if (a->op) {
966         /* fpreg to gpreg */
967         tmp = tcg_temp_new_i32();
968         vfp_load_reg32(tmp, a->vm);
969         store_reg(s, a->rt, tmp);
970         tmp = tcg_temp_new_i32();
971         vfp_load_reg32(tmp, a->vm + 1);
972         store_reg(s, a->rt2, tmp);
973     } else {
974         /* gpreg to fpreg */
975         tmp = load_reg(s, a->rt);
976         vfp_store_reg32(tmp, a->vm);
977         tmp = load_reg(s, a->rt2);
978         vfp_store_reg32(tmp, a->vm + 1);
979     }
980 
981     return true;
982 }
983 
984 static bool trans_VMOV_64_dp(DisasContext *s, arg_VMOV_64_dp *a)
985 {
986     TCGv_i32 tmp;
987 
988     /*
989      * VMOV between two general-purpose registers and one double precision
990      * floating point register.  Note that this does not require support
991      * for double precision arithmetic.
992      */
993     if (!dc_isar_feature(aa32_fpsp_v2, s) && !dc_isar_feature(aa32_mve, s)) {
994         return false;
995     }
996 
997     /* UNDEF accesses to D16-D31 if they don't exist */
998     if (!dc_isar_feature(aa32_simd_r32, s) && (a->vm & 0x10)) {
999         return false;
1000     }
1001 
1002     if (!vfp_access_check(s)) {
1003         return true;
1004     }
1005 
1006     if (a->op) {
1007         /* fpreg to gpreg */
1008         tmp = tcg_temp_new_i32();
1009         vfp_load_reg32(tmp, a->vm * 2);
1010         store_reg(s, a->rt, tmp);
1011         tmp = tcg_temp_new_i32();
1012         vfp_load_reg32(tmp, a->vm * 2 + 1);
1013         store_reg(s, a->rt2, tmp);
1014     } else {
1015         /* gpreg to fpreg */
1016         tmp = load_reg(s, a->rt);
1017         vfp_store_reg32(tmp, a->vm * 2);
1018         tmp = load_reg(s, a->rt2);
1019         vfp_store_reg32(tmp, a->vm * 2 + 1);
1020     }
1021 
1022     return true;
1023 }
1024 
1025 static bool trans_VLDR_VSTR_hp(DisasContext *s, arg_VLDR_VSTR_sp *a)
1026 {
1027     uint32_t offset;
1028     TCGv_i32 addr, tmp;
1029 
1030     if (!dc_isar_feature(aa32_fpsp_v2, s) && !dc_isar_feature(aa32_mve, s)) {
1031         return false;
1032     }
1033 
1034     if (!vfp_access_check(s)) {
1035         return true;
1036     }
1037 
1038     /* imm8 field is offset/2 for fp16, unlike fp32 and fp64 */
1039     offset = a->imm << 1;
1040     if (!a->u) {
1041         offset = -offset;
1042     }
1043 
1044     /* For thumb, use of PC is UNPREDICTABLE.  */
1045     addr = add_reg_for_lit(s, a->rn, offset);
1046     tmp = tcg_temp_new_i32();
1047     if (a->l) {
1048         gen_aa32_ld_i32(s, tmp, addr, get_mem_index(s), MO_UW | MO_ALIGN);
1049         vfp_store_reg32(tmp, a->vd);
1050     } else {
1051         vfp_load_reg32(tmp, a->vd);
1052         gen_aa32_st_i32(s, tmp, addr, get_mem_index(s), MO_UW | MO_ALIGN);
1053     }
1054     return true;
1055 }
1056 
1057 static bool trans_VLDR_VSTR_sp(DisasContext *s, arg_VLDR_VSTR_sp *a)
1058 {
1059     uint32_t offset;
1060     TCGv_i32 addr, tmp;
1061 
1062     if (!dc_isar_feature(aa32_fpsp_v2, s) && !dc_isar_feature(aa32_mve, s)) {
1063         return false;
1064     }
1065 
1066     if (!vfp_access_check(s)) {
1067         return true;
1068     }
1069 
1070     offset = a->imm << 2;
1071     if (!a->u) {
1072         offset = -offset;
1073     }
1074 
1075     /* For thumb, use of PC is UNPREDICTABLE.  */
1076     addr = add_reg_for_lit(s, a->rn, offset);
1077     tmp = tcg_temp_new_i32();
1078     if (a->l) {
1079         gen_aa32_ld_i32(s, tmp, addr, get_mem_index(s), MO_UL | MO_ALIGN);
1080         vfp_store_reg32(tmp, a->vd);
1081     } else {
1082         vfp_load_reg32(tmp, a->vd);
1083         gen_aa32_st_i32(s, tmp, addr, get_mem_index(s), MO_UL | MO_ALIGN);
1084     }
1085     return true;
1086 }
1087 
1088 static bool trans_VLDR_VSTR_dp(DisasContext *s, arg_VLDR_VSTR_dp *a)
1089 {
1090     uint32_t offset;
1091     TCGv_i32 addr;
1092     TCGv_i64 tmp;
1093 
1094     /* Note that this does not require support for double arithmetic.  */
1095     if (!dc_isar_feature(aa32_fpsp_v2, s) && !dc_isar_feature(aa32_mve, s)) {
1096         return false;
1097     }
1098 
1099     /* UNDEF accesses to D16-D31 if they don't exist */
1100     if (!dc_isar_feature(aa32_simd_r32, s) && (a->vd & 0x10)) {
1101         return false;
1102     }
1103 
1104     if (!vfp_access_check(s)) {
1105         return true;
1106     }
1107 
1108     offset = a->imm << 2;
1109     if (!a->u) {
1110         offset = -offset;
1111     }
1112 
1113     /* For thumb, use of PC is UNPREDICTABLE.  */
1114     addr = add_reg_for_lit(s, a->rn, offset);
1115     tmp = tcg_temp_new_i64();
1116     if (a->l) {
1117         gen_aa32_ld_i64(s, tmp, addr, get_mem_index(s), MO_UQ | MO_ALIGN_4);
1118         vfp_store_reg64(tmp, a->vd);
1119     } else {
1120         vfp_load_reg64(tmp, a->vd);
1121         gen_aa32_st_i64(s, tmp, addr, get_mem_index(s), MO_UQ | MO_ALIGN_4);
1122     }
1123     return true;
1124 }
1125 
1126 static bool trans_VLDM_VSTM_sp(DisasContext *s, arg_VLDM_VSTM_sp *a)
1127 {
1128     uint32_t offset;
1129     TCGv_i32 addr, tmp;
1130     int i, n;
1131 
1132     if (!dc_isar_feature(aa32_fpsp_v2, s) && !dc_isar_feature(aa32_mve, s)) {
1133         return false;
1134     }
1135 
1136     n = a->imm;
1137 
1138     if (n == 0 || (a->vd + n) > 32) {
1139         /*
1140          * UNPREDICTABLE cases for bad immediates: we choose to
1141          * UNDEF to avoid generating huge numbers of TCG ops
1142          */
1143         return false;
1144     }
1145     if (a->rn == 15 && a->w) {
1146         /* writeback to PC is UNPREDICTABLE, we choose to UNDEF */
1147         return false;
1148     }
1149 
1150     s->eci_handled = true;
1151 
1152     if (!vfp_access_check(s)) {
1153         return true;
1154     }
1155 
1156     /* For thumb, use of PC is UNPREDICTABLE.  */
1157     addr = add_reg_for_lit(s, a->rn, 0);
1158     if (a->p) {
1159         /* pre-decrement */
1160         tcg_gen_addi_i32(addr, addr, -(a->imm << 2));
1161     }
1162 
1163     if (s->v8m_stackcheck && a->rn == 13 && a->w) {
1164         /*
1165          * Here 'addr' is the lowest address we will store to,
1166          * and is either the old SP (if post-increment) or
1167          * the new SP (if pre-decrement). For post-increment
1168          * where the old value is below the limit and the new
1169          * value is above, it is UNKNOWN whether the limit check
1170          * triggers; we choose to trigger.
1171          */
1172         gen_helper_v8m_stackcheck(tcg_env, addr);
1173     }
1174 
1175     offset = 4;
1176     tmp = tcg_temp_new_i32();
1177     for (i = 0; i < n; i++) {
1178         if (a->l) {
1179             /* load */
1180             gen_aa32_ld_i32(s, tmp, addr, get_mem_index(s), MO_UL | MO_ALIGN);
1181             vfp_store_reg32(tmp, a->vd + i);
1182         } else {
1183             /* store */
1184             vfp_load_reg32(tmp, a->vd + i);
1185             gen_aa32_st_i32(s, tmp, addr, get_mem_index(s), MO_UL | MO_ALIGN);
1186         }
1187         tcg_gen_addi_i32(addr, addr, offset);
1188     }
1189     if (a->w) {
1190         /* writeback */
1191         if (a->p) {
1192             offset = -offset * n;
1193             tcg_gen_addi_i32(addr, addr, offset);
1194         }
1195         store_reg(s, a->rn, addr);
1196     }
1197 
1198     clear_eci_state(s);
1199     return true;
1200 }
1201 
1202 static bool trans_VLDM_VSTM_dp(DisasContext *s, arg_VLDM_VSTM_dp *a)
1203 {
1204     uint32_t offset;
1205     TCGv_i32 addr;
1206     TCGv_i64 tmp;
1207     int i, n;
1208 
1209     /* Note that this does not require support for double arithmetic.  */
1210     if (!dc_isar_feature(aa32_fpsp_v2, s) && !dc_isar_feature(aa32_mve, s)) {
1211         return false;
1212     }
1213 
1214     n = a->imm >> 1;
1215 
1216     if (n == 0 || (a->vd + n) > 32 || n > 16) {
1217         /*
1218          * UNPREDICTABLE cases for bad immediates: we choose to
1219          * UNDEF to avoid generating huge numbers of TCG ops
1220          */
1221         return false;
1222     }
1223     if (a->rn == 15 && a->w) {
1224         /* writeback to PC is UNPREDICTABLE, we choose to UNDEF */
1225         return false;
1226     }
1227 
1228     /* UNDEF accesses to D16-D31 if they don't exist */
1229     if (!dc_isar_feature(aa32_simd_r32, s) && (a->vd + n) > 16) {
1230         return false;
1231     }
1232 
1233     s->eci_handled = true;
1234 
1235     if (!vfp_access_check(s)) {
1236         return true;
1237     }
1238 
1239     /* For thumb, use of PC is UNPREDICTABLE.  */
1240     addr = add_reg_for_lit(s, a->rn, 0);
1241     if (a->p) {
1242         /* pre-decrement */
1243         tcg_gen_addi_i32(addr, addr, -(a->imm << 2));
1244     }
1245 
1246     if (s->v8m_stackcheck && a->rn == 13 && a->w) {
1247         /*
1248          * Here 'addr' is the lowest address we will store to,
1249          * and is either the old SP (if post-increment) or
1250          * the new SP (if pre-decrement). For post-increment
1251          * where the old value is below the limit and the new
1252          * value is above, it is UNKNOWN whether the limit check
1253          * triggers; we choose to trigger.
1254          */
1255         gen_helper_v8m_stackcheck(tcg_env, addr);
1256     }
1257 
1258     offset = 8;
1259     tmp = tcg_temp_new_i64();
1260     for (i = 0; i < n; i++) {
1261         if (a->l) {
1262             /* load */
1263             gen_aa32_ld_i64(s, tmp, addr, get_mem_index(s), MO_UQ | MO_ALIGN_4);
1264             vfp_store_reg64(tmp, a->vd + i);
1265         } else {
1266             /* store */
1267             vfp_load_reg64(tmp, a->vd + i);
1268             gen_aa32_st_i64(s, tmp, addr, get_mem_index(s), MO_UQ | MO_ALIGN_4);
1269         }
1270         tcg_gen_addi_i32(addr, addr, offset);
1271     }
1272     if (a->w) {
1273         /* writeback */
1274         if (a->p) {
1275             offset = -offset * n;
1276         } else if (a->imm & 1) {
1277             offset = 4;
1278         } else {
1279             offset = 0;
1280         }
1281 
1282         if (offset != 0) {
1283             tcg_gen_addi_i32(addr, addr, offset);
1284         }
1285         store_reg(s, a->rn, addr);
1286     }
1287 
1288     clear_eci_state(s);
1289     return true;
1290 }
1291 
1292 /*
1293  * Types for callbacks for do_vfp_3op_sp() and do_vfp_3op_dp().
1294  * The callback should emit code to write a value to vd. If
1295  * do_vfp_3op_{sp,dp}() was passed reads_vd then the TCGv vd
1296  * will contain the old value of the relevant VFP register;
1297  * otherwise it must be written to only.
1298  */
1299 typedef void VFPGen3OpSPFn(TCGv_i32 vd,
1300                            TCGv_i32 vn, TCGv_i32 vm, TCGv_ptr fpst);
1301 typedef void VFPGen3OpDPFn(TCGv_i64 vd,
1302                            TCGv_i64 vn, TCGv_i64 vm, TCGv_ptr fpst);
1303 
1304 /*
1305  * Types for callbacks for do_vfp_2op_sp() and do_vfp_2op_dp().
1306  * The callback should emit code to write a value to vd (which
1307  * should be written to only).
1308  */
1309 typedef void VFPGen2OpSPFn(TCGv_i32 vd, TCGv_i32 vm);
1310 typedef void VFPGen2OpDPFn(TCGv_i64 vd, TCGv_i64 vm);
1311 
1312 /*
1313  * Return true if the specified S reg is in a scalar bank
1314  * (ie if it is s0..s7)
1315  */
1316 static inline bool vfp_sreg_is_scalar(int reg)
1317 {
1318     return (reg & 0x18) == 0;
1319 }
1320 
1321 /*
1322  * Return true if the specified D reg is in a scalar bank
1323  * (ie if it is d0..d3 or d16..d19)
1324  */
1325 static inline bool vfp_dreg_is_scalar(int reg)
1326 {
1327     return (reg & 0xc) == 0;
1328 }
1329 
1330 /*
1331  * Advance the S reg number forwards by delta within its bank
1332  * (ie increment the low 3 bits but leave the rest the same)
1333  */
1334 static inline int vfp_advance_sreg(int reg, int delta)
1335 {
1336     return ((reg + delta) & 0x7) | (reg & ~0x7);
1337 }
1338 
1339 /*
1340  * Advance the D reg number forwards by delta within its bank
1341  * (ie increment the low 2 bits but leave the rest the same)
1342  */
1343 static inline int vfp_advance_dreg(int reg, int delta)
1344 {
1345     return ((reg + delta) & 0x3) | (reg & ~0x3);
1346 }
1347 
1348 /*
1349  * Perform a 3-operand VFP data processing instruction. fn is the
1350  * callback to do the actual operation; this function deals with the
1351  * code to handle looping around for VFP vector processing.
1352  */
1353 static bool do_vfp_3op_sp(DisasContext *s, VFPGen3OpSPFn *fn,
1354                           int vd, int vn, int vm, bool reads_vd)
1355 {
1356     uint32_t delta_m = 0;
1357     uint32_t delta_d = 0;
1358     int veclen = s->vec_len;
1359     TCGv_i32 f0, f1, fd;
1360     TCGv_ptr fpst;
1361 
1362     if (!dc_isar_feature(aa32_fpsp_v2, s)) {
1363         return false;
1364     }
1365 
1366     if (!dc_isar_feature(aa32_fpshvec, s) &&
1367         (veclen != 0 || s->vec_stride != 0)) {
1368         return false;
1369     }
1370 
1371     if (!vfp_access_check(s)) {
1372         return true;
1373     }
1374 
1375     if (veclen > 0) {
1376         /* Figure out what type of vector operation this is.  */
1377         if (vfp_sreg_is_scalar(vd)) {
1378             /* scalar */
1379             veclen = 0;
1380         } else {
1381             delta_d = s->vec_stride + 1;
1382 
1383             if (vfp_sreg_is_scalar(vm)) {
1384                 /* mixed scalar/vector */
1385                 delta_m = 0;
1386             } else {
1387                 /* vector */
1388                 delta_m = delta_d;
1389             }
1390         }
1391     }
1392 
1393     f0 = tcg_temp_new_i32();
1394     f1 = tcg_temp_new_i32();
1395     fd = tcg_temp_new_i32();
1396     fpst = fpstatus_ptr(FPST_FPCR);
1397 
1398     vfp_load_reg32(f0, vn);
1399     vfp_load_reg32(f1, vm);
1400 
1401     for (;;) {
1402         if (reads_vd) {
1403             vfp_load_reg32(fd, vd);
1404         }
1405         fn(fd, f0, f1, fpst);
1406         vfp_store_reg32(fd, vd);
1407 
1408         if (veclen == 0) {
1409             break;
1410         }
1411 
1412         /* Set up the operands for the next iteration */
1413         veclen--;
1414         vd = vfp_advance_sreg(vd, delta_d);
1415         vn = vfp_advance_sreg(vn, delta_d);
1416         vfp_load_reg32(f0, vn);
1417         if (delta_m) {
1418             vm = vfp_advance_sreg(vm, delta_m);
1419             vfp_load_reg32(f1, vm);
1420         }
1421     }
1422     return true;
1423 }
1424 
1425 static bool do_vfp_3op_hp(DisasContext *s, VFPGen3OpSPFn *fn,
1426                           int vd, int vn, int vm, bool reads_vd)
1427 {
1428     /*
1429      * Do a half-precision operation. Functionally this is
1430      * the same as do_vfp_3op_sp(), except:
1431      *  - it uses the FPST_FPCR_F16
1432      *  - it doesn't need the VFP vector handling (fp16 is a
1433      *    v8 feature, and in v8 VFP vectors don't exist)
1434      *  - it does the aa32_fp16_arith feature test
1435      */
1436     TCGv_i32 f0, f1, fd;
1437     TCGv_ptr fpst;
1438 
1439     if (!dc_isar_feature(aa32_fp16_arith, s)) {
1440         return false;
1441     }
1442 
1443     if (s->vec_len != 0 || s->vec_stride != 0) {
1444         return false;
1445     }
1446 
1447     if (!vfp_access_check(s)) {
1448         return true;
1449     }
1450 
1451     f0 = tcg_temp_new_i32();
1452     f1 = tcg_temp_new_i32();
1453     fd = tcg_temp_new_i32();
1454     fpst = fpstatus_ptr(FPST_FPCR_F16);
1455 
1456     vfp_load_reg32(f0, vn);
1457     vfp_load_reg32(f1, vm);
1458 
1459     if (reads_vd) {
1460         vfp_load_reg32(fd, vd);
1461     }
1462     fn(fd, f0, f1, fpst);
1463     vfp_store_reg32(fd, vd);
1464     return true;
1465 }
1466 
1467 static bool do_vfp_3op_dp(DisasContext *s, VFPGen3OpDPFn *fn,
1468                           int vd, int vn, int vm, bool reads_vd)
1469 {
1470     uint32_t delta_m = 0;
1471     uint32_t delta_d = 0;
1472     int veclen = s->vec_len;
1473     TCGv_i64 f0, f1, fd;
1474     TCGv_ptr fpst;
1475 
1476     if (!dc_isar_feature(aa32_fpdp_v2, s)) {
1477         return false;
1478     }
1479 
1480     /* UNDEF accesses to D16-D31 if they don't exist */
1481     if (!dc_isar_feature(aa32_simd_r32, s) && ((vd | vn | vm) & 0x10)) {
1482         return false;
1483     }
1484 
1485     if (!dc_isar_feature(aa32_fpshvec, s) &&
1486         (veclen != 0 || s->vec_stride != 0)) {
1487         return false;
1488     }
1489 
1490     if (!vfp_access_check(s)) {
1491         return true;
1492     }
1493 
1494     if (veclen > 0) {
1495         /* Figure out what type of vector operation this is.  */
1496         if (vfp_dreg_is_scalar(vd)) {
1497             /* scalar */
1498             veclen = 0;
1499         } else {
1500             delta_d = (s->vec_stride >> 1) + 1;
1501 
1502             if (vfp_dreg_is_scalar(vm)) {
1503                 /* mixed scalar/vector */
1504                 delta_m = 0;
1505             } else {
1506                 /* vector */
1507                 delta_m = delta_d;
1508             }
1509         }
1510     }
1511 
1512     f0 = tcg_temp_new_i64();
1513     f1 = tcg_temp_new_i64();
1514     fd = tcg_temp_new_i64();
1515     fpst = fpstatus_ptr(FPST_FPCR);
1516 
1517     vfp_load_reg64(f0, vn);
1518     vfp_load_reg64(f1, vm);
1519 
1520     for (;;) {
1521         if (reads_vd) {
1522             vfp_load_reg64(fd, vd);
1523         }
1524         fn(fd, f0, f1, fpst);
1525         vfp_store_reg64(fd, vd);
1526 
1527         if (veclen == 0) {
1528             break;
1529         }
1530         /* Set up the operands for the next iteration */
1531         veclen--;
1532         vd = vfp_advance_dreg(vd, delta_d);
1533         vn = vfp_advance_dreg(vn, delta_d);
1534         vfp_load_reg64(f0, vn);
1535         if (delta_m) {
1536             vm = vfp_advance_dreg(vm, delta_m);
1537             vfp_load_reg64(f1, vm);
1538         }
1539     }
1540     return true;
1541 }
1542 
1543 static bool do_vfp_2op_sp(DisasContext *s, VFPGen2OpSPFn *fn, int vd, int vm)
1544 {
1545     uint32_t delta_m = 0;
1546     uint32_t delta_d = 0;
1547     int veclen = s->vec_len;
1548     TCGv_i32 f0, fd;
1549 
1550     /* Note that the caller must check the aa32_fpsp_v2 feature. */
1551 
1552     if (!dc_isar_feature(aa32_fpshvec, s) &&
1553         (veclen != 0 || s->vec_stride != 0)) {
1554         return false;
1555     }
1556 
1557     if (!vfp_access_check(s)) {
1558         return true;
1559     }
1560 
1561     if (veclen > 0) {
1562         /* Figure out what type of vector operation this is.  */
1563         if (vfp_sreg_is_scalar(vd)) {
1564             /* scalar */
1565             veclen = 0;
1566         } else {
1567             delta_d = s->vec_stride + 1;
1568 
1569             if (vfp_sreg_is_scalar(vm)) {
1570                 /* mixed scalar/vector */
1571                 delta_m = 0;
1572             } else {
1573                 /* vector */
1574                 delta_m = delta_d;
1575             }
1576         }
1577     }
1578 
1579     f0 = tcg_temp_new_i32();
1580     fd = tcg_temp_new_i32();
1581 
1582     vfp_load_reg32(f0, vm);
1583 
1584     for (;;) {
1585         fn(fd, f0);
1586         vfp_store_reg32(fd, vd);
1587 
1588         if (veclen == 0) {
1589             break;
1590         }
1591 
1592         if (delta_m == 0) {
1593             /* single source one-many */
1594             while (veclen--) {
1595                 vd = vfp_advance_sreg(vd, delta_d);
1596                 vfp_store_reg32(fd, vd);
1597             }
1598             break;
1599         }
1600 
1601         /* Set up the operands for the next iteration */
1602         veclen--;
1603         vd = vfp_advance_sreg(vd, delta_d);
1604         vm = vfp_advance_sreg(vm, delta_m);
1605         vfp_load_reg32(f0, vm);
1606     }
1607     return true;
1608 }
1609 
1610 static bool do_vfp_2op_hp(DisasContext *s, VFPGen2OpSPFn *fn, int vd, int vm)
1611 {
1612     /*
1613      * Do a half-precision operation. Functionally this is
1614      * the same as do_vfp_2op_sp(), except:
1615      *  - it doesn't need the VFP vector handling (fp16 is a
1616      *    v8 feature, and in v8 VFP vectors don't exist)
1617      *  - it does the aa32_fp16_arith feature test
1618      */
1619     TCGv_i32 f0;
1620 
1621     /* Note that the caller must check the aa32_fp16_arith feature */
1622 
1623     if (!dc_isar_feature(aa32_fp16_arith, s)) {
1624         return false;
1625     }
1626 
1627     if (s->vec_len != 0 || s->vec_stride != 0) {
1628         return false;
1629     }
1630 
1631     if (!vfp_access_check(s)) {
1632         return true;
1633     }
1634 
1635     f0 = tcg_temp_new_i32();
1636     vfp_load_reg32(f0, vm);
1637     fn(f0, f0);
1638     vfp_store_reg32(f0, vd);
1639 
1640     return true;
1641 }
1642 
1643 static bool do_vfp_2op_dp(DisasContext *s, VFPGen2OpDPFn *fn, int vd, int vm)
1644 {
1645     uint32_t delta_m = 0;
1646     uint32_t delta_d = 0;
1647     int veclen = s->vec_len;
1648     TCGv_i64 f0, fd;
1649 
1650     /* Note that the caller must check the aa32_fpdp_v2 feature. */
1651 
1652     /* UNDEF accesses to D16-D31 if they don't exist */
1653     if (!dc_isar_feature(aa32_simd_r32, s) && ((vd | vm) & 0x10)) {
1654         return false;
1655     }
1656 
1657     if (!dc_isar_feature(aa32_fpshvec, s) &&
1658         (veclen != 0 || s->vec_stride != 0)) {
1659         return false;
1660     }
1661 
1662     if (!vfp_access_check(s)) {
1663         return true;
1664     }
1665 
1666     if (veclen > 0) {
1667         /* Figure out what type of vector operation this is.  */
1668         if (vfp_dreg_is_scalar(vd)) {
1669             /* scalar */
1670             veclen = 0;
1671         } else {
1672             delta_d = (s->vec_stride >> 1) + 1;
1673 
1674             if (vfp_dreg_is_scalar(vm)) {
1675                 /* mixed scalar/vector */
1676                 delta_m = 0;
1677             } else {
1678                 /* vector */
1679                 delta_m = delta_d;
1680             }
1681         }
1682     }
1683 
1684     f0 = tcg_temp_new_i64();
1685     fd = tcg_temp_new_i64();
1686 
1687     vfp_load_reg64(f0, vm);
1688 
1689     for (;;) {
1690         fn(fd, f0);
1691         vfp_store_reg64(fd, vd);
1692 
1693         if (veclen == 0) {
1694             break;
1695         }
1696 
1697         if (delta_m == 0) {
1698             /* single source one-many */
1699             while (veclen--) {
1700                 vd = vfp_advance_dreg(vd, delta_d);
1701                 vfp_store_reg64(fd, vd);
1702             }
1703             break;
1704         }
1705 
1706         /* Set up the operands for the next iteration */
1707         veclen--;
1708         vd = vfp_advance_dreg(vd, delta_d);
1709         vd = vfp_advance_dreg(vm, delta_m);
1710         vfp_load_reg64(f0, vm);
1711     }
1712     return true;
1713 }
1714 
1715 static void gen_VMLA_hp(TCGv_i32 vd, TCGv_i32 vn, TCGv_i32 vm, TCGv_ptr fpst)
1716 {
1717     /* Note that order of inputs to the add matters for NaNs */
1718     TCGv_i32 tmp = tcg_temp_new_i32();
1719 
1720     gen_helper_vfp_mulh(tmp, vn, vm, fpst);
1721     gen_helper_vfp_addh(vd, vd, tmp, fpst);
1722 }
1723 
1724 static bool trans_VMLA_hp(DisasContext *s, arg_VMLA_sp *a)
1725 {
1726     return do_vfp_3op_hp(s, gen_VMLA_hp, a->vd, a->vn, a->vm, true);
1727 }
1728 
1729 static void gen_VMLA_sp(TCGv_i32 vd, TCGv_i32 vn, TCGv_i32 vm, TCGv_ptr fpst)
1730 {
1731     /* Note that order of inputs to the add matters for NaNs */
1732     TCGv_i32 tmp = tcg_temp_new_i32();
1733 
1734     gen_helper_vfp_muls(tmp, vn, vm, fpst);
1735     gen_helper_vfp_adds(vd, vd, tmp, fpst);
1736 }
1737 
1738 static bool trans_VMLA_sp(DisasContext *s, arg_VMLA_sp *a)
1739 {
1740     return do_vfp_3op_sp(s, gen_VMLA_sp, a->vd, a->vn, a->vm, true);
1741 }
1742 
1743 static void gen_VMLA_dp(TCGv_i64 vd, TCGv_i64 vn, TCGv_i64 vm, TCGv_ptr fpst)
1744 {
1745     /* Note that order of inputs to the add matters for NaNs */
1746     TCGv_i64 tmp = tcg_temp_new_i64();
1747 
1748     gen_helper_vfp_muld(tmp, vn, vm, fpst);
1749     gen_helper_vfp_addd(vd, vd, tmp, fpst);
1750 }
1751 
1752 static bool trans_VMLA_dp(DisasContext *s, arg_VMLA_dp *a)
1753 {
1754     return do_vfp_3op_dp(s, gen_VMLA_dp, a->vd, a->vn, a->vm, true);
1755 }
1756 
1757 static void gen_VMLS_hp(TCGv_i32 vd, TCGv_i32 vn, TCGv_i32 vm, TCGv_ptr fpst)
1758 {
1759     /*
1760      * VMLS: vd = vd + -(vn * vm)
1761      * Note that order of inputs to the add matters for NaNs.
1762      */
1763     TCGv_i32 tmp = tcg_temp_new_i32();
1764 
1765     gen_helper_vfp_mulh(tmp, vn, vm, fpst);
1766     gen_helper_vfp_negh(tmp, tmp);
1767     gen_helper_vfp_addh(vd, vd, tmp, fpst);
1768 }
1769 
1770 static bool trans_VMLS_hp(DisasContext *s, arg_VMLS_sp *a)
1771 {
1772     return do_vfp_3op_hp(s, gen_VMLS_hp, a->vd, a->vn, a->vm, true);
1773 }
1774 
1775 static void gen_VMLS_sp(TCGv_i32 vd, TCGv_i32 vn, TCGv_i32 vm, TCGv_ptr fpst)
1776 {
1777     /*
1778      * VMLS: vd = vd + -(vn * vm)
1779      * Note that order of inputs to the add matters for NaNs.
1780      */
1781     TCGv_i32 tmp = tcg_temp_new_i32();
1782 
1783     gen_helper_vfp_muls(tmp, vn, vm, fpst);
1784     gen_helper_vfp_negs(tmp, tmp);
1785     gen_helper_vfp_adds(vd, vd, tmp, fpst);
1786 }
1787 
1788 static bool trans_VMLS_sp(DisasContext *s, arg_VMLS_sp *a)
1789 {
1790     return do_vfp_3op_sp(s, gen_VMLS_sp, a->vd, a->vn, a->vm, true);
1791 }
1792 
1793 static void gen_VMLS_dp(TCGv_i64 vd, TCGv_i64 vn, TCGv_i64 vm, TCGv_ptr fpst)
1794 {
1795     /*
1796      * VMLS: vd = vd + -(vn * vm)
1797      * Note that order of inputs to the add matters for NaNs.
1798      */
1799     TCGv_i64 tmp = tcg_temp_new_i64();
1800 
1801     gen_helper_vfp_muld(tmp, vn, vm, fpst);
1802     gen_helper_vfp_negd(tmp, tmp);
1803     gen_helper_vfp_addd(vd, vd, tmp, fpst);
1804 }
1805 
1806 static bool trans_VMLS_dp(DisasContext *s, arg_VMLS_dp *a)
1807 {
1808     return do_vfp_3op_dp(s, gen_VMLS_dp, a->vd, a->vn, a->vm, true);
1809 }
1810 
1811 static void gen_VNMLS_hp(TCGv_i32 vd, TCGv_i32 vn, TCGv_i32 vm, TCGv_ptr fpst)
1812 {
1813     /*
1814      * VNMLS: -fd + (fn * fm)
1815      * Note that it isn't valid to replace (-A + B) with (B - A) or similar
1816      * plausible looking simplifications because this will give wrong results
1817      * for NaNs.
1818      */
1819     TCGv_i32 tmp = tcg_temp_new_i32();
1820 
1821     gen_helper_vfp_mulh(tmp, vn, vm, fpst);
1822     gen_helper_vfp_negh(vd, vd);
1823     gen_helper_vfp_addh(vd, vd, tmp, fpst);
1824 }
1825 
1826 static bool trans_VNMLS_hp(DisasContext *s, arg_VNMLS_sp *a)
1827 {
1828     return do_vfp_3op_hp(s, gen_VNMLS_hp, a->vd, a->vn, a->vm, true);
1829 }
1830 
1831 static void gen_VNMLS_sp(TCGv_i32 vd, TCGv_i32 vn, TCGv_i32 vm, TCGv_ptr fpst)
1832 {
1833     /*
1834      * VNMLS: -fd + (fn * fm)
1835      * Note that it isn't valid to replace (-A + B) with (B - A) or similar
1836      * plausible looking simplifications because this will give wrong results
1837      * for NaNs.
1838      */
1839     TCGv_i32 tmp = tcg_temp_new_i32();
1840 
1841     gen_helper_vfp_muls(tmp, vn, vm, fpst);
1842     gen_helper_vfp_negs(vd, vd);
1843     gen_helper_vfp_adds(vd, vd, tmp, fpst);
1844 }
1845 
1846 static bool trans_VNMLS_sp(DisasContext *s, arg_VNMLS_sp *a)
1847 {
1848     return do_vfp_3op_sp(s, gen_VNMLS_sp, a->vd, a->vn, a->vm, true);
1849 }
1850 
1851 static void gen_VNMLS_dp(TCGv_i64 vd, TCGv_i64 vn, TCGv_i64 vm, TCGv_ptr fpst)
1852 {
1853     /*
1854      * VNMLS: -fd + (fn * fm)
1855      * Note that it isn't valid to replace (-A + B) with (B - A) or similar
1856      * plausible looking simplifications because this will give wrong results
1857      * for NaNs.
1858      */
1859     TCGv_i64 tmp = tcg_temp_new_i64();
1860 
1861     gen_helper_vfp_muld(tmp, vn, vm, fpst);
1862     gen_helper_vfp_negd(vd, vd);
1863     gen_helper_vfp_addd(vd, vd, tmp, fpst);
1864 }
1865 
1866 static bool trans_VNMLS_dp(DisasContext *s, arg_VNMLS_dp *a)
1867 {
1868     return do_vfp_3op_dp(s, gen_VNMLS_dp, a->vd, a->vn, a->vm, true);
1869 }
1870 
1871 static void gen_VNMLA_hp(TCGv_i32 vd, TCGv_i32 vn, TCGv_i32 vm, TCGv_ptr fpst)
1872 {
1873     /* VNMLA: -fd + -(fn * fm) */
1874     TCGv_i32 tmp = tcg_temp_new_i32();
1875 
1876     gen_helper_vfp_mulh(tmp, vn, vm, fpst);
1877     gen_helper_vfp_negh(tmp, tmp);
1878     gen_helper_vfp_negh(vd, vd);
1879     gen_helper_vfp_addh(vd, vd, tmp, fpst);
1880 }
1881 
1882 static bool trans_VNMLA_hp(DisasContext *s, arg_VNMLA_sp *a)
1883 {
1884     return do_vfp_3op_hp(s, gen_VNMLA_hp, a->vd, a->vn, a->vm, true);
1885 }
1886 
1887 static void gen_VNMLA_sp(TCGv_i32 vd, TCGv_i32 vn, TCGv_i32 vm, TCGv_ptr fpst)
1888 {
1889     /* VNMLA: -fd + -(fn * fm) */
1890     TCGv_i32 tmp = tcg_temp_new_i32();
1891 
1892     gen_helper_vfp_muls(tmp, vn, vm, fpst);
1893     gen_helper_vfp_negs(tmp, tmp);
1894     gen_helper_vfp_negs(vd, vd);
1895     gen_helper_vfp_adds(vd, vd, tmp, fpst);
1896 }
1897 
1898 static bool trans_VNMLA_sp(DisasContext *s, arg_VNMLA_sp *a)
1899 {
1900     return do_vfp_3op_sp(s, gen_VNMLA_sp, a->vd, a->vn, a->vm, true);
1901 }
1902 
1903 static void gen_VNMLA_dp(TCGv_i64 vd, TCGv_i64 vn, TCGv_i64 vm, TCGv_ptr fpst)
1904 {
1905     /* VNMLA: -fd + (fn * fm) */
1906     TCGv_i64 tmp = tcg_temp_new_i64();
1907 
1908     gen_helper_vfp_muld(tmp, vn, vm, fpst);
1909     gen_helper_vfp_negd(tmp, tmp);
1910     gen_helper_vfp_negd(vd, vd);
1911     gen_helper_vfp_addd(vd, vd, tmp, fpst);
1912 }
1913 
1914 static bool trans_VNMLA_dp(DisasContext *s, arg_VNMLA_dp *a)
1915 {
1916     return do_vfp_3op_dp(s, gen_VNMLA_dp, a->vd, a->vn, a->vm, true);
1917 }
1918 
1919 static bool trans_VMUL_hp(DisasContext *s, arg_VMUL_sp *a)
1920 {
1921     return do_vfp_3op_hp(s, gen_helper_vfp_mulh, a->vd, a->vn, a->vm, false);
1922 }
1923 
1924 static bool trans_VMUL_sp(DisasContext *s, arg_VMUL_sp *a)
1925 {
1926     return do_vfp_3op_sp(s, gen_helper_vfp_muls, a->vd, a->vn, a->vm, false);
1927 }
1928 
1929 static bool trans_VMUL_dp(DisasContext *s, arg_VMUL_dp *a)
1930 {
1931     return do_vfp_3op_dp(s, gen_helper_vfp_muld, a->vd, a->vn, a->vm, false);
1932 }
1933 
1934 static void gen_VNMUL_hp(TCGv_i32 vd, TCGv_i32 vn, TCGv_i32 vm, TCGv_ptr fpst)
1935 {
1936     /* VNMUL: -(fn * fm) */
1937     gen_helper_vfp_mulh(vd, vn, vm, fpst);
1938     gen_helper_vfp_negh(vd, vd);
1939 }
1940 
1941 static bool trans_VNMUL_hp(DisasContext *s, arg_VNMUL_sp *a)
1942 {
1943     return do_vfp_3op_hp(s, gen_VNMUL_hp, a->vd, a->vn, a->vm, false);
1944 }
1945 
1946 static void gen_VNMUL_sp(TCGv_i32 vd, TCGv_i32 vn, TCGv_i32 vm, TCGv_ptr fpst)
1947 {
1948     /* VNMUL: -(fn * fm) */
1949     gen_helper_vfp_muls(vd, vn, vm, fpst);
1950     gen_helper_vfp_negs(vd, vd);
1951 }
1952 
1953 static bool trans_VNMUL_sp(DisasContext *s, arg_VNMUL_sp *a)
1954 {
1955     return do_vfp_3op_sp(s, gen_VNMUL_sp, a->vd, a->vn, a->vm, false);
1956 }
1957 
1958 static void gen_VNMUL_dp(TCGv_i64 vd, TCGv_i64 vn, TCGv_i64 vm, TCGv_ptr fpst)
1959 {
1960     /* VNMUL: -(fn * fm) */
1961     gen_helper_vfp_muld(vd, vn, vm, fpst);
1962     gen_helper_vfp_negd(vd, vd);
1963 }
1964 
1965 static bool trans_VNMUL_dp(DisasContext *s, arg_VNMUL_dp *a)
1966 {
1967     return do_vfp_3op_dp(s, gen_VNMUL_dp, a->vd, a->vn, a->vm, false);
1968 }
1969 
1970 static bool trans_VADD_hp(DisasContext *s, arg_VADD_sp *a)
1971 {
1972     return do_vfp_3op_hp(s, gen_helper_vfp_addh, a->vd, a->vn, a->vm, false);
1973 }
1974 
1975 static bool trans_VADD_sp(DisasContext *s, arg_VADD_sp *a)
1976 {
1977     return do_vfp_3op_sp(s, gen_helper_vfp_adds, a->vd, a->vn, a->vm, false);
1978 }
1979 
1980 static bool trans_VADD_dp(DisasContext *s, arg_VADD_dp *a)
1981 {
1982     return do_vfp_3op_dp(s, gen_helper_vfp_addd, a->vd, a->vn, a->vm, false);
1983 }
1984 
1985 static bool trans_VSUB_hp(DisasContext *s, arg_VSUB_sp *a)
1986 {
1987     return do_vfp_3op_hp(s, gen_helper_vfp_subh, a->vd, a->vn, a->vm, false);
1988 }
1989 
1990 static bool trans_VSUB_sp(DisasContext *s, arg_VSUB_sp *a)
1991 {
1992     return do_vfp_3op_sp(s, gen_helper_vfp_subs, a->vd, a->vn, a->vm, false);
1993 }
1994 
1995 static bool trans_VSUB_dp(DisasContext *s, arg_VSUB_dp *a)
1996 {
1997     return do_vfp_3op_dp(s, gen_helper_vfp_subd, a->vd, a->vn, a->vm, false);
1998 }
1999 
2000 static bool trans_VDIV_hp(DisasContext *s, arg_VDIV_sp *a)
2001 {
2002     return do_vfp_3op_hp(s, gen_helper_vfp_divh, a->vd, a->vn, a->vm, false);
2003 }
2004 
2005 static bool trans_VDIV_sp(DisasContext *s, arg_VDIV_sp *a)
2006 {
2007     return do_vfp_3op_sp(s, gen_helper_vfp_divs, a->vd, a->vn, a->vm, false);
2008 }
2009 
2010 static bool trans_VDIV_dp(DisasContext *s, arg_VDIV_dp *a)
2011 {
2012     return do_vfp_3op_dp(s, gen_helper_vfp_divd, a->vd, a->vn, a->vm, false);
2013 }
2014 
2015 static bool trans_VMINNM_hp(DisasContext *s, arg_VMINNM_sp *a)
2016 {
2017     if (!dc_isar_feature(aa32_vminmaxnm, s)) {
2018         return false;
2019     }
2020     return do_vfp_3op_hp(s, gen_helper_vfp_minnumh,
2021                          a->vd, a->vn, a->vm, false);
2022 }
2023 
2024 static bool trans_VMAXNM_hp(DisasContext *s, arg_VMAXNM_sp *a)
2025 {
2026     if (!dc_isar_feature(aa32_vminmaxnm, s)) {
2027         return false;
2028     }
2029     return do_vfp_3op_hp(s, gen_helper_vfp_maxnumh,
2030                          a->vd, a->vn, a->vm, false);
2031 }
2032 
2033 static bool trans_VMINNM_sp(DisasContext *s, arg_VMINNM_sp *a)
2034 {
2035     if (!dc_isar_feature(aa32_vminmaxnm, s)) {
2036         return false;
2037     }
2038     return do_vfp_3op_sp(s, gen_helper_vfp_minnums,
2039                          a->vd, a->vn, a->vm, false);
2040 }
2041 
2042 static bool trans_VMAXNM_sp(DisasContext *s, arg_VMAXNM_sp *a)
2043 {
2044     if (!dc_isar_feature(aa32_vminmaxnm, s)) {
2045         return false;
2046     }
2047     return do_vfp_3op_sp(s, gen_helper_vfp_maxnums,
2048                          a->vd, a->vn, a->vm, false);
2049 }
2050 
2051 static bool trans_VMINNM_dp(DisasContext *s, arg_VMINNM_dp *a)
2052 {
2053     if (!dc_isar_feature(aa32_vminmaxnm, s)) {
2054         return false;
2055     }
2056     return do_vfp_3op_dp(s, gen_helper_vfp_minnumd,
2057                          a->vd, a->vn, a->vm, false);
2058 }
2059 
2060 static bool trans_VMAXNM_dp(DisasContext *s, arg_VMAXNM_dp *a)
2061 {
2062     if (!dc_isar_feature(aa32_vminmaxnm, s)) {
2063         return false;
2064     }
2065     return do_vfp_3op_dp(s, gen_helper_vfp_maxnumd,
2066                          a->vd, a->vn, a->vm, false);
2067 }
2068 
2069 static bool do_vfm_hp(DisasContext *s, arg_VFMA_sp *a, bool neg_n, bool neg_d)
2070 {
2071     /*
2072      * VFNMA : fd = muladd(-fd,  fn, fm)
2073      * VFNMS : fd = muladd(-fd, -fn, fm)
2074      * VFMA  : fd = muladd( fd,  fn, fm)
2075      * VFMS  : fd = muladd( fd, -fn, fm)
2076      *
2077      * These are fused multiply-add, and must be done as one floating
2078      * point operation with no rounding between the multiplication and
2079      * addition steps.  NB that doing the negations here as separate
2080      * steps is correct : an input NaN should come out with its sign
2081      * bit flipped if it is a negated-input.
2082      */
2083     TCGv_ptr fpst;
2084     TCGv_i32 vn, vm, vd;
2085 
2086     /*
2087      * Present in VFPv4 only, and only with the FP16 extension.
2088      * Note that we can't rely on the SIMDFMAC check alone, because
2089      * in a Neon-no-VFP core that ID register field will be non-zero.
2090      */
2091     if (!dc_isar_feature(aa32_fp16_arith, s) ||
2092         !dc_isar_feature(aa32_simdfmac, s) ||
2093         !dc_isar_feature(aa32_fpsp_v2, s)) {
2094         return false;
2095     }
2096 
2097     if (s->vec_len != 0 || s->vec_stride != 0) {
2098         return false;
2099     }
2100 
2101     if (!vfp_access_check(s)) {
2102         return true;
2103     }
2104 
2105     vn = tcg_temp_new_i32();
2106     vm = tcg_temp_new_i32();
2107     vd = tcg_temp_new_i32();
2108 
2109     vfp_load_reg32(vn, a->vn);
2110     vfp_load_reg32(vm, a->vm);
2111     if (neg_n) {
2112         /* VFNMS, VFMS */
2113         gen_helper_vfp_negh(vn, vn);
2114     }
2115     vfp_load_reg32(vd, a->vd);
2116     if (neg_d) {
2117         /* VFNMA, VFNMS */
2118         gen_helper_vfp_negh(vd, vd);
2119     }
2120     fpst = fpstatus_ptr(FPST_FPCR_F16);
2121     gen_helper_vfp_muladdh(vd, vn, vm, vd, fpst);
2122     vfp_store_reg32(vd, a->vd);
2123     return true;
2124 }
2125 
2126 static bool do_vfm_sp(DisasContext *s, arg_VFMA_sp *a, bool neg_n, bool neg_d)
2127 {
2128     /*
2129      * VFNMA : fd = muladd(-fd,  fn, fm)
2130      * VFNMS : fd = muladd(-fd, -fn, fm)
2131      * VFMA  : fd = muladd( fd,  fn, fm)
2132      * VFMS  : fd = muladd( fd, -fn, fm)
2133      *
2134      * These are fused multiply-add, and must be done as one floating
2135      * point operation with no rounding between the multiplication and
2136      * addition steps.  NB that doing the negations here as separate
2137      * steps is correct : an input NaN should come out with its sign
2138      * bit flipped if it is a negated-input.
2139      */
2140     TCGv_ptr fpst;
2141     TCGv_i32 vn, vm, vd;
2142 
2143     /*
2144      * Present in VFPv4 only.
2145      * Note that we can't rely on the SIMDFMAC check alone, because
2146      * in a Neon-no-VFP core that ID register field will be non-zero.
2147      */
2148     if (!dc_isar_feature(aa32_simdfmac, s) ||
2149         !dc_isar_feature(aa32_fpsp_v2, s)) {
2150         return false;
2151     }
2152     /*
2153      * In v7A, UNPREDICTABLE with non-zero vector length/stride; from
2154      * v8A, must UNDEF. We choose to UNDEF for both v7A and v8A.
2155      */
2156     if (s->vec_len != 0 || s->vec_stride != 0) {
2157         return false;
2158     }
2159 
2160     if (!vfp_access_check(s)) {
2161         return true;
2162     }
2163 
2164     vn = tcg_temp_new_i32();
2165     vm = tcg_temp_new_i32();
2166     vd = tcg_temp_new_i32();
2167 
2168     vfp_load_reg32(vn, a->vn);
2169     vfp_load_reg32(vm, a->vm);
2170     if (neg_n) {
2171         /* VFNMS, VFMS */
2172         gen_helper_vfp_negs(vn, vn);
2173     }
2174     vfp_load_reg32(vd, a->vd);
2175     if (neg_d) {
2176         /* VFNMA, VFNMS */
2177         gen_helper_vfp_negs(vd, vd);
2178     }
2179     fpst = fpstatus_ptr(FPST_FPCR);
2180     gen_helper_vfp_muladds(vd, vn, vm, vd, fpst);
2181     vfp_store_reg32(vd, a->vd);
2182     return true;
2183 }
2184 
2185 static bool do_vfm_dp(DisasContext *s, arg_VFMA_dp *a, bool neg_n, bool neg_d)
2186 {
2187     /*
2188      * VFNMA : fd = muladd(-fd,  fn, fm)
2189      * VFNMS : fd = muladd(-fd, -fn, fm)
2190      * VFMA  : fd = muladd( fd,  fn, fm)
2191      * VFMS  : fd = muladd( fd, -fn, fm)
2192      *
2193      * These are fused multiply-add, and must be done as one floating
2194      * point operation with no rounding between the multiplication and
2195      * addition steps.  NB that doing the negations here as separate
2196      * steps is correct : an input NaN should come out with its sign
2197      * bit flipped if it is a negated-input.
2198      */
2199     TCGv_ptr fpst;
2200     TCGv_i64 vn, vm, vd;
2201 
2202     /*
2203      * Present in VFPv4 only.
2204      * Note that we can't rely on the SIMDFMAC check alone, because
2205      * in a Neon-no-VFP core that ID register field will be non-zero.
2206      */
2207     if (!dc_isar_feature(aa32_simdfmac, s) ||
2208         !dc_isar_feature(aa32_fpdp_v2, s)) {
2209         return false;
2210     }
2211     /*
2212      * In v7A, UNPREDICTABLE with non-zero vector length/stride; from
2213      * v8A, must UNDEF. We choose to UNDEF for both v7A and v8A.
2214      */
2215     if (s->vec_len != 0 || s->vec_stride != 0) {
2216         return false;
2217     }
2218 
2219     /* UNDEF accesses to D16-D31 if they don't exist. */
2220     if (!dc_isar_feature(aa32_simd_r32, s) &&
2221         ((a->vd | a->vn | a->vm) & 0x10)) {
2222         return false;
2223     }
2224 
2225     if (!vfp_access_check(s)) {
2226         return true;
2227     }
2228 
2229     vn = tcg_temp_new_i64();
2230     vm = tcg_temp_new_i64();
2231     vd = tcg_temp_new_i64();
2232 
2233     vfp_load_reg64(vn, a->vn);
2234     vfp_load_reg64(vm, a->vm);
2235     if (neg_n) {
2236         /* VFNMS, VFMS */
2237         gen_helper_vfp_negd(vn, vn);
2238     }
2239     vfp_load_reg64(vd, a->vd);
2240     if (neg_d) {
2241         /* VFNMA, VFNMS */
2242         gen_helper_vfp_negd(vd, vd);
2243     }
2244     fpst = fpstatus_ptr(FPST_FPCR);
2245     gen_helper_vfp_muladdd(vd, vn, vm, vd, fpst);
2246     vfp_store_reg64(vd, a->vd);
2247     return true;
2248 }
2249 
2250 #define MAKE_ONE_VFM_TRANS_FN(INSN, PREC, NEGN, NEGD)                   \
2251     static bool trans_##INSN##_##PREC(DisasContext *s,                  \
2252                                       arg_##INSN##_##PREC *a)           \
2253     {                                                                   \
2254         return do_vfm_##PREC(s, a, NEGN, NEGD);                         \
2255     }
2256 
2257 #define MAKE_VFM_TRANS_FNS(PREC) \
2258     MAKE_ONE_VFM_TRANS_FN(VFMA, PREC, false, false) \
2259     MAKE_ONE_VFM_TRANS_FN(VFMS, PREC, true, false) \
2260     MAKE_ONE_VFM_TRANS_FN(VFNMA, PREC, false, true) \
2261     MAKE_ONE_VFM_TRANS_FN(VFNMS, PREC, true, true)
2262 
2263 MAKE_VFM_TRANS_FNS(hp)
2264 MAKE_VFM_TRANS_FNS(sp)
2265 MAKE_VFM_TRANS_FNS(dp)
2266 
2267 static bool trans_VMOV_imm_hp(DisasContext *s, arg_VMOV_imm_sp *a)
2268 {
2269     if (!dc_isar_feature(aa32_fp16_arith, s)) {
2270         return false;
2271     }
2272 
2273     if (s->vec_len != 0 || s->vec_stride != 0) {
2274         return false;
2275     }
2276 
2277     if (!vfp_access_check(s)) {
2278         return true;
2279     }
2280 
2281     vfp_store_reg32(tcg_constant_i32(vfp_expand_imm(MO_16, a->imm)), a->vd);
2282     return true;
2283 }
2284 
2285 static bool trans_VMOV_imm_sp(DisasContext *s, arg_VMOV_imm_sp *a)
2286 {
2287     uint32_t delta_d = 0;
2288     int veclen = s->vec_len;
2289     TCGv_i32 fd;
2290     uint32_t vd;
2291 
2292     vd = a->vd;
2293 
2294     if (!dc_isar_feature(aa32_fpsp_v3, s)) {
2295         return false;
2296     }
2297 
2298     if (!dc_isar_feature(aa32_fpshvec, s) &&
2299         (veclen != 0 || s->vec_stride != 0)) {
2300         return false;
2301     }
2302 
2303     if (!vfp_access_check(s)) {
2304         return true;
2305     }
2306 
2307     if (veclen > 0) {
2308         /* Figure out what type of vector operation this is.  */
2309         if (vfp_sreg_is_scalar(vd)) {
2310             /* scalar */
2311             veclen = 0;
2312         } else {
2313             delta_d = s->vec_stride + 1;
2314         }
2315     }
2316 
2317     fd = tcg_constant_i32(vfp_expand_imm(MO_32, a->imm));
2318 
2319     for (;;) {
2320         vfp_store_reg32(fd, vd);
2321 
2322         if (veclen == 0) {
2323             break;
2324         }
2325 
2326         /* Set up the operands for the next iteration */
2327         veclen--;
2328         vd = vfp_advance_sreg(vd, delta_d);
2329     }
2330 
2331     return true;
2332 }
2333 
2334 static bool trans_VMOV_imm_dp(DisasContext *s, arg_VMOV_imm_dp *a)
2335 {
2336     uint32_t delta_d = 0;
2337     int veclen = s->vec_len;
2338     TCGv_i64 fd;
2339     uint32_t vd;
2340 
2341     vd = a->vd;
2342 
2343     if (!dc_isar_feature(aa32_fpdp_v3, s)) {
2344         return false;
2345     }
2346 
2347     /* UNDEF accesses to D16-D31 if they don't exist. */
2348     if (!dc_isar_feature(aa32_simd_r32, s) && (vd & 0x10)) {
2349         return false;
2350     }
2351 
2352     if (!dc_isar_feature(aa32_fpshvec, s) &&
2353         (veclen != 0 || s->vec_stride != 0)) {
2354         return false;
2355     }
2356 
2357     if (!vfp_access_check(s)) {
2358         return true;
2359     }
2360 
2361     if (veclen > 0) {
2362         /* Figure out what type of vector operation this is.  */
2363         if (vfp_dreg_is_scalar(vd)) {
2364             /* scalar */
2365             veclen = 0;
2366         } else {
2367             delta_d = (s->vec_stride >> 1) + 1;
2368         }
2369     }
2370 
2371     fd = tcg_constant_i64(vfp_expand_imm(MO_64, a->imm));
2372 
2373     for (;;) {
2374         vfp_store_reg64(fd, vd);
2375 
2376         if (veclen == 0) {
2377             break;
2378         }
2379 
2380         /* Set up the operands for the next iteration */
2381         veclen--;
2382         vd = vfp_advance_dreg(vd, delta_d);
2383     }
2384 
2385     return true;
2386 }
2387 
2388 #define DO_VFP_2OP(INSN, PREC, FN, CHECK)                       \
2389     static bool trans_##INSN##_##PREC(DisasContext *s,          \
2390                                       arg_##INSN##_##PREC *a)   \
2391     {                                                           \
2392         if (!dc_isar_feature(CHECK, s)) {                       \
2393             return false;                                       \
2394         }                                                       \
2395         return do_vfp_2op_##PREC(s, FN, a->vd, a->vm);          \
2396     }
2397 
2398 #define DO_VFP_VMOV(INSN, PREC, FN)                             \
2399     static bool trans_##INSN##_##PREC(DisasContext *s,          \
2400                                       arg_##INSN##_##PREC *a)   \
2401     {                                                           \
2402         if (!dc_isar_feature(aa32_fp##PREC##_v2, s) &&          \
2403             !dc_isar_feature(aa32_mve, s)) {                    \
2404             return false;                                       \
2405         }                                                       \
2406         return do_vfp_2op_##PREC(s, FN, a->vd, a->vm);          \
2407     }
2408 
2409 DO_VFP_VMOV(VMOV_reg, sp, tcg_gen_mov_i32)
2410 DO_VFP_VMOV(VMOV_reg, dp, tcg_gen_mov_i64)
2411 
2412 DO_VFP_2OP(VABS, hp, gen_helper_vfp_absh, aa32_fp16_arith)
2413 DO_VFP_2OP(VABS, sp, gen_helper_vfp_abss, aa32_fpsp_v2)
2414 DO_VFP_2OP(VABS, dp, gen_helper_vfp_absd, aa32_fpdp_v2)
2415 
2416 DO_VFP_2OP(VNEG, hp, gen_helper_vfp_negh, aa32_fp16_arith)
2417 DO_VFP_2OP(VNEG, sp, gen_helper_vfp_negs, aa32_fpsp_v2)
2418 DO_VFP_2OP(VNEG, dp, gen_helper_vfp_negd, aa32_fpdp_v2)
2419 
2420 static void gen_VSQRT_hp(TCGv_i32 vd, TCGv_i32 vm)
2421 {
2422     gen_helper_vfp_sqrth(vd, vm, tcg_env);
2423 }
2424 
2425 static void gen_VSQRT_sp(TCGv_i32 vd, TCGv_i32 vm)
2426 {
2427     gen_helper_vfp_sqrts(vd, vm, tcg_env);
2428 }
2429 
2430 static void gen_VSQRT_dp(TCGv_i64 vd, TCGv_i64 vm)
2431 {
2432     gen_helper_vfp_sqrtd(vd, vm, tcg_env);
2433 }
2434 
2435 DO_VFP_2OP(VSQRT, hp, gen_VSQRT_hp, aa32_fp16_arith)
2436 DO_VFP_2OP(VSQRT, sp, gen_VSQRT_sp, aa32_fpsp_v2)
2437 DO_VFP_2OP(VSQRT, dp, gen_VSQRT_dp, aa32_fpdp_v2)
2438 
2439 static bool trans_VCMP_hp(DisasContext *s, arg_VCMP_sp *a)
2440 {
2441     TCGv_i32 vd, vm;
2442 
2443     if (!dc_isar_feature(aa32_fp16_arith, s)) {
2444         return false;
2445     }
2446 
2447     /* Vm/M bits must be zero for the Z variant */
2448     if (a->z && a->vm != 0) {
2449         return false;
2450     }
2451 
2452     if (!vfp_access_check(s)) {
2453         return true;
2454     }
2455 
2456     vd = tcg_temp_new_i32();
2457     vm = tcg_temp_new_i32();
2458 
2459     vfp_load_reg32(vd, a->vd);
2460     if (a->z) {
2461         tcg_gen_movi_i32(vm, 0);
2462     } else {
2463         vfp_load_reg32(vm, a->vm);
2464     }
2465 
2466     if (a->e) {
2467         gen_helper_vfp_cmpeh(vd, vm, tcg_env);
2468     } else {
2469         gen_helper_vfp_cmph(vd, vm, tcg_env);
2470     }
2471     return true;
2472 }
2473 
2474 static bool trans_VCMP_sp(DisasContext *s, arg_VCMP_sp *a)
2475 {
2476     TCGv_i32 vd, vm;
2477 
2478     if (!dc_isar_feature(aa32_fpsp_v2, s)) {
2479         return false;
2480     }
2481 
2482     /* Vm/M bits must be zero for the Z variant */
2483     if (a->z && a->vm != 0) {
2484         return false;
2485     }
2486 
2487     if (!vfp_access_check(s)) {
2488         return true;
2489     }
2490 
2491     vd = tcg_temp_new_i32();
2492     vm = tcg_temp_new_i32();
2493 
2494     vfp_load_reg32(vd, a->vd);
2495     if (a->z) {
2496         tcg_gen_movi_i32(vm, 0);
2497     } else {
2498         vfp_load_reg32(vm, a->vm);
2499     }
2500 
2501     if (a->e) {
2502         gen_helper_vfp_cmpes(vd, vm, tcg_env);
2503     } else {
2504         gen_helper_vfp_cmps(vd, vm, tcg_env);
2505     }
2506     return true;
2507 }
2508 
2509 static bool trans_VCMP_dp(DisasContext *s, arg_VCMP_dp *a)
2510 {
2511     TCGv_i64 vd, vm;
2512 
2513     if (!dc_isar_feature(aa32_fpdp_v2, s)) {
2514         return false;
2515     }
2516 
2517     /* Vm/M bits must be zero for the Z variant */
2518     if (a->z && a->vm != 0) {
2519         return false;
2520     }
2521 
2522     /* UNDEF accesses to D16-D31 if they don't exist. */
2523     if (!dc_isar_feature(aa32_simd_r32, s) && ((a->vd | a->vm) & 0x10)) {
2524         return false;
2525     }
2526 
2527     if (!vfp_access_check(s)) {
2528         return true;
2529     }
2530 
2531     vd = tcg_temp_new_i64();
2532     vm = tcg_temp_new_i64();
2533 
2534     vfp_load_reg64(vd, a->vd);
2535     if (a->z) {
2536         tcg_gen_movi_i64(vm, 0);
2537     } else {
2538         vfp_load_reg64(vm, a->vm);
2539     }
2540 
2541     if (a->e) {
2542         gen_helper_vfp_cmped(vd, vm, tcg_env);
2543     } else {
2544         gen_helper_vfp_cmpd(vd, vm, tcg_env);
2545     }
2546     return true;
2547 }
2548 
2549 static bool trans_VCVT_f32_f16(DisasContext *s, arg_VCVT_f32_f16 *a)
2550 {
2551     TCGv_ptr fpst;
2552     TCGv_i32 ahp_mode;
2553     TCGv_i32 tmp;
2554 
2555     if (!dc_isar_feature(aa32_fp16_spconv, s)) {
2556         return false;
2557     }
2558 
2559     if (!vfp_access_check(s)) {
2560         return true;
2561     }
2562 
2563     fpst = fpstatus_ptr(FPST_FPCR);
2564     ahp_mode = get_ahp_flag();
2565     tmp = tcg_temp_new_i32();
2566     /* The T bit tells us if we want the low or high 16 bits of Vm */
2567     tcg_gen_ld16u_i32(tmp, tcg_env, vfp_f16_offset(a->vm, a->t));
2568     gen_helper_vfp_fcvt_f16_to_f32(tmp, tmp, fpst, ahp_mode);
2569     vfp_store_reg32(tmp, a->vd);
2570     return true;
2571 }
2572 
2573 static bool trans_VCVT_f64_f16(DisasContext *s, arg_VCVT_f64_f16 *a)
2574 {
2575     TCGv_ptr fpst;
2576     TCGv_i32 ahp_mode;
2577     TCGv_i32 tmp;
2578     TCGv_i64 vd;
2579 
2580     if (!dc_isar_feature(aa32_fpdp_v2, s)) {
2581         return false;
2582     }
2583 
2584     if (!dc_isar_feature(aa32_fp16_dpconv, s)) {
2585         return false;
2586     }
2587 
2588     /* UNDEF accesses to D16-D31 if they don't exist. */
2589     if (!dc_isar_feature(aa32_simd_r32, s) && (a->vd  & 0x10)) {
2590         return false;
2591     }
2592 
2593     if (!vfp_access_check(s)) {
2594         return true;
2595     }
2596 
2597     fpst = fpstatus_ptr(FPST_FPCR);
2598     ahp_mode = get_ahp_flag();
2599     tmp = tcg_temp_new_i32();
2600     /* The T bit tells us if we want the low or high 16 bits of Vm */
2601     tcg_gen_ld16u_i32(tmp, tcg_env, vfp_f16_offset(a->vm, a->t));
2602     vd = tcg_temp_new_i64();
2603     gen_helper_vfp_fcvt_f16_to_f64(vd, tmp, fpst, ahp_mode);
2604     vfp_store_reg64(vd, a->vd);
2605     return true;
2606 }
2607 
2608 static bool trans_VCVT_b16_f32(DisasContext *s, arg_VCVT_b16_f32 *a)
2609 {
2610     TCGv_ptr fpst;
2611     TCGv_i32 tmp;
2612 
2613     if (!dc_isar_feature(aa32_bf16, s)) {
2614         return false;
2615     }
2616 
2617     if (!vfp_access_check(s)) {
2618         return true;
2619     }
2620 
2621     fpst = fpstatus_ptr(FPST_FPCR);
2622     tmp = tcg_temp_new_i32();
2623 
2624     vfp_load_reg32(tmp, a->vm);
2625     gen_helper_bfcvt(tmp, tmp, fpst);
2626     tcg_gen_st16_i32(tmp, tcg_env, vfp_f16_offset(a->vd, a->t));
2627     return true;
2628 }
2629 
2630 static bool trans_VCVT_f16_f32(DisasContext *s, arg_VCVT_f16_f32 *a)
2631 {
2632     TCGv_ptr fpst;
2633     TCGv_i32 ahp_mode;
2634     TCGv_i32 tmp;
2635 
2636     if (!dc_isar_feature(aa32_fp16_spconv, s)) {
2637         return false;
2638     }
2639 
2640     if (!vfp_access_check(s)) {
2641         return true;
2642     }
2643 
2644     fpst = fpstatus_ptr(FPST_FPCR);
2645     ahp_mode = get_ahp_flag();
2646     tmp = tcg_temp_new_i32();
2647 
2648     vfp_load_reg32(tmp, a->vm);
2649     gen_helper_vfp_fcvt_f32_to_f16(tmp, tmp, fpst, ahp_mode);
2650     tcg_gen_st16_i32(tmp, tcg_env, vfp_f16_offset(a->vd, a->t));
2651     return true;
2652 }
2653 
2654 static bool trans_VCVT_f16_f64(DisasContext *s, arg_VCVT_f16_f64 *a)
2655 {
2656     TCGv_ptr fpst;
2657     TCGv_i32 ahp_mode;
2658     TCGv_i32 tmp;
2659     TCGv_i64 vm;
2660 
2661     if (!dc_isar_feature(aa32_fpdp_v2, s)) {
2662         return false;
2663     }
2664 
2665     if (!dc_isar_feature(aa32_fp16_dpconv, s)) {
2666         return false;
2667     }
2668 
2669     /* UNDEF accesses to D16-D31 if they don't exist. */
2670     if (!dc_isar_feature(aa32_simd_r32, s) && (a->vm  & 0x10)) {
2671         return false;
2672     }
2673 
2674     if (!vfp_access_check(s)) {
2675         return true;
2676     }
2677 
2678     fpst = fpstatus_ptr(FPST_FPCR);
2679     ahp_mode = get_ahp_flag();
2680     tmp = tcg_temp_new_i32();
2681     vm = tcg_temp_new_i64();
2682 
2683     vfp_load_reg64(vm, a->vm);
2684     gen_helper_vfp_fcvt_f64_to_f16(tmp, vm, fpst, ahp_mode);
2685     tcg_gen_st16_i32(tmp, tcg_env, vfp_f16_offset(a->vd, a->t));
2686     return true;
2687 }
2688 
2689 static bool trans_VRINTR_hp(DisasContext *s, arg_VRINTR_sp *a)
2690 {
2691     TCGv_ptr fpst;
2692     TCGv_i32 tmp;
2693 
2694     if (!dc_isar_feature(aa32_fp16_arith, s)) {
2695         return false;
2696     }
2697 
2698     if (!vfp_access_check(s)) {
2699         return true;
2700     }
2701 
2702     tmp = tcg_temp_new_i32();
2703     vfp_load_reg32(tmp, a->vm);
2704     fpst = fpstatus_ptr(FPST_FPCR_F16);
2705     gen_helper_rinth(tmp, tmp, fpst);
2706     vfp_store_reg32(tmp, a->vd);
2707     return true;
2708 }
2709 
2710 static bool trans_VRINTR_sp(DisasContext *s, arg_VRINTR_sp *a)
2711 {
2712     TCGv_ptr fpst;
2713     TCGv_i32 tmp;
2714 
2715     if (!dc_isar_feature(aa32_vrint, s)) {
2716         return false;
2717     }
2718 
2719     if (!vfp_access_check(s)) {
2720         return true;
2721     }
2722 
2723     tmp = tcg_temp_new_i32();
2724     vfp_load_reg32(tmp, a->vm);
2725     fpst = fpstatus_ptr(FPST_FPCR);
2726     gen_helper_rints(tmp, tmp, fpst);
2727     vfp_store_reg32(tmp, a->vd);
2728     return true;
2729 }
2730 
2731 static bool trans_VRINTR_dp(DisasContext *s, arg_VRINTR_dp *a)
2732 {
2733     TCGv_ptr fpst;
2734     TCGv_i64 tmp;
2735 
2736     if (!dc_isar_feature(aa32_fpdp_v2, s)) {
2737         return false;
2738     }
2739 
2740     if (!dc_isar_feature(aa32_vrint, s)) {
2741         return false;
2742     }
2743 
2744     /* UNDEF accesses to D16-D31 if they don't exist. */
2745     if (!dc_isar_feature(aa32_simd_r32, s) && ((a->vd | a->vm) & 0x10)) {
2746         return false;
2747     }
2748 
2749     if (!vfp_access_check(s)) {
2750         return true;
2751     }
2752 
2753     tmp = tcg_temp_new_i64();
2754     vfp_load_reg64(tmp, a->vm);
2755     fpst = fpstatus_ptr(FPST_FPCR);
2756     gen_helper_rintd(tmp, tmp, fpst);
2757     vfp_store_reg64(tmp, a->vd);
2758     return true;
2759 }
2760 
2761 static bool trans_VRINTZ_hp(DisasContext *s, arg_VRINTZ_sp *a)
2762 {
2763     TCGv_ptr fpst;
2764     TCGv_i32 tmp;
2765     TCGv_i32 tcg_rmode;
2766 
2767     if (!dc_isar_feature(aa32_fp16_arith, s)) {
2768         return false;
2769     }
2770 
2771     if (!vfp_access_check(s)) {
2772         return true;
2773     }
2774 
2775     tmp = tcg_temp_new_i32();
2776     vfp_load_reg32(tmp, a->vm);
2777     fpst = fpstatus_ptr(FPST_FPCR_F16);
2778     tcg_rmode = gen_set_rmode(FPROUNDING_ZERO, fpst);
2779     gen_helper_rinth(tmp, tmp, fpst);
2780     gen_restore_rmode(tcg_rmode, fpst);
2781     vfp_store_reg32(tmp, a->vd);
2782     return true;
2783 }
2784 
2785 static bool trans_VRINTZ_sp(DisasContext *s, arg_VRINTZ_sp *a)
2786 {
2787     TCGv_ptr fpst;
2788     TCGv_i32 tmp;
2789     TCGv_i32 tcg_rmode;
2790 
2791     if (!dc_isar_feature(aa32_vrint, s)) {
2792         return false;
2793     }
2794 
2795     if (!vfp_access_check(s)) {
2796         return true;
2797     }
2798 
2799     tmp = tcg_temp_new_i32();
2800     vfp_load_reg32(tmp, a->vm);
2801     fpst = fpstatus_ptr(FPST_FPCR);
2802     tcg_rmode = gen_set_rmode(FPROUNDING_ZERO, fpst);
2803     gen_helper_rints(tmp, tmp, fpst);
2804     gen_restore_rmode(tcg_rmode, fpst);
2805     vfp_store_reg32(tmp, a->vd);
2806     return true;
2807 }
2808 
2809 static bool trans_VRINTZ_dp(DisasContext *s, arg_VRINTZ_dp *a)
2810 {
2811     TCGv_ptr fpst;
2812     TCGv_i64 tmp;
2813     TCGv_i32 tcg_rmode;
2814 
2815     if (!dc_isar_feature(aa32_fpdp_v2, s)) {
2816         return false;
2817     }
2818 
2819     if (!dc_isar_feature(aa32_vrint, s)) {
2820         return false;
2821     }
2822 
2823     /* UNDEF accesses to D16-D31 if they don't exist. */
2824     if (!dc_isar_feature(aa32_simd_r32, s) && ((a->vd | a->vm) & 0x10)) {
2825         return false;
2826     }
2827 
2828     if (!vfp_access_check(s)) {
2829         return true;
2830     }
2831 
2832     tmp = tcg_temp_new_i64();
2833     vfp_load_reg64(tmp, a->vm);
2834     fpst = fpstatus_ptr(FPST_FPCR);
2835     tcg_rmode = gen_set_rmode(FPROUNDING_ZERO, fpst);
2836     gen_helper_rintd(tmp, tmp, fpst);
2837     gen_restore_rmode(tcg_rmode, fpst);
2838     vfp_store_reg64(tmp, a->vd);
2839     return true;
2840 }
2841 
2842 static bool trans_VRINTX_hp(DisasContext *s, arg_VRINTX_sp *a)
2843 {
2844     TCGv_ptr fpst;
2845     TCGv_i32 tmp;
2846 
2847     if (!dc_isar_feature(aa32_fp16_arith, s)) {
2848         return false;
2849     }
2850 
2851     if (!vfp_access_check(s)) {
2852         return true;
2853     }
2854 
2855     tmp = tcg_temp_new_i32();
2856     vfp_load_reg32(tmp, a->vm);
2857     fpst = fpstatus_ptr(FPST_FPCR_F16);
2858     gen_helper_rinth_exact(tmp, tmp, fpst);
2859     vfp_store_reg32(tmp, a->vd);
2860     return true;
2861 }
2862 
2863 static bool trans_VRINTX_sp(DisasContext *s, arg_VRINTX_sp *a)
2864 {
2865     TCGv_ptr fpst;
2866     TCGv_i32 tmp;
2867 
2868     if (!dc_isar_feature(aa32_vrint, s)) {
2869         return false;
2870     }
2871 
2872     if (!vfp_access_check(s)) {
2873         return true;
2874     }
2875 
2876     tmp = tcg_temp_new_i32();
2877     vfp_load_reg32(tmp, a->vm);
2878     fpst = fpstatus_ptr(FPST_FPCR);
2879     gen_helper_rints_exact(tmp, tmp, fpst);
2880     vfp_store_reg32(tmp, a->vd);
2881     return true;
2882 }
2883 
2884 static bool trans_VRINTX_dp(DisasContext *s, arg_VRINTX_dp *a)
2885 {
2886     TCGv_ptr fpst;
2887     TCGv_i64 tmp;
2888 
2889     if (!dc_isar_feature(aa32_fpdp_v2, s)) {
2890         return false;
2891     }
2892 
2893     if (!dc_isar_feature(aa32_vrint, s)) {
2894         return false;
2895     }
2896 
2897     /* UNDEF accesses to D16-D31 if they don't exist. */
2898     if (!dc_isar_feature(aa32_simd_r32, s) && ((a->vd | a->vm) & 0x10)) {
2899         return false;
2900     }
2901 
2902     if (!vfp_access_check(s)) {
2903         return true;
2904     }
2905 
2906     tmp = tcg_temp_new_i64();
2907     vfp_load_reg64(tmp, a->vm);
2908     fpst = fpstatus_ptr(FPST_FPCR);
2909     gen_helper_rintd_exact(tmp, tmp, fpst);
2910     vfp_store_reg64(tmp, a->vd);
2911     return true;
2912 }
2913 
2914 static bool trans_VCVT_sp(DisasContext *s, arg_VCVT_sp *a)
2915 {
2916     TCGv_i64 vd;
2917     TCGv_i32 vm;
2918 
2919     if (!dc_isar_feature(aa32_fpdp_v2, s)) {
2920         return false;
2921     }
2922 
2923     /* UNDEF accesses to D16-D31 if they don't exist. */
2924     if (!dc_isar_feature(aa32_simd_r32, s) && (a->vd & 0x10)) {
2925         return false;
2926     }
2927 
2928     if (!vfp_access_check(s)) {
2929         return true;
2930     }
2931 
2932     vm = tcg_temp_new_i32();
2933     vd = tcg_temp_new_i64();
2934     vfp_load_reg32(vm, a->vm);
2935     gen_helper_vfp_fcvtds(vd, vm, tcg_env);
2936     vfp_store_reg64(vd, a->vd);
2937     return true;
2938 }
2939 
2940 static bool trans_VCVT_dp(DisasContext *s, arg_VCVT_dp *a)
2941 {
2942     TCGv_i64 vm;
2943     TCGv_i32 vd;
2944 
2945     if (!dc_isar_feature(aa32_fpdp_v2, s)) {
2946         return false;
2947     }
2948 
2949     /* UNDEF accesses to D16-D31 if they don't exist. */
2950     if (!dc_isar_feature(aa32_simd_r32, s) && (a->vm & 0x10)) {
2951         return false;
2952     }
2953 
2954     if (!vfp_access_check(s)) {
2955         return true;
2956     }
2957 
2958     vd = tcg_temp_new_i32();
2959     vm = tcg_temp_new_i64();
2960     vfp_load_reg64(vm, a->vm);
2961     gen_helper_vfp_fcvtsd(vd, vm, tcg_env);
2962     vfp_store_reg32(vd, a->vd);
2963     return true;
2964 }
2965 
2966 static bool trans_VCVT_int_hp(DisasContext *s, arg_VCVT_int_sp *a)
2967 {
2968     TCGv_i32 vm;
2969     TCGv_ptr fpst;
2970 
2971     if (!dc_isar_feature(aa32_fp16_arith, s)) {
2972         return false;
2973     }
2974 
2975     if (!vfp_access_check(s)) {
2976         return true;
2977     }
2978 
2979     vm = tcg_temp_new_i32();
2980     vfp_load_reg32(vm, a->vm);
2981     fpst = fpstatus_ptr(FPST_FPCR_F16);
2982     if (a->s) {
2983         /* i32 -> f16 */
2984         gen_helper_vfp_sitoh(vm, vm, fpst);
2985     } else {
2986         /* u32 -> f16 */
2987         gen_helper_vfp_uitoh(vm, vm, fpst);
2988     }
2989     vfp_store_reg32(vm, a->vd);
2990     return true;
2991 }
2992 
2993 static bool trans_VCVT_int_sp(DisasContext *s, arg_VCVT_int_sp *a)
2994 {
2995     TCGv_i32 vm;
2996     TCGv_ptr fpst;
2997 
2998     if (!dc_isar_feature(aa32_fpsp_v2, s)) {
2999         return false;
3000     }
3001 
3002     if (!vfp_access_check(s)) {
3003         return true;
3004     }
3005 
3006     vm = tcg_temp_new_i32();
3007     vfp_load_reg32(vm, a->vm);
3008     fpst = fpstatus_ptr(FPST_FPCR);
3009     if (a->s) {
3010         /* i32 -> f32 */
3011         gen_helper_vfp_sitos(vm, vm, fpst);
3012     } else {
3013         /* u32 -> f32 */
3014         gen_helper_vfp_uitos(vm, vm, fpst);
3015     }
3016     vfp_store_reg32(vm, a->vd);
3017     return true;
3018 }
3019 
3020 static bool trans_VCVT_int_dp(DisasContext *s, arg_VCVT_int_dp *a)
3021 {
3022     TCGv_i32 vm;
3023     TCGv_i64 vd;
3024     TCGv_ptr fpst;
3025 
3026     if (!dc_isar_feature(aa32_fpdp_v2, s)) {
3027         return false;
3028     }
3029 
3030     /* UNDEF accesses to D16-D31 if they don't exist. */
3031     if (!dc_isar_feature(aa32_simd_r32, s) && (a->vd & 0x10)) {
3032         return false;
3033     }
3034 
3035     if (!vfp_access_check(s)) {
3036         return true;
3037     }
3038 
3039     vm = tcg_temp_new_i32();
3040     vd = tcg_temp_new_i64();
3041     vfp_load_reg32(vm, a->vm);
3042     fpst = fpstatus_ptr(FPST_FPCR);
3043     if (a->s) {
3044         /* i32 -> f64 */
3045         gen_helper_vfp_sitod(vd, vm, fpst);
3046     } else {
3047         /* u32 -> f64 */
3048         gen_helper_vfp_uitod(vd, vm, fpst);
3049     }
3050     vfp_store_reg64(vd, a->vd);
3051     return true;
3052 }
3053 
3054 static bool trans_VJCVT(DisasContext *s, arg_VJCVT *a)
3055 {
3056     TCGv_i32 vd;
3057     TCGv_i64 vm;
3058 
3059     if (!dc_isar_feature(aa32_fpdp_v2, s)) {
3060         return false;
3061     }
3062 
3063     if (!dc_isar_feature(aa32_jscvt, s)) {
3064         return false;
3065     }
3066 
3067     /* UNDEF accesses to D16-D31 if they don't exist. */
3068     if (!dc_isar_feature(aa32_simd_r32, s) && (a->vm & 0x10)) {
3069         return false;
3070     }
3071 
3072     if (!vfp_access_check(s)) {
3073         return true;
3074     }
3075 
3076     vm = tcg_temp_new_i64();
3077     vd = tcg_temp_new_i32();
3078     vfp_load_reg64(vm, a->vm);
3079     gen_helper_vjcvt(vd, vm, tcg_env);
3080     vfp_store_reg32(vd, a->vd);
3081     return true;
3082 }
3083 
3084 static bool trans_VCVT_fix_hp(DisasContext *s, arg_VCVT_fix_sp *a)
3085 {
3086     TCGv_i32 vd, shift;
3087     TCGv_ptr fpst;
3088     int frac_bits;
3089 
3090     if (!dc_isar_feature(aa32_fp16_arith, s)) {
3091         return false;
3092     }
3093 
3094     if (!vfp_access_check(s)) {
3095         return true;
3096     }
3097 
3098     frac_bits = (a->opc & 1) ? (32 - a->imm) : (16 - a->imm);
3099 
3100     vd = tcg_temp_new_i32();
3101     vfp_load_reg32(vd, a->vd);
3102 
3103     fpst = fpstatus_ptr(FPST_FPCR_F16);
3104     shift = tcg_constant_i32(frac_bits);
3105 
3106     /* Switch on op:U:sx bits */
3107     switch (a->opc) {
3108     case 0:
3109         gen_helper_vfp_shtoh_round_to_nearest(vd, vd, shift, fpst);
3110         break;
3111     case 1:
3112         gen_helper_vfp_sltoh_round_to_nearest(vd, vd, shift, fpst);
3113         break;
3114     case 2:
3115         gen_helper_vfp_uhtoh_round_to_nearest(vd, vd, shift, fpst);
3116         break;
3117     case 3:
3118         gen_helper_vfp_ultoh_round_to_nearest(vd, vd, shift, fpst);
3119         break;
3120     case 4:
3121         gen_helper_vfp_toshh_round_to_zero(vd, vd, shift, fpst);
3122         break;
3123     case 5:
3124         gen_helper_vfp_toslh_round_to_zero(vd, vd, shift, fpst);
3125         break;
3126     case 6:
3127         gen_helper_vfp_touhh_round_to_zero(vd, vd, shift, fpst);
3128         break;
3129     case 7:
3130         gen_helper_vfp_toulh_round_to_zero(vd, vd, shift, fpst);
3131         break;
3132     default:
3133         g_assert_not_reached();
3134     }
3135 
3136     vfp_store_reg32(vd, a->vd);
3137     return true;
3138 }
3139 
3140 static bool trans_VCVT_fix_sp(DisasContext *s, arg_VCVT_fix_sp *a)
3141 {
3142     TCGv_i32 vd, shift;
3143     TCGv_ptr fpst;
3144     int frac_bits;
3145 
3146     if (!dc_isar_feature(aa32_fpsp_v3, s)) {
3147         return false;
3148     }
3149 
3150     if (!vfp_access_check(s)) {
3151         return true;
3152     }
3153 
3154     frac_bits = (a->opc & 1) ? (32 - a->imm) : (16 - a->imm);
3155 
3156     vd = tcg_temp_new_i32();
3157     vfp_load_reg32(vd, a->vd);
3158 
3159     fpst = fpstatus_ptr(FPST_FPCR);
3160     shift = tcg_constant_i32(frac_bits);
3161 
3162     /* Switch on op:U:sx bits */
3163     switch (a->opc) {
3164     case 0:
3165         gen_helper_vfp_shtos_round_to_nearest(vd, vd, shift, fpst);
3166         break;
3167     case 1:
3168         gen_helper_vfp_sltos_round_to_nearest(vd, vd, shift, fpst);
3169         break;
3170     case 2:
3171         gen_helper_vfp_uhtos_round_to_nearest(vd, vd, shift, fpst);
3172         break;
3173     case 3:
3174         gen_helper_vfp_ultos_round_to_nearest(vd, vd, shift, fpst);
3175         break;
3176     case 4:
3177         gen_helper_vfp_toshs_round_to_zero(vd, vd, shift, fpst);
3178         break;
3179     case 5:
3180         gen_helper_vfp_tosls_round_to_zero(vd, vd, shift, fpst);
3181         break;
3182     case 6:
3183         gen_helper_vfp_touhs_round_to_zero(vd, vd, shift, fpst);
3184         break;
3185     case 7:
3186         gen_helper_vfp_touls_round_to_zero(vd, vd, shift, fpst);
3187         break;
3188     default:
3189         g_assert_not_reached();
3190     }
3191 
3192     vfp_store_reg32(vd, a->vd);
3193     return true;
3194 }
3195 
3196 static bool trans_VCVT_fix_dp(DisasContext *s, arg_VCVT_fix_dp *a)
3197 {
3198     TCGv_i64 vd;
3199     TCGv_i32 shift;
3200     TCGv_ptr fpst;
3201     int frac_bits;
3202 
3203     if (!dc_isar_feature(aa32_fpdp_v3, s)) {
3204         return false;
3205     }
3206 
3207     /* UNDEF accesses to D16-D31 if they don't exist. */
3208     if (!dc_isar_feature(aa32_simd_r32, s) && (a->vd & 0x10)) {
3209         return false;
3210     }
3211 
3212     if (!vfp_access_check(s)) {
3213         return true;
3214     }
3215 
3216     frac_bits = (a->opc & 1) ? (32 - a->imm) : (16 - a->imm);
3217 
3218     vd = tcg_temp_new_i64();
3219     vfp_load_reg64(vd, a->vd);
3220 
3221     fpst = fpstatus_ptr(FPST_FPCR);
3222     shift = tcg_constant_i32(frac_bits);
3223 
3224     /* Switch on op:U:sx bits */
3225     switch (a->opc) {
3226     case 0:
3227         gen_helper_vfp_shtod_round_to_nearest(vd, vd, shift, fpst);
3228         break;
3229     case 1:
3230         gen_helper_vfp_sltod_round_to_nearest(vd, vd, shift, fpst);
3231         break;
3232     case 2:
3233         gen_helper_vfp_uhtod_round_to_nearest(vd, vd, shift, fpst);
3234         break;
3235     case 3:
3236         gen_helper_vfp_ultod_round_to_nearest(vd, vd, shift, fpst);
3237         break;
3238     case 4:
3239         gen_helper_vfp_toshd_round_to_zero(vd, vd, shift, fpst);
3240         break;
3241     case 5:
3242         gen_helper_vfp_tosld_round_to_zero(vd, vd, shift, fpst);
3243         break;
3244     case 6:
3245         gen_helper_vfp_touhd_round_to_zero(vd, vd, shift, fpst);
3246         break;
3247     case 7:
3248         gen_helper_vfp_tould_round_to_zero(vd, vd, shift, fpst);
3249         break;
3250     default:
3251         g_assert_not_reached();
3252     }
3253 
3254     vfp_store_reg64(vd, a->vd);
3255     return true;
3256 }
3257 
3258 static bool trans_VCVT_hp_int(DisasContext *s, arg_VCVT_sp_int *a)
3259 {
3260     TCGv_i32 vm;
3261     TCGv_ptr fpst;
3262 
3263     if (!dc_isar_feature(aa32_fp16_arith, s)) {
3264         return false;
3265     }
3266 
3267     if (!vfp_access_check(s)) {
3268         return true;
3269     }
3270 
3271     fpst = fpstatus_ptr(FPST_FPCR_F16);
3272     vm = tcg_temp_new_i32();
3273     vfp_load_reg32(vm, a->vm);
3274 
3275     if (a->s) {
3276         if (a->rz) {
3277             gen_helper_vfp_tosizh(vm, vm, fpst);
3278         } else {
3279             gen_helper_vfp_tosih(vm, vm, fpst);
3280         }
3281     } else {
3282         if (a->rz) {
3283             gen_helper_vfp_touizh(vm, vm, fpst);
3284         } else {
3285             gen_helper_vfp_touih(vm, vm, fpst);
3286         }
3287     }
3288     vfp_store_reg32(vm, a->vd);
3289     return true;
3290 }
3291 
3292 static bool trans_VCVT_sp_int(DisasContext *s, arg_VCVT_sp_int *a)
3293 {
3294     TCGv_i32 vm;
3295     TCGv_ptr fpst;
3296 
3297     if (!dc_isar_feature(aa32_fpsp_v2, s)) {
3298         return false;
3299     }
3300 
3301     if (!vfp_access_check(s)) {
3302         return true;
3303     }
3304 
3305     fpst = fpstatus_ptr(FPST_FPCR);
3306     vm = tcg_temp_new_i32();
3307     vfp_load_reg32(vm, a->vm);
3308 
3309     if (a->s) {
3310         if (a->rz) {
3311             gen_helper_vfp_tosizs(vm, vm, fpst);
3312         } else {
3313             gen_helper_vfp_tosis(vm, vm, fpst);
3314         }
3315     } else {
3316         if (a->rz) {
3317             gen_helper_vfp_touizs(vm, vm, fpst);
3318         } else {
3319             gen_helper_vfp_touis(vm, vm, fpst);
3320         }
3321     }
3322     vfp_store_reg32(vm, a->vd);
3323     return true;
3324 }
3325 
3326 static bool trans_VCVT_dp_int(DisasContext *s, arg_VCVT_dp_int *a)
3327 {
3328     TCGv_i32 vd;
3329     TCGv_i64 vm;
3330     TCGv_ptr fpst;
3331 
3332     if (!dc_isar_feature(aa32_fpdp_v2, s)) {
3333         return false;
3334     }
3335 
3336     /* UNDEF accesses to D16-D31 if they don't exist. */
3337     if (!dc_isar_feature(aa32_simd_r32, s) && (a->vm & 0x10)) {
3338         return false;
3339     }
3340 
3341     if (!vfp_access_check(s)) {
3342         return true;
3343     }
3344 
3345     fpst = fpstatus_ptr(FPST_FPCR);
3346     vm = tcg_temp_new_i64();
3347     vd = tcg_temp_new_i32();
3348     vfp_load_reg64(vm, a->vm);
3349 
3350     if (a->s) {
3351         if (a->rz) {
3352             gen_helper_vfp_tosizd(vd, vm, fpst);
3353         } else {
3354             gen_helper_vfp_tosid(vd, vm, fpst);
3355         }
3356     } else {
3357         if (a->rz) {
3358             gen_helper_vfp_touizd(vd, vm, fpst);
3359         } else {
3360             gen_helper_vfp_touid(vd, vm, fpst);
3361         }
3362     }
3363     vfp_store_reg32(vd, a->vd);
3364     return true;
3365 }
3366 
3367 static bool trans_VINS(DisasContext *s, arg_VINS *a)
3368 {
3369     TCGv_i32 rd, rm;
3370 
3371     if (!dc_isar_feature(aa32_fp16_arith, s)) {
3372         return false;
3373     }
3374 
3375     if (s->vec_len != 0 || s->vec_stride != 0) {
3376         return false;
3377     }
3378 
3379     if (!vfp_access_check(s)) {
3380         return true;
3381     }
3382 
3383     /* Insert low half of Vm into high half of Vd */
3384     rm = tcg_temp_new_i32();
3385     rd = tcg_temp_new_i32();
3386     vfp_load_reg32(rm, a->vm);
3387     vfp_load_reg32(rd, a->vd);
3388     tcg_gen_deposit_i32(rd, rd, rm, 16, 16);
3389     vfp_store_reg32(rd, a->vd);
3390     return true;
3391 }
3392 
3393 static bool trans_VMOVX(DisasContext *s, arg_VINS *a)
3394 {
3395     TCGv_i32 rm;
3396 
3397     if (!dc_isar_feature(aa32_fp16_arith, s)) {
3398         return false;
3399     }
3400 
3401     if (s->vec_len != 0 || s->vec_stride != 0) {
3402         return false;
3403     }
3404 
3405     if (!vfp_access_check(s)) {
3406         return true;
3407     }
3408 
3409     /* Set Vd to high half of Vm */
3410     rm = tcg_temp_new_i32();
3411     vfp_load_reg32(rm, a->vm);
3412     tcg_gen_shri_i32(rm, rm, 16);
3413     vfp_store_reg32(rm, a->vd);
3414     return true;
3415 }
3416