mirror of https://github.com/xemu-project/xemu.git
target/ppc: Machine check on invalid real address access on POWER9/10
ppc currently silently accepts invalid real address access. Catch these and turn them into machine checks on POWER9/10 machines. Signed-off-by: Nicholas Piggin <npiggin@gmail.com> Reviewed-by: Cédric Le Goater <clg@kaod.org> Message-ID: <20230703120301.45313-1-npiggin@gmail.com> Signed-off-by: Daniel Henrique Barboza <danielhb413@gmail.com>
This commit is contained in:
parent
e1a821d471
commit
55a7fa34f8
|
@ -7335,6 +7335,7 @@ static const struct TCGCPUOps ppc_tcg_ops = {
|
||||||
.cpu_exec_enter = ppc_cpu_exec_enter,
|
.cpu_exec_enter = ppc_cpu_exec_enter,
|
||||||
.cpu_exec_exit = ppc_cpu_exec_exit,
|
.cpu_exec_exit = ppc_cpu_exec_exit,
|
||||||
.do_unaligned_access = ppc_cpu_do_unaligned_access,
|
.do_unaligned_access = ppc_cpu_do_unaligned_access,
|
||||||
|
.do_transaction_failed = ppc_cpu_do_transaction_failed,
|
||||||
#endif /* !CONFIG_USER_ONLY */
|
#endif /* !CONFIG_USER_ONLY */
|
||||||
};
|
};
|
||||||
#endif /* CONFIG_TCG */
|
#endif /* CONFIG_TCG */
|
||||||
|
|
|
@ -1428,7 +1428,9 @@ static void powerpc_excp_books(PowerPCCPU *cpu, int excp)
|
||||||
/* machine check exceptions don't have ME set */
|
/* machine check exceptions don't have ME set */
|
||||||
new_msr &= ~((target_ulong)1 << MSR_ME);
|
new_msr &= ~((target_ulong)1 << MSR_ME);
|
||||||
|
|
||||||
|
msr |= env->error_code;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case POWERPC_EXCP_DSI: /* Data storage exception */
|
case POWERPC_EXCP_DSI: /* Data storage exception */
|
||||||
trace_ppc_excp_dsi(env->spr[SPR_DSISR], env->spr[SPR_DAR]);
|
trace_ppc_excp_dsi(env->spr[SPR_DSISR], env->spr[SPR_DAR]);
|
||||||
break;
|
break;
|
||||||
|
@ -3188,5 +3190,52 @@ void ppc_cpu_do_unaligned_access(CPUState *cs, vaddr vaddr,
|
||||||
env->error_code = insn & 0x03FF0000;
|
env->error_code = insn & 0x03FF0000;
|
||||||
cpu_loop_exit(cs);
|
cpu_loop_exit(cs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ppc_cpu_do_transaction_failed(CPUState *cs, hwaddr physaddr,
|
||||||
|
vaddr vaddr, unsigned size,
|
||||||
|
MMUAccessType access_type,
|
||||||
|
int mmu_idx, MemTxAttrs attrs,
|
||||||
|
MemTxResult response, uintptr_t retaddr)
|
||||||
|
{
|
||||||
|
CPUPPCState *env = cs->env_ptr;
|
||||||
|
|
||||||
|
switch (env->excp_model) {
|
||||||
|
#if defined(TARGET_PPC64)
|
||||||
|
case POWERPC_EXCP_POWER9:
|
||||||
|
case POWERPC_EXCP_POWER10:
|
||||||
|
/*
|
||||||
|
* Machine check codes can be found in processor User Manual or
|
||||||
|
* Linux or skiboot source.
|
||||||
|
*/
|
||||||
|
if (access_type == MMU_DATA_LOAD) {
|
||||||
|
env->spr[SPR_DAR] = vaddr;
|
||||||
|
env->spr[SPR_DSISR] = PPC_BIT(57);
|
||||||
|
env->error_code = PPC_BIT(42);
|
||||||
|
|
||||||
|
} else if (access_type == MMU_DATA_STORE) {
|
||||||
|
/*
|
||||||
|
* MCE for stores in POWER is asynchronous so hardware does
|
||||||
|
* not set DAR, but QEMU can do better.
|
||||||
|
*/
|
||||||
|
env->spr[SPR_DAR] = vaddr;
|
||||||
|
env->error_code = PPC_BIT(36) | PPC_BIT(43) | PPC_BIT(45);
|
||||||
|
env->error_code |= PPC_BIT(42);
|
||||||
|
|
||||||
|
} else { /* Fetch */
|
||||||
|
env->error_code = PPC_BIT(36) | PPC_BIT(44) | PPC_BIT(45);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
default:
|
||||||
|
/*
|
||||||
|
* TODO: Check behaviour for other CPUs, for now do nothing.
|
||||||
|
* Could add a basic MCE even if real hardware ignores.
|
||||||
|
*/
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
cs->exception_index = POWERPC_EXCP_MCHECK;
|
||||||
|
cpu_loop_exit_restore(cs, retaddr);
|
||||||
|
}
|
||||||
#endif /* CONFIG_TCG */
|
#endif /* CONFIG_TCG */
|
||||||
#endif /* !CONFIG_USER_ONLY */
|
#endif /* !CONFIG_USER_ONLY */
|
||||||
|
|
|
@ -296,6 +296,11 @@ bool ppc_cpu_tlb_fill(CPUState *cs, vaddr address, int size,
|
||||||
G_NORETURN void ppc_cpu_do_unaligned_access(CPUState *cs, vaddr addr,
|
G_NORETURN void ppc_cpu_do_unaligned_access(CPUState *cs, vaddr addr,
|
||||||
MMUAccessType access_type, int mmu_idx,
|
MMUAccessType access_type, int mmu_idx,
|
||||||
uintptr_t retaddr);
|
uintptr_t retaddr);
|
||||||
|
void ppc_cpu_do_transaction_failed(CPUState *cs, hwaddr physaddr,
|
||||||
|
vaddr addr, unsigned size,
|
||||||
|
MMUAccessType access_type,
|
||||||
|
int mmu_idx, MemTxAttrs attrs,
|
||||||
|
MemTxResult response, uintptr_t retaddr);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
FIELD(GER_MSK, XMSK, 0, 4)
|
FIELD(GER_MSK, XMSK, 0, 4)
|
||||||
|
|
Loading…
Reference in New Issue