target/alpha: Fix user-only floating-point exceptions
Record the software fp control register, as set by the osf_setsysinfo syscall. Add those masked exceptions to fpcr_exc_enable. Do not raise a signal for masked fp exceptions. Fixes: https://bugs.launchpad.net/bugs/1701835 Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
This commit is contained in:
parent
4a24793290
commit
21ba856499
@ -10223,18 +10223,11 @@ static abi_long do_syscall1(void *cpu_env, int num, abi_long arg1,
|
|||||||
switch (arg1) {
|
switch (arg1) {
|
||||||
case TARGET_GSI_IEEE_FP_CONTROL:
|
case TARGET_GSI_IEEE_FP_CONTROL:
|
||||||
{
|
{
|
||||||
uint64_t swcr, fpcr = cpu_alpha_load_fpcr (cpu_env);
|
uint64_t fpcr = cpu_alpha_load_fpcr(cpu_env);
|
||||||
|
uint64_t swcr = ((CPUAlphaState *)cpu_env)->swcr;
|
||||||
|
|
||||||
/* Copied from linux ieee_fpcr_to_swcr. */
|
swcr &= ~SWCR_STATUS_MASK;
|
||||||
swcr = (fpcr >> 35) & SWCR_STATUS_MASK;
|
swcr |= (fpcr >> 35) & SWCR_STATUS_MASK;
|
||||||
swcr |= (fpcr >> 36) & SWCR_MAP_DMZ;
|
|
||||||
swcr |= (~fpcr >> 48) & (SWCR_TRAP_ENABLE_INV
|
|
||||||
| SWCR_TRAP_ENABLE_DZE
|
|
||||||
| SWCR_TRAP_ENABLE_OVF);
|
|
||||||
swcr |= (~fpcr >> 57) & (SWCR_TRAP_ENABLE_UNF
|
|
||||||
| SWCR_TRAP_ENABLE_INE);
|
|
||||||
swcr |= (fpcr >> 47) & SWCR_MAP_UMZ;
|
|
||||||
swcr |= (~fpcr >> 41) & SWCR_TRAP_ENABLE_DNO;
|
|
||||||
|
|
||||||
if (put_user_u64 (swcr, arg2))
|
if (put_user_u64 (swcr, arg2))
|
||||||
return -TARGET_EFAULT;
|
return -TARGET_EFAULT;
|
||||||
@ -10261,25 +10254,24 @@ static abi_long do_syscall1(void *cpu_env, int num, abi_long arg1,
|
|||||||
switch (arg1) {
|
switch (arg1) {
|
||||||
case TARGET_SSI_IEEE_FP_CONTROL:
|
case TARGET_SSI_IEEE_FP_CONTROL:
|
||||||
{
|
{
|
||||||
uint64_t swcr, fpcr, orig_fpcr;
|
uint64_t swcr, fpcr;
|
||||||
|
|
||||||
if (get_user_u64 (swcr, arg2)) {
|
if (get_user_u64 (swcr, arg2)) {
|
||||||
return -TARGET_EFAULT;
|
return -TARGET_EFAULT;
|
||||||
}
|
}
|
||||||
orig_fpcr = cpu_alpha_load_fpcr(cpu_env);
|
|
||||||
fpcr = orig_fpcr & FPCR_DYN_MASK;
|
|
||||||
|
|
||||||
/* Copied from linux ieee_swcr_to_fpcr. */
|
/*
|
||||||
fpcr |= (swcr & SWCR_STATUS_MASK) << 35;
|
* The kernel calls swcr_update_status to update the
|
||||||
fpcr |= (swcr & SWCR_MAP_DMZ) << 36;
|
* status bits from the fpcr at every point that it
|
||||||
fpcr |= (~swcr & (SWCR_TRAP_ENABLE_INV
|
* could be queried. Therefore, we store the status
|
||||||
| SWCR_TRAP_ENABLE_DZE
|
* bits only in FPCR.
|
||||||
| SWCR_TRAP_ENABLE_OVF)) << 48;
|
*/
|
||||||
fpcr |= (~swcr & (SWCR_TRAP_ENABLE_UNF
|
((CPUAlphaState *)cpu_env)->swcr
|
||||||
| SWCR_TRAP_ENABLE_INE)) << 57;
|
= swcr & (SWCR_TRAP_ENABLE_MASK | SWCR_MAP_MASK);
|
||||||
fpcr |= (swcr & SWCR_MAP_UMZ ? FPCR_UNDZ | FPCR_UNFD : 0);
|
|
||||||
fpcr |= (~swcr & SWCR_TRAP_ENABLE_DNO) << 41;
|
|
||||||
|
|
||||||
|
fpcr = cpu_alpha_load_fpcr(cpu_env);
|
||||||
|
fpcr &= ((uint64_t)FPCR_DYN_MASK << 32);
|
||||||
|
fpcr |= alpha_ieee_swcr_to_fpcr(swcr);
|
||||||
cpu_alpha_store_fpcr(cpu_env, fpcr);
|
cpu_alpha_store_fpcr(cpu_env, fpcr);
|
||||||
ret = 0;
|
ret = 0;
|
||||||
}
|
}
|
||||||
@ -10287,44 +10279,47 @@ static abi_long do_syscall1(void *cpu_env, int num, abi_long arg1,
|
|||||||
|
|
||||||
case TARGET_SSI_IEEE_RAISE_EXCEPTION:
|
case TARGET_SSI_IEEE_RAISE_EXCEPTION:
|
||||||
{
|
{
|
||||||
uint64_t exc, fpcr, orig_fpcr;
|
uint64_t exc, fpcr, fex;
|
||||||
int si_code;
|
|
||||||
|
|
||||||
if (get_user_u64(exc, arg2)) {
|
if (get_user_u64(exc, arg2)) {
|
||||||
return -TARGET_EFAULT;
|
return -TARGET_EFAULT;
|
||||||
}
|
}
|
||||||
|
exc &= SWCR_STATUS_MASK;
|
||||||
orig_fpcr = cpu_alpha_load_fpcr(cpu_env);
|
fpcr = cpu_alpha_load_fpcr(cpu_env);
|
||||||
|
|
||||||
/* We only add to the exception status here. */
|
|
||||||
fpcr = orig_fpcr | ((exc & SWCR_STATUS_MASK) << 35);
|
|
||||||
|
|
||||||
cpu_alpha_store_fpcr(cpu_env, fpcr);
|
|
||||||
ret = 0;
|
|
||||||
|
|
||||||
/* Old exceptions are not signaled. */
|
/* Old exceptions are not signaled. */
|
||||||
fpcr &= ~(orig_fpcr & FPCR_STATUS_MASK);
|
fex = alpha_ieee_fpcr_to_swcr(fpcr);
|
||||||
|
fex = exc & ~fex;
|
||||||
|
fex >>= SWCR_STATUS_TO_EXCSUM_SHIFT;
|
||||||
|
fex &= ((CPUArchState *)cpu_env)->swcr;
|
||||||
|
|
||||||
/* If any exceptions set by this call,
|
/* Update the hardware fpcr. */
|
||||||
and are unmasked, send a signal. */
|
fpcr |= alpha_ieee_swcr_to_fpcr(exc);
|
||||||
si_code = 0;
|
cpu_alpha_store_fpcr(cpu_env, fpcr);
|
||||||
if ((fpcr & (FPCR_INE | FPCR_INED)) == FPCR_INE) {
|
|
||||||
si_code = TARGET_FPE_FLTRES;
|
if (fex) {
|
||||||
}
|
int si_code = TARGET_FPE_FLTUNK;
|
||||||
if ((fpcr & (FPCR_UNF | FPCR_UNFD)) == FPCR_UNF) {
|
target_siginfo_t info;
|
||||||
|
|
||||||
|
if (fex & SWCR_TRAP_ENABLE_DNO) {
|
||||||
si_code = TARGET_FPE_FLTUND;
|
si_code = TARGET_FPE_FLTUND;
|
||||||
}
|
}
|
||||||
if ((fpcr & (FPCR_OVF | FPCR_OVFD)) == FPCR_OVF) {
|
if (fex & SWCR_TRAP_ENABLE_INE) {
|
||||||
|
si_code = TARGET_FPE_FLTRES;
|
||||||
|
}
|
||||||
|
if (fex & SWCR_TRAP_ENABLE_UNF) {
|
||||||
|
si_code = TARGET_FPE_FLTUND;
|
||||||
|
}
|
||||||
|
if (fex & SWCR_TRAP_ENABLE_OVF) {
|
||||||
si_code = TARGET_FPE_FLTOVF;
|
si_code = TARGET_FPE_FLTOVF;
|
||||||
}
|
}
|
||||||
if ((fpcr & (FPCR_DZE | FPCR_DZED)) == FPCR_DZE) {
|
if (fex & SWCR_TRAP_ENABLE_DZE) {
|
||||||
si_code = TARGET_FPE_FLTDIV;
|
si_code = TARGET_FPE_FLTDIV;
|
||||||
}
|
}
|
||||||
if ((fpcr & (FPCR_INV | FPCR_INVD)) == FPCR_INV) {
|
if (fex & SWCR_TRAP_ENABLE_INV) {
|
||||||
si_code = TARGET_FPE_FLTINV;
|
si_code = TARGET_FPE_FLTINV;
|
||||||
}
|
}
|
||||||
if (si_code != 0) {
|
|
||||||
target_siginfo_t info;
|
|
||||||
info.si_signo = SIGFPE;
|
info.si_signo = SIGFPE;
|
||||||
info.si_errno = 0;
|
info.si_errno = 0;
|
||||||
info.si_code = si_code;
|
info.si_code = si_code;
|
||||||
@ -10333,6 +10328,7 @@ static abi_long do_syscall1(void *cpu_env, int num, abi_long arg1,
|
|||||||
queue_signal((CPUArchState *)cpu_env, info.si_signo,
|
queue_signal((CPUArchState *)cpu_env, info.si_signo,
|
||||||
QEMU_SI_FAULT, &info);
|
QEMU_SI_FAULT, &info);
|
||||||
}
|
}
|
||||||
|
ret = 0;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
@ -635,7 +635,8 @@ typedef struct target_siginfo {
|
|||||||
#define TARGET_FPE_FLTRES (6) /* floating point inexact result */
|
#define TARGET_FPE_FLTRES (6) /* floating point inexact result */
|
||||||
#define TARGET_FPE_FLTINV (7) /* floating point invalid operation */
|
#define TARGET_FPE_FLTINV (7) /* floating point invalid operation */
|
||||||
#define TARGET_FPE_FLTSUB (8) /* subscript out of range */
|
#define TARGET_FPE_FLTSUB (8) /* subscript out of range */
|
||||||
#define TARGET_NSIGFPE 8
|
#define TARGET_FPE_FLTUNK (14) /* undiagnosed fp exception */
|
||||||
|
#define TARGET_NSIGFPE 15
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* SIGSEGV si_codes
|
* SIGSEGV si_codes
|
||||||
|
@ -198,6 +198,8 @@ enum {
|
|||||||
#define SWCR_STATUS_DNO (1U << 22)
|
#define SWCR_STATUS_DNO (1U << 22)
|
||||||
#define SWCR_STATUS_MASK ((1U << 23) - (1U << 17))
|
#define SWCR_STATUS_MASK ((1U << 23) - (1U << 17))
|
||||||
|
|
||||||
|
#define SWCR_STATUS_TO_EXCSUM_SHIFT 16
|
||||||
|
|
||||||
#define SWCR_MASK (SWCR_TRAP_ENABLE_MASK | SWCR_MAP_MASK | SWCR_STATUS_MASK)
|
#define SWCR_MASK (SWCR_TRAP_ENABLE_MASK | SWCR_MAP_MASK | SWCR_STATUS_MASK)
|
||||||
|
|
||||||
/* MMU modes definitions */
|
/* MMU modes definitions */
|
||||||
@ -235,6 +237,9 @@ struct CPUAlphaState {
|
|||||||
|
|
||||||
/* The FPCR, and disassembled portions thereof. */
|
/* The FPCR, and disassembled portions thereof. */
|
||||||
uint32_t fpcr;
|
uint32_t fpcr;
|
||||||
|
#ifdef CONFIG_USER_ONLY
|
||||||
|
uint32_t swcr;
|
||||||
|
#endif
|
||||||
uint32_t fpcr_exc_enable;
|
uint32_t fpcr_exc_enable;
|
||||||
float_status fp_status;
|
float_status fp_status;
|
||||||
uint8_t fpcr_dyn_round;
|
uint8_t fpcr_dyn_round;
|
||||||
@ -501,4 +506,41 @@ static inline void cpu_get_tb_cpu_state(CPUAlphaState *env, target_ulong *pc,
|
|||||||
*pflags = env->flags & ENV_FLAG_TB_MASK;
|
*pflags = env->flags & ENV_FLAG_TB_MASK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef CONFIG_USER_ONLY
|
||||||
|
/* Copied from linux ieee_swcr_to_fpcr. */
|
||||||
|
static inline uint64_t alpha_ieee_swcr_to_fpcr(uint64_t swcr)
|
||||||
|
{
|
||||||
|
uint64_t fpcr = 0;
|
||||||
|
|
||||||
|
fpcr |= (swcr & SWCR_STATUS_MASK) << 35;
|
||||||
|
fpcr |= (swcr & SWCR_MAP_DMZ) << 36;
|
||||||
|
fpcr |= (~swcr & (SWCR_TRAP_ENABLE_INV
|
||||||
|
| SWCR_TRAP_ENABLE_DZE
|
||||||
|
| SWCR_TRAP_ENABLE_OVF)) << 48;
|
||||||
|
fpcr |= (~swcr & (SWCR_TRAP_ENABLE_UNF
|
||||||
|
| SWCR_TRAP_ENABLE_INE)) << 57;
|
||||||
|
fpcr |= (swcr & SWCR_MAP_UMZ ? FPCR_UNDZ | FPCR_UNFD : 0);
|
||||||
|
fpcr |= (~swcr & SWCR_TRAP_ENABLE_DNO) << 41;
|
||||||
|
|
||||||
|
return fpcr;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Copied from linux ieee_fpcr_to_swcr. */
|
||||||
|
static inline uint64_t alpha_ieee_fpcr_to_swcr(uint64_t fpcr)
|
||||||
|
{
|
||||||
|
uint64_t swcr = 0;
|
||||||
|
|
||||||
|
swcr |= (fpcr >> 35) & SWCR_STATUS_MASK;
|
||||||
|
swcr |= (fpcr >> 36) & SWCR_MAP_DMZ;
|
||||||
|
swcr |= (~fpcr >> 48) & (SWCR_TRAP_ENABLE_INV
|
||||||
|
| SWCR_TRAP_ENABLE_DZE
|
||||||
|
| SWCR_TRAP_ENABLE_OVF);
|
||||||
|
swcr |= (~fpcr >> 57) & (SWCR_TRAP_ENABLE_UNF | SWCR_TRAP_ENABLE_INE);
|
||||||
|
swcr |= (fpcr >> 47) & SWCR_MAP_UMZ;
|
||||||
|
swcr |= (~fpcr >> 41) & SWCR_TRAP_ENABLE_DNO;
|
||||||
|
|
||||||
|
return swcr;
|
||||||
|
}
|
||||||
|
#endif /* CONFIG_USER_ONLY */
|
||||||
|
|
||||||
#endif /* ALPHA_CPU_H */
|
#endif /* ALPHA_CPU_H */
|
||||||
|
@ -91,11 +91,26 @@ void helper_fp_exc_raise_s(CPUAlphaState *env, uint32_t ignore, uint32_t regno)
|
|||||||
if (exc) {
|
if (exc) {
|
||||||
env->fpcr |= exc;
|
env->fpcr |= exc;
|
||||||
exc &= ~ignore;
|
exc &= ~ignore;
|
||||||
if (exc) {
|
#ifdef CONFIG_USER_ONLY
|
||||||
|
/*
|
||||||
|
* In user mode, the kernel's software handler only
|
||||||
|
* delivers a signal if the exception is enabled.
|
||||||
|
*/
|
||||||
|
if (!(exc & env->fpcr_exc_enable)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
/*
|
||||||
|
* In system mode, the software handler gets invoked
|
||||||
|
* for any non-ignored exception.
|
||||||
|
*/
|
||||||
|
if (!exc) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
exc &= env->fpcr_exc_enable;
|
exc &= env->fpcr_exc_enable;
|
||||||
fp_exc_raise1(env, GETPC(), exc, regno, EXC_M_SWC);
|
fp_exc_raise1(env, GETPC(), exc, regno, EXC_M_SWC);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Input handing without software completion. Trap for all
|
/* Input handing without software completion. Trap for all
|
||||||
|
@ -29,12 +29,12 @@
|
|||||||
#define CONVERT_BIT(X, SRC, DST) \
|
#define CONVERT_BIT(X, SRC, DST) \
|
||||||
(SRC > DST ? (X) / (SRC / DST) & (DST) : ((X) & SRC) * (DST / SRC))
|
(SRC > DST ? (X) / (SRC / DST) & (DST) : ((X) & SRC) * (DST / SRC))
|
||||||
|
|
||||||
uint64_t cpu_alpha_load_fpcr (CPUAlphaState *env)
|
uint64_t cpu_alpha_load_fpcr(CPUAlphaState *env)
|
||||||
{
|
{
|
||||||
return (uint64_t)env->fpcr << 32;
|
return (uint64_t)env->fpcr << 32;
|
||||||
}
|
}
|
||||||
|
|
||||||
void cpu_alpha_store_fpcr (CPUAlphaState *env, uint64_t val)
|
void cpu_alpha_store_fpcr(CPUAlphaState *env, uint64_t val)
|
||||||
{
|
{
|
||||||
uint32_t fpcr = val >> 32;
|
uint32_t fpcr = val >> 32;
|
||||||
uint32_t t = 0;
|
uint32_t t = 0;
|
||||||
@ -67,6 +67,22 @@ void cpu_alpha_store_fpcr (CPUAlphaState *env, uint64_t val)
|
|||||||
|
|
||||||
env->fpcr_flush_to_zero = (fpcr & FPCR_UNFD) && (fpcr & FPCR_UNDZ);
|
env->fpcr_flush_to_zero = (fpcr & FPCR_UNFD) && (fpcr & FPCR_UNDZ);
|
||||||
env->fp_status.flush_inputs_to_zero = (fpcr & FPCR_DNZ) != 0;
|
env->fp_status.flush_inputs_to_zero = (fpcr & FPCR_DNZ) != 0;
|
||||||
|
|
||||||
|
#ifdef CONFIG_USER_ONLY
|
||||||
|
/*
|
||||||
|
* Override some of these bits with the contents of ENV->SWCR.
|
||||||
|
* In system mode, some of these would trap to the kernel, at
|
||||||
|
* which point the kernel's handler would emulate and apply
|
||||||
|
* the software exception mask.
|
||||||
|
*/
|
||||||
|
if (env->swcr & SWCR_MAP_DMZ) {
|
||||||
|
env->fp_status.flush_inputs_to_zero = 1;
|
||||||
|
}
|
||||||
|
if (env->swcr & SWCR_MAP_UMZ) {
|
||||||
|
env->fp_status.flush_to_zero = 1;
|
||||||
|
}
|
||||||
|
env->fpcr_exc_enable &= ~(alpha_ieee_swcr_to_fpcr(env->swcr) >> 32);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
uint64_t helper_load_fpcr(CPUAlphaState *env)
|
uint64_t helper_load_fpcr(CPUAlphaState *env)
|
||||||
|
Loading…
Reference in New Issue
Block a user