target/nios2: Split mmu_write
Create three separate functions for the three separate registers. Avoid extra dispatch through op_helper.c. Dispatch to the correct function in translation. Clean up the ifdefs in wrctl. Reviewed-by: Peter Maydell <peter.maydell@linaro.org> Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
This commit is contained in:
parent
0b6e8f5b23
commit
304c05df7c
@ -21,6 +21,8 @@
|
|||||||
DEF_HELPER_FLAGS_2(raise_exception, TCG_CALL_NO_WG, noreturn, env, i32)
|
DEF_HELPER_FLAGS_2(raise_exception, TCG_CALL_NO_WG, noreturn, env, i32)
|
||||||
|
|
||||||
#if !defined(CONFIG_USER_ONLY)
|
#if !defined(CONFIG_USER_ONLY)
|
||||||
DEF_HELPER_3(mmu_write, void, env, i32, i32)
|
DEF_HELPER_2(mmu_write_tlbacc, void, env, i32)
|
||||||
|
DEF_HELPER_2(mmu_write_tlbmisc, void, env, i32)
|
||||||
|
DEF_HELPER_2(mmu_write_pteaddr, void, env, i32)
|
||||||
DEF_HELPER_1(check_interrupts, void, env)
|
DEF_HELPER_1(check_interrupts, void, env)
|
||||||
#endif
|
#endif
|
||||||
|
@ -23,6 +23,7 @@
|
|||||||
#include "cpu.h"
|
#include "cpu.h"
|
||||||
#include "exec/exec-all.h"
|
#include "exec/exec-all.h"
|
||||||
#include "mmu.h"
|
#include "mmu.h"
|
||||||
|
#include "exec/helper-proto.h"
|
||||||
#include "trace/trace-target_nios2.h"
|
#include "trace/trace-target_nios2.h"
|
||||||
|
|
||||||
|
|
||||||
@ -80,106 +81,103 @@ static void mmu_flush_pid(CPUNios2State *env, uint32_t pid)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void mmu_write(CPUNios2State *env, uint32_t rn, uint32_t v)
|
void helper_mmu_write_tlbacc(CPUNios2State *env, uint32_t v)
|
||||||
{
|
{
|
||||||
CPUState *cs = env_cpu(env);
|
CPUState *cs = env_cpu(env);
|
||||||
Nios2CPU *cpu = env_archcpu(env);
|
Nios2CPU *cpu = env_archcpu(env);
|
||||||
|
|
||||||
switch (rn) {
|
trace_nios2_mmu_write_tlbacc(v >> CR_TLBACC_IGN_SHIFT,
|
||||||
case CR_TLBACC:
|
(v & CR_TLBACC_C) ? 'C' : '.',
|
||||||
trace_nios2_mmu_write_tlbacc(v >> CR_TLBACC_IGN_SHIFT,
|
(v & CR_TLBACC_R) ? 'R' : '.',
|
||||||
(v & CR_TLBACC_C) ? 'C' : '.',
|
(v & CR_TLBACC_W) ? 'W' : '.',
|
||||||
(v & CR_TLBACC_R) ? 'R' : '.',
|
(v & CR_TLBACC_X) ? 'X' : '.',
|
||||||
(v & CR_TLBACC_W) ? 'W' : '.',
|
(v & CR_TLBACC_G) ? 'G' : '.',
|
||||||
(v & CR_TLBACC_X) ? 'X' : '.',
|
v & CR_TLBACC_PFN_MASK);
|
||||||
(v & CR_TLBACC_G) ? 'G' : '.',
|
|
||||||
v & CR_TLBACC_PFN_MASK);
|
|
||||||
|
|
||||||
/* if tlbmisc.WE == 1 then trigger a TLB write on writes to TLBACC */
|
/* if tlbmisc.WE == 1 then trigger a TLB write on writes to TLBACC */
|
||||||
if (env->regs[CR_TLBMISC] & CR_TLBMISC_WR) {
|
if (env->regs[CR_TLBMISC] & CR_TLBMISC_WR) {
|
||||||
int way = (env->regs[CR_TLBMISC] >> CR_TLBMISC_WAY_SHIFT);
|
int way = (env->regs[CR_TLBMISC] >> CR_TLBMISC_WAY_SHIFT);
|
||||||
int vpn = (env->mmu.pteaddr_wr & CR_PTEADDR_VPN_MASK) >> 2;
|
int vpn = (env->mmu.pteaddr_wr & CR_PTEADDR_VPN_MASK) >> 2;
|
||||||
int pid = (env->mmu.tlbmisc_wr & CR_TLBMISC_PID_MASK) >> 4;
|
int pid = (env->mmu.tlbmisc_wr & CR_TLBMISC_PID_MASK) >> 4;
|
||||||
int g = (v & CR_TLBACC_G) ? 1 : 0;
|
int g = (v & CR_TLBACC_G) ? 1 : 0;
|
||||||
int valid = ((vpn & CR_TLBACC_PFN_MASK) < 0xC0000) ? 1 : 0;
|
int valid = ((vpn & CR_TLBACC_PFN_MASK) < 0xC0000) ? 1 : 0;
|
||||||
Nios2TLBEntry *entry =
|
Nios2TLBEntry *entry =
|
||||||
&env->mmu.tlb[(way * cpu->tlb_num_ways) +
|
&env->mmu.tlb[(way * cpu->tlb_num_ways) +
|
||||||
(vpn & env->mmu.tlb_entry_mask)];
|
(vpn & env->mmu.tlb_entry_mask)];
|
||||||
uint32_t newTag = (vpn << 12) | (g << 11) | (valid << 10) | pid;
|
uint32_t newTag = (vpn << 12) | (g << 11) | (valid << 10) | pid;
|
||||||
uint32_t newData = v & (CR_TLBACC_C | CR_TLBACC_R | CR_TLBACC_W |
|
uint32_t newData = v & (CR_TLBACC_C | CR_TLBACC_R | CR_TLBACC_W |
|
||||||
CR_TLBACC_X | CR_TLBACC_PFN_MASK);
|
CR_TLBACC_X | CR_TLBACC_PFN_MASK);
|
||||||
|
|
||||||
if ((entry->tag != newTag) || (entry->data != newData)) {
|
if ((entry->tag != newTag) || (entry->data != newData)) {
|
||||||
if (entry->tag & (1 << 10)) {
|
if (entry->tag & (1 << 10)) {
|
||||||
/* Flush existing entry */
|
/* Flush existing entry */
|
||||||
tlb_flush_page(cs, entry->tag & TARGET_PAGE_MASK);
|
tlb_flush_page(cs, entry->tag & TARGET_PAGE_MASK);
|
||||||
}
|
|
||||||
entry->tag = newTag;
|
|
||||||
entry->data = newData;
|
|
||||||
}
|
}
|
||||||
/* Auto-increment tlbmisc.WAY */
|
entry->tag = newTag;
|
||||||
env->regs[CR_TLBMISC] =
|
entry->data = newData;
|
||||||
(env->regs[CR_TLBMISC] & ~CR_TLBMISC_WAY_MASK) |
|
|
||||||
(((way + 1) & (cpu->tlb_num_ways - 1)) <<
|
|
||||||
CR_TLBMISC_WAY_SHIFT);
|
|
||||||
}
|
}
|
||||||
|
/* Auto-increment tlbmisc.WAY */
|
||||||
/* Writes to TLBACC don't change the read-back value */
|
env->regs[CR_TLBMISC] =
|
||||||
env->mmu.tlbacc_wr = v;
|
(env->regs[CR_TLBMISC] & ~CR_TLBMISC_WAY_MASK) |
|
||||||
break;
|
(((way + 1) & (cpu->tlb_num_ways - 1)) <<
|
||||||
|
CR_TLBMISC_WAY_SHIFT);
|
||||||
case CR_TLBMISC:
|
|
||||||
trace_nios2_mmu_write_tlbmisc(v >> CR_TLBMISC_WAY_SHIFT,
|
|
||||||
(v & CR_TLBMISC_RD) ? 'R' : '.',
|
|
||||||
(v & CR_TLBMISC_WR) ? 'W' : '.',
|
|
||||||
(v & CR_TLBMISC_DBL) ? '2' : '.',
|
|
||||||
(v & CR_TLBMISC_BAD) ? 'B' : '.',
|
|
||||||
(v & CR_TLBMISC_PERM) ? 'P' : '.',
|
|
||||||
(v & CR_TLBMISC_D) ? 'D' : '.',
|
|
||||||
(v & CR_TLBMISC_PID_MASK) >> 4);
|
|
||||||
|
|
||||||
if ((v & CR_TLBMISC_PID_MASK) !=
|
|
||||||
(env->mmu.tlbmisc_wr & CR_TLBMISC_PID_MASK)) {
|
|
||||||
mmu_flush_pid(env, (env->mmu.tlbmisc_wr & CR_TLBMISC_PID_MASK) >>
|
|
||||||
CR_TLBMISC_PID_SHIFT);
|
|
||||||
}
|
|
||||||
/* if tlbmisc.RD == 1 then trigger a TLB read on writes to TLBMISC */
|
|
||||||
if (v & CR_TLBMISC_RD) {
|
|
||||||
int way = (v >> CR_TLBMISC_WAY_SHIFT);
|
|
||||||
int vpn = (env->mmu.pteaddr_wr & CR_PTEADDR_VPN_MASK) >> 2;
|
|
||||||
Nios2TLBEntry *entry =
|
|
||||||
&env->mmu.tlb[(way * cpu->tlb_num_ways) +
|
|
||||||
(vpn & env->mmu.tlb_entry_mask)];
|
|
||||||
|
|
||||||
env->regs[CR_TLBACC] &= CR_TLBACC_IGN_MASK;
|
|
||||||
env->regs[CR_TLBACC] |= entry->data;
|
|
||||||
env->regs[CR_TLBACC] |= (entry->tag & (1 << 11)) ? CR_TLBACC_G : 0;
|
|
||||||
env->regs[CR_TLBMISC] =
|
|
||||||
(v & ~CR_TLBMISC_PID_MASK) |
|
|
||||||
((entry->tag & ((1 << cpu->pid_num_bits) - 1)) <<
|
|
||||||
CR_TLBMISC_PID_SHIFT);
|
|
||||||
env->regs[CR_PTEADDR] &= ~CR_PTEADDR_VPN_MASK;
|
|
||||||
env->regs[CR_PTEADDR] |= (entry->tag >> 12) << CR_PTEADDR_VPN_SHIFT;
|
|
||||||
} else {
|
|
||||||
env->regs[CR_TLBMISC] = v;
|
|
||||||
}
|
|
||||||
|
|
||||||
env->mmu.tlbmisc_wr = v;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case CR_PTEADDR:
|
|
||||||
trace_nios2_mmu_write_pteaddr(v >> CR_PTEADDR_PTBASE_SHIFT,
|
|
||||||
(v & CR_PTEADDR_VPN_MASK) >> CR_PTEADDR_VPN_SHIFT);
|
|
||||||
|
|
||||||
/* Writes to PTEADDR don't change the read-back VPN value */
|
|
||||||
env->regs[CR_PTEADDR] = (v & ~CR_PTEADDR_VPN_MASK) |
|
|
||||||
(env->regs[CR_PTEADDR] & CR_PTEADDR_VPN_MASK);
|
|
||||||
env->mmu.pteaddr_wr = v;
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Writes to TLBACC don't change the read-back value */
|
||||||
|
env->mmu.tlbacc_wr = v;
|
||||||
|
}
|
||||||
|
|
||||||
|
void helper_mmu_write_tlbmisc(CPUNios2State *env, uint32_t v)
|
||||||
|
{
|
||||||
|
Nios2CPU *cpu = env_archcpu(env);
|
||||||
|
|
||||||
|
trace_nios2_mmu_write_tlbmisc(v >> CR_TLBMISC_WAY_SHIFT,
|
||||||
|
(v & CR_TLBMISC_RD) ? 'R' : '.',
|
||||||
|
(v & CR_TLBMISC_WR) ? 'W' : '.',
|
||||||
|
(v & CR_TLBMISC_DBL) ? '2' : '.',
|
||||||
|
(v & CR_TLBMISC_BAD) ? 'B' : '.',
|
||||||
|
(v & CR_TLBMISC_PERM) ? 'P' : '.',
|
||||||
|
(v & CR_TLBMISC_D) ? 'D' : '.',
|
||||||
|
(v & CR_TLBMISC_PID_MASK) >> 4);
|
||||||
|
|
||||||
|
if ((v & CR_TLBMISC_PID_MASK) !=
|
||||||
|
(env->mmu.tlbmisc_wr & CR_TLBMISC_PID_MASK)) {
|
||||||
|
mmu_flush_pid(env, (env->mmu.tlbmisc_wr & CR_TLBMISC_PID_MASK) >>
|
||||||
|
CR_TLBMISC_PID_SHIFT);
|
||||||
|
}
|
||||||
|
/* if tlbmisc.RD == 1 then trigger a TLB read on writes to TLBMISC */
|
||||||
|
if (v & CR_TLBMISC_RD) {
|
||||||
|
int way = (v >> CR_TLBMISC_WAY_SHIFT);
|
||||||
|
int vpn = (env->mmu.pteaddr_wr & CR_PTEADDR_VPN_MASK) >> 2;
|
||||||
|
Nios2TLBEntry *entry =
|
||||||
|
&env->mmu.tlb[(way * cpu->tlb_num_ways) +
|
||||||
|
(vpn & env->mmu.tlb_entry_mask)];
|
||||||
|
|
||||||
|
env->regs[CR_TLBACC] &= CR_TLBACC_IGN_MASK;
|
||||||
|
env->regs[CR_TLBACC] |= entry->data;
|
||||||
|
env->regs[CR_TLBACC] |= (entry->tag & (1 << 11)) ? CR_TLBACC_G : 0;
|
||||||
|
env->regs[CR_TLBMISC] =
|
||||||
|
(v & ~CR_TLBMISC_PID_MASK) |
|
||||||
|
((entry->tag & ((1 << cpu->pid_num_bits) - 1)) <<
|
||||||
|
CR_TLBMISC_PID_SHIFT);
|
||||||
|
env->regs[CR_PTEADDR] &= ~CR_PTEADDR_VPN_MASK;
|
||||||
|
env->regs[CR_PTEADDR] |= (entry->tag >> 12) << CR_PTEADDR_VPN_SHIFT;
|
||||||
|
} else {
|
||||||
|
env->regs[CR_TLBMISC] = v;
|
||||||
|
}
|
||||||
|
|
||||||
|
env->mmu.tlbmisc_wr = v;
|
||||||
|
}
|
||||||
|
|
||||||
|
void helper_mmu_write_pteaddr(CPUNios2State *env, uint32_t v)
|
||||||
|
{
|
||||||
|
trace_nios2_mmu_write_pteaddr(v >> CR_PTEADDR_PTBASE_SHIFT,
|
||||||
|
(v & CR_PTEADDR_VPN_MASK) >> CR_PTEADDR_VPN_SHIFT);
|
||||||
|
|
||||||
|
/* Writes to PTEADDR don't change the read-back VPN value */
|
||||||
|
env->regs[CR_PTEADDR] = (v & ~CR_PTEADDR_VPN_MASK) |
|
||||||
|
(env->regs[CR_PTEADDR] & CR_PTEADDR_VPN_MASK);
|
||||||
|
env->mmu.pteaddr_wr = v;
|
||||||
}
|
}
|
||||||
|
|
||||||
void mmu_init(CPUNios2State *env)
|
void mmu_init(CPUNios2State *env)
|
||||||
|
@ -26,11 +26,6 @@
|
|||||||
#include "qemu/main-loop.h"
|
#include "qemu/main-loop.h"
|
||||||
|
|
||||||
#if !defined(CONFIG_USER_ONLY)
|
#if !defined(CONFIG_USER_ONLY)
|
||||||
void helper_mmu_write(CPUNios2State *env, uint32_t rn, uint32_t v)
|
|
||||||
{
|
|
||||||
mmu_write(env, rn, v);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void nios2_check_interrupts(CPUNios2State *env)
|
static void nios2_check_interrupts(CPUNios2State *env)
|
||||||
{
|
{
|
||||||
if (env->irq_pending &&
|
if (env->irq_pending &&
|
||||||
|
@ -461,30 +461,28 @@ static void rdctl(DisasContext *dc, uint32_t code, uint32_t flags)
|
|||||||
/* ctlN <- rA */
|
/* ctlN <- rA */
|
||||||
static void wrctl(DisasContext *dc, uint32_t code, uint32_t flags)
|
static void wrctl(DisasContext *dc, uint32_t code, uint32_t flags)
|
||||||
{
|
{
|
||||||
R_TYPE(instr, code);
|
|
||||||
|
|
||||||
gen_check_supervisor(dc);
|
gen_check_supervisor(dc);
|
||||||
|
|
||||||
|
#ifndef CONFIG_USER_ONLY
|
||||||
|
R_TYPE(instr, code);
|
||||||
|
TCGv v = load_gpr(dc, instr.a);
|
||||||
|
|
||||||
switch (instr.imm5 + CR_BASE) {
|
switch (instr.imm5 + CR_BASE) {
|
||||||
case CR_PTEADDR:
|
case CR_PTEADDR:
|
||||||
case CR_TLBACC:
|
gen_helper_mmu_write_pteaddr(cpu_env, v);
|
||||||
case CR_TLBMISC:
|
break;
|
||||||
{
|
case CR_TLBACC:
|
||||||
#if !defined(CONFIG_USER_ONLY)
|
gen_helper_mmu_write_tlbacc(cpu_env, v);
|
||||||
TCGv_i32 tmp = tcg_const_i32(instr.imm5 + CR_BASE);
|
break;
|
||||||
gen_helper_mmu_write(cpu_env, tmp, load_gpr(dc, instr.a));
|
case CR_TLBMISC:
|
||||||
tcg_temp_free_i32(tmp);
|
gen_helper_mmu_write_tlbmisc(cpu_env, v);
|
||||||
#endif
|
|
||||||
break;
|
break;
|
||||||
}
|
|
||||||
|
|
||||||
default:
|
default:
|
||||||
tcg_gen_mov_tl(cpu_R[instr.imm5 + CR_BASE], load_gpr(dc, instr.a));
|
tcg_gen_mov_tl(cpu_R[instr.imm5 + CR_BASE], v);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If interrupts were enabled using WRCTL, trigger them. */
|
/* If interrupts were enabled using WRCTL, trigger them. */
|
||||||
#if !defined(CONFIG_USER_ONLY)
|
|
||||||
if ((instr.imm5 + CR_BASE) == CR_STATUS) {
|
if ((instr.imm5 + CR_BASE) == CR_STATUS) {
|
||||||
if (tb_cflags(dc->base.tb) & CF_USE_ICOUNT) {
|
if (tb_cflags(dc->base.tb) & CF_USE_ICOUNT) {
|
||||||
gen_io_start();
|
gen_io_start();
|
||||||
|
Loading…
Reference in New Issue
Block a user