emailrelay/src/glib/gcleanup_unix.cpp
Graeme Walker ed23b3c747 v1.5
2007-04-10 12:00:00 +00:00

184 lines
4.1 KiB
C++

//
// Copyright (C) 2001-2007 Graeme Walker <graeme_walker@users.sourceforge.net>
//
// 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 02139, USA.
//
// ===
//
// gcleanup_unix.cpp
//
#include "gdef.h"
#include "gcleanup.h"
#include "gprocess.h"
#include "groot.h"
#include "glog.h"
#include <signal.h>
extern "C"
{
void gcleanup_unix_handler_( int signum ) ;
typedef void (*Handler)( int ) ;
}
namespace G
{
class CleanupImp ;
}
/// \class G::CleanupImp
/// A private implementation class used by G::Cleanup.
///
class G::CleanupImp
{
public:
static void add( void (*fn)(const char*) , const char * ) ;
static void installDefault( int ) ;
static void callHandlers() ;
static void ignore( int ) ;
private:
static void init() ;
static void install( int , Handler ) ;
static void installHandler( int ) ;
static bool ignored( int ) ;
private:
typedef Cleanup::Error Error ;
/// A private linked-list structure used by G::CleanupImp.
struct Link
{
void (*fn)(const char*) ;
const char * arg ;
Link * next ;
} ;
static Link * m_head ;
static Link * m_tail ;
} ;
G::CleanupImp::Link * G::CleanupImp::m_head = NULL ;
G::CleanupImp::Link * G::CleanupImp::m_tail = NULL ;
// ===
void G::Cleanup::init()
{
CleanupImp::ignore( SIGPIPE ) ;
}
void G::Cleanup::add( void (*fn)(const char*) , const char * arg )
{
if( arg != NULL )
CleanupImp::add( fn , arg ) ;
}
// ===
void G::CleanupImp::init()
{
// install our meta-handler for signals that normally terminate the process,
// except for sigpipe which we ignore
//
ignore( SIGPIPE ) ;
installHandler( SIGTERM ) ;
installHandler( SIGINT ) ;
installHandler( SIGHUP ) ;
installHandler( SIGQUIT ) ;
//installHandler( SIGUSR1 ) ;
//installHandler( SIGUSR2 ) ;
}
void G::CleanupImp::add( void (*fn)(const char*) , const char * arg )
{
Link * p = new Link ;
p->fn = fn ;
p->arg = arg ;
p->next = NULL ;
// (this bit should be protected from signals)
if( m_head == NULL ) init() ;
if( m_tail != NULL ) m_tail->next = p ;
m_tail = p ;
if( m_head == NULL ) m_head = p ;
}
void G::CleanupImp::installHandler( int signum )
{
if( ignored(signum) )
G_DEBUG( "G::CleanupImp::installHandler: signal " << signum << " is ignored" ) ;
else
install( signum , gcleanup_unix_handler_ ) ;
}
void G::CleanupImp::installDefault( int signum )
{
install( signum , NULL ) ;
}
bool G::CleanupImp::ignored( int signum )
{
static struct sigaction zero_action ;
struct sigaction action( zero_action ) ;
if( ::sigaction( signum , NULL , &action ) )
throw Error( "sigaction" ) ;
return action.sa_handler == SIG_IGN ;
}
void G::CleanupImp::install( int signum , Handler fn )
{
// install the given handler, or the system default if null
static struct sigaction zero_action ;
struct sigaction action( zero_action ) ;
action.sa_handler = fn ? fn : SIG_DFL ;
if( ::sigaction( signum , &action , NULL ) && fn != NULL )
throw Error( "sigaction" ) ;
}
void G::CleanupImp::ignore( int signum )
{
static struct sigaction zero_action ;
struct sigaction action( zero_action ) ;
action.sa_handler = SIG_IGN ;
if( ::sigaction( signum , &action , NULL ) )
throw Error( "sigaction" ) ;
}
void G::CleanupImp::callHandlers()
{
G::Root claim_root ;
for( const Link * p = m_head ; p != NULL ; p = p->next )
{
(*(p->fn))(p->arg) ;
}
}
extern "C"
void gcleanup_unix_handler_( int signum )
{
// call the registered handler(s) and then do the system default action
try
{
G::CleanupImp::callHandlers() ;
G::CleanupImp::installDefault( signum ) ;
::raise( signum ) ;
}
catch(...)
{
}
}