提交 24c32852 编写于 作者: R Richard Henderson

target/openrisc: Tidy ppc/npc implementation

The NPC SPR is really only supposed to be used for FPGA debugging.
It contains the same contents as PC, unless one plays games.  Follow
the or1ksim implementation in flushing delayed branch state when it
is changed.

The PPC SPR need not be updated every instruction, merely when we
exit the TB or attempt to read its contents.
Signed-off-by: NRichard Henderson <rth@twiddle.net>
上级 a8000cb4
...@@ -58,6 +58,7 @@ typedef struct OpenRISCCPUClass { ...@@ -58,6 +58,7 @@ typedef struct OpenRISCCPUClass {
} OpenRISCCPUClass; } OpenRISCCPUClass;
#define NB_MMU_MODES 3 #define NB_MMU_MODES 3
#define TARGET_INSN_START_EXTRA_WORDS 1
enum { enum {
MMU_NOMMU_IDX = 0, MMU_NOMMU_IDX = 0,
...@@ -273,7 +274,6 @@ typedef struct CPUOpenRISCTLBContext { ...@@ -273,7 +274,6 @@ typedef struct CPUOpenRISCTLBContext {
typedef struct CPUOpenRISCState { typedef struct CPUOpenRISCState {
target_ulong gpr[32]; /* General registers */ target_ulong gpr[32]; /* General registers */
target_ulong pc; /* Program counter */ target_ulong pc; /* Program counter */
target_ulong npc; /* Next PC */
target_ulong ppc; /* Prev PC */ target_ulong ppc; /* Prev PC */
target_ulong jmp_pc; /* Jump PC */ target_ulong jmp_pc; /* Jump PC */
......
...@@ -34,8 +34,8 @@ int openrisc_cpu_gdb_read_register(CPUState *cs, uint8_t *mem_buf, int n) ...@@ -34,8 +34,8 @@ int openrisc_cpu_gdb_read_register(CPUState *cs, uint8_t *mem_buf, int n)
case 32: /* PPC */ case 32: /* PPC */
return gdb_get_reg32(mem_buf, env->ppc); return gdb_get_reg32(mem_buf, env->ppc);
case 33: /* NPC */ case 33: /* NPC (equals PC) */
return gdb_get_reg32(mem_buf, env->npc); return gdb_get_reg32(mem_buf, env->pc);
case 34: /* SR */ case 34: /* SR */
return gdb_get_reg32(mem_buf, cpu_get_sr(env)); return gdb_get_reg32(mem_buf, cpu_get_sr(env));
...@@ -68,8 +68,13 @@ int openrisc_cpu_gdb_write_register(CPUState *cs, uint8_t *mem_buf, int n) ...@@ -68,8 +68,13 @@ int openrisc_cpu_gdb_write_register(CPUState *cs, uint8_t *mem_buf, int n)
env->ppc = tmp; env->ppc = tmp;
break; break;
case 33: /* NPC */ case 33: /* NPC (equals PC) */
env->npc = tmp; /* If setting PC to something different,
also clear delayed branch status. */
if (env->pc != tmp) {
env->pc = tmp;
env->flags = 0;
}
break; break;
case 34: /* SR */ case 34: /* SR */
......
...@@ -32,7 +32,6 @@ void HELPER(rfe)(CPUOpenRISCState *env) ...@@ -32,7 +32,6 @@ void HELPER(rfe)(CPUOpenRISCState *env)
(cpu->env.esr & (SR_SM | SR_IME | SR_DME)); (cpu->env.esr & (SR_SM | SR_IME | SR_DME));
#endif #endif
cpu->env.pc = cpu->env.epcr; cpu->env.pc = cpu->env.epcr;
cpu->env.npc = cpu->env.epcr;
cpu_set_sr(&cpu->env, cpu->env.esr); cpu_set_sr(&cpu->env, cpu->env.esr);
cpu->env.lock_addr = -1; cpu->env.lock_addr = -1;
......
...@@ -47,12 +47,11 @@ static const VMStateInfo vmstate_sr = { ...@@ -47,12 +47,11 @@ static const VMStateInfo vmstate_sr = {
static const VMStateDescription vmstate_env = { static const VMStateDescription vmstate_env = {
.name = "env", .name = "env",
.version_id = 3, .version_id = 4,
.minimum_version_id = 3, .minimum_version_id = 4,
.fields = (VMStateField[]) { .fields = (VMStateField[]) {
VMSTATE_UINTTL_ARRAY(gpr, CPUOpenRISCState, 32), VMSTATE_UINTTL_ARRAY(gpr, CPUOpenRISCState, 32),
VMSTATE_UINTTL(pc, CPUOpenRISCState), VMSTATE_UINTTL(pc, CPUOpenRISCState),
VMSTATE_UINTTL(npc, CPUOpenRISCState),
VMSTATE_UINTTL(ppc, CPUOpenRISCState), VMSTATE_UINTTL(ppc, CPUOpenRISCState),
VMSTATE_UINTTL(jmp_pc, CPUOpenRISCState), VMSTATE_UINTTL(jmp_pc, CPUOpenRISCState),
VMSTATE_UINTTL(lock_addr, CPUOpenRISCState), VMSTATE_UINTTL(lock_addr, CPUOpenRISCState),
......
...@@ -29,11 +29,10 @@ void HELPER(mtspr)(CPUOpenRISCState *env, ...@@ -29,11 +29,10 @@ void HELPER(mtspr)(CPUOpenRISCState *env,
target_ulong ra, target_ulong rb, target_ulong offset) target_ulong ra, target_ulong rb, target_ulong offset)
{ {
#ifndef CONFIG_USER_ONLY #ifndef CONFIG_USER_ONLY
int spr = (ra | offset);
int idx;
OpenRISCCPU *cpu = openrisc_env_get_cpu(env); OpenRISCCPU *cpu = openrisc_env_get_cpu(env);
CPUState *cs = CPU(cpu); CPUState *cs = CPU(cpu);
int spr = (ra | offset);
int idx;
switch (spr) { switch (spr) {
case TO_SPR(0, 0): /* VR */ case TO_SPR(0, 0): /* VR */
...@@ -41,7 +40,14 @@ void HELPER(mtspr)(CPUOpenRISCState *env, ...@@ -41,7 +40,14 @@ void HELPER(mtspr)(CPUOpenRISCState *env,
break; break;
case TO_SPR(0, 16): /* NPC */ case TO_SPR(0, 16): /* NPC */
env->npc = rb; cpu_restore_state(cs, GETPC());
/* ??? Mirror or1ksim in not trashing delayed branch state
when "jumping" to the current instruction. */
if (env->pc != rb) {
env->pc = rb;
env->flags = 0;
cpu_loop_exit(cs);
}
break; break;
case TO_SPR(0, 17): /* SR */ case TO_SPR(0, 17): /* SR */
...@@ -170,7 +176,6 @@ void HELPER(mtspr)(CPUOpenRISCState *env, ...@@ -170,7 +176,6 @@ void HELPER(mtspr)(CPUOpenRISCState *env,
cpu_openrisc_timer_update(cpu); cpu_openrisc_timer_update(cpu);
break; break;
default: default:
break; break;
} }
#endif #endif
...@@ -180,11 +185,11 @@ target_ulong HELPER(mfspr)(CPUOpenRISCState *env, ...@@ -180,11 +185,11 @@ target_ulong HELPER(mfspr)(CPUOpenRISCState *env,
target_ulong rd, target_ulong ra, uint32_t offset) target_ulong rd, target_ulong ra, uint32_t offset)
{ {
#ifndef CONFIG_USER_ONLY #ifndef CONFIG_USER_ONLY
OpenRISCCPU *cpu = openrisc_env_get_cpu(env);
CPUState *cs = CPU(cpu);
int spr = (ra | offset); int spr = (ra | offset);
int idx; int idx;
OpenRISCCPU *cpu = openrisc_env_get_cpu(env);
switch (spr) { switch (spr) {
case TO_SPR(0, 0): /* VR */ case TO_SPR(0, 0): /* VR */
return env->vr & SPR_VR; return env->vr & SPR_VR;
...@@ -201,13 +206,15 @@ target_ulong HELPER(mfspr)(CPUOpenRISCState *env, ...@@ -201,13 +206,15 @@ target_ulong HELPER(mfspr)(CPUOpenRISCState *env,
case TO_SPR(0, 4): /* IMMUCFGR */ case TO_SPR(0, 4): /* IMMUCFGR */
return env->immucfgr; return env->immucfgr;
case TO_SPR(0, 16): /* NPC */ case TO_SPR(0, 16): /* NPC (equals PC) */
return env->npc; cpu_restore_state(cs, GETPC());
return env->pc;
case TO_SPR(0, 17): /* SR */ case TO_SPR(0, 17): /* SR */
return cpu_get_sr(env); return cpu_get_sr(env);
case TO_SPR(0, 18): /* PPC */ case TO_SPR(0, 18): /* PPC */
cpu_restore_state(cs, GETPC());
return env->ppc; return env->ppc;
case TO_SPR(0, 32): /* EPCR */ case TO_SPR(0, 32): /* EPCR */
...@@ -276,25 +283,6 @@ target_ulong HELPER(mfspr)(CPUOpenRISCState *env, ...@@ -276,25 +283,6 @@ target_ulong HELPER(mfspr)(CPUOpenRISCState *env,
} }
#endif #endif
/*If we later need to add tracepoints (or debug printfs) for the return
value, it may be useful to structure the code like this:
target_ulong ret = 0;
switch() {
case x:
ret = y;
break;
case z:
ret = 42;
break;
...
}
later something like trace_spr_read(ret);
return ret;*/
/* for rd is passed in, if rd unchanged, just keep it back. */ /* for rd is passed in, if rd unchanged, just keep it back. */
return rd; return rd;
} }
...@@ -39,7 +39,7 @@ ...@@ -39,7 +39,7 @@
typedef struct DisasContext { typedef struct DisasContext {
TranslationBlock *tb; TranslationBlock *tb;
target_ulong pc, ppc, npc; target_ulong pc;
uint32_t tb_flags, synced_flags, flags; uint32_t tb_flags, synced_flags, flags;
uint32_t is_jmp; uint32_t is_jmp;
uint32_t mem_idx; uint32_t mem_idx;
...@@ -52,7 +52,6 @@ static TCGv cpu_sr; ...@@ -52,7 +52,6 @@ static TCGv cpu_sr;
static TCGv cpu_R[32]; static TCGv cpu_R[32];
static TCGv cpu_pc; static TCGv cpu_pc;
static TCGv jmp_pc; /* l.jr/l.jalr temp pc */ static TCGv jmp_pc; /* l.jr/l.jalr temp pc */
static TCGv cpu_npc;
static TCGv cpu_ppc; static TCGv cpu_ppc;
static TCGv cpu_sr_f; /* bf/bnf, F flag taken */ static TCGv cpu_sr_f; /* bf/bnf, F flag taken */
static TCGv cpu_sr_cy; /* carry (unsigned overflow) */ static TCGv cpu_sr_cy; /* carry (unsigned overflow) */
...@@ -83,8 +82,6 @@ void openrisc_translate_init(void) ...@@ -83,8 +82,6 @@ void openrisc_translate_init(void)
"flags"); "flags");
cpu_pc = tcg_global_mem_new(cpu_env, cpu_pc = tcg_global_mem_new(cpu_env,
offsetof(CPUOpenRISCState, pc), "pc"); offsetof(CPUOpenRISCState, pc), "pc");
cpu_npc = tcg_global_mem_new(cpu_env,
offsetof(CPUOpenRISCState, npc), "npc");
cpu_ppc = tcg_global_mem_new(cpu_env, cpu_ppc = tcg_global_mem_new(cpu_env,
offsetof(CPUOpenRISCState, ppc), "ppc"); offsetof(CPUOpenRISCState, ppc), "ppc");
jmp_pc = tcg_global_mem_new(cpu_env, jmp_pc = tcg_global_mem_new(cpu_env,
...@@ -1514,7 +1511,6 @@ void gen_intermediate_code(CPUOpenRISCState *env, struct TranslationBlock *tb) ...@@ -1514,7 +1511,6 @@ void gen_intermediate_code(CPUOpenRISCState *env, struct TranslationBlock *tb)
dc->tb = tb; dc->tb = tb;
dc->is_jmp = DISAS_NEXT; dc->is_jmp = DISAS_NEXT;
dc->ppc = pc_start;
dc->pc = pc_start; dc->pc = pc_start;
dc->flags = cpu->env.cpucfgr; dc->flags = cpu->env.cpucfgr;
dc->mem_idx = cpu_mmu_index(&cpu->env, false); dc->mem_idx = cpu_mmu_index(&cpu->env, false);
...@@ -1543,7 +1539,7 @@ void gen_intermediate_code(CPUOpenRISCState *env, struct TranslationBlock *tb) ...@@ -1543,7 +1539,7 @@ void gen_intermediate_code(CPUOpenRISCState *env, struct TranslationBlock *tb)
gen_tb_start(tb); gen_tb_start(tb);
do { do {
tcg_gen_insn_start(dc->pc); tcg_gen_insn_start(dc->pc, num_insns != 0);
num_insns++; num_insns++;
if (unlikely(cpu_breakpoint_test(cs, dc->pc, BP_ANY))) { if (unlikely(cpu_breakpoint_test(cs, dc->pc, BP_ANY))) {
...@@ -1561,12 +1557,9 @@ void gen_intermediate_code(CPUOpenRISCState *env, struct TranslationBlock *tb) ...@@ -1561,12 +1557,9 @@ void gen_intermediate_code(CPUOpenRISCState *env, struct TranslationBlock *tb)
if (num_insns == max_insns && (tb->cflags & CF_LAST_IO)) { if (num_insns == max_insns && (tb->cflags & CF_LAST_IO)) {
gen_io_start(); gen_io_start();
} }
dc->ppc = dc->pc - 4;
dc->npc = dc->pc + 4;
tcg_gen_movi_tl(cpu_ppc, dc->ppc);
tcg_gen_movi_tl(cpu_npc, dc->npc);
disas_openrisc_insn(dc, cpu); disas_openrisc_insn(dc, cpu);
dc->pc = dc->npc; dc->pc = dc->pc + 4;
/* delay slot */ /* delay slot */
if (dc->delayed_branch) { if (dc->delayed_branch) {
dc->delayed_branch--; dc->delayed_branch--;
...@@ -1574,10 +1567,8 @@ void gen_intermediate_code(CPUOpenRISCState *env, struct TranslationBlock *tb) ...@@ -1574,10 +1567,8 @@ void gen_intermediate_code(CPUOpenRISCState *env, struct TranslationBlock *tb)
dc->tb_flags &= ~D_FLAG; dc->tb_flags &= ~D_FLAG;
gen_sync_flags(dc); gen_sync_flags(dc);
tcg_gen_mov_tl(cpu_pc, jmp_pc); tcg_gen_mov_tl(cpu_pc, jmp_pc);
tcg_gen_mov_tl(cpu_npc, jmp_pc); tcg_gen_discard_tl(jmp_pc);
tcg_gen_movi_tl(jmp_pc, 0); dc->is_jmp = DISAS_UPDATE;
tcg_gen_exit_tb(0);
dc->is_jmp = DISAS_JUMP;
break; break;
} }
} }
...@@ -1591,14 +1582,13 @@ void gen_intermediate_code(CPUOpenRISCState *env, struct TranslationBlock *tb) ...@@ -1591,14 +1582,13 @@ void gen_intermediate_code(CPUOpenRISCState *env, struct TranslationBlock *tb)
if (tb->cflags & CF_LAST_IO) { if (tb->cflags & CF_LAST_IO) {
gen_io_end(); gen_io_end();
} }
tcg_gen_movi_tl(cpu_ppc, dc->pc - 4);
if (dc->is_jmp == DISAS_NEXT) { if (dc->is_jmp == DISAS_NEXT) {
dc->is_jmp = DISAS_UPDATE; dc->is_jmp = DISAS_UPDATE;
tcg_gen_movi_tl(cpu_pc, dc->pc); tcg_gen_movi_tl(cpu_pc, dc->pc);
} }
if (unlikely(cs->singlestep_enabled)) { if (unlikely(cs->singlestep_enabled)) {
if (dc->is_jmp == DISAS_NEXT) {
tcg_gen_movi_tl(cpu_pc, dc->pc);
}
gen_exception(dc, EXCP_DEBUG); gen_exception(dc, EXCP_DEBUG);
} else { } else {
switch (dc->is_jmp) { switch (dc->is_jmp) {
...@@ -1651,4 +1641,7 @@ void restore_state_to_opc(CPUOpenRISCState *env, TranslationBlock *tb, ...@@ -1651,4 +1641,7 @@ void restore_state_to_opc(CPUOpenRISCState *env, TranslationBlock *tb,
target_ulong *data) target_ulong *data)
{ {
env->pc = data[0]; env->pc = data[0];
if (data[1]) {
env->ppc = env->pc - 4;
}
} }
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册