The patch titled
i386: fix incorrect FP signal delivery
has been added to the -mm tree. Its filename is
i386-fix-incorrect-fp-signal-delivery.patch
Patches currently in -mm which might be from [EMAIL PROTECTED] are
i386-fix-incorrect-fp-signal-delivery.patch
docs-fix-misinformation-about.patch
From: Chuck Ebbert <[EMAIL PROTECTED]>
This patch fixes a problem with incorrect floating-point exception signal
delivery on i386 kernels. In some cases, an error code of zero is
delivered instead of the correct code, as the output from my test program
shows:
Before patch:
$ ./fpsig
handler: signum = 8, errno = 0, code = 0
handler: fpu cwd = 0xb40, fpu swd = 0xbaa0
After:
$ ./fpsig
handler: signum = 8, errno = 0, code = 6
handler: fpu cwd = 0xb40, fpu swd = 0xbaa0
2.4 also has this problem; the patch applies with offsets on 2.4.31
but I didn't test it beyond that. Patch also applies to 2.6.13-rc6-mm1
with offsets.
x86-64 also looks to be affected but I have no way to test it
Test program:
/* i387 fp signal test */
#define _GNU_SOURCE
#include <stdlib.h>
#include <unistd.h>
#include <stdio.h>
#include <signal.h>
#include <errno.h>
__attribute__ ((aligned(4096))) unsigned char altstack[4096];
unsigned short cw = 0x0b40; /* unmask all exceptions, round up */
struct sigaction sa;
stack_t ss = {
.ss_sp = &altstack[2047],
.ss_size = sizeof(altstack)/2,
};
static void handler(int nr, siginfo_t *si, void *uc)
{
printf("handler: signum = %d, errno = %d, code = %d\n",
si->si_signo, si->si_errno, si->si_code);
printf("handler: fpu cwd = 0x%hx, fpu swd = 0x%hx\n",
*(unsigned short *)&altstack[0xd84],
*(unsigned short *)&altstack[0xd88]);
exit(1);
}
int main(int argc, char * const argv[])
{
sa.sa_sigaction = handler;
sa.sa_flags = SA_ONSTACK | SA_SIGINFO;
if (sigaltstack(&ss, 0))
perror("sigaltstack");
if (sigaction(SIGFPE, &sa, NULL))
perror("sigaction");
asm volatile ("fnclex ; fldcw %0" : : "m" (cw));
asm volatile ( /* st(1) = 3.0, st = 1.0 */
"fld1 ; fld1 ; faddp ; fld1 ; faddp ; fld1");
asm volatile (
"fdivp ; fwait"); /* 1.0 / 3.0 */
return 0;
}
Signed-off-by: Chuck Ebbert <[EMAIL PROTECTED]>
Signed-off-by: Andrew Morton <[EMAIL PROTECTED]>
---
arch/i386/kernel/traps.c | 8 ++++++--
1 files changed, 6 insertions(+), 2 deletions(-)
diff -puN arch/i386/kernel/traps.c~i386-fix-incorrect-fp-signal-delivery
arch/i386/kernel/traps.c
--- devel/arch/i386/kernel/traps.c~i386-fix-incorrect-fp-signal-delivery
2005-08-22 16:53:30.000000000 -0700
+++ devel-akpm/arch/i386/kernel/traps.c 2005-08-22 16:53:30.000000000 -0700
@@ -873,7 +873,7 @@ void math_error(void __user *eip)
{
struct task_struct * task;
siginfo_t info;
- unsigned short cwd, swd;
+ unsigned short cwd, swd, wd;
/*
* Save the info for the exception handler and clear the error.
@@ -898,7 +898,11 @@ void math_error(void __user *eip)
*/
cwd = get_fpu_cwd(task);
swd = get_fpu_swd(task);
- switch (((~cwd) & swd & 0x3f) | (swd & 0x240)) {
+ wd = swd & 0x3f & ~cwd;
+ if (wd & 1)
+ wd |= swd & 0x240;
+
+ switch (wd) {
case 0x000:
default:
break;
_
-
To unsubscribe from this list: send the line "unsubscribe mm-commits" in
the body of a message to [EMAIL PROTECTED]
More majordomo info at http://vger.kernel.org/majordomo-info.html