xref: /qemu/target/arm/tcg/translate-m-nocp.c (revision 9a5071abbce614c52d0e72bdbd688cd4e2a9ee46)
1*9a5071abSPeter Maydell /*
2*9a5071abSPeter Maydell  *  ARM translation: M-profile NOCP special-case instructions
3*9a5071abSPeter Maydell  *
4*9a5071abSPeter Maydell  *  Copyright (c) 2020 Linaro, Ltd.
5*9a5071abSPeter Maydell  *
6*9a5071abSPeter Maydell  * This library is free software; you can redistribute it and/or
7*9a5071abSPeter Maydell  * modify it under the terms of the GNU Lesser General Public
8*9a5071abSPeter Maydell  * License as published by the Free Software Foundation; either
9*9a5071abSPeter Maydell  * version 2.1 of the License, or (at your option) any later version.
10*9a5071abSPeter Maydell  *
11*9a5071abSPeter Maydell  * This library is distributed in the hope that it will be useful,
12*9a5071abSPeter Maydell  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13*9a5071abSPeter Maydell  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14*9a5071abSPeter Maydell  * Lesser General Public License for more details.
15*9a5071abSPeter Maydell  *
16*9a5071abSPeter Maydell  * You should have received a copy of the GNU Lesser General Public
17*9a5071abSPeter Maydell  * License along with this library; if not, see <http://www.gnu.org/licenses/>.
18*9a5071abSPeter Maydell  */
19*9a5071abSPeter Maydell 
20*9a5071abSPeter Maydell #include "qemu/osdep.h"
21*9a5071abSPeter Maydell #include "tcg/tcg-op.h"
22*9a5071abSPeter Maydell #include "translate.h"
23*9a5071abSPeter Maydell #include "translate-a32.h"
24*9a5071abSPeter Maydell 
25*9a5071abSPeter Maydell #include "decode-m-nocp.c.inc"
26*9a5071abSPeter Maydell 
27*9a5071abSPeter Maydell /*
28*9a5071abSPeter Maydell  * Decode VLLDM and VLSTM are nonstandard because:
29*9a5071abSPeter Maydell  *  * if there is no FPU then these insns must NOP in
30*9a5071abSPeter Maydell  *    Secure state and UNDEF in Nonsecure state
31*9a5071abSPeter Maydell  *  * if there is an FPU then these insns do not have
32*9a5071abSPeter Maydell  *    the usual behaviour that vfp_access_check() provides of
33*9a5071abSPeter Maydell  *    being controlled by CPACR/NSACR enable bits or the
34*9a5071abSPeter Maydell  *    lazy-stacking logic.
35*9a5071abSPeter Maydell  */
36*9a5071abSPeter Maydell static bool trans_VLLDM_VLSTM(DisasContext *s, arg_VLLDM_VLSTM *a)
37*9a5071abSPeter Maydell {
38*9a5071abSPeter Maydell     TCGv_i32 fptr;
39*9a5071abSPeter Maydell 
40*9a5071abSPeter Maydell     if (!arm_dc_feature(s, ARM_FEATURE_M) ||
41*9a5071abSPeter Maydell         !arm_dc_feature(s, ARM_FEATURE_V8)) {
42*9a5071abSPeter Maydell         return false;
43*9a5071abSPeter Maydell     }
44*9a5071abSPeter Maydell 
45*9a5071abSPeter Maydell     if (a->op) {
46*9a5071abSPeter Maydell         /*
47*9a5071abSPeter Maydell          * T2 encoding ({D0-D31} reglist): v8.1M and up. We choose not
48*9a5071abSPeter Maydell          * to take the IMPDEF option to make memory accesses to the stack
49*9a5071abSPeter Maydell          * slots that correspond to the D16-D31 registers (discarding
50*9a5071abSPeter Maydell          * read data and writing UNKNOWN values), so for us the T2
51*9a5071abSPeter Maydell          * encoding behaves identically to the T1 encoding.
52*9a5071abSPeter Maydell          */
53*9a5071abSPeter Maydell         if (!arm_dc_feature(s, ARM_FEATURE_V8_1M)) {
54*9a5071abSPeter Maydell             return false;
55*9a5071abSPeter Maydell         }
56*9a5071abSPeter Maydell     } else {
57*9a5071abSPeter Maydell         /*
58*9a5071abSPeter Maydell          * T1 encoding ({D0-D15} reglist); undef if we have 32 Dregs.
59*9a5071abSPeter Maydell          * This is currently architecturally impossible, but we add the
60*9a5071abSPeter Maydell          * check to stay in line with the pseudocode. Note that we must
61*9a5071abSPeter Maydell          * emit code for the UNDEF so it takes precedence over the NOCP.
62*9a5071abSPeter Maydell          */
63*9a5071abSPeter Maydell         if (dc_isar_feature(aa32_simd_r32, s)) {
64*9a5071abSPeter Maydell             unallocated_encoding(s);
65*9a5071abSPeter Maydell             return true;
66*9a5071abSPeter Maydell         }
67*9a5071abSPeter Maydell     }
68*9a5071abSPeter Maydell 
69*9a5071abSPeter Maydell     /*
70*9a5071abSPeter Maydell      * If not secure, UNDEF. We must emit code for this
71*9a5071abSPeter Maydell      * rather than returning false so that this takes
72*9a5071abSPeter Maydell      * precedence over the m-nocp.decode NOCP fallback.
73*9a5071abSPeter Maydell      */
74*9a5071abSPeter Maydell     if (!s->v8m_secure) {
75*9a5071abSPeter Maydell         unallocated_encoding(s);
76*9a5071abSPeter Maydell         return true;
77*9a5071abSPeter Maydell     }
78*9a5071abSPeter Maydell     /* If no fpu, NOP. */
79*9a5071abSPeter Maydell     if (!dc_isar_feature(aa32_vfp, s)) {
80*9a5071abSPeter Maydell         return true;
81*9a5071abSPeter Maydell     }
82*9a5071abSPeter Maydell 
83*9a5071abSPeter Maydell     fptr = load_reg(s, a->rn);
84*9a5071abSPeter Maydell     if (a->l) {
85*9a5071abSPeter Maydell         gen_helper_v7m_vlldm(cpu_env, fptr);
86*9a5071abSPeter Maydell     } else {
87*9a5071abSPeter Maydell         gen_helper_v7m_vlstm(cpu_env, fptr);
88*9a5071abSPeter Maydell     }
89*9a5071abSPeter Maydell     tcg_temp_free_i32(fptr);
90*9a5071abSPeter Maydell 
91*9a5071abSPeter Maydell     /* End the TB, because we have updated FP control bits */
92*9a5071abSPeter Maydell     s->base.is_jmp = DISAS_UPDATE_EXIT;
93*9a5071abSPeter Maydell     return true;
94*9a5071abSPeter Maydell }
95*9a5071abSPeter Maydell 
96*9a5071abSPeter Maydell static bool trans_VSCCLRM(DisasContext *s, arg_VSCCLRM *a)
97*9a5071abSPeter Maydell {
98*9a5071abSPeter Maydell     int btmreg, topreg;
99*9a5071abSPeter Maydell     TCGv_i64 zero;
100*9a5071abSPeter Maydell     TCGv_i32 aspen, sfpa;
101*9a5071abSPeter Maydell 
102*9a5071abSPeter Maydell     if (!dc_isar_feature(aa32_m_sec_state, s)) {
103*9a5071abSPeter Maydell         /* Before v8.1M, fall through in decode to NOCP check */
104*9a5071abSPeter Maydell         return false;
105*9a5071abSPeter Maydell     }
106*9a5071abSPeter Maydell 
107*9a5071abSPeter Maydell     /* Explicitly UNDEF because this takes precedence over NOCP */
108*9a5071abSPeter Maydell     if (!arm_dc_feature(s, ARM_FEATURE_M_MAIN) || !s->v8m_secure) {
109*9a5071abSPeter Maydell         unallocated_encoding(s);
110*9a5071abSPeter Maydell         return true;
111*9a5071abSPeter Maydell     }
112*9a5071abSPeter Maydell 
113*9a5071abSPeter Maydell     if (!dc_isar_feature(aa32_vfp_simd, s)) {
114*9a5071abSPeter Maydell         /* NOP if we have neither FP nor MVE */
115*9a5071abSPeter Maydell         return true;
116*9a5071abSPeter Maydell     }
117*9a5071abSPeter Maydell 
118*9a5071abSPeter Maydell     /*
119*9a5071abSPeter Maydell      * If FPCCR.ASPEN != 0 && CONTROL_S.SFPA == 0 then there is no
120*9a5071abSPeter Maydell      * active floating point context so we must NOP (without doing
121*9a5071abSPeter Maydell      * any lazy state preservation or the NOCP check).
122*9a5071abSPeter Maydell      */
123*9a5071abSPeter Maydell     aspen = load_cpu_field(v7m.fpccr[M_REG_S]);
124*9a5071abSPeter Maydell     sfpa = load_cpu_field(v7m.control[M_REG_S]);
125*9a5071abSPeter Maydell     tcg_gen_andi_i32(aspen, aspen, R_V7M_FPCCR_ASPEN_MASK);
126*9a5071abSPeter Maydell     tcg_gen_xori_i32(aspen, aspen, R_V7M_FPCCR_ASPEN_MASK);
127*9a5071abSPeter Maydell     tcg_gen_andi_i32(sfpa, sfpa, R_V7M_CONTROL_SFPA_MASK);
128*9a5071abSPeter Maydell     tcg_gen_or_i32(sfpa, sfpa, aspen);
129*9a5071abSPeter Maydell     arm_gen_condlabel(s);
130*9a5071abSPeter Maydell     tcg_gen_brcondi_i32(TCG_COND_EQ, sfpa, 0, s->condlabel);
131*9a5071abSPeter Maydell 
132*9a5071abSPeter Maydell     if (s->fp_excp_el != 0) {
133*9a5071abSPeter Maydell         gen_exception_insn(s, s->pc_curr, EXCP_NOCP,
134*9a5071abSPeter Maydell                            syn_uncategorized(), s->fp_excp_el);
135*9a5071abSPeter Maydell         return true;
136*9a5071abSPeter Maydell     }
137*9a5071abSPeter Maydell 
138*9a5071abSPeter Maydell     topreg = a->vd + a->imm - 1;
139*9a5071abSPeter Maydell     btmreg = a->vd;
140*9a5071abSPeter Maydell 
141*9a5071abSPeter Maydell     /* Convert to Sreg numbers if the insn specified in Dregs */
142*9a5071abSPeter Maydell     if (a->size == 3) {
143*9a5071abSPeter Maydell         topreg = topreg * 2 + 1;
144*9a5071abSPeter Maydell         btmreg *= 2;
145*9a5071abSPeter Maydell     }
146*9a5071abSPeter Maydell 
147*9a5071abSPeter Maydell     if (topreg > 63 || (topreg > 31 && !(topreg & 1))) {
148*9a5071abSPeter Maydell         /* UNPREDICTABLE: we choose to undef */
149*9a5071abSPeter Maydell         unallocated_encoding(s);
150*9a5071abSPeter Maydell         return true;
151*9a5071abSPeter Maydell     }
152*9a5071abSPeter Maydell 
153*9a5071abSPeter Maydell     /* Silently ignore requests to clear D16-D31 if they don't exist */
154*9a5071abSPeter Maydell     if (topreg > 31 && !dc_isar_feature(aa32_simd_r32, s)) {
155*9a5071abSPeter Maydell         topreg = 31;
156*9a5071abSPeter Maydell     }
157*9a5071abSPeter Maydell 
158*9a5071abSPeter Maydell     if (!vfp_access_check(s)) {
159*9a5071abSPeter Maydell         return true;
160*9a5071abSPeter Maydell     }
161*9a5071abSPeter Maydell 
162*9a5071abSPeter Maydell     /* Zero the Sregs from btmreg to topreg inclusive. */
163*9a5071abSPeter Maydell     zero = tcg_const_i64(0);
164*9a5071abSPeter Maydell     if (btmreg & 1) {
165*9a5071abSPeter Maydell         write_neon_element64(zero, btmreg >> 1, 1, MO_32);
166*9a5071abSPeter Maydell         btmreg++;
167*9a5071abSPeter Maydell     }
168*9a5071abSPeter Maydell     for (; btmreg + 1 <= topreg; btmreg += 2) {
169*9a5071abSPeter Maydell         write_neon_element64(zero, btmreg >> 1, 0, MO_64);
170*9a5071abSPeter Maydell     }
171*9a5071abSPeter Maydell     if (btmreg == topreg) {
172*9a5071abSPeter Maydell         write_neon_element64(zero, btmreg >> 1, 0, MO_32);
173*9a5071abSPeter Maydell         btmreg++;
174*9a5071abSPeter Maydell     }
175*9a5071abSPeter Maydell     assert(btmreg == topreg + 1);
176*9a5071abSPeter Maydell     /* TODO: when MVE is implemented, zero VPR here */
177*9a5071abSPeter Maydell     return true;
178*9a5071abSPeter Maydell }
179*9a5071abSPeter Maydell 
180*9a5071abSPeter Maydell static bool trans_NOCP(DisasContext *s, arg_nocp *a)
181*9a5071abSPeter Maydell {
182*9a5071abSPeter Maydell     /*
183*9a5071abSPeter Maydell      * Handle M-profile early check for disabled coprocessor:
184*9a5071abSPeter Maydell      * all we need to do here is emit the NOCP exception if
185*9a5071abSPeter Maydell      * the coprocessor is disabled. Otherwise we return false
186*9a5071abSPeter Maydell      * and the real VFP/etc decode will handle the insn.
187*9a5071abSPeter Maydell      */
188*9a5071abSPeter Maydell     assert(arm_dc_feature(s, ARM_FEATURE_M));
189*9a5071abSPeter Maydell 
190*9a5071abSPeter Maydell     if (a->cp == 11) {
191*9a5071abSPeter Maydell         a->cp = 10;
192*9a5071abSPeter Maydell     }
193*9a5071abSPeter Maydell     if (arm_dc_feature(s, ARM_FEATURE_V8_1M) &&
194*9a5071abSPeter Maydell         (a->cp == 8 || a->cp == 9 || a->cp == 14 || a->cp == 15)) {
195*9a5071abSPeter Maydell         /* in v8.1M cp 8, 9, 14, 15 also are governed by the cp10 enable */
196*9a5071abSPeter Maydell         a->cp = 10;
197*9a5071abSPeter Maydell     }
198*9a5071abSPeter Maydell 
199*9a5071abSPeter Maydell     if (a->cp != 10) {
200*9a5071abSPeter Maydell         gen_exception_insn(s, s->pc_curr, EXCP_NOCP,
201*9a5071abSPeter Maydell                            syn_uncategorized(), default_exception_el(s));
202*9a5071abSPeter Maydell         return true;
203*9a5071abSPeter Maydell     }
204*9a5071abSPeter Maydell 
205*9a5071abSPeter Maydell     if (s->fp_excp_el != 0) {
206*9a5071abSPeter Maydell         gen_exception_insn(s, s->pc_curr, EXCP_NOCP,
207*9a5071abSPeter Maydell                            syn_uncategorized(), s->fp_excp_el);
208*9a5071abSPeter Maydell         return true;
209*9a5071abSPeter Maydell     }
210*9a5071abSPeter Maydell 
211*9a5071abSPeter Maydell     return false;
212*9a5071abSPeter Maydell }
213*9a5071abSPeter Maydell 
214*9a5071abSPeter Maydell static bool trans_NOCP_8_1(DisasContext *s, arg_nocp *a)
215*9a5071abSPeter Maydell {
216*9a5071abSPeter Maydell     /* This range needs a coprocessor check for v8.1M and later only */
217*9a5071abSPeter Maydell     if (!arm_dc_feature(s, ARM_FEATURE_V8_1M)) {
218*9a5071abSPeter Maydell         return false;
219*9a5071abSPeter Maydell     }
220*9a5071abSPeter Maydell     return trans_NOCP(s, a);
221*9a5071abSPeter Maydell }
222