Signed-off-by: Bastian Koppelmann <kbastian@mail.uni-paderborn.de>
---
target/tricore/translate.c | 117 +++++++++++++++++++++++--------------
1 file changed, 74 insertions(+), 43 deletions(-)
diff --git a/target/tricore/translate.c b/target/tricore/translate.c
index b4e332777a..f3b297639a 100644
--- a/target/tricore/translate.c
+++ b/target/tricore/translate.c
@@ -8763,7 +8763,7 @@ static void decode_32Bit_opc(DisasContext *ctx)
}
}
-static void decode_opc(DisasContext *ctx, int *is_branch)
+static void decode_opc(DisasContext *ctx)
{
/* 16-Bit Instruction */
if ((ctx->opcode & 0x1) == 0) {
@@ -8776,56 +8776,87 @@ static void decode_opc(DisasContext *ctx, int *is_branch)
}
}
-void gen_intermediate_code(CPUState *cs, TranslationBlock *tb, int max_insns)
+static void tricore_tr_init_disas_context(DisasContextBase *dcbase,
+ CPUState *cs)
{
+ DisasContext *ctx = container_of(dcbase, DisasContext, base);
CPUTriCoreState *env = cs->env_ptr;
- DisasContext ctx;
- target_ulong pc_start;
- int num_insns = 0;
-
- pc_start = tb->pc;
- ctx.base.pc_next = pc_start;
- ctx.base.tb = tb;
- ctx.base.singlestep_enabled = cs->singlestep_enabled;
- ctx.base.is_jmp = DISAS_NEXT;
- ctx.mem_idx = cpu_mmu_index(env, false);
- ctx.env = env;
-
- tcg_clear_temp_count();
- gen_tb_start(tb);
- while (ctx.base.is_jmp == DISAS_NEXT) {
- tcg_gen_insn_start(ctx.base.pc_next);
- num_insns++;
-
- ctx.opcode = cpu_ldl_code(env, ctx.base.pc_next);
- decode_opc(&ctx, 0);
-
- if (num_insns >= max_insns || tcg_op_buf_full()) {
- gen_save_pc(ctx.pc_succ_insn);
- tcg_gen_exit_tb(NULL, 0);
- break;
+ ctx->base.pc_next = ctx->base.pc_first;
+ ctx->mem_idx = cpu_mmu_index(env, false);
+}
+
+static void tricore_tr_tb_start(DisasContextBase *db, CPUState *cpu)
+{
+}
+
+static void tricore_tr_insn_start(DisasContextBase *dcbase, CPUState *cpu)
+{
+ DisasContext *ctx = container_of(dcbase, DisasContext, base);
+
+ tcg_gen_insn_start(ctx->base.pc_next);
+}
+
+static bool tricore_tr_breakpoint_check(DisasContextBase *dcbase, CPUState *cpu,
+ const CPUBreakpoint *bp)
+{
+ return true;
+}
+
+static void tricore_tr_translate_insn(DisasContextBase *dcbase, CPUState *cpu)
+{
+ DisasContext *ctx = container_of(dcbase, DisasContext, base);
+ CPUTriCoreState *env = cpu->env_ptr;
+
+ ctx->opcode = cpu_ldl_code(env, ctx->base.pc_next);
+ decode_opc(ctx);
+ ctx->base.pc_next = ctx->pc_succ_insn;
+
+ if (ctx->base.is_jmp == DISAS_NEXT) {
+ target_ulong page_start;
+
+ page_start = ctx->base.pc_first & TARGET_PAGE_MASK;
+ if (ctx->base.pc_next - page_start >= TARGET_PAGE_SIZE) {
+ ctx->base.is_jmp = DISAS_TOO_MANY;
}
- ctx.base.pc_next = ctx.pc_succ_insn;
}
+}
- gen_tb_end(tb, num_insns);
- tb->size = ctx.base.pc_next - pc_start;
- tb->icount = num_insns;
+static void tricore_tr_tb_stop(DisasContextBase *dcbase, CPUState *cpu)
+{
+ DisasContext *ctx = container_of(dcbase, DisasContext, base);
- if (tcg_check_temp_count()) {
- printf("LEAK at %08x\n", env->PC);
+ switch (ctx->base.is_jmp) {
+ case DISAS_TOO_MANY:
+ gen_goto_tb(ctx, 0, ctx->base.pc_next);
+ break;
+ case DISAS_NORETURN:
+ break;
+ default:
+ g_assert_not_reached();
}
+}
-#ifdef DEBUG_DISAS
- if (qemu_loglevel_mask(CPU_LOG_TB_IN_ASM)
- && qemu_log_in_addr_range(pc_start)) {
- qemu_log_lock();
- qemu_log("IN: %s\n", lookup_symbol(pc_start));
- log_target_disas(cs, pc_start, ctx.base.pc_next - pc_start);
- qemu_log("\n");
- qemu_log_unlock();
- }
-#endif
+static void tricore_tr_disas_log(const DisasContextBase *dcbase, CPUState *cpu)
+{
+ qemu_log("IN: %s\n", lookup_symbol(dcbase->pc_first));
+ log_target_disas(cpu, dcbase->pc_first, dcbase->tb->size);
+}
+
+static const TranslatorOps tricore_tr_ops = {
+ .init_disas_context = tricore_tr_init_disas_context,
+ .tb_start = tricore_tr_tb_start,
+ .insn_start = tricore_tr_insn_start,
+ .breakpoint_check = tricore_tr_breakpoint_check,
+ .translate_insn = tricore_tr_translate_insn,
+ .tb_stop = tricore_tr_tb_stop,
+ .disas_log = tricore_tr_disas_log,
+};
+
+
+void gen_intermediate_code(CPUState *cs, TranslationBlock *tb, int max_insns)
+{
+ DisasContext ctx;
+ translator_loop(&tricore_tr_ops, &ctx.base, cs, tb, max_insns);
}
void
--
2.22.0
On 6/17/19 7:35 AM, Bastian Koppelmann wrote: > +static void tricore_tr_init_disas_context(DisasContextBase *dcbase, > + CPUState *cs) > { > + DisasContext *ctx = container_of(dcbase, DisasContext, base); > CPUTriCoreState *env = cs->env_ptr; > + ctx->base.pc_next = ctx->base.pc_first; This is already done in generic code. I don't see an initialization of hflags & saved_hflags? Although I don't see that either before or afterward... > +static bool tricore_tr_breakpoint_check(DisasContextBase *dcbase, CPUState *cpu, > + const CPUBreakpoint *bp) > +{ > + return true; > +} Not supporting breakpoints, I think it's better to return false here. Although it's not difficult -- just raise EXCP_DEBUG as an exception. It'd be nice to follow up and fix this afterward. > +static void tricore_tr_translate_insn(DisasContextBase *dcbase, CPUState *cpu) > +{ > + DisasContext *ctx = container_of(dcbase, DisasContext, base); > + CPUTriCoreState *env = cpu->env_ptr; > + > + ctx->opcode = cpu_ldl_code(env, ctx->base.pc_next); > + decode_opc(ctx); > + ctx->base.pc_next = ctx->pc_succ_insn; > + > + if (ctx->base.is_jmp == DISAS_NEXT) { > + target_ulong page_start; > + > + page_start = ctx->base.pc_first & TARGET_PAGE_MASK; > + if (ctx->base.pc_next - page_start >= TARGET_PAGE_SIZE) { > + ctx->base.is_jmp = DISAS_TOO_MANY; > } This isn't perfect as an ending, but you didn't seem to have one at all before, so I guess improvements can come incrementally afterward. Have a look at the end of thumb_tr_translate_insn & insn_crosses_page to see how to handle this properly. r~
On 6/17/19 6:45 PM, Richard Henderson wrote: > On 6/17/19 7:35 AM, Bastian Koppelmann wrote: >> +static void tricore_tr_init_disas_context(DisasContextBase *dcbase, >> + CPUState *cs) >> { >> + DisasContext *ctx = container_of(dcbase, DisasContext, base); >> CPUTriCoreState *env = cs->env_ptr; >> + ctx->base.pc_next = ctx->base.pc_first; > This is already done in generic code. > > I don't see an initialization of hflags & saved_hflags? > Although I don't see that either before or afterward... Yes, I mentioned this problem in David's patch (https://lists.gnu.org/archive/html/qemu-devel/2019-06/msg01058.html). If he doesn't fix it, I will in a follow-up patch. > >> +static bool tricore_tr_breakpoint_check(DisasContextBase *dcbase, CPUState *cpu, >> + const CPUBreakpoint *bp) >> +{ >> + return true; >> +} > Not supporting breakpoints, I think it's better to return false here. > > Although it's not difficult -- just raise EXCP_DEBUG as an exception. > It'd be nice to follow up and fix this afterward. Yes, I will do that. > >> +static void tricore_tr_translate_insn(DisasContextBase *dcbase, CPUState *cpu) >> +{ >> + DisasContext *ctx = container_of(dcbase, DisasContext, base); >> + CPUTriCoreState *env = cpu->env_ptr; >> + >> + ctx->opcode = cpu_ldl_code(env, ctx->base.pc_next); >> + decode_opc(ctx); >> + ctx->base.pc_next = ctx->pc_succ_insn; >> + >> + if (ctx->base.is_jmp == DISAS_NEXT) { >> + target_ulong page_start; >> + >> + page_start = ctx->base.pc_first & TARGET_PAGE_MASK; >> + if (ctx->base.pc_next - page_start >= TARGET_PAGE_SIZE) { >> + ctx->base.is_jmp = DISAS_TOO_MANY; >> } > This isn't perfect as an ending, but you didn't seem to have one at all before, > so I guess improvements can come incrementally afterward. > > Have a look at the end of thumb_tr_translate_insn & insn_crosses_page to see > how to handle this properly. I copied it more or less from target/riscv. I guess that needs fixing as well :) Cheers, Bastian
On 6/18/19 5:06 AM, Bastian Koppelmann wrote: >> Have a look at the end of thumb_tr_translate_insn & insn_crosses_page to see >> how to handle this properly. > > I copied it more or less from target/riscv. I guess that needs fixing as well :) Yes, I noticed the riscv problem during review of the plugin api. :-) r~
On 6/17/19 7:35 AM, Bastian Koppelmann wrote: > +static void tricore_tr_translate_insn(DisasContextBase *dcbase, CPUState *cpu) > +{ > + DisasContext *ctx = container_of(dcbase, DisasContext, base); > + CPUTriCoreState *env = cpu->env_ptr; > + > + ctx->opcode = cpu_ldl_code(env, ctx->base.pc_next); > + decode_opc(ctx); I'll note that there's an existing bug here, always reading 4 bytes with ldl. You need to load 2 bytes, look at the low bit as in decode_opc. If 16-bit, read nothing more; if 32-bit, read 2 more bytes. r~
© 2016 - 2025 Red Hat, Inc.