mirror of
https://github.com/yuzu-emu/unicorn.git
synced 2025-01-20 18:41:14 +00:00
target/riscv: Remove the hyp load and store functions
Remove the special Virtulisation load and store functions and just use the standard tcg tcg_gen_qemu_ld_tl() and tcg_gen_qemu_st_tl() functions instead. As part of this change we ensure we still run an access check to make sure we can perform the operations. Backports 743077b35b1ed88ed243daefafe9403d88a958f6
This commit is contained in:
parent
bd81c057ed
commit
4762dcda3c
|
@ -6260,8 +6260,6 @@ riscv_symbols = (
|
||||||
'helper_fsub_d',
|
'helper_fsub_d',
|
||||||
'helper_fsub_s',
|
'helper_fsub_s',
|
||||||
'helper_hyp_gvma_tlb_flush',
|
'helper_hyp_gvma_tlb_flush',
|
||||||
'helper_hyp_load',
|
|
||||||
'helper_hyp_store',
|
|
||||||
'helper_hyp_tlb_flush',
|
'helper_hyp_tlb_flush',
|
||||||
'helper_hyp_x_load',
|
'helper_hyp_x_load',
|
||||||
'helper_mret',
|
'helper_mret',
|
||||||
|
|
|
@ -3696,8 +3696,6 @@
|
||||||
#define helper_fsub_d helper_fsub_d_riscv32
|
#define helper_fsub_d helper_fsub_d_riscv32
|
||||||
#define helper_fsub_s helper_fsub_s_riscv32
|
#define helper_fsub_s helper_fsub_s_riscv32
|
||||||
#define helper_hyp_gvma_tlb_flush helper_hyp_gvma_tlb_flush_riscv32
|
#define helper_hyp_gvma_tlb_flush helper_hyp_gvma_tlb_flush_riscv32
|
||||||
#define helper_hyp_load helper_hyp_load_riscv32
|
|
||||||
#define helper_hyp_store helper_hyp_store_riscv32
|
|
||||||
#define helper_hyp_tlb_flush helper_hyp_tlb_flush_riscv32
|
#define helper_hyp_tlb_flush helper_hyp_tlb_flush_riscv32
|
||||||
#define helper_hyp_x_load helper_hyp_x_load_riscv32
|
#define helper_hyp_x_load helper_hyp_x_load_riscv32
|
||||||
#define helper_mret helper_mret_riscv32
|
#define helper_mret helper_mret_riscv32
|
||||||
|
|
|
@ -3696,8 +3696,6 @@
|
||||||
#define helper_fsub_d helper_fsub_d_riscv64
|
#define helper_fsub_d helper_fsub_d_riscv64
|
||||||
#define helper_fsub_s helper_fsub_s_riscv64
|
#define helper_fsub_s helper_fsub_s_riscv64
|
||||||
#define helper_hyp_gvma_tlb_flush helper_hyp_gvma_tlb_flush_riscv64
|
#define helper_hyp_gvma_tlb_flush helper_hyp_gvma_tlb_flush_riscv64
|
||||||
#define helper_hyp_load helper_hyp_load_riscv64
|
|
||||||
#define helper_hyp_store helper_hyp_store_riscv64
|
|
||||||
#define helper_hyp_tlb_flush helper_hyp_tlb_flush_riscv64
|
#define helper_hyp_tlb_flush helper_hyp_tlb_flush_riscv64
|
||||||
#define helper_hyp_x_load helper_hyp_x_load_riscv64
|
#define helper_hyp_x_load helper_hyp_x_load_riscv64
|
||||||
#define helper_mret helper_mret_riscv64
|
#define helper_mret helper_mret_riscv64
|
||||||
|
|
|
@ -377,6 +377,8 @@ FIELD(TB_FLAGS, VL_EQ_VLMAX, 2, 1)
|
||||||
FIELD(TB_FLAGS, LMUL, 3, 2)
|
FIELD(TB_FLAGS, LMUL, 3, 2)
|
||||||
FIELD(TB_FLAGS, SEW, 5, 3)
|
FIELD(TB_FLAGS, SEW, 5, 3)
|
||||||
FIELD(TB_FLAGS, VILL, 8, 1)
|
FIELD(TB_FLAGS, VILL, 8, 1)
|
||||||
|
/* Is a Hypervisor instruction load/store allowed? */
|
||||||
|
FIELD(TB_FLAGS, HLSX, 9, 1)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* A simplification for VLMAX
|
* A simplification for VLMAX
|
||||||
|
@ -423,7 +425,17 @@ static inline void cpu_get_tb_cpu_state(CPURISCVState *env, target_ulong *pc,
|
||||||
if (riscv_cpu_fp_enabled(env)) {
|
if (riscv_cpu_fp_enabled(env)) {
|
||||||
flags |= env->mstatus & MSTATUS_FS;
|
flags |= env->mstatus & MSTATUS_FS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (riscv_has_ext(env, RVH)) {
|
||||||
|
if (env->priv == PRV_M ||
|
||||||
|
(env->priv == PRV_S && !riscv_cpu_virt_enabled(env)) ||
|
||||||
|
(env->priv == PRV_U && !riscv_cpu_virt_enabled(env) &&
|
||||||
|
get_field(env->hstatus, HSTATUS_HU))) {
|
||||||
|
flags = FIELD_DP32(flags, TB_FLAGS, HLSX, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
*pflags = flags;
|
*pflags = flags;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -84,8 +84,6 @@ DEF_HELPER_1(tlb_flush, void, env)
|
||||||
#ifndef CONFIG_USER_ONLY
|
#ifndef CONFIG_USER_ONLY
|
||||||
DEF_HELPER_1(hyp_tlb_flush, void, env)
|
DEF_HELPER_1(hyp_tlb_flush, void, env)
|
||||||
DEF_HELPER_1(hyp_gvma_tlb_flush, void, env)
|
DEF_HELPER_1(hyp_gvma_tlb_flush, void, env)
|
||||||
DEF_HELPER_4(hyp_load, tl, env, tl, tl, tl)
|
|
||||||
DEF_HELPER_5(hyp_store, void, env, tl, tl, tl, tl)
|
|
||||||
DEF_HELPER_4(hyp_x_load, tl, env, tl, tl, tl)
|
DEF_HELPER_4(hyp_x_load, tl, env, tl, tl, tl)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -16,6 +16,18 @@
|
||||||
* this program. If not, see <http://www.gnu.org/licenses/>.
|
* this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#ifndef CONFIG_USER_ONLY
|
||||||
|
static void check_access(DisasContext *ctx) {
|
||||||
|
if (!ctx->hlsx) {
|
||||||
|
if (ctx->virt_enabled) {
|
||||||
|
generate_exception(ctx, RISCV_EXCP_VIRT_INSTRUCTION_FAULT);
|
||||||
|
} else {
|
||||||
|
generate_exception(ctx, RISCV_EXCP_ILLEGAL_INST);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
static bool trans_hlv_b(DisasContext *ctx, arg_hlv_b *a)
|
static bool trans_hlv_b(DisasContext *ctx, arg_hlv_b *a)
|
||||||
{
|
{
|
||||||
REQUIRE_EXT(ctx, RVH);
|
REQUIRE_EXT(ctx, RVH);
|
||||||
|
@ -23,20 +35,15 @@ static bool trans_hlv_b(DisasContext *ctx, arg_hlv_b *a)
|
||||||
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
||||||
TCGv t0 = tcg_temp_new(tcg_ctx);
|
TCGv t0 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv t1 = tcg_temp_new(tcg_ctx);
|
TCGv t1 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv mem_idx = tcg_temp_new(tcg_ctx);
|
|
||||||
TCGv memop = tcg_temp_new(tcg_ctx);
|
check_access(ctx);
|
||||||
|
|
||||||
gen_get_gpr(ctx, t0, a->rs1);
|
gen_get_gpr(ctx, t0, a->rs1);
|
||||||
tcg_gen_movi_tl(tcg_ctx, mem_idx, ctx->mem_idx);
|
tcg_gen_qemu_ld_tl(ctx->uc, t1, t0, ctx->mem_idx | TB_FLAGS_PRIV_HYP_ACCESS_MASK, MO_SB);
|
||||||
tcg_gen_movi_tl(tcg_ctx, memop, MO_SB);
|
|
||||||
|
|
||||||
gen_helper_hyp_load(tcg_ctx, t1, tcg_ctx->cpu_env, t0, mem_idx, memop);
|
|
||||||
gen_set_gpr(ctx, a->rd, t1);
|
gen_set_gpr(ctx, a->rd, t1);
|
||||||
|
|
||||||
tcg_temp_free(tcg_ctx, t0);
|
tcg_temp_free(tcg_ctx, t0);
|
||||||
tcg_temp_free(tcg_ctx, t1);
|
tcg_temp_free(tcg_ctx, t1);
|
||||||
tcg_temp_free(tcg_ctx, mem_idx);
|
|
||||||
tcg_temp_free(tcg_ctx, memop);
|
|
||||||
return true;
|
return true;
|
||||||
#else
|
#else
|
||||||
return false;
|
return false;
|
||||||
|
@ -50,20 +57,15 @@ static bool trans_hlv_h(DisasContext *ctx, arg_hlv_h *a)
|
||||||
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
||||||
TCGv t0 = tcg_temp_new(tcg_ctx);
|
TCGv t0 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv t1 = tcg_temp_new(tcg_ctx);
|
TCGv t1 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv mem_idx = tcg_temp_new(tcg_ctx);
|
|
||||||
TCGv memop = tcg_temp_new(tcg_ctx);
|
check_access(ctx);
|
||||||
|
|
||||||
gen_get_gpr(ctx, t0, a->rs1);
|
gen_get_gpr(ctx, t0, a->rs1);
|
||||||
tcg_gen_movi_tl(tcg_ctx, mem_idx, ctx->mem_idx);
|
tcg_gen_qemu_ld_tl(ctx->uc, t1, t0, ctx->mem_idx | TB_FLAGS_PRIV_HYP_ACCESS_MASK, MO_TESW);
|
||||||
tcg_gen_movi_tl(tcg_ctx, memop, MO_TESW);
|
|
||||||
|
|
||||||
gen_helper_hyp_load(tcg_ctx, t1, tcg_ctx->cpu_env, t0, mem_idx, memop);
|
|
||||||
gen_set_gpr(ctx, a->rd, t1);
|
gen_set_gpr(ctx, a->rd, t1);
|
||||||
|
|
||||||
tcg_temp_free(tcg_ctx, t0);
|
tcg_temp_free(tcg_ctx, t0);
|
||||||
tcg_temp_free(tcg_ctx, t1);
|
tcg_temp_free(tcg_ctx, t1);
|
||||||
tcg_temp_free(tcg_ctx, mem_idx);
|
|
||||||
tcg_temp_free(tcg_ctx, memop);
|
|
||||||
return true;
|
return true;
|
||||||
#else
|
#else
|
||||||
return false;
|
return false;
|
||||||
|
@ -77,20 +79,15 @@ static bool trans_hlv_w(DisasContext *ctx, arg_hlv_w *a)
|
||||||
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
||||||
TCGv t0 = tcg_temp_new(tcg_ctx);
|
TCGv t0 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv t1 = tcg_temp_new(tcg_ctx);
|
TCGv t1 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv mem_idx = tcg_temp_new(tcg_ctx);
|
|
||||||
TCGv memop = tcg_temp_new(tcg_ctx);
|
check_access(ctx);
|
||||||
|
|
||||||
gen_get_gpr(ctx, t0, a->rs1);
|
gen_get_gpr(ctx, t0, a->rs1);
|
||||||
tcg_gen_movi_tl(tcg_ctx, mem_idx, ctx->mem_idx);
|
tcg_gen_qemu_ld_tl(ctx->uc, t1, t0, ctx->mem_idx | TB_FLAGS_PRIV_HYP_ACCESS_MASK, MO_TESL);
|
||||||
tcg_gen_movi_tl(tcg_ctx, memop, MO_TESL);
|
|
||||||
|
|
||||||
gen_helper_hyp_load(tcg_ctx, t1, tcg_ctx->cpu_env, t0, mem_idx, memop);
|
|
||||||
gen_set_gpr(ctx, a->rd, t1);
|
gen_set_gpr(ctx, a->rd, t1);
|
||||||
|
|
||||||
tcg_temp_free(tcg_ctx, t0);
|
tcg_temp_free(tcg_ctx, t0);
|
||||||
tcg_temp_free(tcg_ctx, t1);
|
tcg_temp_free(tcg_ctx, t1);
|
||||||
tcg_temp_free(tcg_ctx, mem_idx);
|
|
||||||
tcg_temp_free(tcg_ctx, memop);
|
|
||||||
return true;
|
return true;
|
||||||
#else
|
#else
|
||||||
return false;
|
return false;
|
||||||
|
@ -104,20 +101,15 @@ static bool trans_hlv_bu(DisasContext *ctx, arg_hlv_bu *a)
|
||||||
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
||||||
TCGv t0 = tcg_temp_new(tcg_ctx);
|
TCGv t0 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv t1 = tcg_temp_new(tcg_ctx);
|
TCGv t1 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv mem_idx = tcg_temp_new(tcg_ctx);
|
|
||||||
TCGv memop = tcg_temp_new(tcg_ctx);
|
check_access(ctx);
|
||||||
|
|
||||||
gen_get_gpr(ctx, t0, a->rs1);
|
gen_get_gpr(ctx, t0, a->rs1);
|
||||||
tcg_gen_movi_tl(tcg_ctx, mem_idx, ctx->mem_idx);
|
tcg_gen_qemu_ld_tl(ctx->uc, t1, t0, ctx->mem_idx | TB_FLAGS_PRIV_HYP_ACCESS_MASK, MO_UB);
|
||||||
tcg_gen_movi_tl(tcg_ctx, memop, MO_UB);
|
|
||||||
|
|
||||||
gen_helper_hyp_load(tcg_ctx, t1, tcg_ctx->cpu_env, t0, mem_idx, memop);
|
|
||||||
gen_set_gpr(ctx, a->rd, t1);
|
gen_set_gpr(ctx, a->rd, t1);
|
||||||
|
|
||||||
tcg_temp_free(tcg_ctx, t0);
|
tcg_temp_free(tcg_ctx, t0);
|
||||||
tcg_temp_free(tcg_ctx, t1);
|
tcg_temp_free(tcg_ctx, t1);
|
||||||
tcg_temp_free(tcg_ctx, mem_idx);
|
|
||||||
tcg_temp_free(tcg_ctx, memop);
|
|
||||||
return true;
|
return true;
|
||||||
#else
|
#else
|
||||||
return false;
|
return false;
|
||||||
|
@ -131,20 +123,15 @@ static bool trans_hlv_hu(DisasContext *ctx, arg_hlv_hu *a)
|
||||||
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
||||||
TCGv t0 = tcg_temp_new(tcg_ctx);
|
TCGv t0 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv t1 = tcg_temp_new(tcg_ctx);
|
TCGv t1 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv mem_idx = tcg_temp_new(tcg_ctx);
|
|
||||||
TCGv memop = tcg_temp_new(tcg_ctx);
|
check_access(ctx);
|
||||||
|
|
||||||
gen_get_gpr(ctx, t0, a->rs1);
|
gen_get_gpr(ctx, t0, a->rs1);
|
||||||
tcg_gen_movi_tl(tcg_ctx, mem_idx, ctx->mem_idx);
|
tcg_gen_qemu_ld_tl(ctx->uc, t1, t0, ctx->mem_idx | TB_FLAGS_PRIV_HYP_ACCESS_MASK, MO_TEUW);
|
||||||
tcg_gen_movi_tl(tcg_ctx, memop, MO_TEUW);
|
|
||||||
|
|
||||||
gen_helper_hyp_load(tcg_ctx, t1, tcg_ctx->cpu_env, t0, mem_idx, memop);
|
|
||||||
gen_set_gpr(ctx, a->rd, t1);
|
gen_set_gpr(ctx, a->rd, t1);
|
||||||
|
|
||||||
tcg_temp_free(tcg_ctx, t0);
|
tcg_temp_free(tcg_ctx, t0);
|
||||||
tcg_temp_free(tcg_ctx, t1);
|
tcg_temp_free(tcg_ctx, t1);
|
||||||
tcg_temp_free(tcg_ctx, mem_idx);
|
|
||||||
tcg_temp_free(tcg_ctx, memop);
|
|
||||||
return true;
|
return true;
|
||||||
#else
|
#else
|
||||||
return false;
|
return false;
|
||||||
|
@ -158,20 +145,15 @@ static bool trans_hsv_b(DisasContext *ctx, arg_hsv_b *a)
|
||||||
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
||||||
TCGv t0 = tcg_temp_new(tcg_ctx);
|
TCGv t0 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv dat = tcg_temp_new(tcg_ctx);
|
TCGv dat = tcg_temp_new(tcg_ctx);
|
||||||
TCGv mem_idx = tcg_temp_new(tcg_ctx);
|
|
||||||
TCGv memop = tcg_temp_new(tcg_ctx);
|
check_access(ctx);
|
||||||
|
|
||||||
gen_get_gpr(ctx, t0, a->rs1);
|
gen_get_gpr(ctx, t0, a->rs1);
|
||||||
gen_get_gpr(ctx, dat, a->rs2);
|
gen_get_gpr(ctx, dat, a->rs2);
|
||||||
tcg_gen_movi_tl(tcg_ctx, mem_idx, ctx->mem_idx);
|
tcg_gen_qemu_st_tl(ctx->uc, dat, t0, ctx->mem_idx | TB_FLAGS_PRIV_HYP_ACCESS_MASK, MO_SB);
|
||||||
tcg_gen_movi_tl(tcg_ctx, memop, MO_SB);
|
|
||||||
|
|
||||||
gen_helper_hyp_store(tcg_ctx, tcg_ctx->cpu_env, t0, dat, mem_idx, memop);
|
|
||||||
|
|
||||||
tcg_temp_free(tcg_ctx, t0);
|
tcg_temp_free(tcg_ctx, t0);
|
||||||
tcg_temp_free(tcg_ctx, dat);
|
tcg_temp_free(tcg_ctx, dat);
|
||||||
tcg_temp_free(tcg_ctx, mem_idx);
|
|
||||||
tcg_temp_free(tcg_ctx, memop);
|
|
||||||
return true;
|
return true;
|
||||||
#else
|
#else
|
||||||
return false;
|
return false;
|
||||||
|
@ -185,20 +167,15 @@ static bool trans_hsv_h(DisasContext *ctx, arg_hsv_h *a)
|
||||||
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
||||||
TCGv t0 = tcg_temp_new(tcg_ctx);
|
TCGv t0 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv dat = tcg_temp_new(tcg_ctx);
|
TCGv dat = tcg_temp_new(tcg_ctx);
|
||||||
TCGv mem_idx = tcg_temp_new(tcg_ctx);
|
|
||||||
TCGv memop = tcg_temp_new(tcg_ctx);
|
check_access(ctx);
|
||||||
|
|
||||||
gen_get_gpr(ctx, t0, a->rs1);
|
gen_get_gpr(ctx, t0, a->rs1);
|
||||||
gen_get_gpr(ctx, dat, a->rs2);
|
gen_get_gpr(ctx, dat, a->rs2);
|
||||||
tcg_gen_movi_tl(tcg_ctx, mem_idx, ctx->mem_idx);
|
tcg_gen_qemu_st_tl(ctx->uc, dat, t0, ctx->mem_idx | TB_FLAGS_PRIV_HYP_ACCESS_MASK, MO_TESW);
|
||||||
tcg_gen_movi_tl(tcg_ctx, memop, MO_TESW);
|
|
||||||
|
|
||||||
gen_helper_hyp_store(tcg_ctx, tcg_ctx->cpu_env, t0, dat, mem_idx, memop);
|
|
||||||
|
|
||||||
tcg_temp_free(tcg_ctx, t0);
|
tcg_temp_free(tcg_ctx, t0);
|
||||||
tcg_temp_free(tcg_ctx, dat);
|
tcg_temp_free(tcg_ctx, dat);
|
||||||
tcg_temp_free(tcg_ctx, mem_idx);
|
|
||||||
tcg_temp_free(tcg_ctx, memop);
|
|
||||||
return true;
|
return true;
|
||||||
#else
|
#else
|
||||||
return false;
|
return false;
|
||||||
|
@ -212,20 +189,15 @@ static bool trans_hsv_w(DisasContext *ctx, arg_hsv_w *a)
|
||||||
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
||||||
TCGv t0 = tcg_temp_new(tcg_ctx);
|
TCGv t0 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv dat = tcg_temp_new(tcg_ctx);
|
TCGv dat = tcg_temp_new(tcg_ctx);
|
||||||
TCGv mem_idx = tcg_temp_new(tcg_ctx);
|
|
||||||
TCGv memop = tcg_temp_new(tcg_ctx);
|
check_access(ctx);
|
||||||
|
|
||||||
gen_get_gpr(ctx, t0, a->rs1);
|
gen_get_gpr(ctx, t0, a->rs1);
|
||||||
gen_get_gpr(ctx, dat, a->rs2);
|
gen_get_gpr(ctx, dat, a->rs2);
|
||||||
tcg_gen_movi_tl(tcg_ctx, mem_idx, ctx->mem_idx);
|
tcg_gen_qemu_st_tl(ctx->uc, dat, t0, ctx->mem_idx | TB_FLAGS_PRIV_HYP_ACCESS_MASK, MO_TESL);
|
||||||
tcg_gen_movi_tl(tcg_ctx, memop, MO_TESL);
|
|
||||||
|
|
||||||
gen_helper_hyp_store(tcg_ctx, tcg_ctx->cpu_env, t0, dat, mem_idx, memop);
|
|
||||||
|
|
||||||
tcg_temp_free(tcg_ctx, t0);
|
tcg_temp_free(tcg_ctx, t0);
|
||||||
tcg_temp_free(tcg_ctx, dat);
|
tcg_temp_free(tcg_ctx, dat);
|
||||||
tcg_temp_free(tcg_ctx, mem_idx);
|
|
||||||
tcg_temp_free(tcg_ctx, memop);
|
|
||||||
return true;
|
return true;
|
||||||
#else
|
#else
|
||||||
return false;
|
return false;
|
||||||
|
@ -240,20 +212,15 @@ static bool trans_hlv_wu(DisasContext *ctx, arg_hlv_wu *a)
|
||||||
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
||||||
TCGv t0 = tcg_temp_new(tcg_ctx);
|
TCGv t0 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv t1 = tcg_temp_new(tcg_ctx);
|
TCGv t1 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv mem_idx = tcg_temp_new(tcg_ctx);
|
|
||||||
TCGv memop = tcg_temp_new(tcg_ctx);
|
check_access(ctx);
|
||||||
|
|
||||||
gen_get_gpr(ctx, t0, a->rs1);
|
gen_get_gpr(ctx, t0, a->rs1);
|
||||||
tcg_gen_movi_tl(tcg_ctx, mem_idx, ctx->mem_idx);
|
tcg_gen_qemu_ld_tl(ctx->uc, t1, t0, ctx->mem_idx | TB_FLAGS_PRIV_HYP_ACCESS_MASK, MO_TEUL);
|
||||||
tcg_gen_movi_tl(tcg_ctx, memop, MO_TEUL);
|
|
||||||
|
|
||||||
gen_helper_hyp_load(tcg_ctx, t1, tcg_ctx->cpu_env, t0, mem_idx, memop);
|
|
||||||
gen_set_gpr(ctx, a->rd, t1);
|
gen_set_gpr(ctx, a->rd, t1);
|
||||||
|
|
||||||
tcg_temp_free(tcg_ctx, t0);
|
tcg_temp_free(tcg_ctx, t0);
|
||||||
tcg_temp_free(tcg_ctx, t1);
|
tcg_temp_free(tcg_ctx, t1);
|
||||||
tcg_temp_free(tcg_ctx, mem_idx);
|
|
||||||
tcg_temp_free(tcg_ctx, memop);
|
|
||||||
return true;
|
return true;
|
||||||
#else
|
#else
|
||||||
return false;
|
return false;
|
||||||
|
@ -267,20 +234,15 @@ static bool trans_hlv_d(DisasContext *ctx, arg_hlv_d *a)
|
||||||
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
||||||
TCGv t0 = tcg_temp_new(tcg_ctx);
|
TCGv t0 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv t1 = tcg_temp_new(tcg_ctx);
|
TCGv t1 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv mem_idx = tcg_temp_new(tcg_ctx);
|
|
||||||
TCGv memop = tcg_temp_new(tcg_ctx);
|
check_access(ctx);
|
||||||
|
|
||||||
gen_get_gpr(ctx, t0, a->rs1);
|
gen_get_gpr(ctx, t0, a->rs1);
|
||||||
tcg_gen_movi_tl(tcg_ctx, mem_idx, ctx->mem_idx);
|
tcg_gen_qemu_ld_tl(ctx->uc, t1, t0, ctx->mem_idx | TB_FLAGS_PRIV_HYP_ACCESS_MASK, MO_TEQ);
|
||||||
tcg_gen_movi_tl(tcg_ctx, memop, MO_TEQ);
|
|
||||||
|
|
||||||
gen_helper_hyp_load(tcg_ctx, t1, tcg_ctx->cpu_env, t0, mem_idx, memop);
|
|
||||||
gen_set_gpr(ctx, a->rd, t1);
|
gen_set_gpr(ctx, a->rd, t1);
|
||||||
|
|
||||||
tcg_temp_free(tcg_ctx, t0);
|
tcg_temp_free(tcg_ctx, t0);
|
||||||
tcg_temp_free(tcg_ctx, t1);
|
tcg_temp_free(tcg_ctx, t1);
|
||||||
tcg_temp_free(tcg_ctx, mem_idx);
|
|
||||||
tcg_temp_free(tcg_ctx, memop);
|
|
||||||
return true;
|
return true;
|
||||||
#else
|
#else
|
||||||
return false;
|
return false;
|
||||||
|
@ -294,20 +256,15 @@ static bool trans_hsv_d(DisasContext *ctx, arg_hsv_d *a)
|
||||||
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
TCGContext *tcg_ctx = ctx->uc->tcg_ctx;
|
||||||
TCGv t0 = tcg_temp_new(tcg_ctx);
|
TCGv t0 = tcg_temp_new(tcg_ctx);
|
||||||
TCGv dat = tcg_temp_new(tcg_ctx);
|
TCGv dat = tcg_temp_new(tcg_ctx);
|
||||||
TCGv mem_idx = tcg_temp_new(tcg_ctx);
|
|
||||||
TCGv memop = tcg_temp_new(tcg_ctx);
|
check_access(ctx);
|
||||||
|
|
||||||
gen_get_gpr(ctx, t0, a->rs1);
|
gen_get_gpr(ctx, t0, a->rs1);
|
||||||
gen_get_gpr(ctx, dat, a->rs2);
|
gen_get_gpr(ctx, dat, a->rs2);
|
||||||
tcg_gen_movi_tl(tcg_ctx, mem_idx, ctx->mem_idx);
|
tcg_gen_qemu_st_tl(ctx->uc, dat, t0, ctx->mem_idx | TB_FLAGS_PRIV_HYP_ACCESS_MASK, MO_TEQ);
|
||||||
tcg_gen_movi_tl(tcg_ctx, memop, MO_TEQ);
|
|
||||||
|
|
||||||
gen_helper_hyp_store(tcg_ctx, tcg_ctx->cpu_env, t0, dat, mem_idx, memop);
|
|
||||||
|
|
||||||
tcg_temp_free(tcg_ctx, t0);
|
tcg_temp_free(tcg_ctx, t0);
|
||||||
tcg_temp_free(tcg_ctx, dat);
|
tcg_temp_free(tcg_ctx, dat);
|
||||||
tcg_temp_free(tcg_ctx, mem_idx);
|
|
||||||
tcg_temp_free(tcg_ctx, memop);
|
|
||||||
return true;
|
return true;
|
||||||
#else
|
#else
|
||||||
return false;
|
return false;
|
||||||
|
|
|
@ -227,92 +227,6 @@ void helper_hyp_gvma_tlb_flush(CPURISCVState *env)
|
||||||
helper_hyp_tlb_flush(env);
|
helper_hyp_tlb_flush(env);
|
||||||
}
|
}
|
||||||
|
|
||||||
target_ulong helper_hyp_load(CPURISCVState *env, target_ulong address,
|
|
||||||
target_ulong attrs, target_ulong memop)
|
|
||||||
{
|
|
||||||
if (env->priv == PRV_M ||
|
|
||||||
(env->priv == PRV_S && !riscv_cpu_virt_enabled(env)) ||
|
|
||||||
(env->priv == PRV_U && !riscv_cpu_virt_enabled(env) &&
|
|
||||||
get_field(env->hstatus, HSTATUS_HU))) {
|
|
||||||
target_ulong pte;
|
|
||||||
int mmu_idx = cpu_mmu_index(env, false) | TB_FLAGS_PRIV_HYP_ACCESS_MASK;
|
|
||||||
|
|
||||||
switch (memop) {
|
|
||||||
case MO_SB:
|
|
||||||
pte = cpu_ldsb_mmuidx_ra(env, address, mmu_idx, GETPC());
|
|
||||||
break;
|
|
||||||
case MO_UB:
|
|
||||||
pte = cpu_ldub_mmuidx_ra(env, address, mmu_idx, GETPC());
|
|
||||||
break;
|
|
||||||
case MO_TESW:
|
|
||||||
pte = cpu_ldsw_mmuidx_ra(env, address, mmu_idx, GETPC());
|
|
||||||
break;
|
|
||||||
case MO_TEUW:
|
|
||||||
pte = cpu_lduw_mmuidx_ra(env, address, mmu_idx, GETPC());
|
|
||||||
break;
|
|
||||||
case MO_TESL:
|
|
||||||
pte = cpu_ldl_mmuidx_ra(env, address, mmu_idx, GETPC());
|
|
||||||
break;
|
|
||||||
case MO_TEUL:
|
|
||||||
pte = cpu_ldl_mmuidx_ra(env, address, mmu_idx, GETPC());
|
|
||||||
break;
|
|
||||||
case MO_TEQ:
|
|
||||||
pte = cpu_ldq_mmuidx_ra(env, address, mmu_idx, GETPC());
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
g_assert_not_reached();
|
|
||||||
}
|
|
||||||
|
|
||||||
return pte;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (riscv_cpu_virt_enabled(env)) {
|
|
||||||
riscv_raise_exception(env, RISCV_EXCP_VIRT_INSTRUCTION_FAULT, GETPC());
|
|
||||||
} else {
|
|
||||||
riscv_raise_exception(env, RISCV_EXCP_ILLEGAL_INST, GETPC());
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void helper_hyp_store(CPURISCVState *env, target_ulong address,
|
|
||||||
target_ulong val, target_ulong attrs, target_ulong memop)
|
|
||||||
{
|
|
||||||
if (env->priv == PRV_M ||
|
|
||||||
(env->priv == PRV_S && !riscv_cpu_virt_enabled(env)) ||
|
|
||||||
(env->priv == PRV_U && !riscv_cpu_virt_enabled(env) &&
|
|
||||||
get_field(env->hstatus, HSTATUS_HU))) {
|
|
||||||
int mmu_idx = cpu_mmu_index(env, false) | TB_FLAGS_PRIV_HYP_ACCESS_MASK;
|
|
||||||
|
|
||||||
switch (memop) {
|
|
||||||
case MO_SB:
|
|
||||||
case MO_UB:
|
|
||||||
cpu_stb_mmuidx_ra(env, address, val, mmu_idx, GETPC());
|
|
||||||
break;
|
|
||||||
case MO_TESW:
|
|
||||||
case MO_TEUW:
|
|
||||||
cpu_stw_mmuidx_ra(env, address, val, mmu_idx, GETPC());
|
|
||||||
break;
|
|
||||||
case MO_TESL:
|
|
||||||
case MO_TEUL:
|
|
||||||
cpu_stl_mmuidx_ra(env, address, val, mmu_idx, GETPC());
|
|
||||||
break;
|
|
||||||
case MO_TEQ:
|
|
||||||
cpu_stq_mmuidx_ra(env, address, val, mmu_idx, GETPC());
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
g_assert_not_reached();
|
|
||||||
}
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (riscv_cpu_virt_enabled(env)) {
|
|
||||||
riscv_raise_exception(env, RISCV_EXCP_VIRT_INSTRUCTION_FAULT, GETPC());
|
|
||||||
} else {
|
|
||||||
riscv_raise_exception(env, RISCV_EXCP_ILLEGAL_INST, GETPC());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
target_ulong helper_hyp_x_load(CPURISCVState *env, target_ulong address,
|
target_ulong helper_hyp_x_load(CPURISCVState *env, target_ulong address,
|
||||||
target_ulong attrs, target_ulong memop)
|
target_ulong attrs, target_ulong memop)
|
||||||
{
|
{
|
||||||
|
|
|
@ -54,6 +54,7 @@ typedef struct DisasContext {
|
||||||
to reset this known value. */
|
to reset this known value. */
|
||||||
int frm;
|
int frm;
|
||||||
bool ext_ifencei;
|
bool ext_ifencei;
|
||||||
|
bool hlsx;
|
||||||
/* vector extension */
|
/* vector extension */
|
||||||
bool vill;
|
bool vill;
|
||||||
uint8_t lmul;
|
uint8_t lmul;
|
||||||
|
@ -838,6 +839,7 @@ static void riscv_tr_init_disas_context(DisasContextBase *dcbase, CPUState *cs)
|
||||||
ctx->frm = -1; /* unknown rounding mode */
|
ctx->frm = -1; /* unknown rounding mode */
|
||||||
ctx->ext_ifencei = cpu->cfg.ext_ifencei;
|
ctx->ext_ifencei = cpu->cfg.ext_ifencei;
|
||||||
ctx->vlen = cpu->cfg.vlen;
|
ctx->vlen = cpu->cfg.vlen;
|
||||||
|
ctx->hlsx = FIELD_EX32(tb_flags, TB_FLAGS, HLSX);
|
||||||
ctx->vill = FIELD_EX32(tb_flags, TB_FLAGS, VILL);
|
ctx->vill = FIELD_EX32(tb_flags, TB_FLAGS, VILL);
|
||||||
ctx->sew = FIELD_EX32(tb_flags, TB_FLAGS, SEW);
|
ctx->sew = FIELD_EX32(tb_flags, TB_FLAGS, SEW);
|
||||||
ctx->lmul = FIELD_EX32(tb_flags, TB_FLAGS, LMUL);
|
ctx->lmul = FIELD_EX32(tb_flags, TB_FLAGS, LMUL);
|
||||||
|
|
Loading…
Reference in a new issue