Commit ee3d9bd4de
("uml: simplify SIGSEGV
handling"), while greatly simplifying the kernel SIGSEGV handler that
runs in the process address space, introduced a bug which corrupts FP
state in the process.
Previously, the SIGSEGV handler called the sigreturn system call by hand - it
couldn't return through the restorer provided to it because that could try to
call the libc restorer which likely wouldn't exist in the process address
space. So, it blocked off some signals, including SIGUSR1, on entry to the
SIGSEGV handler, queued a SIGUSR1 to itself, and invoked sigreturn. The
SIGUSR1 was delivered, and was visible to the UML kernel after sigreturn
finished.
The commit eliminated the signal masking and the call to sigreturn. The
handler simply hits itself with a SIGTRAP to let the UML kernel know that it
is finished. UML then restores the process registers, which effectively
longjmps the process out of the signal handler, skipping sigreturn's restoring
of register state and the signal mask.
The bug is that the host apparently sets used_fp to 0 when it saves the
process FP state in the sigcontext on the process signal stack. Thus, when
the process is longjmped out of the handler, its FP state is corrupt because
it wasn't saved on the context switch to the UML kernel.
This manifested itself as sleep hanging. For some reason, sleep uses floating
point in order to calculate the sleep interval. When a page fault corrupts
its FP state, it is faked into essentially sleeping forever.
This patch saves the FP state before entering the SIGSEGV handler and restores
it afterwards.
Signed-off-by: Jeff Dike <jdike@linux.intel.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
79 lines
1.7 KiB
C
79 lines
1.7 KiB
C
/*
|
|
* Copyright (C) 2002 Jeff Dike (jdike@karaya.com)
|
|
* Licensed under the GPL
|
|
*/
|
|
|
|
#ifndef __SYSDEP_I386_PTRACE_USER_H__
|
|
#define __SYSDEP_I386_PTRACE_USER_H__
|
|
|
|
#include <sys/ptrace.h>
|
|
#include <linux/ptrace.h>
|
|
#include <asm/ptrace.h>
|
|
#include "user_constants.h"
|
|
|
|
#define PT_OFFSET(r) ((r) * sizeof(long))
|
|
|
|
#define PT_SYSCALL_NR(regs) ((regs)[ORIG_EAX])
|
|
#define PT_SYSCALL_NR_OFFSET PT_OFFSET(ORIG_EAX)
|
|
|
|
#define PT_SYSCALL_ARG1_OFFSET PT_OFFSET(EBX)
|
|
#define PT_SYSCALL_ARG2_OFFSET PT_OFFSET(ECX)
|
|
#define PT_SYSCALL_ARG3_OFFSET PT_OFFSET(EDX)
|
|
#define PT_SYSCALL_ARG4_OFFSET PT_OFFSET(ESI)
|
|
#define PT_SYSCALL_ARG5_OFFSET PT_OFFSET(EDI)
|
|
#define PT_SYSCALL_ARG6_OFFSET PT_OFFSET(EBP)
|
|
|
|
#define PT_SYSCALL_RET_OFFSET PT_OFFSET(EAX)
|
|
|
|
#define REGS_SYSCALL_NR EAX /* This is used before a system call */
|
|
#define REGS_SYSCALL_ARG1 EBX
|
|
#define REGS_SYSCALL_ARG2 ECX
|
|
#define REGS_SYSCALL_ARG3 EDX
|
|
#define REGS_SYSCALL_ARG4 ESI
|
|
#define REGS_SYSCALL_ARG5 EDI
|
|
#define REGS_SYSCALL_ARG6 EBP
|
|
|
|
#define REGS_IP_INDEX EIP
|
|
#define REGS_SP_INDEX UESP
|
|
|
|
#define PT_IP_OFFSET PT_OFFSET(EIP)
|
|
#define PT_IP(regs) ((regs)[EIP])
|
|
#define PT_SP_OFFSET PT_OFFSET(UESP)
|
|
#define PT_SP(regs) ((regs)[UESP])
|
|
|
|
#define FP_SIZE ((HOST_XFP_SIZE > HOST_FP_SIZE) ? HOST_XFP_SIZE : HOST_FP_SIZE)
|
|
|
|
#ifndef FRAME_SIZE
|
|
#define FRAME_SIZE (17)
|
|
#endif
|
|
#define FRAME_SIZE_OFFSET (FRAME_SIZE * sizeof(unsigned long))
|
|
|
|
#define FP_FRAME_SIZE (27)
|
|
#define FPX_FRAME_SIZE (128)
|
|
|
|
#ifdef PTRACE_GETREGS
|
|
#define UM_HAVE_GETREGS
|
|
#endif
|
|
|
|
#ifdef PTRACE_SETREGS
|
|
#define UM_HAVE_SETREGS
|
|
#endif
|
|
|
|
#ifdef PTRACE_GETFPREGS
|
|
#define UM_HAVE_GETFPREGS
|
|
#endif
|
|
|
|
#ifdef PTRACE_SETFPREGS
|
|
#define UM_HAVE_SETFPREGS
|
|
#endif
|
|
|
|
#ifdef PTRACE_GETFPXREGS
|
|
#define UM_HAVE_GETFPXREGS
|
|
#endif
|
|
|
|
#ifdef PTRACE_SETFPXREGS
|
|
#define UM_HAVE_SETFPXREGS
|
|
#endif
|
|
|
|
#endif
|