2005-06-23 22:01:16 -07:00
/*
2007-05-31 17:49:32 -07:00
* arch / xtensa / kernel / signal . c
2005-06-23 22:01:16 -07:00
*
2007-05-31 17:49:32 -07:00
* Default platform functions .
2005-06-23 22:01:16 -07:00
*
2007-05-31 17:49:32 -07:00
* 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 .
2005-06-23 22:01:16 -07:00
*
2007-05-31 17:49:32 -07:00
* Copyright ( C ) 2005 , 2006 Tensilica Inc .
* Copyright ( C ) 1991 , 1992 Linus Torvalds
* 1997 - 11 - 28 Modified for POSIX .1 b signals by Richard Henderson
2005-06-23 22:01:16 -07:00
*
2007-05-31 17:49:32 -07:00
* Chris Zankel < chris @ zankel . net >
* Joe Taylor < joe @ tensilica . com >
2005-06-23 22:01:16 -07:00
*/
# include <linux/signal.h>
# include <linux/errno.h>
# include <linux/ptrace.h>
# include <linux/personality.h>
2007-05-31 17:49:32 -07:00
# include <linux/freezer.h>
2005-06-23 22:01:16 -07:00
# include <asm/ucontext.h>
# include <asm/uaccess.h>
# include <asm/cacheflush.h>
2007-05-31 17:49:32 -07:00
# include <asm/coprocessor.h>
# include <asm/unistd.h>
2005-06-23 22:01:16 -07:00
# define DEBUG_SIG 0
# define _BLOCKABLE (~(sigmask(SIGKILL) | sigmask(SIGSTOP)))
extern struct task_struct * coproc_owners [ ] ;
2007-05-31 17:49:32 -07:00
struct rt_sigframe
{
struct siginfo info ;
struct ucontext uc ;
2008-02-12 13:17:07 -08:00
struct {
xtregs_opt_t opt ;
xtregs_user_t user ;
# if XTENSA_HAVE_COPROCESSORS
xtregs_coprocessor_t cp ;
# endif
} xtregs ;
2007-05-31 17:49:32 -07:00
unsigned char retcode [ 6 ] ;
unsigned int window [ 4 ] ;
} ;
/*
* Flush register windows stored in pt_regs to stack .
* Returns 1 for errors .
2005-06-23 22:01:16 -07:00
*/
2007-05-31 17:49:32 -07:00
int
flush_window_regs_user ( struct pt_regs * regs )
2005-06-23 22:01:16 -07:00
{
2007-05-31 17:49:32 -07:00
const unsigned long ws = regs - > windowstart ;
const unsigned long wb = regs - > windowbase ;
unsigned long sp = 0 ;
unsigned long wm ;
int err = 1 ;
int base ;
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
/* Return if no other frames. */
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
if ( regs - > wmask = = 1 )
return 0 ;
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
/* Rotate windowmask and skip empty frames. */
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
wm = ( ws > > wb ) | ( ws < < ( XCHAL_NUM_AREGS / 4 - wb ) ) ;
base = ( XCHAL_NUM_AREGS / 4 ) - ( regs - > wmask > > 4 ) ;
/* For call8 or call12 frames, we need the previous stack pointer. */
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
if ( ( regs - > wmask & 2 ) = = 0 )
if ( __get_user ( sp , ( int * ) ( regs - > areg [ base * 4 + 1 ] - 12 ) ) )
goto errout ;
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
/* Spill frames to stack. */
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
while ( base < XCHAL_NUM_AREGS / 4 ) {
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
int m = ( wm > > base ) ;
int inc = 0 ;
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
/* Save registers a4..a7 (call8) or a4...a11 (call12) */
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
if ( m & 2 ) { /* call4 */
inc = 1 ;
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
} else if ( m & 4 ) { /* call8 */
if ( copy_to_user ( ( void * ) ( sp - 32 ) ,
& regs - > areg [ ( base + 1 ) * 4 ] , 16 ) )
goto errout ;
inc = 2 ;
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
} else if ( m & 8 ) { /* call12 */
if ( copy_to_user ( ( void * ) ( sp - 48 ) ,
& regs - > areg [ ( base + 1 ) * 4 ] , 32 ) )
goto errout ;
inc = 3 ;
}
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
/* Save current frame a0..a3 under next SP */
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
sp = regs - > areg [ ( ( base + inc ) * 4 + 1 ) % XCHAL_NUM_AREGS ] ;
if ( copy_to_user ( ( void * ) ( sp - 16 ) , & regs - > areg [ base * 4 ] , 16 ) )
goto errout ;
/* Get current stack pointer for next loop iteration. */
sp = regs - > areg [ base * 4 + 1 ] ;
base + = inc ;
}
2008-02-12 12:22:15 -08:00
regs - > wmask = 1 ;
regs - > windowstart = 1 < < wb ;
2007-05-31 17:49:32 -07:00
return 0 ;
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
errout :
return err ;
}
2005-06-23 22:01:16 -07:00
/*
2007-05-31 17:49:32 -07:00
* Note : We don ' t copy double exception ' regs ' , we have to finish double exc .
* first before we return to signal handler ! This dbl . exc . handler might cause
* another double exception , but I think we are fine as the situation is the
* same as if we had returned to the signal handerl and got an interrupt
* immediately . . .
2005-06-23 22:01:16 -07:00
*/
2007-05-31 17:49:32 -07:00
static int
2008-02-12 13:17:07 -08:00
setup_sigcontext ( struct rt_sigframe __user * frame , struct pt_regs * regs )
2005-06-23 22:01:16 -07:00
{
2008-02-12 13:17:07 -08:00
struct sigcontext __user * sc = & frame - > uc . uc_mcontext ;
struct thread_info * ti = current_thread_info ( ) ;
2007-05-31 17:49:32 -07:00
int err = 0 ;
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
# define COPY(x) err |= __put_user(regs->x, &sc->sc_##x)
COPY ( pc ) ;
COPY ( ps ) ;
COPY ( lbeg ) ;
COPY ( lend ) ;
COPY ( lcount ) ;
COPY ( sar ) ;
# undef COPY
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
err | = flush_window_regs_user ( regs ) ;
err | = __copy_to_user ( sc - > sc_a , regs - > areg , 16 * 4 ) ;
2008-02-12 13:17:07 -08:00
err | = __put_user ( 0 , & sc - > sc_xtregs ) ;
2005-06-23 22:01:16 -07:00
2008-02-12 13:17:07 -08:00
if ( err )
return err ;
2005-06-23 22:01:16 -07:00
2008-02-12 13:17:07 -08:00
# if XTENSA_HAVE_COPROCESSORS
coprocessor_flush_all ( ti ) ;
coprocessor_release_all ( ti ) ;
err | = __copy_to_user ( & frame - > xtregs . cp , & ti - > xtregs_cp ,
sizeof ( frame - > xtregs . cp ) ) ;
2005-06-23 22:01:16 -07:00
# endif
2008-02-12 13:17:07 -08:00
err | = __copy_to_user ( & frame - > xtregs . opt , & regs - > xtregs_opt ,
sizeof ( xtregs_opt_t ) ) ;
err | = __copy_to_user ( & frame - > xtregs . user , & ti - > xtregs_user ,
sizeof ( xtregs_user_t ) ) ;
err | = __put_user ( err ? NULL : & frame - > xtregs , & sc - > sc_xtregs ) ;
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
return err ;
}
2005-06-23 22:01:16 -07:00
static int
2008-02-12 13:17:07 -08:00
restore_sigcontext ( struct pt_regs * regs , struct rt_sigframe __user * frame )
2005-06-23 22:01:16 -07:00
{
2008-02-12 13:17:07 -08:00
struct sigcontext __user * sc = & frame - > uc . uc_mcontext ;
struct thread_info * ti = current_thread_info ( ) ;
2005-06-23 22:01:16 -07:00
unsigned int err = 0 ;
unsigned long ps ;
# define COPY(x) err |= __get_user(regs->x, &sc->sc_##x)
COPY ( pc ) ;
COPY ( lbeg ) ;
COPY ( lend ) ;
COPY ( lcount ) ;
COPY ( sar ) ;
# undef COPY
2007-05-31 17:49:32 -07:00
/* All registers were flushed to stack. Start with a prestine frame. */
regs - > wmask = 1 ;
regs - > windowbase = 0 ;
regs - > windowstart = 1 ;
2008-02-12 13:17:07 -08:00
regs - > syscall = - 1 ; /* disable syscall checks */
2005-06-23 22:01:16 -07:00
/* For PS, restore only PS.CALLINC.
* Assume that all other bits are either the same as for the signal
* handler , or the user mode value doesn ' t matter ( e . g . PS . OWB ) .
*/
err | = __get_user ( ps , & sc - > sc_ps ) ;
2007-05-31 17:49:32 -07:00
regs - > ps = ( regs - > ps & ~ PS_CALLINC_MASK ) | ( ps & PS_CALLINC_MASK ) ;
2005-06-23 22:01:16 -07:00
/* Additional corruption checks */
if ( ( regs - > lcount > 0 )
2007-05-31 17:49:32 -07:00
& & ( ( regs - > lbeg > TASK_SIZE ) | | ( regs - > lend > TASK_SIZE ) ) )
2005-06-23 22:01:16 -07:00
err = 1 ;
2007-05-31 17:49:32 -07:00
err | = __copy_from_user ( regs - > areg , sc - > sc_a , 16 * 4 ) ;
2005-06-23 22:01:16 -07:00
2008-02-12 13:17:07 -08:00
if ( err )
return err ;
2007-05-31 17:49:32 -07:00
/* The signal handler may have used coprocessors in which
* case they are still enabled . We disable them to force a
* reloading of the original task ' s CP state by the lazy
* context - switching mechanisms of CP exception handling .
* Also , we essentially discard any coprocessor state that the
* signal handler created . */
2005-06-23 22:01:16 -07:00
2008-02-12 13:17:07 -08:00
# if XTENSA_HAVE_COPROCESSORS
coprocessor_release_all ( ti ) ;
err | = __copy_from_user ( & ti - > xtregs_cp , & frame - > xtregs . cp ,
sizeof ( frame - > xtregs . cp ) ) ;
2005-06-23 22:01:16 -07:00
# endif
2008-02-12 13:17:07 -08:00
err | = __copy_from_user ( & ti - > xtregs_user , & frame - > xtregs . user ,
sizeof ( xtregs_user_t ) ) ;
err | = __copy_from_user ( & regs - > xtregs_opt , & frame - > xtregs . opt ,
sizeof ( xtregs_opt_t ) ) ;
2005-06-23 22:01:16 -07:00
return err ;
}
2007-05-31 17:49:32 -07:00
/*
* Do a signal return ; undo the signal stack .
*/
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
asmlinkage long xtensa_rt_sigreturn ( long a0 , long a1 , long a2 , long a3 ,
long a4 , long a5 , struct pt_regs * regs )
2005-06-23 22:01:16 -07:00
{
2007-05-31 17:49:32 -07:00
struct rt_sigframe __user * frame ;
2005-06-23 22:01:16 -07:00
sigset_t set ;
int ret ;
2007-05-31 17:49:32 -07:00
2012-04-22 03:35:28 -04:00
/* Always make any pending restarted system calls return -EINTR */
current_thread_info ( ) - > restart_block . fn = do_no_restart_syscall ;
2005-06-23 22:01:16 -07:00
if ( regs - > depc > 64 )
2007-05-31 17:49:32 -07:00
panic ( " rt_sigreturn in double exception! \n " ) ;
frame = ( struct rt_sigframe __user * ) regs - > areg [ 1 ] ;
2005-06-23 22:01:16 -07:00
2006-06-23 02:05:04 -07:00
if ( ! access_ok ( VERIFY_READ , frame , sizeof ( * frame ) ) )
2005-06-23 22:01:16 -07:00
goto badframe ;
if ( __copy_from_user ( & set , & frame - > uc . uc_sigmask , sizeof ( set ) ) )
goto badframe ;
sigdelsetmask ( & set , ~ _BLOCKABLE ) ;
2012-03-21 16:33:45 -07:00
set_current_blocked ( & set ) ;
2005-06-23 22:01:16 -07:00
2008-02-12 13:17:07 -08:00
if ( restore_sigcontext ( regs , frame ) )
2005-06-23 22:01:16 -07:00
goto badframe ;
2007-05-31 17:49:32 -07:00
2005-06-23 22:01:16 -07:00
ret = regs - > areg [ 2 ] ;
2007-05-31 17:49:32 -07:00
if ( do_sigaltstack ( & frame - > uc . uc_stack , NULL , regs - > areg [ 1 ] ) = = - EFAULT )
2005-06-23 22:01:16 -07:00
goto badframe ;
return ret ;
badframe :
force_sig ( SIGSEGV , current ) ;
return 0 ;
}
2007-05-31 17:49:32 -07:00
2005-06-23 22:01:16 -07:00
/*
2007-05-31 17:49:32 -07:00
* Set up a signal frame .
2005-06-23 22:01:16 -07:00
*/
static int
2007-05-31 17:49:32 -07:00
gen_return_code ( unsigned char * codemem )
2005-06-23 22:01:16 -07:00
{
int err = 0 ;
2007-05-31 17:49:32 -07:00
/*
* The 12 - bit immediate is really split up within the 24 - bit MOVI
* instruction . As long as the above system call numbers fit within
* 8 - bits , the following code works fine . See the Xtensa ISA for
* details .
2005-06-23 22:01:16 -07:00
*/
2007-05-31 17:49:32 -07:00
# if __NR_rt_sigreturn > 255
# error Generating the MOVI instruction below breaks!
2005-06-23 22:01:16 -07:00
# endif
# ifdef __XTENSA_EB__ /* Big Endian version */
2007-05-31 17:49:32 -07:00
/* Generate instruction: MOVI a2, __NR_rt_sigreturn */
err | = __put_user ( 0x22 , & codemem [ 0 ] ) ;
err | = __put_user ( 0x0a , & codemem [ 1 ] ) ;
err | = __put_user ( __NR_rt_sigreturn , & codemem [ 2 ] ) ;
/* Generate instruction: SYSCALL */
err | = __put_user ( 0x00 , & codemem [ 3 ] ) ;
err | = __put_user ( 0x05 , & codemem [ 4 ] ) ;
err | = __put_user ( 0x00 , & codemem [ 5 ] ) ;
2005-06-23 22:01:16 -07:00
# elif defined __XTENSA_EL__ /* Little Endian version */
2007-05-31 17:49:32 -07:00
/* Generate instruction: MOVI a2, __NR_rt_sigreturn */
err | = __put_user ( 0x22 , & codemem [ 0 ] ) ;
err | = __put_user ( 0xa0 , & codemem [ 1 ] ) ;
err | = __put_user ( __NR_rt_sigreturn , & codemem [ 2 ] ) ;
/* Generate instruction: SYSCALL */
err | = __put_user ( 0x00 , & codemem [ 3 ] ) ;
err | = __put_user ( 0x50 , & codemem [ 4 ] ) ;
err | = __put_user ( 0x00 , & codemem [ 5 ] ) ;
2005-06-23 22:01:16 -07:00
# else
2007-05-31 17:49:32 -07:00
# error Must use compiler for Xtensa processors.
2005-06-23 22:01:16 -07:00
# endif
/* Flush generated code out of the data cache */
2006-12-10 02:18:48 -08:00
if ( err = = 0 ) {
__invalidate_icache_range ( ( unsigned long ) codemem , 6UL ) ;
__flush_invalidate_dcache_range ( ( unsigned long ) codemem , 6UL ) ;
}
2005-06-23 22:01:16 -07:00
return err ;
}
2012-03-21 16:33:45 -07:00
static int setup_frame ( int sig , struct k_sigaction * ka , siginfo_t * info ,
sigset_t * set , struct pt_regs * regs )
2005-06-23 22:01:16 -07:00
{
2007-05-31 17:49:32 -07:00
struct rt_sigframe * frame ;
2005-06-23 22:01:16 -07:00
int err = 0 ;
int signal ;
2007-05-31 17:49:32 -07:00
unsigned long sp , ra ;
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
sp = regs - > areg [ 1 ] ;
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
if ( ( ka - > sa . sa_flags & SA_ONSTACK ) ! = 0 & & ! on_sig_stack ( sp ) ) {
sp = current - > sas_ss_sp + current - > sas_ss_size ;
2005-06-23 22:01:16 -07:00
}
2007-05-31 17:49:32 -07:00
frame = ( void * ) ( ( sp - sizeof ( * frame ) ) & - 16ul ) ;
2005-06-23 22:01:16 -07:00
if ( regs - > depc > 64 )
panic ( " Double exception sys_sigreturn \n " ) ;
2007-05-31 17:49:32 -07:00
if ( ! access_ok ( VERIFY_WRITE , frame , sizeof ( * frame ) ) ) {
2005-06-23 22:01:16 -07:00
goto give_sigsegv ;
2007-05-31 17:49:32 -07:00
}
2005-06-23 22:01:16 -07:00
signal = current_thread_info ( ) - > exec_domain
& & current_thread_info ( ) - > exec_domain - > signal_invmap
& & sig < 32
? current_thread_info ( ) - > exec_domain - > signal_invmap [ sig ]
: sig ;
2007-05-31 17:49:32 -07:00
if ( ka - > sa . sa_flags & SA_SIGINFO ) {
err | = copy_siginfo_to_user ( & frame - > info , info ) ;
}
/* Create the user context. */
2005-06-23 22:01:16 -07:00
err | = __put_user ( 0 , & frame - > uc . uc_flags ) ;
err | = __put_user ( 0 , & frame - > uc . uc_link ) ;
err | = __put_user ( ( void * ) current - > sas_ss_sp ,
& frame - > uc . uc_stack . ss_sp ) ;
err | = __put_user ( sas_ss_flags ( regs - > areg [ 1 ] ) ,
& frame - > uc . uc_stack . ss_flags ) ;
err | = __put_user ( current - > sas_ss_size , & frame - > uc . uc_stack . ss_size ) ;
2008-02-12 13:17:07 -08:00
err | = setup_sigcontext ( frame , regs ) ;
2005-06-23 22:01:16 -07:00
err | = __copy_to_user ( & frame - > uc . uc_sigmask , set , sizeof ( * set ) ) ;
2008-01-11 11:44:17 -08:00
if ( ka - > sa . sa_flags & SA_RESTORER ) {
ra = ( unsigned long ) ka - > sa . sa_restorer ;
} else {
2005-06-23 22:01:16 -07:00
2008-01-11 11:44:17 -08:00
/* Create sys_rt_sigreturn syscall in stack frame */
2007-05-31 17:49:32 -07:00
2008-01-11 11:44:17 -08:00
err | = gen_return_code ( frame - > retcode ) ;
if ( err ) {
goto give_sigsegv ;
}
ra = ( unsigned long ) frame - > retcode ;
2007-05-31 17:49:32 -07:00
}
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
/*
* Create signal handler execution context .
2005-06-23 22:01:16 -07:00
* Return context not modified until this point .
*/
2007-05-31 17:49:32 -07:00
/* Set up registers for signal handler */
start_thread ( regs , ( unsigned long ) ka - > sa . sa_handler ,
( unsigned long ) frame ) ;
/* Set up a stack frame for a call4
* Note : PS . CALLINC is set to one by start_thread
*/
regs - > areg [ 4 ] = ( ( ( unsigned long ) ra ) & 0x3fffffff ) | 0x40000000 ;
regs - > areg [ 6 ] = ( unsigned long ) signal ;
regs - > areg [ 7 ] = ( unsigned long ) & frame - > info ;
regs - > areg [ 8 ] = ( unsigned long ) & frame - > uc ;
2005-06-23 22:01:16 -07:00
/* Set access mode to USER_DS. Nomenclature is outdated, but
* functionality is used in uaccess . h
*/
set_fs ( USER_DS ) ;
# if DEBUG_SIG
printk ( " SIG rt deliver (%s:%d): signal=%d sp=%p pc=%08x \n " ,
current - > comm , current - > pid , signal , frame , regs - > pc ) ;
# endif
2012-03-21 16:33:45 -07:00
return 0 ;
2005-06-23 22:01:16 -07:00
give_sigsegv :
2012-03-21 16:33:44 -07:00
force_sigsegv ( sig , current ) ;
2012-03-21 16:33:45 -07:00
return - EFAULT ;
2005-06-23 22:01:16 -07:00
}
2007-05-31 17:49:32 -07:00
asmlinkage long xtensa_sigaltstack ( const stack_t __user * uss ,
stack_t __user * uoss ,
long a2 , long a3 , long a4 , long a5 ,
struct pt_regs * regs )
{
return do_sigaltstack ( uss , uoss , regs - > areg [ 1 ] ) ;
}
2005-06-23 22:01:16 -07: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-22 00:59:07 -04:00
void do_signal ( struct pt_regs * regs )
2005-06-23 22:01:16 -07:00
{
siginfo_t info ;
int signr ;
struct k_sigaction ka ;
2012-04-22 00:59:07 -04:00
sigset_t oldset ;
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
if ( ! user_mode ( regs ) )
2012-04-22 00:59:07 -04:00
return ;
2007-05-31 17:49:32 -07:00
if ( try_to_freeze ( ) )
goto no_signal ;
2012-04-22 00:59:07 -04:00
if ( test_thread_flag ( TIF_RESTORE_SIGMASK ) )
oldset = & current - > saved_sigmask ;
else
2005-06-23 22:01:16 -07:00
oldset = & current - > blocked ;
2007-05-31 17:49:32 -07:00
task_pt_regs ( current ) - > icountlevel = 0 ;
2005-06-23 22:01:16 -07:00
signr = get_signal_to_deliver ( & info , & ka , regs , NULL ) ;
2007-05-31 17:49:32 -07:00
if ( signr > 0 ) {
2012-04-18 16:17:13 -04:00
int ret ;
2007-05-31 17:49:32 -07:00
/* Are we from a system call? */
if ( ( signed ) regs - > syscall > = 0 ) {
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
/* If so, check system call restarting.. */
switch ( regs - > areg [ 2 ] ) {
case - ERESTARTNOHAND :
case - ERESTART_RESTARTBLOCK :
2005-06-23 22:01:16 -07:00
regs - > areg [ 2 ] = - EINTR ;
break ;
2007-05-31 17:49:32 -07:00
case - ERESTARTSYS :
if ( ! ( ka . sa . sa_flags & SA_RESTART ) ) {
regs - > areg [ 2 ] = - EINTR ;
break ;
}
/* fallthrough */
case - ERESTARTNOINTR :
regs - > areg [ 2 ] = regs - > syscall ;
regs - > pc - = 3 ;
break ;
default :
/* nothing to do */
if ( regs - > areg [ 2 ] ! = 0 )
break ;
}
2005-06-23 22:01:16 -07:00
}
2007-05-31 17:49:32 -07:00
/* Whee! Actually deliver the signal. */
/* Set up the stack frame */
2012-03-21 16:33:45 -07:00
ret = setup_frame ( signr , & ka , & info , oldset , regs ) ;
if ( ret )
2012-04-22 00:59:07 -04:00
return ;
2005-06-23 22:01:16 -07:00
2012-04-22 00:59:07 -04:00
clear_thread_flag ( TIF_RESTORE_SIGMASK ) ;
2012-03-21 16:33:45 -07:00
block_sigmask ( & ka , signr ) ;
2007-05-31 17:49:32 -07:00
if ( current - > ptrace & PT_SINGLESTEP )
task_pt_regs ( current ) - > icountlevel = 1 ;
2005-06-23 22:01:16 -07:00
2012-04-22 00:59:07 -04:00
return ;
2007-05-31 17:49:32 -07:00
}
2005-06-23 22:01:16 -07:00
2007-05-31 17:49:32 -07:00
no_signal :
/* Did we come from a system call? */
if ( ( signed ) regs - > syscall > = 0 ) {
/* Restart the system call - no handlers present */
switch ( regs - > areg [ 2 ] ) {
case - ERESTARTNOHAND :
case - ERESTARTSYS :
case - ERESTARTNOINTR :
regs - > areg [ 2 ] = regs - > syscall ;
regs - > pc - = 3 ;
break ;
case - ERESTART_RESTARTBLOCK :
regs - > areg [ 2 ] = __NR_restart_syscall ;
regs - > pc - = 3 ;
break ;
}
}
2012-04-22 00:59:07 -04:00
/* If there's no signal to deliver, we just restore the saved mask. */
if ( test_and_clear_thread_flag ( TIF_RESTORE_SIGMASK ) )
set_current_blocked ( & current - > saved_sigmask ) ;
2007-05-31 17:49:32 -07:00
if ( current - > ptrace & PT_SINGLESTEP )
task_pt_regs ( current ) - > icountlevel = 1 ;
2012-04-22 00:59:07 -04:00
return ;
2005-06-23 22:01:16 -07:00
}
2007-05-31 17:49:32 -07:00