target/mips: Convert to CPUClass::tlb_fill
Note that env->active_tc.PC is removed from the qemu_log as that value is garbage. The PC isn't recovered until cpu_restore_state, called from cpu_loop_exit_restore, called from do_raise_exception_err. Cc: Aleksandar Markovic <amarkovic@wavecomp.com> Cc: Aleksandar Rikalo <arikalo@wavecomp.com> Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com> Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
This commit is contained in:
parent
e38f4eb630
commit
931d019f5b
@ -197,9 +197,7 @@ static void mips_cpu_class_init(ObjectClass *c, void *data)
|
|||||||
cc->synchronize_from_tb = mips_cpu_synchronize_from_tb;
|
cc->synchronize_from_tb = mips_cpu_synchronize_from_tb;
|
||||||
cc->gdb_read_register = mips_cpu_gdb_read_register;
|
cc->gdb_read_register = mips_cpu_gdb_read_register;
|
||||||
cc->gdb_write_register = mips_cpu_gdb_write_register;
|
cc->gdb_write_register = mips_cpu_gdb_write_register;
|
||||||
#ifdef CONFIG_USER_ONLY
|
#ifndef CONFIG_USER_ONLY
|
||||||
cc->handle_mmu_fault = mips_cpu_handle_mmu_fault;
|
|
||||||
#else
|
|
||||||
cc->do_unassigned_access = mips_cpu_unassigned_access;
|
cc->do_unassigned_access = mips_cpu_unassigned_access;
|
||||||
cc->do_unaligned_access = mips_cpu_do_unaligned_access;
|
cc->do_unaligned_access = mips_cpu_do_unaligned_access;
|
||||||
cc->get_phys_page_debug = mips_cpu_get_phys_page_debug;
|
cc->get_phys_page_debug = mips_cpu_get_phys_page_debug;
|
||||||
@ -208,6 +206,7 @@ static void mips_cpu_class_init(ObjectClass *c, void *data)
|
|||||||
cc->disas_set_info = mips_cpu_disas_set_info;
|
cc->disas_set_info = mips_cpu_disas_set_info;
|
||||||
#ifdef CONFIG_TCG
|
#ifdef CONFIG_TCG
|
||||||
cc->tcg_initialize = mips_tcg_init;
|
cc->tcg_initialize = mips_tcg_init;
|
||||||
|
cc->tlb_fill = mips_cpu_tlb_fill;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
cc->gdb_num_core_regs = 73;
|
cc->gdb_num_core_regs = 73;
|
||||||
|
@ -874,31 +874,25 @@ refill:
|
|||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
int mips_cpu_handle_mmu_fault(CPUState *cs, vaddr address, int size, int rw,
|
bool mips_cpu_tlb_fill(CPUState *cs, vaddr address, int size,
|
||||||
int mmu_idx)
|
MMUAccessType access_type, int mmu_idx,
|
||||||
|
bool probe, uintptr_t retaddr)
|
||||||
{
|
{
|
||||||
MIPSCPU *cpu = MIPS_CPU(cs);
|
MIPSCPU *cpu = MIPS_CPU(cs);
|
||||||
CPUMIPSState *env = &cpu->env;
|
CPUMIPSState *env = &cpu->env;
|
||||||
#if !defined(CONFIG_USER_ONLY)
|
#if !defined(CONFIG_USER_ONLY)
|
||||||
hwaddr physical;
|
hwaddr physical;
|
||||||
int prot;
|
int prot;
|
||||||
int access_type;
|
int mips_access_type;
|
||||||
#endif
|
#endif
|
||||||
int ret = TLBRET_BADADDR;
|
int ret = TLBRET_BADADDR;
|
||||||
|
|
||||||
#if 0
|
|
||||||
log_cpu_state(cs, 0);
|
|
||||||
#endif
|
|
||||||
qemu_log_mask(CPU_LOG_MMU,
|
|
||||||
"%s pc " TARGET_FMT_lx " ad %" VADDR_PRIx " rw %d mmu_idx %d\n",
|
|
||||||
__func__, env->active_tc.PC, address, rw, mmu_idx);
|
|
||||||
|
|
||||||
/* data access */
|
/* data access */
|
||||||
#if !defined(CONFIG_USER_ONLY)
|
#if !defined(CONFIG_USER_ONLY)
|
||||||
/* XXX: put correct access by using cpu_restore_state() correctly */
|
/* XXX: put correct access by using cpu_restore_state() correctly */
|
||||||
access_type = ACCESS_INT;
|
mips_access_type = ACCESS_INT;
|
||||||
ret = get_physical_address(env, &physical, &prot,
|
ret = get_physical_address(env, &physical, &prot, address,
|
||||||
address, rw, access_type, mmu_idx);
|
access_type, mips_access_type, mmu_idx);
|
||||||
switch (ret) {
|
switch (ret) {
|
||||||
case TLBRET_MATCH:
|
case TLBRET_MATCH:
|
||||||
qemu_log_mask(CPU_LOG_MMU,
|
qemu_log_mask(CPU_LOG_MMU,
|
||||||
@ -915,7 +909,7 @@ int mips_cpu_handle_mmu_fault(CPUState *cs, vaddr address, int size, int rw,
|
|||||||
tlb_set_page(cs, address & TARGET_PAGE_MASK,
|
tlb_set_page(cs, address & TARGET_PAGE_MASK,
|
||||||
physical & TARGET_PAGE_MASK, prot | PAGE_EXEC,
|
physical & TARGET_PAGE_MASK, prot | PAGE_EXEC,
|
||||||
mmu_idx, TARGET_PAGE_SIZE);
|
mmu_idx, TARGET_PAGE_SIZE);
|
||||||
return 0;
|
return true;
|
||||||
}
|
}
|
||||||
#if !defined(TARGET_MIPS64)
|
#if !defined(TARGET_MIPS64)
|
||||||
if ((ret == TLBRET_NOMATCH) && (env->tlb->nb_tlb > 1)) {
|
if ((ret == TLBRET_NOMATCH) && (env->tlb->nb_tlb > 1)) {
|
||||||
@ -926,27 +920,36 @@ int mips_cpu_handle_mmu_fault(CPUState *cs, vaddr address, int size, int rw,
|
|||||||
int mode = (env->hflags & MIPS_HFLAG_KSU);
|
int mode = (env->hflags & MIPS_HFLAG_KSU);
|
||||||
bool ret_walker;
|
bool ret_walker;
|
||||||
env->hflags &= ~MIPS_HFLAG_KSU;
|
env->hflags &= ~MIPS_HFLAG_KSU;
|
||||||
ret_walker = page_table_walk_refill(env, address, rw, mmu_idx);
|
ret_walker = page_table_walk_refill(env, address, access_type, mmu_idx);
|
||||||
env->hflags |= mode;
|
env->hflags |= mode;
|
||||||
if (ret_walker) {
|
if (ret_walker) {
|
||||||
ret = get_physical_address(env, &physical, &prot,
|
ret = get_physical_address(env, &physical, &prot, address,
|
||||||
address, rw, access_type, mmu_idx);
|
access_type, mips_access_type, mmu_idx);
|
||||||
if (ret == TLBRET_MATCH) {
|
if (ret == TLBRET_MATCH) {
|
||||||
tlb_set_page(cs, address & TARGET_PAGE_MASK,
|
tlb_set_page(cs, address & TARGET_PAGE_MASK,
|
||||||
physical & TARGET_PAGE_MASK, prot | PAGE_EXEC,
|
physical & TARGET_PAGE_MASK, prot | PAGE_EXEC,
|
||||||
mmu_idx, TARGET_PAGE_SIZE);
|
mmu_idx, TARGET_PAGE_SIZE);
|
||||||
return 0;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
if (probe) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
raise_mmu_exception(env, address, rw, ret);
|
raise_mmu_exception(env, address, access_type, ret);
|
||||||
return 1;
|
do_raise_exception_err(env, cs->exception_index, env->error_code, retaddr);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef CONFIG_USER_ONLY
|
||||||
|
void tlb_fill(CPUState *cs, target_ulong addr, int size,
|
||||||
|
MMUAccessType access_type, int mmu_idx, uintptr_t retaddr)
|
||||||
|
{
|
||||||
|
mips_cpu_tlb_fill(cs, addr, size, access_type, mmu_idx, false, retaddr);
|
||||||
}
|
}
|
||||||
|
|
||||||
#if !defined(CONFIG_USER_ONLY)
|
|
||||||
hwaddr cpu_mips_translate_address(CPUMIPSState *env, target_ulong address, int rw)
|
hwaddr cpu_mips_translate_address(CPUMIPSState *env, target_ulong address, int rw)
|
||||||
{
|
{
|
||||||
hwaddr physical;
|
hwaddr physical;
|
||||||
|
@ -202,8 +202,9 @@ void cpu_mips_start_count(CPUMIPSState *env);
|
|||||||
void cpu_mips_stop_count(CPUMIPSState *env);
|
void cpu_mips_stop_count(CPUMIPSState *env);
|
||||||
|
|
||||||
/* helper.c */
|
/* helper.c */
|
||||||
int mips_cpu_handle_mmu_fault(CPUState *cpu, vaddr address, int size, int rw,
|
bool mips_cpu_tlb_fill(CPUState *cs, vaddr address, int size,
|
||||||
int mmu_idx);
|
MMUAccessType access_type, int mmu_idx,
|
||||||
|
bool probe, uintptr_t retaddr);
|
||||||
|
|
||||||
/* op_helper.c */
|
/* op_helper.c */
|
||||||
uint32_t float_class_s(uint32_t arg, float_status *fst);
|
uint32_t float_class_s(uint32_t arg, float_status *fst);
|
||||||
|
@ -2669,21 +2669,6 @@ void mips_cpu_do_unaligned_access(CPUState *cs, vaddr addr,
|
|||||||
do_raise_exception_err(env, excp, error_code, retaddr);
|
do_raise_exception_err(env, excp, error_code, retaddr);
|
||||||
}
|
}
|
||||||
|
|
||||||
void tlb_fill(CPUState *cs, target_ulong addr, int size,
|
|
||||||
MMUAccessType access_type, int mmu_idx, uintptr_t retaddr)
|
|
||||||
{
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
ret = mips_cpu_handle_mmu_fault(cs, addr, size, access_type, mmu_idx);
|
|
||||||
if (ret) {
|
|
||||||
MIPSCPU *cpu = MIPS_CPU(cs);
|
|
||||||
CPUMIPSState *env = &cpu->env;
|
|
||||||
|
|
||||||
do_raise_exception_err(env, cs->exception_index,
|
|
||||||
env->error_code, retaddr);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void mips_cpu_unassigned_access(CPUState *cs, hwaddr addr,
|
void mips_cpu_unassigned_access(CPUState *cs, hwaddr addr,
|
||||||
bool is_write, bool is_exec, int unused,
|
bool is_write, bool is_exec, int unused,
|
||||||
unsigned size)
|
unsigned size)
|
||||||
|
Loading…
Reference in New Issue
Block a user