Bug 308 - Added handlers for fatal signals, with backtracing facility.

git-svn-id: http://svn.zoneminder.com/svn/zm/trunk@1914 e3e1d417-86f3-4887-817a-d78f3d33393f
pull/27/merge
stan 2006-04-05 12:20:46 +00:00
parent cd4bb06965
commit e22e410320
2 changed files with 229 additions and 0 deletions

177
src/zm_signal.cpp Normal file
View File

@ -0,0 +1,177 @@
//
// ZoneMinder Signal Handling Implementation, $Date$, $Revision$
// Copyright (C) 2003, 2004, 2005, 2006 Philip Coombes
//
// 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
//
#include <string.h>
#include <stdlib.h>
#include "zm.h"
#include "zm_signal.h"
bool zm_reload = false;
RETSIGTYPE zm_hup_handler( int signal )
{
#if HAVE_DECL_STRSIGNAL
char * error = strsignal(signal);
size_t errorStringSize = strlen(error) + strlen("Got signal (), exiting.");
char * errorString =(char *) malloc(errorStringSize + 1); // plus 1 for termination char.
(void) snprintf(errorString, errorStringSize, "Got signal (%s), exiting.", error);
Info(( (const char *)errorString ));
free(errorString);
#else // HAVE_DECL_STRSIGNAL
Info(( "Got HUP signal, reloading" ));
#endif // HAVE_DECL_STRSIGNAL
zm_reload = true;
}
bool zm_terminate = false;
RETSIGTYPE zm_term_handler( int signal )
{
#if HAVE_DECL_STRSIGNAL
char * error = strsignal(signal);
size_t errorStringSize = strlen(error) + strlen("Got signal (), exiting.");
char * errorString =(char *) malloc(errorStringSize + 1); // plus 1 for termination char.
(void) snprintf(errorString, errorStringSize, "Got signal (%s), exiting.", error);
Info(( (const char *)errorString ));
free(errorString);
#else // HAVE_DECL_STRSIGNAL
Info(( "Got TERM signal, exiting" ));
#endif // HAVE_DECL_STRSIGNAL
zm_terminate = true;
}
#if HAVE_STRUCT_SIGCONTEXT
RETSIGTYPE zm_die_handler( int signal, struct sigcontext context )
#elif ( HAVE_SIGINFO_T && HAVE_UCONTEXT_T )
#include <ucontext.h>
RETSIGTYPE zm_die_handler( int signal, siginfo_t *info, void *context )
#else
RETSIGTYPE zm_die_handler( int signal )
#endif
{
#if HAVE_DECL_STRSIGNAL
char* error = strsignal(signal);
size_t errorStringSize = strlen(error) + strlen("Got signal (), crashing.");
char* errorString =(char *)malloc(errorStringSize+1); // plus 1 for termination char.
snprintf(errorString, errorStringSize, "Got signal (%s), crashing.", error );
Error(( (const char *)errorString ));
free(errorString);
#else // HAVE_DECL_STRSIGNAL
Error(( "Got signal %d, crashing", signal ));
#endif // HAVE_DECL_STRSIGNAL
#ifndef ZM_NO_CRASHTRACE
#if ( ( HAVE_SIGINFO_T && HAVE_UCONTEXT_T ) || HAVE_STRUCT_SIGCONTEXT )
void *trace[16];
int trace_size = 0;
#if HAVE_STRUCT_SIGCONTEXT
Error(( "Signal address is %p, from %p\n", context.cr2, context.eip ));
trace_size = backtrace( trace, 16 );
// overwrite sigaction with caller's address
trace[1] = (void *)context.eip;
#else // HAVE_STRUCT_SIGCONTEXT
if ( info && context )
{
ucontext_t *uc = (ucontext_t *)context;
Error(( "Signal address is %p, from %p\n", info->si_addr, uc->uc_mcontext.gregs[REG_EIP] ));
trace_size = backtrace( trace, 16 );
// overwrite sigaction with caller's address
trace[1] = (void *) uc->uc_mcontext.gregs[REG_EIP];
}
#endif // HAVE_STRUCT_SIGCONTEXT
#if HAVE_DECL_BACKTRACE
char **messages = (char **)NULL;
messages = backtrace_symbols( trace, trace_size );
// skip first stack frame (points here)
for ( int i=1; i < trace_size; ++i )
Error(( "Backtrace: %s", messages[i] ));
Info(( "Backtrace complete" ));
#endif // HAVE_DECL_BACKTRACE
#endif // ( HAVE_SIGINFO_T && HAVE_UCONTEXT_T ) || HAVE_STRUCT_SIGCONTEXT
#endif // ZM_NO_CRASHTRACE
exit( signal );
}
void zmSetHupHandler( SigHandler *handler )
{
sigset_t block_set;
sigemptyset( &block_set );
struct sigaction action, old_action;
action.sa_handler = (SigHandler *)handler;
action.sa_mask = block_set;
action.sa_flags = 0;
sigaction( SIGHUP, &action, &old_action );
}
void zmSetTermHandler( SigHandler *handler )
{
sigset_t block_set;
sigemptyset( &block_set );
struct sigaction action, old_action;
action.sa_handler = (SigHandler *)handler;
action.sa_mask = block_set;
action.sa_flags = 0;
sigaction( SIGTERM, &action, &old_action );
}
void zmSetDieHandler( SigHandler *handler )
{
sigset_t block_set;
sigemptyset( &block_set );
struct sigaction action, old_action;
action.sa_handler = (SigHandler *)handler;
action.sa_mask = block_set;
action.sa_flags = 0;
sigaction( SIGBUS, &action, &old_action );
sigaction( SIGSEGV, &action, &old_action );
sigaction( SIGABRT, &action, &old_action );
sigaction( SIGILL, &action, &old_action );
sigaction( SIGFPE, &action, &old_action );
}
void zmSetDefaultHupHandler()
{
zmSetHupHandler( (SigHandler *)zm_hup_handler );
}
void zmSetDefaultTermHandler()
{
zmSetTermHandler( (SigHandler *)zm_term_handler );
}
void zmSetDefaultDieHandler()
{
zmSetDieHandler( (SigHandler *)zm_die_handler );
}

52
src/zm_signal.h Normal file
View File

@ -0,0 +1,52 @@
/*
* ZoneMinder Signal Handling Interface, $Date$, $Revision$
* Copyright (C) 2003, 2004, 2005, 2006 Philip Coombes
*
* 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
#ifndef ZM_SIGNAL_H
#define ZM_SIGNAL_H
#include <signal.h>
#include <execinfo.h>
#include "zm.h"
typedef RETSIGTYPE (SigHandler)( int );
extern bool zm_reload;
extern bool zm_terminate;
RETSIGTYPE zmc_hup_handler( int signal );
RETSIGTYPE zmc_term_handler( int signal );
#if HAVE_STRUCT_SIGCONTEXT
RETSIGTYPE zmc_die_handler( int signal, struct sigcontext context );
#elif ( HAVE_SIGINFO_T && HAVE_UCONTEXT_T )
#include <ucontext.h>
RETSIGTYPE zmc_die_handler( int signal, siginfo_t *info, void *context );
#else
RETSIGTYPE zmc_die_handler( int signal );
#endif
void zmSetHupHandler( SigHandler *handler );
void zmSetTermHandler( SigHandler *handler );
void zmSetDieHandler( SigHandler *handler );
void zmSetDefaultHupHandler();
void zmSetDefaultTermHandler();
void zmSetDefaultDieHandler();
#endif // ZM_SIGNAL_H