mirror of https://github.com/xemu-project/xemu.git
target/riscv: Introduce DisasExtend and new helpers
Introduce get_gpr, dest_gpr, temp_new -- new helpers that do not force tcg globals into temps, returning a constant 0 for $zero as source and a new temp for $zero as destination. Introduce ctx->w for simplifying word operations, such as addw. Reviewed-by: Bin Meng <bmeng.cn@gmail.com> Reviewed-by: Alistair Francis <alistair.francis@wdc.com> Signed-off-by: Richard Henderson <richard.henderson@linaro.org> Message-id: 20210823195529.560295-6-richard.henderson@linaro.org Signed-off-by: Alistair Francis <alistair.francis@wdc.com>
This commit is contained in:
parent
867c81968a
commit
ecda15d137
|
@ -39,15 +39,25 @@ static TCGv load_val;
|
||||||
|
|
||||||
#include "exec/gen-icount.h"
|
#include "exec/gen-icount.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If an operation is being performed on less than TARGET_LONG_BITS,
|
||||||
|
* it may require the inputs to be sign- or zero-extended; which will
|
||||||
|
* depend on the exact operation being performed.
|
||||||
|
*/
|
||||||
|
typedef enum {
|
||||||
|
EXT_NONE,
|
||||||
|
EXT_SIGN,
|
||||||
|
EXT_ZERO,
|
||||||
|
} DisasExtend;
|
||||||
|
|
||||||
typedef struct DisasContext {
|
typedef struct DisasContext {
|
||||||
DisasContextBase base;
|
DisasContextBase base;
|
||||||
/* pc_succ_insn points to the instruction following base.pc_next */
|
/* pc_succ_insn points to the instruction following base.pc_next */
|
||||||
target_ulong pc_succ_insn;
|
target_ulong pc_succ_insn;
|
||||||
target_ulong priv_ver;
|
target_ulong priv_ver;
|
||||||
bool virt_enabled;
|
target_ulong misa;
|
||||||
uint32_t opcode;
|
uint32_t opcode;
|
||||||
uint32_t mstatus_fs;
|
uint32_t mstatus_fs;
|
||||||
target_ulong misa;
|
|
||||||
uint32_t mem_idx;
|
uint32_t mem_idx;
|
||||||
/* Remember the rounding mode encoded in the previous fp instruction,
|
/* Remember the rounding mode encoded in the previous fp instruction,
|
||||||
which we have already installed into env->fp_status. Or -1 for
|
which we have already installed into env->fp_status. Or -1 for
|
||||||
|
@ -55,6 +65,8 @@ typedef struct DisasContext {
|
||||||
to any system register, which includes CSR_FRM, so we do not have
|
to any system register, which includes CSR_FRM, so we do not have
|
||||||
to reset this known value. */
|
to reset this known value. */
|
||||||
int frm;
|
int frm;
|
||||||
|
bool w;
|
||||||
|
bool virt_enabled;
|
||||||
bool ext_ifencei;
|
bool ext_ifencei;
|
||||||
bool hlsx;
|
bool hlsx;
|
||||||
/* vector extension */
|
/* vector extension */
|
||||||
|
@ -64,7 +76,11 @@ typedef struct DisasContext {
|
||||||
uint16_t vlen;
|
uint16_t vlen;
|
||||||
uint16_t mlen;
|
uint16_t mlen;
|
||||||
bool vl_eq_vlmax;
|
bool vl_eq_vlmax;
|
||||||
|
uint8_t ntemp;
|
||||||
CPUState *cs;
|
CPUState *cs;
|
||||||
|
TCGv zero;
|
||||||
|
/* Space for 3 operands plus 1 extra for address computation. */
|
||||||
|
TCGv temp[4];
|
||||||
} DisasContext;
|
} DisasContext;
|
||||||
|
|
||||||
static inline bool has_ext(DisasContext *ctx, uint32_t ext)
|
static inline bool has_ext(DisasContext *ctx, uint32_t ext)
|
||||||
|
@ -172,27 +188,64 @@ static void gen_goto_tb(DisasContext *ctx, int n, target_ulong dest)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Wrapper for getting reg values - need to check of reg is zero since
|
/*
|
||||||
* cpu_gpr[0] is not actually allocated
|
* Wrappers for getting reg values.
|
||||||
|
*
|
||||||
|
* The $zero register does not have cpu_gpr[0] allocated -- we supply the
|
||||||
|
* constant zero as a source, and an uninitialized sink as destination.
|
||||||
|
*
|
||||||
|
* Further, we may provide an extension for word operations.
|
||||||
*/
|
*/
|
||||||
static void gen_get_gpr(DisasContext *ctx, TCGv t, int reg_num)
|
static TCGv temp_new(DisasContext *ctx)
|
||||||
{
|
{
|
||||||
if (reg_num == 0) {
|
assert(ctx->ntemp < ARRAY_SIZE(ctx->temp));
|
||||||
tcg_gen_movi_tl(t, 0);
|
return ctx->temp[ctx->ntemp++] = tcg_temp_new();
|
||||||
} else {
|
|
||||||
tcg_gen_mov_tl(t, cpu_gpr[reg_num]);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Wrapper for setting reg values - need to check of reg is zero since
|
static TCGv get_gpr(DisasContext *ctx, int reg_num, DisasExtend ext)
|
||||||
* cpu_gpr[0] is not actually allocated. this is more for safety purposes,
|
|
||||||
* since we usually avoid calling the OP_TYPE_gen function if we see a write to
|
|
||||||
* $zero
|
|
||||||
*/
|
|
||||||
static void gen_set_gpr(DisasContext *ctx, int reg_num_dst, TCGv t)
|
|
||||||
{
|
{
|
||||||
if (reg_num_dst != 0) {
|
TCGv t;
|
||||||
tcg_gen_mov_tl(cpu_gpr[reg_num_dst], t);
|
|
||||||
|
if (reg_num == 0) {
|
||||||
|
return ctx->zero;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (ctx->w ? ext : EXT_NONE) {
|
||||||
|
case EXT_NONE:
|
||||||
|
return cpu_gpr[reg_num];
|
||||||
|
case EXT_SIGN:
|
||||||
|
t = temp_new(ctx);
|
||||||
|
tcg_gen_ext32s_tl(t, cpu_gpr[reg_num]);
|
||||||
|
return t;
|
||||||
|
case EXT_ZERO:
|
||||||
|
t = temp_new(ctx);
|
||||||
|
tcg_gen_ext32u_tl(t, cpu_gpr[reg_num]);
|
||||||
|
return t;
|
||||||
|
}
|
||||||
|
g_assert_not_reached();
|
||||||
|
}
|
||||||
|
|
||||||
|
static void gen_get_gpr(DisasContext *ctx, TCGv t, int reg_num)
|
||||||
|
{
|
||||||
|
tcg_gen_mov_tl(t, get_gpr(ctx, reg_num, EXT_NONE));
|
||||||
|
}
|
||||||
|
|
||||||
|
static TCGv __attribute__((unused)) dest_gpr(DisasContext *ctx, int reg_num)
|
||||||
|
{
|
||||||
|
if (reg_num == 0 || ctx->w) {
|
||||||
|
return temp_new(ctx);
|
||||||
|
}
|
||||||
|
return cpu_gpr[reg_num];
|
||||||
|
}
|
||||||
|
|
||||||
|
static void gen_set_gpr(DisasContext *ctx, int reg_num, TCGv t)
|
||||||
|
{
|
||||||
|
if (reg_num != 0) {
|
||||||
|
if (ctx->w) {
|
||||||
|
tcg_gen_ext32s_tl(cpu_gpr[reg_num], t);
|
||||||
|
} else {
|
||||||
|
tcg_gen_mov_tl(cpu_gpr[reg_num], t);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -940,6 +993,11 @@ static void riscv_tr_init_disas_context(DisasContextBase *dcbase, CPUState *cs)
|
||||||
ctx->mlen = 1 << (ctx->sew + 3 - ctx->lmul);
|
ctx->mlen = 1 << (ctx->sew + 3 - ctx->lmul);
|
||||||
ctx->vl_eq_vlmax = FIELD_EX32(tb_flags, TB_FLAGS, VL_EQ_VLMAX);
|
ctx->vl_eq_vlmax = FIELD_EX32(tb_flags, TB_FLAGS, VL_EQ_VLMAX);
|
||||||
ctx->cs = cs;
|
ctx->cs = cs;
|
||||||
|
ctx->w = false;
|
||||||
|
ctx->ntemp = 0;
|
||||||
|
memset(ctx->temp, 0, sizeof(ctx->temp));
|
||||||
|
|
||||||
|
ctx->zero = tcg_constant_tl(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void riscv_tr_tb_start(DisasContextBase *db, CPUState *cpu)
|
static void riscv_tr_tb_start(DisasContextBase *db, CPUState *cpu)
|
||||||
|
@ -961,6 +1019,13 @@ static void riscv_tr_translate_insn(DisasContextBase *dcbase, CPUState *cpu)
|
||||||
|
|
||||||
decode_opc(env, ctx, opcode16);
|
decode_opc(env, ctx, opcode16);
|
||||||
ctx->base.pc_next = ctx->pc_succ_insn;
|
ctx->base.pc_next = ctx->pc_succ_insn;
|
||||||
|
ctx->w = false;
|
||||||
|
|
||||||
|
for (int i = ctx->ntemp - 1; i >= 0; --i) {
|
||||||
|
tcg_temp_free(ctx->temp[i]);
|
||||||
|
ctx->temp[i] = NULL;
|
||||||
|
}
|
||||||
|
ctx->ntemp = 0;
|
||||||
|
|
||||||
if (ctx->base.is_jmp == DISAS_NEXT) {
|
if (ctx->base.is_jmp == DISAS_NEXT) {
|
||||||
target_ulong page_start;
|
target_ulong page_start;
|
||||||
|
|
Loading…
Reference in New Issue