/* * Copyright (C) 2002 - 2003 Jeff Dike (jdike@addtoit.com) * Licensed under the GPL */ #include #include #include "kern_util.h" #include "as-layout.h" #include "task.h" #include "sigcontext.h" #include "skas.h" #include "ptrace_user.h" #include "sysdep/ptrace.h" #include "sysdep/ptrace_user.h" #include "os.h" static union uml_pt_regs ksig_regs[UM_NR_CPUS]; void sig_handler_common_skas(int sig, void *sc_ptr) { struct sigcontext *sc = sc_ptr; union uml_pt_regs *r; void (*handler)(int, union uml_pt_regs *); int save_user, save_errno = errno; /* This is done because to allow SIGSEGV to be delivered inside a SEGV * handler. This can happen in copy_user, and if SEGV is disabled, * the process will die. * XXX Figure out why this is better than SA_NODEFER */ if(sig == SIGSEGV) { change_sig(SIGSEGV, 1); /* For segfaults, we want the data from the * sigcontext. In this case, we don't want to mangle * the process registers, so use a static set of * registers. For other signals, the process * registers are OK. */ r = &ksig_regs[cpu()]; copy_sc(r, sc_ptr); } else r = TASK_REGS(get_current()); save_user = r->skas.is_user; r->skas.is_user = 0; if ( sig == SIGFPE || sig == SIGSEGV || sig == SIGBUS || sig == SIGILL || sig == SIGTRAP ) { GET_FAULTINFO_FROM_SC(r->skas.faultinfo, sc); } change_sig(SIGUSR1, 1); handler = sig_info[sig]; /* unblock SIGALRM, SIGVTALRM, SIGIO if sig isn't IRQ signal */ if (sig != SIGIO && sig != SIGWINCH && sig != SIGVTALRM && sig != SIGALRM) unblock_signals(); handler(sig, r); errno = save_errno; r->skas.is_user = save_user; }