1e0a3054fSWeiwei Li/* 2ce3af0bbSWeiwei Li * RISC-V translation routines for the Zc[b,mp,mt] Standard Extensions. 3e0a3054fSWeiwei Li * 4e0a3054fSWeiwei Li * Copyright (c) 2021-2022 PLCT Lab 5e0a3054fSWeiwei Li * 6e0a3054fSWeiwei Li * This program is free software; you can redistribute it and/or modify it 7e0a3054fSWeiwei Li * under the terms and conditions of the GNU General Public License, 8e0a3054fSWeiwei Li * version 2 or later, as published by the Free Software Foundation. 9e0a3054fSWeiwei Li * 10e0a3054fSWeiwei Li * This program is distributed in the hope it will be useful, but WITHOUT 11e0a3054fSWeiwei Li * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 12e0a3054fSWeiwei Li * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for 13e0a3054fSWeiwei Li * more details. 14e0a3054fSWeiwei Li * 15e0a3054fSWeiwei Li * You should have received a copy of the GNU General Public License along with 16e0a3054fSWeiwei Li * this program. If not, see <http://www.gnu.org/licenses/>. 17e0a3054fSWeiwei Li */ 18e0a3054fSWeiwei Li 19e0a3054fSWeiwei Li#define REQUIRE_ZCB(ctx) do { \ 20e0a3054fSWeiwei Li if (!ctx->cfg_ptr->ext_zcb) \ 21e0a3054fSWeiwei Li return false; \ 22e0a3054fSWeiwei Li} while (0) 23e0a3054fSWeiwei Li 24193eb522SWeiwei Li#define REQUIRE_ZCMP(ctx) do { \ 25193eb522SWeiwei Li if (!ctx->cfg_ptr->ext_zcmp) \ 26193eb522SWeiwei Li return false; \ 27193eb522SWeiwei Li} while (0) 28193eb522SWeiwei Li 29ce3af0bbSWeiwei Li#define REQUIRE_ZCMT(ctx) do { \ 30ce3af0bbSWeiwei Li if (!ctx->cfg_ptr->ext_zcmt) \ 31ce3af0bbSWeiwei Li return false; \ 32ce3af0bbSWeiwei Li} while (0) 33ce3af0bbSWeiwei Li 34e0a3054fSWeiwei Listatic bool trans_c_zext_b(DisasContext *ctx, arg_c_zext_b *a) 35e0a3054fSWeiwei Li{ 36e0a3054fSWeiwei Li REQUIRE_ZCB(ctx); 37e0a3054fSWeiwei Li return gen_unary(ctx, a, EXT_NONE, tcg_gen_ext8u_tl); 38e0a3054fSWeiwei Li} 39e0a3054fSWeiwei Li 40e0a3054fSWeiwei Listatic bool trans_c_zext_h(DisasContext *ctx, arg_c_zext_h *a) 41e0a3054fSWeiwei Li{ 42e0a3054fSWeiwei Li REQUIRE_ZCB(ctx); 43e0a3054fSWeiwei Li REQUIRE_ZBB(ctx); 44e0a3054fSWeiwei Li return gen_unary(ctx, a, EXT_NONE, tcg_gen_ext16u_tl); 45e0a3054fSWeiwei Li} 46e0a3054fSWeiwei Li 47e0a3054fSWeiwei Listatic bool trans_c_sext_b(DisasContext *ctx, arg_c_sext_b *a) 48e0a3054fSWeiwei Li{ 49e0a3054fSWeiwei Li REQUIRE_ZCB(ctx); 50e0a3054fSWeiwei Li REQUIRE_ZBB(ctx); 51e0a3054fSWeiwei Li return gen_unary(ctx, a, EXT_NONE, tcg_gen_ext8s_tl); 52e0a3054fSWeiwei Li} 53e0a3054fSWeiwei Li 54e0a3054fSWeiwei Listatic bool trans_c_sext_h(DisasContext *ctx, arg_c_sext_h *a) 55e0a3054fSWeiwei Li{ 56e0a3054fSWeiwei Li REQUIRE_ZCB(ctx); 57e0a3054fSWeiwei Li REQUIRE_ZBB(ctx); 58e0a3054fSWeiwei Li return gen_unary(ctx, a, EXT_NONE, tcg_gen_ext16s_tl); 59e0a3054fSWeiwei Li} 60e0a3054fSWeiwei Li 61e0a3054fSWeiwei Listatic bool trans_c_zext_w(DisasContext *ctx, arg_c_zext_w *a) 62e0a3054fSWeiwei Li{ 63e0a3054fSWeiwei Li REQUIRE_64BIT(ctx); 64e0a3054fSWeiwei Li REQUIRE_ZCB(ctx); 65e0a3054fSWeiwei Li REQUIRE_ZBA(ctx); 66e0a3054fSWeiwei Li return gen_unary(ctx, a, EXT_NONE, tcg_gen_ext32u_tl); 67e0a3054fSWeiwei Li} 68e0a3054fSWeiwei Li 69e0a3054fSWeiwei Listatic bool trans_c_not(DisasContext *ctx, arg_c_not *a) 70e0a3054fSWeiwei Li{ 71e0a3054fSWeiwei Li REQUIRE_ZCB(ctx); 72e0a3054fSWeiwei Li return gen_unary(ctx, a, EXT_NONE, tcg_gen_not_tl); 73e0a3054fSWeiwei Li} 74e0a3054fSWeiwei Li 75e0a3054fSWeiwei Listatic bool trans_c_mul(DisasContext *ctx, arg_c_mul *a) 76e0a3054fSWeiwei Li{ 77e0a3054fSWeiwei Li REQUIRE_ZCB(ctx); 78e0a3054fSWeiwei Li REQUIRE_M_OR_ZMMUL(ctx); 79e0a3054fSWeiwei Li return gen_arith(ctx, a, EXT_NONE, tcg_gen_mul_tl, NULL); 80e0a3054fSWeiwei Li} 81e0a3054fSWeiwei Li 82e0a3054fSWeiwei Listatic bool trans_c_lbu(DisasContext *ctx, arg_c_lbu *a) 83e0a3054fSWeiwei Li{ 84e0a3054fSWeiwei Li REQUIRE_ZCB(ctx); 85e0a3054fSWeiwei Li return gen_load(ctx, a, MO_UB); 86e0a3054fSWeiwei Li} 87e0a3054fSWeiwei Li 88e0a3054fSWeiwei Listatic bool trans_c_lhu(DisasContext *ctx, arg_c_lhu *a) 89e0a3054fSWeiwei Li{ 90e0a3054fSWeiwei Li REQUIRE_ZCB(ctx); 91e0a3054fSWeiwei Li return gen_load(ctx, a, MO_UW); 92e0a3054fSWeiwei Li} 93e0a3054fSWeiwei Li 94e0a3054fSWeiwei Listatic bool trans_c_lh(DisasContext *ctx, arg_c_lh *a) 95e0a3054fSWeiwei Li{ 96e0a3054fSWeiwei Li REQUIRE_ZCB(ctx); 97e0a3054fSWeiwei Li return gen_load(ctx, a, MO_SW); 98e0a3054fSWeiwei Li} 99e0a3054fSWeiwei Li 100e0a3054fSWeiwei Listatic bool trans_c_sb(DisasContext *ctx, arg_c_sb *a) 101e0a3054fSWeiwei Li{ 102e0a3054fSWeiwei Li REQUIRE_ZCB(ctx); 103e0a3054fSWeiwei Li return gen_store(ctx, a, MO_UB); 104e0a3054fSWeiwei Li} 105e0a3054fSWeiwei Li 106e0a3054fSWeiwei Listatic bool trans_c_sh(DisasContext *ctx, arg_c_sh *a) 107e0a3054fSWeiwei Li{ 108e0a3054fSWeiwei Li REQUIRE_ZCB(ctx); 109e0a3054fSWeiwei Li return gen_store(ctx, a, MO_UW); 110e0a3054fSWeiwei Li} 111193eb522SWeiwei Li 112193eb522SWeiwei Li#define X_S0 8 113193eb522SWeiwei Li#define X_S1 9 114193eb522SWeiwei Li#define X_Sn 16 115193eb522SWeiwei Li 116193eb522SWeiwei Listatic uint32_t decode_push_pop_list(DisasContext *ctx, target_ulong rlist) 117193eb522SWeiwei Li{ 118193eb522SWeiwei Li uint32_t reg_bitmap = 0; 119193eb522SWeiwei Li 120427d8e7dSDaniel Henrique Barboza if (has_ext(ctx, RVE) && rlist > 6) { 121193eb522SWeiwei Li return 0; 122193eb522SWeiwei Li } 123193eb522SWeiwei Li 124193eb522SWeiwei Li switch (rlist) { 125193eb522SWeiwei Li case 15: 126193eb522SWeiwei Li reg_bitmap |= 1 << (X_Sn + 11) ; 127193eb522SWeiwei Li reg_bitmap |= 1 << (X_Sn + 10) ; 128193eb522SWeiwei Li /* FALL THROUGH */ 129193eb522SWeiwei Li case 14: 130193eb522SWeiwei Li reg_bitmap |= 1 << (X_Sn + 9) ; 131193eb522SWeiwei Li /* FALL THROUGH */ 132193eb522SWeiwei Li case 13: 133193eb522SWeiwei Li reg_bitmap |= 1 << (X_Sn + 8) ; 134193eb522SWeiwei Li /* FALL THROUGH */ 135193eb522SWeiwei Li case 12: 136193eb522SWeiwei Li reg_bitmap |= 1 << (X_Sn + 7) ; 137193eb522SWeiwei Li /* FALL THROUGH */ 138193eb522SWeiwei Li case 11: 139193eb522SWeiwei Li reg_bitmap |= 1 << (X_Sn + 6) ; 140193eb522SWeiwei Li /* FALL THROUGH */ 141193eb522SWeiwei Li case 10: 142193eb522SWeiwei Li reg_bitmap |= 1 << (X_Sn + 5) ; 143193eb522SWeiwei Li /* FALL THROUGH */ 144193eb522SWeiwei Li case 9: 145193eb522SWeiwei Li reg_bitmap |= 1 << (X_Sn + 4) ; 146193eb522SWeiwei Li /* FALL THROUGH */ 147193eb522SWeiwei Li case 8: 148193eb522SWeiwei Li reg_bitmap |= 1 << (X_Sn + 3) ; 149193eb522SWeiwei Li /* FALL THROUGH */ 150193eb522SWeiwei Li case 7: 151193eb522SWeiwei Li reg_bitmap |= 1 << (X_Sn + 2) ; 152193eb522SWeiwei Li /* FALL THROUGH */ 153193eb522SWeiwei Li case 6: 154193eb522SWeiwei Li reg_bitmap |= 1 << X_S1 ; 155193eb522SWeiwei Li /* FALL THROUGH */ 156193eb522SWeiwei Li case 5: 157193eb522SWeiwei Li reg_bitmap |= 1 << X_S0; 158193eb522SWeiwei Li /* FALL THROUGH */ 159193eb522SWeiwei Li case 4: 160193eb522SWeiwei Li reg_bitmap |= 1 << xRA; 161193eb522SWeiwei Li break; 162193eb522SWeiwei Li default: 163193eb522SWeiwei Li break; 164193eb522SWeiwei Li } 165193eb522SWeiwei Li 166193eb522SWeiwei Li return reg_bitmap; 167193eb522SWeiwei Li} 168193eb522SWeiwei Li 169193eb522SWeiwei Listatic bool gen_pop(DisasContext *ctx, arg_cmpp *a, bool ret, bool ret_val) 170193eb522SWeiwei Li{ 171193eb522SWeiwei Li REQUIRE_ZCMP(ctx); 172193eb522SWeiwei Li 173193eb522SWeiwei Li uint32_t reg_bitmap = decode_push_pop_list(ctx, a->urlist); 174193eb522SWeiwei Li if (reg_bitmap == 0) { 175193eb522SWeiwei Li return false; 176193eb522SWeiwei Li } 177193eb522SWeiwei Li 178193eb522SWeiwei Li MemOp memop = get_ol(ctx) == MXL_RV32 ? MO_TEUL : MO_TEUQ; 179193eb522SWeiwei Li int reg_size = memop_size(memop); 180193eb522SWeiwei Li target_ulong stack_adj = ROUND_UP(ctpop32(reg_bitmap) * reg_size, 16) + 181193eb522SWeiwei Li a->spimm; 182193eb522SWeiwei Li TCGv sp = dest_gpr(ctx, xSP); 183193eb522SWeiwei Li TCGv addr = tcg_temp_new(); 184193eb522SWeiwei Li int i; 185193eb522SWeiwei Li 186193eb522SWeiwei Li tcg_gen_addi_tl(addr, sp, stack_adj - reg_size); 187193eb522SWeiwei Li 188193eb522SWeiwei Li for (i = X_Sn + 11; i >= 0; i--) { 189193eb522SWeiwei Li if (reg_bitmap & (1 << i)) { 190193eb522SWeiwei Li TCGv dest = dest_gpr(ctx, i); 191193eb522SWeiwei Li tcg_gen_qemu_ld_tl(dest, addr, ctx->mem_idx, memop); 192193eb522SWeiwei Li gen_set_gpr(ctx, i, dest); 193193eb522SWeiwei Li tcg_gen_subi_tl(addr, addr, reg_size); 194193eb522SWeiwei Li } 195193eb522SWeiwei Li } 196193eb522SWeiwei Li 197193eb522SWeiwei Li tcg_gen_addi_tl(sp, sp, stack_adj); 198193eb522SWeiwei Li gen_set_gpr(ctx, xSP, sp); 199193eb522SWeiwei Li 200193eb522SWeiwei Li if (ret_val) { 201193eb522SWeiwei Li gen_set_gpr(ctx, xA0, ctx->zero); 202193eb522SWeiwei Li } 203193eb522SWeiwei Li 204193eb522SWeiwei Li if (ret) { 205bfc4f9e3SWeiwei Li TCGv ret_addr = get_gpr(ctx, xRA, EXT_SIGN); 206bfc4f9e3SWeiwei Li tcg_gen_mov_tl(cpu_pc, ret_addr); 207193eb522SWeiwei Li tcg_gen_lookup_and_goto_ptr(); 208193eb522SWeiwei Li ctx->base.is_jmp = DISAS_NORETURN; 209193eb522SWeiwei Li } 210193eb522SWeiwei Li 211193eb522SWeiwei Li return true; 212193eb522SWeiwei Li} 213193eb522SWeiwei Li 214193eb522SWeiwei Listatic bool trans_cm_push(DisasContext *ctx, arg_cm_push *a) 215193eb522SWeiwei Li{ 216193eb522SWeiwei Li REQUIRE_ZCMP(ctx); 217193eb522SWeiwei Li 218193eb522SWeiwei Li uint32_t reg_bitmap = decode_push_pop_list(ctx, a->urlist); 219193eb522SWeiwei Li if (reg_bitmap == 0) { 220193eb522SWeiwei Li return false; 221193eb522SWeiwei Li } 222193eb522SWeiwei Li 223193eb522SWeiwei Li MemOp memop = get_ol(ctx) == MXL_RV32 ? MO_TEUL : MO_TEUQ; 224193eb522SWeiwei Li int reg_size = memop_size(memop); 225193eb522SWeiwei Li target_ulong stack_adj = ROUND_UP(ctpop32(reg_bitmap) * reg_size, 16) + 226193eb522SWeiwei Li a->spimm; 227193eb522SWeiwei Li TCGv sp = dest_gpr(ctx, xSP); 228193eb522SWeiwei Li TCGv addr = tcg_temp_new(); 229193eb522SWeiwei Li int i; 230193eb522SWeiwei Li 231193eb522SWeiwei Li tcg_gen_subi_tl(addr, sp, reg_size); 232193eb522SWeiwei Li 233193eb522SWeiwei Li for (i = X_Sn + 11; i >= 0; i--) { 234193eb522SWeiwei Li if (reg_bitmap & (1 << i)) { 235193eb522SWeiwei Li TCGv val = get_gpr(ctx, i, EXT_NONE); 236193eb522SWeiwei Li tcg_gen_qemu_st_tl(val, addr, ctx->mem_idx, memop); 237193eb522SWeiwei Li tcg_gen_subi_tl(addr, addr, reg_size); 238193eb522SWeiwei Li } 239193eb522SWeiwei Li } 240193eb522SWeiwei Li 241193eb522SWeiwei Li tcg_gen_subi_tl(sp, sp, stack_adj); 242193eb522SWeiwei Li gen_set_gpr(ctx, xSP, sp); 243193eb522SWeiwei Li 244193eb522SWeiwei Li return true; 245193eb522SWeiwei Li} 246193eb522SWeiwei Li 247193eb522SWeiwei Listatic bool trans_cm_pop(DisasContext *ctx, arg_cm_pop *a) 248193eb522SWeiwei Li{ 249193eb522SWeiwei Li return gen_pop(ctx, a, false, false); 250193eb522SWeiwei Li} 251193eb522SWeiwei Li 252193eb522SWeiwei Listatic bool trans_cm_popret(DisasContext *ctx, arg_cm_popret *a) 253193eb522SWeiwei Li{ 254193eb522SWeiwei Li return gen_pop(ctx, a, true, false); 255193eb522SWeiwei Li} 256193eb522SWeiwei Li 257193eb522SWeiwei Listatic bool trans_cm_popretz(DisasContext *ctx, arg_cm_popret *a) 258193eb522SWeiwei Li{ 259193eb522SWeiwei Li return gen_pop(ctx, a, true, true); 260193eb522SWeiwei Li} 261193eb522SWeiwei Li 262193eb522SWeiwei Listatic bool trans_cm_mva01s(DisasContext *ctx, arg_cm_mva01s *a) 263193eb522SWeiwei Li{ 264193eb522SWeiwei Li REQUIRE_ZCMP(ctx); 265193eb522SWeiwei Li 266193eb522SWeiwei Li TCGv src1 = get_gpr(ctx, a->rs1, EXT_NONE); 267193eb522SWeiwei Li TCGv src2 = get_gpr(ctx, a->rs2, EXT_NONE); 268193eb522SWeiwei Li 269193eb522SWeiwei Li gen_set_gpr(ctx, xA0, src1); 270193eb522SWeiwei Li gen_set_gpr(ctx, xA1, src2); 271193eb522SWeiwei Li 272193eb522SWeiwei Li return true; 273193eb522SWeiwei Li} 274193eb522SWeiwei Li 275193eb522SWeiwei Listatic bool trans_cm_mvsa01(DisasContext *ctx, arg_cm_mvsa01 *a) 276193eb522SWeiwei Li{ 277193eb522SWeiwei Li REQUIRE_ZCMP(ctx); 278193eb522SWeiwei Li 279193eb522SWeiwei Li if (a->rs1 == a->rs2) { 280193eb522SWeiwei Li return false; 281193eb522SWeiwei Li } 282193eb522SWeiwei Li 283193eb522SWeiwei Li TCGv a0 = get_gpr(ctx, xA0, EXT_NONE); 284193eb522SWeiwei Li TCGv a1 = get_gpr(ctx, xA1, EXT_NONE); 285193eb522SWeiwei Li 286193eb522SWeiwei Li gen_set_gpr(ctx, a->rs1, a0); 287193eb522SWeiwei Li gen_set_gpr(ctx, a->rs2, a1); 288193eb522SWeiwei Li 289193eb522SWeiwei Li return true; 290193eb522SWeiwei Li} 291ce3af0bbSWeiwei Li 292ce3af0bbSWeiwei Listatic bool trans_cm_jalt(DisasContext *ctx, arg_cm_jalt *a) 293ce3af0bbSWeiwei Li{ 294ce3af0bbSWeiwei Li REQUIRE_ZCMT(ctx); 295ce3af0bbSWeiwei Li 296*3011c1ddSJason Chien TCGv addr = tcg_temp_new(); 297*3011c1ddSJason Chien 298ce3af0bbSWeiwei Li /* 299ce3af0bbSWeiwei Li * Update pc to current for the non-unwinding exception 300ce3af0bbSWeiwei Li * that might come from cpu_ld*_code() in the helper. 301ce3af0bbSWeiwei Li */ 302227fb82fSWeiwei Li gen_update_pc(ctx, 0); 303*3011c1ddSJason Chien gen_helper_cm_jalt(addr, tcg_env, tcg_constant_i32(a->index)); 304ce3af0bbSWeiwei Li 305ce3af0bbSWeiwei Li /* c.jt vs c.jalt depends on the index. */ 306ce3af0bbSWeiwei Li if (a->index >= 32) { 307356c13f9SWeiwei Li TCGv succ_pc = dest_gpr(ctx, xRA); 308356c13f9SWeiwei Li gen_pc_plus_diff(succ_pc, ctx, ctx->cur_insn_len); 309356c13f9SWeiwei Li gen_set_gpr(ctx, xRA, succ_pc); 310ce3af0bbSWeiwei Li } 311ce3af0bbSWeiwei Li 312*3011c1ddSJason Chien tcg_gen_mov_tl(cpu_pc, addr); 313*3011c1ddSJason Chien 314ce3af0bbSWeiwei Li tcg_gen_lookup_and_goto_ptr(); 315ce3af0bbSWeiwei Li ctx->base.is_jmp = DISAS_NORETURN; 316ce3af0bbSWeiwei Li return true; 317ce3af0bbSWeiwei Li} 318