target/openrisc: Implement lwa, swa
Signed-off-by: Richard Henderson <rth@twiddle.net>
This commit is contained in:
parent
c56e3b8670
commit
930c3d0074
@ -48,6 +48,7 @@ static void openrisc_cpu_reset(CPUState *s)
|
|||||||
|
|
||||||
cpu->env.pc = 0x100;
|
cpu->env.pc = 0x100;
|
||||||
cpu->env.sr = SR_FO | SR_SM;
|
cpu->env.sr = SR_FO | SR_SM;
|
||||||
|
cpu->env.lock_addr = -1;
|
||||||
s->exception_index = -1;
|
s->exception_index = -1;
|
||||||
|
|
||||||
cpu->env.upr = UPR_UP | UPR_DMP | UPR_IMP | UPR_PICP | UPR_TTP;
|
cpu->env.upr = UPR_UP | UPR_DMP | UPR_IMP | UPR_PICP | UPR_TTP;
|
||||||
|
@ -296,6 +296,9 @@ typedef struct CPUOpenRISCState {
|
|||||||
uint32_t fpcsr; /* Float register */
|
uint32_t fpcsr; /* Float register */
|
||||||
float_status fp_status;
|
float_status fp_status;
|
||||||
|
|
||||||
|
target_ulong lock_addr;
|
||||||
|
target_ulong lock_value;
|
||||||
|
|
||||||
uint32_t flags; /* cpu_flags, we only use it for exception
|
uint32_t flags; /* cpu_flags, we only use it for exception
|
||||||
in solt so far. */
|
in solt so far. */
|
||||||
uint32_t btaken; /* the SR_F bit */
|
uint32_t btaken; /* the SR_F bit */
|
||||||
|
@ -62,6 +62,7 @@ void openrisc_cpu_do_interrupt(CPUState *cs)
|
|||||||
env->sr &= ~SR_TEE;
|
env->sr &= ~SR_TEE;
|
||||||
env->tlb->cpu_openrisc_map_address_data = &cpu_openrisc_get_phys_nommu;
|
env->tlb->cpu_openrisc_map_address_data = &cpu_openrisc_get_phys_nommu;
|
||||||
env->tlb->cpu_openrisc_map_address_code = &cpu_openrisc_get_phys_nommu;
|
env->tlb->cpu_openrisc_map_address_code = &cpu_openrisc_get_phys_nommu;
|
||||||
|
env->lock_addr = -1;
|
||||||
|
|
||||||
if (cs->exception_index > 0 && cs->exception_index < EXCP_NR) {
|
if (cs->exception_index > 0 && cs->exception_index < EXCP_NR) {
|
||||||
env->pc = (cs->exception_index << 8);
|
env->pc = (cs->exception_index << 8);
|
||||||
|
@ -34,6 +34,7 @@ void HELPER(rfe)(CPUOpenRISCState *env)
|
|||||||
cpu->env.pc = cpu->env.epcr;
|
cpu->env.pc = cpu->env.epcr;
|
||||||
cpu->env.npc = cpu->env.epcr;
|
cpu->env.npc = cpu->env.epcr;
|
||||||
cpu->env.sr = cpu->env.esr;
|
cpu->env.sr = cpu->env.esr;
|
||||||
|
cpu->env.lock_addr = -1;
|
||||||
|
|
||||||
#ifndef CONFIG_USER_ONLY
|
#ifndef CONFIG_USER_ONLY
|
||||||
if (cpu->env.sr & SR_DME) {
|
if (cpu->env.sr & SR_DME) {
|
||||||
|
@ -26,18 +26,26 @@
|
|||||||
|
|
||||||
static const VMStateDescription vmstate_env = {
|
static const VMStateDescription vmstate_env = {
|
||||||
.name = "env",
|
.name = "env",
|
||||||
.version_id = 1,
|
.version_id = 2,
|
||||||
.minimum_version_id = 1,
|
.minimum_version_id = 2,
|
||||||
.fields = (VMStateField[]) {
|
.fields = (VMStateField[]) {
|
||||||
VMSTATE_UINT32_ARRAY(gpr, CPUOpenRISCState, 32),
|
VMSTATE_UINTTL_ARRAY(gpr, CPUOpenRISCState, 32),
|
||||||
|
VMSTATE_UINTTL(pc, CPUOpenRISCState),
|
||||||
|
VMSTATE_UINTTL(npc, CPUOpenRISCState),
|
||||||
|
VMSTATE_UINTTL(ppc, CPUOpenRISCState),
|
||||||
|
VMSTATE_UINTTL(jmp_pc, CPUOpenRISCState),
|
||||||
|
VMSTATE_UINTTL(lock_addr, CPUOpenRISCState),
|
||||||
|
VMSTATE_UINTTL(lock_value, CPUOpenRISCState),
|
||||||
|
VMSTATE_UINTTL(epcr, CPUOpenRISCState),
|
||||||
|
VMSTATE_UINTTL(eear, CPUOpenRISCState),
|
||||||
VMSTATE_UINT32(sr, CPUOpenRISCState),
|
VMSTATE_UINT32(sr, CPUOpenRISCState),
|
||||||
VMSTATE_UINT32(epcr, CPUOpenRISCState),
|
VMSTATE_UINT32(vr, CPUOpenRISCState),
|
||||||
VMSTATE_UINT32(eear, CPUOpenRISCState),
|
VMSTATE_UINT32(upr, CPUOpenRISCState),
|
||||||
|
VMSTATE_UINT32(cpucfgr, CPUOpenRISCState),
|
||||||
|
VMSTATE_UINT32(dmmucfgr, CPUOpenRISCState),
|
||||||
|
VMSTATE_UINT32(immucfgr, CPUOpenRISCState),
|
||||||
VMSTATE_UINT32(esr, CPUOpenRISCState),
|
VMSTATE_UINT32(esr, CPUOpenRISCState),
|
||||||
VMSTATE_UINT32(fpcsr, CPUOpenRISCState),
|
VMSTATE_UINT32(fpcsr, CPUOpenRISCState),
|
||||||
VMSTATE_UINT32(pc, CPUOpenRISCState),
|
|
||||||
VMSTATE_UINT32(npc, CPUOpenRISCState),
|
|
||||||
VMSTATE_UINT32(ppc, CPUOpenRISCState),
|
|
||||||
VMSTATE_END_OF_LIST()
|
VMSTATE_END_OF_LIST()
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -174,6 +174,7 @@ static void cpu_openrisc_raise_mmu_exception(OpenRISCCPU *cpu,
|
|||||||
|
|
||||||
cs->exception_index = exception;
|
cs->exception_index = exception;
|
||||||
cpu->env.eear = address;
|
cpu->env.eear = address;
|
||||||
|
cpu->env.lock_addr = -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifndef CONFIG_USER_ONLY
|
#ifndef CONFIG_USER_ONLY
|
||||||
|
@ -61,6 +61,8 @@ static TCGv jmp_pc; /* l.jr/l.jalr temp pc */
|
|||||||
static TCGv cpu_npc;
|
static TCGv cpu_npc;
|
||||||
static TCGv cpu_ppc;
|
static TCGv cpu_ppc;
|
||||||
static TCGv_i32 env_btaken; /* bf/bnf , F flag taken */
|
static TCGv_i32 env_btaken; /* bf/bnf , F flag taken */
|
||||||
|
static TCGv cpu_lock_addr;
|
||||||
|
static TCGv cpu_lock_value;
|
||||||
static TCGv_i32 fpcsr;
|
static TCGv_i32 fpcsr;
|
||||||
static TCGv machi, maclo;
|
static TCGv machi, maclo;
|
||||||
static TCGv fpmaddhi, fpmaddlo;
|
static TCGv fpmaddhi, fpmaddlo;
|
||||||
@ -95,6 +97,12 @@ void openrisc_translate_init(void)
|
|||||||
env_btaken = tcg_global_mem_new_i32(cpu_env,
|
env_btaken = tcg_global_mem_new_i32(cpu_env,
|
||||||
offsetof(CPUOpenRISCState, btaken),
|
offsetof(CPUOpenRISCState, btaken),
|
||||||
"btaken");
|
"btaken");
|
||||||
|
cpu_lock_addr = tcg_global_mem_new(cpu_env,
|
||||||
|
offsetof(CPUOpenRISCState, lock_addr),
|
||||||
|
"lock_addr");
|
||||||
|
cpu_lock_value = tcg_global_mem_new(cpu_env,
|
||||||
|
offsetof(CPUOpenRISCState, lock_value),
|
||||||
|
"lock_value");
|
||||||
fpcsr = tcg_global_mem_new_i32(cpu_env,
|
fpcsr = tcg_global_mem_new_i32(cpu_env,
|
||||||
offsetof(CPUOpenRISCState, fpcsr),
|
offsetof(CPUOpenRISCState, fpcsr),
|
||||||
"fpcsr");
|
"fpcsr");
|
||||||
@ -265,6 +273,46 @@ static void gen_jump(DisasContext *dc, uint32_t imm, uint32_t reg, uint32_t op0)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void gen_lwa(DisasContext *dc, TCGv rd, TCGv ra, int32_t ofs)
|
||||||
|
{
|
||||||
|
TCGv ea = tcg_temp_new();
|
||||||
|
|
||||||
|
tcg_gen_addi_tl(ea, ra, ofs);
|
||||||
|
tcg_gen_qemu_ld_tl(rd, ea, dc->mem_idx, MO_TEUL);
|
||||||
|
tcg_gen_mov_tl(cpu_lock_addr, ea);
|
||||||
|
tcg_gen_mov_tl(cpu_lock_value, rd);
|
||||||
|
tcg_temp_free(ea);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void gen_swa(DisasContext *dc, TCGv rb, TCGv ra, int32_t ofs)
|
||||||
|
{
|
||||||
|
TCGv ea, val;
|
||||||
|
TCGLabel *lab_fail, *lab_done;
|
||||||
|
|
||||||
|
ea = tcg_temp_new();
|
||||||
|
tcg_gen_addi_tl(ea, ra, ofs);
|
||||||
|
|
||||||
|
lab_fail = gen_new_label();
|
||||||
|
lab_done = gen_new_label();
|
||||||
|
tcg_gen_brcond_tl(TCG_COND_NE, ea, cpu_lock_addr, lab_fail);
|
||||||
|
tcg_temp_free(ea);
|
||||||
|
|
||||||
|
val = tcg_temp_new();
|
||||||
|
tcg_gen_atomic_cmpxchg_tl(val, cpu_lock_addr, cpu_lock_value,
|
||||||
|
rb, dc->mem_idx, MO_TEUL);
|
||||||
|
tcg_gen_setcond_tl(TCG_COND_EQ, env_btaken, val, cpu_lock_value);
|
||||||
|
tcg_temp_free(val);
|
||||||
|
|
||||||
|
tcg_gen_br(lab_done);
|
||||||
|
|
||||||
|
gen_set_label(lab_fail);
|
||||||
|
tcg_gen_movi_tl(env_btaken, 0);
|
||||||
|
|
||||||
|
gen_set_label(lab_done);
|
||||||
|
tcg_gen_movi_tl(cpu_lock_addr, -1);
|
||||||
|
wb_SR_F();
|
||||||
|
}
|
||||||
|
|
||||||
static void dec_calc(DisasContext *dc, uint32_t insn)
|
static void dec_calc(DisasContext *dc, uint32_t insn)
|
||||||
{
|
{
|
||||||
uint32_t op0, op1, op2;
|
uint32_t op0, op1, op2;
|
||||||
@ -819,6 +867,11 @@ static void dec_misc(DisasContext *dc, uint32_t insn)
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case 0x1b: /* l.lwa */
|
||||||
|
LOG_DIS("l.lwa r%d, r%d, %d\n", rd, ra, I16);
|
||||||
|
gen_lwa(dc, cpu_R[rd], cpu_R[ra], I16);
|
||||||
|
break;
|
||||||
|
|
||||||
case 0x1c: /* l.cust1 */
|
case 0x1c: /* l.cust1 */
|
||||||
LOG_DIS("l.cust1\n");
|
LOG_DIS("l.cust1\n");
|
||||||
break;
|
break;
|
||||||
@ -1029,6 +1082,11 @@ static void dec_misc(DisasContext *dc, uint32_t insn)
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case 0x33: /* l.swa */
|
||||||
|
LOG_DIS("l.swa %d, r%d, r%d, %d\n", I5, ra, rb, I11);
|
||||||
|
gen_swa(dc, cpu_R[rb], cpu_R[ra], sign_extend(tmp, 16));
|
||||||
|
break;
|
||||||
|
|
||||||
/* not used yet, open it when we need or64. */
|
/* not used yet, open it when we need or64. */
|
||||||
/*#ifdef TARGET_OPENRISC64
|
/*#ifdef TARGET_OPENRISC64
|
||||||
case 0x34: l.sd
|
case 0x34: l.sd
|
||||||
|
Loading…
Reference in New Issue
Block a user