mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-31 05:51:53 -06:00
target/arm: Implement arm_cpu_record_sigbus
Because of the complexity of setting ESR, re-use the existing arm_cpu_do_unaligned_access function. This means we have to handle the exception ourselves in cpu_loop, transforming it to the appropriate signal. Reviewed-by: Warner Losh <imp@bsdimp.com> Reviewed-by: Peter Maydell <peter.maydell@linaro.org> Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
This commit is contained in:
parent
e7424abc20
commit
39a099ca25
6 changed files with 45 additions and 7 deletions
|
@ -79,7 +79,7 @@
|
||||||
void cpu_loop(CPUARMState *env)
|
void cpu_loop(CPUARMState *env)
|
||||||
{
|
{
|
||||||
CPUState *cs = env_cpu(env);
|
CPUState *cs = env_cpu(env);
|
||||||
int trapnr, ec, fsc, si_code;
|
int trapnr, ec, fsc, si_code, si_signo;
|
||||||
abi_long ret;
|
abi_long ret;
|
||||||
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
|
@ -121,20 +121,26 @@ void cpu_loop(CPUARMState *env)
|
||||||
fsc = extract32(env->exception.syndrome, 0, 6);
|
fsc = extract32(env->exception.syndrome, 0, 6);
|
||||||
switch (fsc) {
|
switch (fsc) {
|
||||||
case 0x04 ... 0x07: /* Translation fault, level {0-3} */
|
case 0x04 ... 0x07: /* Translation fault, level {0-3} */
|
||||||
|
si_signo = TARGET_SIGSEGV;
|
||||||
si_code = TARGET_SEGV_MAPERR;
|
si_code = TARGET_SEGV_MAPERR;
|
||||||
break;
|
break;
|
||||||
case 0x09 ... 0x0b: /* Access flag fault, level {1-3} */
|
case 0x09 ... 0x0b: /* Access flag fault, level {1-3} */
|
||||||
case 0x0d ... 0x0f: /* Permission fault, level {1-3} */
|
case 0x0d ... 0x0f: /* Permission fault, level {1-3} */
|
||||||
|
si_signo = TARGET_SIGSEGV;
|
||||||
si_code = TARGET_SEGV_ACCERR;
|
si_code = TARGET_SEGV_ACCERR;
|
||||||
break;
|
break;
|
||||||
case 0x11: /* Synchronous Tag Check Fault */
|
case 0x11: /* Synchronous Tag Check Fault */
|
||||||
|
si_signo = TARGET_SIGSEGV;
|
||||||
si_code = TARGET_SEGV_MTESERR;
|
si_code = TARGET_SEGV_MTESERR;
|
||||||
break;
|
break;
|
||||||
|
case 0x21: /* Alignment fault */
|
||||||
|
si_signo = TARGET_SIGBUS;
|
||||||
|
si_code = TARGET_BUS_ADRALN;
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
g_assert_not_reached();
|
g_assert_not_reached();
|
||||||
}
|
}
|
||||||
|
force_sig_fault(si_signo, si_code, env->exception.vaddress);
|
||||||
force_sig_fault(TARGET_SIGSEGV, si_code, env->exception.vaddress);
|
|
||||||
break;
|
break;
|
||||||
case EXCP_DEBUG:
|
case EXCP_DEBUG:
|
||||||
case EXCP_BKPT:
|
case EXCP_BKPT:
|
||||||
|
|
|
@ -25,6 +25,7 @@
|
||||||
#include "cpu_loop-common.h"
|
#include "cpu_loop-common.h"
|
||||||
#include "signal-common.h"
|
#include "signal-common.h"
|
||||||
#include "semihosting/common-semi.h"
|
#include "semihosting/common-semi.h"
|
||||||
|
#include "target/arm/syndrome.h"
|
||||||
|
|
||||||
#define get_user_code_u32(x, gaddr, env) \
|
#define get_user_code_u32(x, gaddr, env) \
|
||||||
({ abi_long __r = get_user_u32((x), (gaddr)); \
|
({ abi_long __r = get_user_u32((x), (gaddr)); \
|
||||||
|
@ -280,7 +281,7 @@ static bool emulate_arm_fpa11(CPUARMState *env, uint32_t opcode)
|
||||||
void cpu_loop(CPUARMState *env)
|
void cpu_loop(CPUARMState *env)
|
||||||
{
|
{
|
||||||
CPUState *cs = env_cpu(env);
|
CPUState *cs = env_cpu(env);
|
||||||
int trapnr;
|
int trapnr, si_signo, si_code;
|
||||||
unsigned int n, insn;
|
unsigned int n, insn;
|
||||||
abi_ulong ret;
|
abi_ulong ret;
|
||||||
|
|
||||||
|
@ -423,9 +424,30 @@ void cpu_loop(CPUARMState *env)
|
||||||
break;
|
break;
|
||||||
case EXCP_PREFETCH_ABORT:
|
case EXCP_PREFETCH_ABORT:
|
||||||
case EXCP_DATA_ABORT:
|
case EXCP_DATA_ABORT:
|
||||||
/* XXX: check env->error_code */
|
/* For user-only we don't set TTBCR_EAE, so look at the FSR. */
|
||||||
force_sig_fault(TARGET_SIGSEGV, TARGET_SEGV_MAPERR,
|
switch (env->exception.fsr & 0x1f) {
|
||||||
env->exception.vaddress);
|
case 0x1: /* Alignment */
|
||||||
|
si_signo = TARGET_SIGBUS;
|
||||||
|
si_code = TARGET_BUS_ADRALN;
|
||||||
|
break;
|
||||||
|
case 0x3: /* Access flag fault, level 1 */
|
||||||
|
case 0x6: /* Access flag fault, level 2 */
|
||||||
|
case 0x9: /* Domain fault, level 1 */
|
||||||
|
case 0xb: /* Domain fault, level 2 */
|
||||||
|
case 0xd: /* Permision fault, level 1 */
|
||||||
|
case 0xf: /* Permision fault, level 2 */
|
||||||
|
si_signo = TARGET_SIGSEGV;
|
||||||
|
si_code = TARGET_SEGV_ACCERR;
|
||||||
|
break;
|
||||||
|
case 0x5: /* Translation fault, level 1 */
|
||||||
|
case 0x7: /* Translation fault, level 2 */
|
||||||
|
si_signo = TARGET_SIGSEGV;
|
||||||
|
si_code = TARGET_SEGV_MAPERR;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
g_assert_not_reached();
|
||||||
|
}
|
||||||
|
force_sig_fault(si_signo, si_code, env->exception.vaddress);
|
||||||
break;
|
break;
|
||||||
case EXCP_DEBUG:
|
case EXCP_DEBUG:
|
||||||
case EXCP_BKPT:
|
case EXCP_BKPT:
|
||||||
|
|
|
@ -2035,6 +2035,7 @@ static const struct TCGCPUOps arm_tcg_ops = {
|
||||||
|
|
||||||
#ifdef CONFIG_USER_ONLY
|
#ifdef CONFIG_USER_ONLY
|
||||||
.record_sigsegv = arm_cpu_record_sigsegv,
|
.record_sigsegv = arm_cpu_record_sigsegv,
|
||||||
|
.record_sigbus = arm_cpu_record_sigbus,
|
||||||
#else
|
#else
|
||||||
.tlb_fill = arm_cpu_tlb_fill,
|
.tlb_fill = arm_cpu_tlb_fill,
|
||||||
.cpu_exec_interrupt = arm_cpu_exec_interrupt,
|
.cpu_exec_interrupt = arm_cpu_exec_interrupt,
|
||||||
|
|
|
@ -902,6 +902,7 @@ static const struct TCGCPUOps arm_v7m_tcg_ops = {
|
||||||
|
|
||||||
#ifdef CONFIG_USER_ONLY
|
#ifdef CONFIG_USER_ONLY
|
||||||
.record_sigsegv = arm_cpu_record_sigsegv,
|
.record_sigsegv = arm_cpu_record_sigsegv,
|
||||||
|
.record_sigbus = arm_cpu_record_sigbus,
|
||||||
#else
|
#else
|
||||||
.tlb_fill = arm_cpu_tlb_fill,
|
.tlb_fill = arm_cpu_tlb_fill,
|
||||||
.cpu_exec_interrupt = arm_v7m_cpu_exec_interrupt,
|
.cpu_exec_interrupt = arm_v7m_cpu_exec_interrupt,
|
||||||
|
|
|
@ -548,6 +548,8 @@ static inline bool arm_extabort_type(MemTxResult result)
|
||||||
void arm_cpu_record_sigsegv(CPUState *cpu, vaddr addr,
|
void arm_cpu_record_sigsegv(CPUState *cpu, vaddr addr,
|
||||||
MMUAccessType access_type,
|
MMUAccessType access_type,
|
||||||
bool maperr, uintptr_t ra);
|
bool maperr, uintptr_t ra);
|
||||||
|
void arm_cpu_record_sigbus(CPUState *cpu, vaddr addr,
|
||||||
|
MMUAccessType access_type, uintptr_t ra);
|
||||||
#else
|
#else
|
||||||
bool arm_cpu_tlb_fill(CPUState *cs, vaddr address, int size,
|
bool arm_cpu_tlb_fill(CPUState *cs, vaddr address, int size,
|
||||||
MMUAccessType access_type, int mmu_idx,
|
MMUAccessType access_type, int mmu_idx,
|
||||||
|
|
|
@ -213,4 +213,10 @@ void arm_cpu_record_sigsegv(CPUState *cs, vaddr addr,
|
||||||
cpu_restore_state(cs, ra, true);
|
cpu_restore_state(cs, ra, true);
|
||||||
arm_deliver_fault(cpu, addr, access_type, MMU_USER_IDX, &fi);
|
arm_deliver_fault(cpu, addr, access_type, MMU_USER_IDX, &fi);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void arm_cpu_record_sigbus(CPUState *cs, vaddr addr,
|
||||||
|
MMUAccessType access_type, uintptr_t ra)
|
||||||
|
{
|
||||||
|
arm_cpu_do_unaligned_access(cs, addr, access_type, MMU_USER_IDX, ra);
|
||||||
|
}
|
||||||
#endif /* !defined(CONFIG_USER_ONLY) */
|
#endif /* !defined(CONFIG_USER_ONLY) */
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue