2006-04-05 20:20:46 +08:00
|
|
|
//
|
|
|
|
// 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);
|
2006-05-24 21:51:23 +08:00
|
|
|
size_t errorStringSize = strlen(error) + strlen("Got signal (), reloading.");
|
2006-04-05 20:20:46 +08:00
|
|
|
char * errorString =(char *) malloc(errorStringSize + 1); // plus 1 for termination char.
|
2006-05-24 21:51:23 +08:00
|
|
|
(void) snprintf(errorString, errorStringSize, "Got signal (%s), reloading.", error);
|
2006-04-05 20:20:46 +08:00
|
|
|
|
|
|
|
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;
|
|
|
|
|
2006-05-05 19:23:26 +08:00
|
|
|
#if HAVE_STRUCT_SIGCONTEXT_EIP
|
2008-03-13 21:36:12 +08:00
|
|
|
Error(( "Signal address is %p, from %p\n", (void *)context.cr2, (void *)context.eip ));
|
2006-04-05 20:20:46 +08:00
|
|
|
|
|
|
|
trace_size = backtrace( trace, 16 );
|
|
|
|
// overwrite sigaction with caller's address
|
|
|
|
trace[1] = (void *)context.eip;
|
2006-05-05 19:23:26 +08:00
|
|
|
#elif HAVE_STRUCT_SIGCONTEXT
|
|
|
|
Error(( "Signal address is %p, no eip\n", context.cr2 ));
|
2006-04-05 20:20:46 +08:00
|
|
|
|
2006-05-05 19:23:26 +08:00
|
|
|
trace_size = backtrace( trace, 16 );
|
2006-04-05 20:20:46 +08:00
|
|
|
#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 );
|
|
|
|
}
|
|
|
|
|