1998-07-29 03:08:05 +00:00
/*
Unix SMB / Netbios implementation .
Version 1.9 .
signal handling functions
Copyright ( C ) Andrew Tridgell 1998
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include "includes.h"
/****************************************************************************
1999-12-13 13:27:58 +00:00
Catch child exits and reap the child zombie status .
1998-07-29 03:08:05 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 13:27:58 +00:00
1998-07-29 03:08:05 +00:00
static void sig_cld ( int signum )
{
1999-12-13 13:27:58 +00:00
while ( sys_waitpid ( ( pid_t ) - 1 , ( int * ) NULL , WNOHANG ) > 0 )
;
/*
* Turns out it ' s * really * important not to
* restore the signal handler here if we have real POSIX
* signal handling . If we do , then we get the signal re - delivered
* immediately - hey presto - instant loop ! JRA .
*/
1998-07-29 03:08:05 +00:00
1999-12-13 13:27:58 +00:00
# if !defined(HAVE_SIGACTION)
1998-08-30 12:32:45 +00:00
CatchSignal ( SIGCLD , sig_cld ) ;
1999-12-13 13:27:58 +00:00
# endif
1998-07-29 03:08:05 +00:00
}
1999-12-13 13:27:58 +00:00
/****************************************************************************
catch child exits - leave status ;
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-07-29 03:08:05 +00:00
1999-12-13 13:27:58 +00:00
static void sig_cld_leave_status ( int signum )
{
/*
* Turns out it ' s * really * important not to
* restore the signal handler here if we have real POSIX
* signal handling . If we do , then we get the signal re - delivered
* immediately - hey presto - instant loop ! JRA .
*/
# if !defined(HAVE_SIGACTION)
CatchSignal ( SIGCLD , sig_cld_leave_status ) ;
# else
;
# endif
}
1998-07-29 03:08:05 +00:00
/*******************************************************************
1999-12-13 13:27:58 +00:00
Block sigs .
1998-07-29 03:08:05 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 13:27:58 +00:00
1998-07-29 03:08:05 +00:00
void BlockSignals ( BOOL block , int signum )
{
# ifdef HAVE_SIGPROCMASK
sigset_t set ;
sigemptyset ( & set ) ;
sigaddset ( & set , signum ) ;
sigprocmask ( block ? SIG_BLOCK : SIG_UNBLOCK , & set , NULL ) ;
# elif defined(HAVE_SIGBLOCK)
int block_mask = sigmask ( signum ) ;
static int oldmask = 0 ;
if ( block ) {
oldmask = sigblock ( block_mask ) ;
} else {
sigsetmask ( oldmask ) ;
}
# else
/* yikes! This platform can't block signals? */
static int done ;
if ( ! done ) {
DEBUG ( 0 , ( " WARNING: No signal blocking available \n " ) ) ;
done = 1 ;
}
# endif
}
/*******************************************************************
1999-12-13 13:27:58 +00:00
Catch a signal . This should implement the following semantics :
1998-07-29 03:08:05 +00:00
1999-12-13 13:27:58 +00:00
1 ) The handler remains installed after being called .
2 ) The signal should be blocked during handler execution .
1998-07-29 03:08:05 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 13:27:58 +00:00
1998-07-29 03:08:05 +00:00
void CatchSignal ( int signum , void ( * handler ) ( int ) )
{
# ifdef HAVE_SIGACTION
struct sigaction act ;
1998-09-05 13:24:20 +00:00
ZERO_STRUCT ( act ) ;
1998-07-29 03:08:05 +00:00
act . sa_handler = handler ;
# ifdef SA_RESTART
1999-12-13 13:27:58 +00:00
/*
* We * want * SIGALRM to interrupt a system call .
*/
if ( signum ! = SIGALRM )
act . sa_flags = SA_RESTART ;
1998-07-29 03:08:05 +00:00
# endif
sigemptyset ( & act . sa_mask ) ;
sigaddset ( & act . sa_mask , signum ) ;
sigaction ( signum , & act , NULL ) ;
1999-12-13 13:27:58 +00:00
# else /* !HAVE_SIGACTION */
1998-07-29 03:08:05 +00:00
/* FIXME: need to handle sigvec and systems with broken signal() */
signal ( signum , handler ) ;
# endif
}
/*******************************************************************
1999-12-13 13:27:58 +00:00
Ignore SIGCLD via whatever means is necessary for this OS .
1998-07-29 03:08:05 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 13:27:58 +00:00
1998-07-29 03:08:05 +00:00
void CatchChild ( void )
{
CatchSignal ( SIGCLD , sig_cld ) ;
}
1999-12-13 13:27:58 +00:00
/*******************************************************************
Catch SIGCLD but leave the child around so it ' s status can be reaped .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void CatchChildLeaveStatus ( void )
{
CatchSignal ( SIGCLD , sig_cld_leave_status ) ;
}