xref: /openbmc/qemu/target/riscv/insn_trans/trans_rvzce.c.inc (revision cbccded4a2b5d685a426a437e25f67d3a375b292)
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