2009-03-27 14:25:14 +01:00
/*
* Signal handling
*
* Copyright ( C ) 2008 - 2009 Michal Simek < monstr @ monstr . eu >
* Copyright ( C ) 2008 - 2009 PetaLogix
* Copyright ( C ) 2003 , 2004 John Williams < jwilliams @ itee . uq . edu . au >
* Copyright ( C ) 2001 NEC Corporation
* Copyright ( C ) 2001 Miles Bader < miles @ gnu . org >
* Copyright ( C ) 1999 , 2000 Niibe Yutaka & Kaz Kojima
* Copyright ( C ) 1991 , 1992 Linus Torvalds
*
* 1997 - 11 - 28 Modified for POSIX .1 b signals by Richard Henderson
*
* This file was was derived from the sh version , arch / sh / kernel / signal . c
*
* This file is subject to the terms and conditions of the GNU General
* Public License . See the file COPYING in the main directory of this
* archive for more details .
*/
# include <linux/sched.h>
# include <linux/mm.h>
# include <linux/smp.h>
# include <linux/kernel.h>
# include <linux/signal.h>
# include <linux/errno.h>
# include <linux/wait.h>
# include <linux/ptrace.h>
# include <linux/unistd.h>
# include <linux/stddef.h>
# include <linux/personality.h>
# include <linux/percpu.h>
# include <linux/linkage.h>
2012-04-24 02:03:06 -04:00
# include <linux/tracehook.h>
2009-03-27 14:25:14 +01:00
# include <asm/entry.h>
# include <asm/ucontext.h>
# include <linux/uaccess.h>
# include <asm/pgtable.h>
# include <asm/pgalloc.h>
# include <linux/syscalls.h>
# include <asm/cacheflush.h>
# include <asm/syscalls.h>
/*
* Do a signal return ; undo the signal stack .
*/
struct sigframe {
struct sigcontext sc ;
unsigned long extramask [ _NSIG_WORDS - 1 ] ;
unsigned long tramp [ 2 ] ; /* signal trampoline */
} ;
struct rt_sigframe {
struct siginfo info ;
struct ucontext uc ;
unsigned long tramp [ 2 ] ; /* signal trampoline */
} ;
2009-05-01 13:37:46 +00:00
static int restore_sigcontext ( struct pt_regs * regs ,
struct sigcontext __user * sc , int * rval_p )
2009-03-27 14:25:14 +01:00
{
unsigned int err = 0 ;
# define COPY(x) {err |= __get_user(regs->x, &sc->regs.x); }
COPY ( r0 ) ;
COPY ( r1 ) ;
COPY ( r2 ) ; COPY ( r3 ) ; COPY ( r4 ) ; COPY ( r5 ) ;
COPY ( r6 ) ; COPY ( r7 ) ; COPY ( r8 ) ; COPY ( r9 ) ;
COPY ( r10 ) ; COPY ( r11 ) ; COPY ( r12 ) ; COPY ( r13 ) ;
COPY ( r14 ) ; COPY ( r15 ) ; COPY ( r16 ) ; COPY ( r17 ) ;
COPY ( r18 ) ; COPY ( r19 ) ; COPY ( r20 ) ; COPY ( r21 ) ;
COPY ( r22 ) ; COPY ( r23 ) ; COPY ( r24 ) ; COPY ( r25 ) ;
COPY ( r26 ) ; COPY ( r27 ) ; COPY ( r28 ) ; COPY ( r29 ) ;
COPY ( r30 ) ; COPY ( r31 ) ;
COPY ( pc ) ; COPY ( ear ) ; COPY ( esr ) ; COPY ( fsr ) ;
# undef COPY
* rval_p = regs - > r3 ;
return err ;
}
2009-06-18 19:55:29 +02:00
asmlinkage long sys_rt_sigreturn ( struct pt_regs * regs )
2009-03-27 14:25:14 +01:00
{
2009-05-01 13:37:46 +00:00
struct rt_sigframe __user * frame =
2011-01-31 15:10:04 +01:00
( struct rt_sigframe __user * ) ( regs - > r1 ) ;
2009-04-21 14:08:47 +02:00
2009-03-27 14:25:14 +01:00
sigset_t set ;
int rval ;
2012-04-22 03:32:18 -04:00
/* Always make any pending restarted system calls return -EINTR */
current_thread_info ( ) - > restart_block . fn = do_no_restart_syscall ;
2009-03-27 14:25:14 +01:00
if ( ! access_ok ( VERIFY_READ , frame , sizeof ( * frame ) ) )
goto badframe ;
if ( __copy_from_user ( & set , & frame - > uc . uc_sigmask , sizeof ( set ) ) )
goto badframe ;
2012-05-11 10:58:05 +10:00
set_current_blocked ( & set ) ;
2009-03-27 14:25:14 +01:00
if ( restore_sigcontext ( regs , & frame - > uc . uc_mcontext , & rval ) )
goto badframe ;
2012-12-23 03:08:53 -05:00
if ( restore_altstack ( & frame - > uc . uc_stack ) )
2009-05-01 13:37:46 +00:00
goto badframe ;
2009-03-27 14:25:14 +01:00
return rval ;
badframe :
force_sig ( SIGSEGV , current ) ;
return 0 ;
}
/*
* Set up a signal frame .
*/
static int
2009-05-01 13:37:46 +00:00
setup_sigcontext ( struct sigcontext __user * sc , struct pt_regs * regs ,
2009-03-27 14:25:14 +01:00
unsigned long mask )
{
int err = 0 ;
# define COPY(x) {err |= __put_user(regs->x, &sc->regs.x); }
COPY ( r0 ) ;
COPY ( r1 ) ;
COPY ( r2 ) ; COPY ( r3 ) ; COPY ( r4 ) ; COPY ( r5 ) ;
COPY ( r6 ) ; COPY ( r7 ) ; COPY ( r8 ) ; COPY ( r9 ) ;
COPY ( r10 ) ; COPY ( r11 ) ; COPY ( r12 ) ; COPY ( r13 ) ;
COPY ( r14 ) ; COPY ( r15 ) ; COPY ( r16 ) ; COPY ( r17 ) ;
COPY ( r18 ) ; COPY ( r19 ) ; COPY ( r20 ) ; COPY ( r21 ) ;
COPY ( r22 ) ; COPY ( r23 ) ; COPY ( r24 ) ; COPY ( r25 ) ;
COPY ( r26 ) ; COPY ( r27 ) ; COPY ( r28 ) ; COPY ( r29 ) ;
COPY ( r30 ) ; COPY ( r31 ) ;
COPY ( pc ) ; COPY ( ear ) ; COPY ( esr ) ; COPY ( fsr ) ;
# undef COPY
err | = __put_user ( mask , & sc - > oldmask ) ;
return err ;
}
/*
* Determine which stack to use . .
*/
2009-05-01 13:37:46 +00:00
static inline void __user *
2009-03-27 14:25:14 +01:00
get_sigframe ( struct k_sigaction * ka , struct pt_regs * regs , size_t frame_size )
{
/* Default to using normal stack */
unsigned long sp = regs - > r1 ;
if ( ( ka - > sa . sa_flags & SA_ONSTACK ) ! = 0 & & ! on_sig_stack ( sp ) )
sp = current - > sas_ss_sp + current - > sas_ss_size ;
2009-05-01 13:37:46 +00:00
return ( void __user * ) ( ( sp - frame_size ) & - 8UL ) ;
2009-03-27 14:25:14 +01:00
}
2012-05-11 10:58:04 +10:00
static int setup_rt_frame ( int sig , struct k_sigaction * ka , siginfo_t * info ,
2009-03-27 14:25:14 +01:00
sigset_t * set , struct pt_regs * regs )
{
2009-05-01 13:37:46 +00:00
struct rt_sigframe __user * frame ;
2009-03-27 14:25:14 +01:00
int err = 0 ;
int signal ;
2009-12-10 11:43:57 +01:00
unsigned long address = 0 ;
# ifdef CONFIG_MMU
pmd_t * pmdp ;
pte_t * ptep ;
# endif
2009-03-27 14:25:14 +01:00
frame = get_sigframe ( ka , regs , sizeof ( * frame ) ) ;
if ( ! access_ok ( VERIFY_WRITE , frame , sizeof ( * frame ) ) )
goto give_sigsegv ;
signal = current_thread_info ( ) - > exec_domain
& & current_thread_info ( ) - > exec_domain - > signal_invmap
& & sig < 32
? current_thread_info ( ) - > exec_domain - > signal_invmap [ sig ]
: sig ;
2009-05-12 12:10:52 +02:00
if ( info )
err | = copy_siginfo_to_user ( & frame - > info , info ) ;
2009-03-27 14:25:14 +01:00
/* Create the ucontext. */
err | = __put_user ( 0 , & frame - > uc . uc_flags ) ;
2011-02-07 12:25:37 +01:00
err | = __put_user ( NULL , & frame - > uc . uc_link ) ;
2012-12-23 03:08:53 -05:00
err | = __save_altstack ( & frame - > uc . uc_stack , regs - > r1 ) ;
2009-03-27 14:25:14 +01:00
err | = setup_sigcontext ( & frame - > uc . uc_mcontext ,
regs , set - > sig [ 0 ] ) ;
err | = __copy_to_user ( & frame - > uc . uc_sigmask , set , sizeof ( * set ) ) ;
/* Set up to return from userspace. If provided, use a stub
already in userspace . */
/* minus 8 is offset to cater for "rtsd r15,8" */
2009-06-18 19:55:29 +02:00
/* addi r12, r0, __NR_sigreturn */
err | = __put_user ( 0x31800000 | __NR_rt_sigreturn ,
frame - > tramp + 0 ) ;
/* brki r14, 0x8 */
err | = __put_user ( 0xb9cc0008 , frame - > tramp + 1 ) ;
/* Return from sighandler will jump to the tramp.
Negative 8 offset because return is rtsd r15 , 8 */
regs - > r15 = ( ( unsigned long ) frame - > tramp ) - 8 ;
2009-12-10 11:43:57 +01:00
address = ( ( unsigned long ) frame - > tramp ) ;
# ifdef CONFIG_MMU
pmdp = pmd_offset ( pud_offset (
pgd_offset ( current - > mm , address ) ,
address ) , address ) ;
preempt_disable ( ) ;
ptep = pte_offset_map ( pmdp , address ) ;
if ( pte_present ( * ptep ) ) {
address = ( unsigned long ) page_address ( pte_page ( * ptep ) ) ;
/* MS: I need add offset in page */
address + = ( ( unsigned long ) frame - > tramp ) & ~ PAGE_MASK ;
/* MS address is virtual */
address = virt_to_phys ( address ) ;
invalidate_icache_range ( address , address + 8 ) ;
flush_dcache_range ( address , address + 8 ) ;
}
pte_unmap ( ptep ) ;
preempt_enable ( ) ;
# else
flush_icache_range ( address , address + 8 ) ;
flush_dcache_range ( address , address + 8 ) ;
# endif
2009-03-27 14:25:14 +01:00
if ( err )
goto give_sigsegv ;
/* Set up registers for signal handler */
2011-01-31 15:10:04 +01:00
regs - > r1 = ( unsigned long ) frame ;
2009-04-21 14:08:47 +02:00
2009-03-27 14:25:14 +01:00
/* Signal handler args: */
regs - > r5 = signal ; /* arg 0: signum */
regs - > r6 = ( unsigned long ) & frame - > info ; /* arg 1: siginfo */
regs - > r7 = ( unsigned long ) & frame - > uc ; /* arg2: ucontext */
/* Offset to handle microblaze rtid r14, 0 */
regs - > pc = ( unsigned long ) ka - > sa . sa_handler ;
set_fs ( USER_DS ) ;
# ifdef DEBUG_SIG
2012-12-27 10:40:38 +01:00
pr_info ( " SIG deliver (%s:%d): sp=%p pc=%08lx \n " ,
2009-03-27 14:25:14 +01:00
current - > comm , current - > pid , frame , regs - > pc ) ;
# endif
2012-05-11 10:58:04 +10:00
return 0 ;
2009-03-27 14:25:14 +01:00
give_sigsegv :
2012-05-11 10:58:03 +10:00
force_sigsegv ( sig , current ) ;
2012-05-11 10:58:04 +10:00
return - EFAULT ;
2009-03-27 14:25:14 +01:00
}
/* Handle restarting system calls */
static inline void
handle_restart ( struct pt_regs * regs , struct k_sigaction * ka , int has_handler )
{
switch ( regs - > r3 ) {
case - ERESTART_RESTARTBLOCK :
case - ERESTARTNOHAND :
if ( ! has_handler )
goto do_restart ;
regs - > r3 = - EINTR ;
break ;
case - ERESTARTSYS :
if ( has_handler & & ! ( ka - > sa . sa_flags & SA_RESTART ) ) {
regs - > r3 = - EINTR ;
break ;
}
/* fallthrough */
case - ERESTARTNOINTR :
do_restart :
/* offset of 4 bytes to re-execute trap (brki) instruction */
regs - > pc - = 4 ;
break ;
}
}
/*
* OK , we ' re invoking a handler
*/
2012-05-21 23:42:15 -04:00
static void
2009-03-27 14:25:14 +01:00
handle_signal ( unsigned long sig , struct k_sigaction * ka ,
2012-05-02 09:59:21 -04:00
siginfo_t * info , struct pt_regs * regs )
2009-03-27 14:25:14 +01:00
{
2012-05-02 09:59:21 -04:00
sigset_t * oldset = sigmask_to_save ( ) ;
2012-05-11 10:58:04 +10:00
int ret ;
2009-03-27 14:25:14 +01:00
/* Set up the stack frame */
if ( ka - > sa . sa_flags & SA_SIGINFO )
2012-05-11 10:58:04 +10:00
ret = setup_rt_frame ( sig , ka , info , oldset , regs ) ;
2009-03-27 14:25:14 +01:00
else
2012-05-11 10:58:04 +10:00
ret = setup_rt_frame ( sig , ka , NULL , oldset , regs ) ;
if ( ret )
2012-05-21 23:42:15 -04:00
return ;
2009-03-27 14:25:14 +01:00
2012-09-17 18:42:01 -04:00
signal_delivered ( sig , info , ka , regs ,
test_thread_flag ( TIF_SINGLESTEP ) ) ;
2009-03-27 14:25:14 +01:00
}
/*
* Note that ' init ' is a special process : it doesn ' t get signals it doesn ' t
* want to handle . Thus you cannot kill init even with a SIGKILL even by
* mistake .
*
* Note that we go through the signals twice : once to check the signals that
* the kernel can handle , and then we build all the user - level signal handling
* stack - frames in one go after that .
*/
2012-04-24 02:21:18 -04:00
static void do_signal ( struct pt_regs * regs , int in_syscall )
2009-03-27 14:25:14 +01:00
{
siginfo_t info ;
int signr ;
struct k_sigaction ka ;
# ifdef DEBUG_SIG
2012-12-27 10:40:38 +01:00
pr_info ( " do signal: %p %d \n " , regs , in_syscall ) ;
pr_info ( " do signal2: %lx %lx %ld [%lx] \n " , regs - > pc , regs - > r1 ,
2009-03-27 14:25:14 +01:00
regs - > r12 , current_thread_info ( ) - > flags ) ;
# endif
signr = get_signal_to_deliver ( & info , & ka , regs , NULL ) ;
if ( signr > 0 ) {
/* Whee! Actually deliver the signal. */
if ( in_syscall )
handle_restart ( regs , & ka , 1 ) ;
2012-05-21 23:42:15 -04:00
handle_signal ( signr , & ka , & info , regs ) ;
2012-04-24 02:21:18 -04:00
return ;
2009-03-27 14:25:14 +01:00
}
if ( in_syscall )
handle_restart ( regs , NULL , 0 ) ;
2009-06-18 19:55:29 +02:00
/*
* If there ' s no signal to deliver , we just put the saved sigmask
* back .
*/
2012-05-21 23:33:55 -04:00
restore_saved_sigmask ( ) ;
2009-03-27 14:25:14 +01:00
}
2012-04-24 02:03:06 -04:00
2012-12-13 18:03:50 +01:00
asmlinkage void do_notify_resume ( struct pt_regs * regs , int in_syscall )
2012-04-24 02:03:06 -04:00
{
if ( test_thread_flag ( TIF_SIGPENDING ) )
2012-04-24 02:21:18 -04:00
do_signal ( regs , in_syscall ) ;
2012-04-24 02:03:06 -04:00
2012-05-23 14:44:37 -04:00
if ( test_and_clear_thread_flag ( TIF_NOTIFY_RESUME ) )
2012-04-24 02:03:06 -04:00
tracehook_notify_resume ( regs ) ;
}