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