📄 signalhandler.cpp
字号:
#include "../crashDefines.h"
#include "SignalHandler.h"
#include "GetSysInfo.h"
#if defined(LINUX)
#include "LinuxThreadHelpers.h"
#endif
#include <signal.h>
#include <cstdlib>
#include <unistd.h>
#include <sys/mman.h>
#include <cerrno>
#if defined(DARWIN)
extern "C" int sigaltstack( const struct sigaltstack *ss, struct sigaltstack *oss );
#define MAP_ANONYMOUS MAP_ANON
#endif
static vector<SignalHandler::handler> handlers;
SaveSignals *saved_sigs;
static int signals[] =
{
SIGALRM, SIGBUS, SIGFPE, SIGHUP, SIGILL, SIGINT, SIGABRT,
SIGQUIT, SIGSEGV, SIGTRAP, SIGTERM, SIGVTALRM, SIGXCPU, SIGXFSZ,
#if defined(HAVE_DECL_SIGPWR) && HAVE_DECL_SIGPWR
SIGPWR,
#endif
-1
};
/*
* Store signals and restore them. This lets us clean up after libraries
* that like to mess with our signal handler. (Of course, if we undo
* signal handlers for libraries, we need to be sure to handle whatever it
* was cleaning up by hand.)
*/
SaveSignals::SaveSignals()
{
/* Store the old signal handlers. */
for(int i = 0; signals[i] != -1; ++i)
{
struct sigaction sa;
sigaction(signals[i], NULL, &sa);
old_handlers.push_back(sa);
}
}
SaveSignals::~SaveSignals()
{
/* Restore the old signal handlers. */
for(unsigned i = 0; i < old_handlers.size(); ++i)
sigaction(signals[i], &old_handlers[i], NULL);
}
static void SigHandler( int signal, siginfo_t *si, void *ucp )
{
for(unsigned i = 0; i < handlers.size(); ++i)
handlers[i]( signal, si, (const ucontext_t *) ucp );
}
static int find_stack_direction2( char *p )
{
char c;
return (&c > p) ? +1:-1;
}
static int find_stack_direction()
{
char c;
return find_stack_direction2( &c );
}
/* Create a stack with a barrier page at the end to guard against stack overflow. */
static void *CreateStack( int size )
{
const long PageSize = getpagesize();
/* Round size up to the nearest PageSize. */
size += PageSize-1;
size -= (size%PageSize);
/* mmap always returns page-aligned data.
*
* mmap entries always show up individually in /proc/#/maps. We could use posix_memalign as
* a fallback, but we'd have to put a barrier page on both sides to guarantee that. */
char *p = NULL;
p = (char *) mmap( NULL, size+PageSize, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0 );
if( p == (void *) -1 )
return NULL;
// if( posix_memalign( (void**) &p, PageSize, RealSize ) != 0 )
// return NULL;
if( find_stack_direction() < 0 )
{
/* The stack grows towards smaller addresses. Protect the first page. */
mprotect( p, PageSize, PROT_NONE );
p += PageSize;
}
else
{
/* The stack grows towards larger addresses. Protect the last page. */
mprotect( p+size, PageSize, PROT_NONE );
}
return p;
}
/* Hook up events to fatal signals, so we can clean up if we're killed. */
void SignalHandler::OnClose(handler h)
{
if(saved_sigs == NULL)
{
saved_sigs = new SaveSignals;
bool bUseAltSigStack = true;
#if defined(LINUX)
/* Linuxthreads (pre-NPTL) sigaltstack is broken. */
if( !UsingNPTL() )
bUseAltSigStack = false;
#endif
/* Allocate a separate signal stack. This makes the crash handler work
* if we run out of stack space. */
const int AltStackSize = 1024*64;
void *p = NULL;
if( bUseAltSigStack )
p = CreateStack( AltStackSize );
if( p != NULL )
{
stack_t ss;
ss.ss_sp = (char*)p; /* cast for Darwin */
ss.ss_size = AltStackSize;
ss.ss_flags = 0;
if( sigaltstack( &ss, NULL ) == -1 )
{
//LOG->Info( "sigaltstack failed: %s", strerror(errno)
p = NULL; /* no SA_ONSTACK */
}
}
struct sigaction sa;
sa.sa_flags = 0;
if( p != NULL )
sa.sa_flags |= SA_ONSTACK;
sa.sa_flags |= SA_NODEFER;
sa.sa_flags |= SA_SIGINFO;
sigemptyset(&sa.sa_mask);
/* Set up our signal handlers. */
sa.sa_sigaction = SigHandler;
for(int i = 0; signals[i] != -1; ++i)
sigaction(signals[i], &sa, NULL);
/* Block SIGPIPE, so we get EPIPE. */
sa.sa_handler = SIG_IGN;
sigaction( SIGPIPE, &sa, NULL );
}
handlers.push_back(h);
}
/* Revert to the default signal handler. This is called */
void SignalHandler::ResetSignalHandlers()
{
struct sigaction sa;
sa.sa_flags = 0;
sigemptyset(&sa.sa_mask);
/* Set up the default signal handler. */
sa.sa_handler = SIG_DFL;
for(int i = 0; signals[i] != -1; ++i)
sigaction( signals[i], &sa, NULL );
}
/*
* (c) 2003-2004 Glenn Maynard
* All rights reserved.
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the
* "Software"), to deal in the Software without restriction, including
* without limitation the rights to use, copy, modify, merge, publish,
* distribute, and/or sell copies of the Software, and to permit persons to
* whom the Software is furnished to do so, provided that the above
* copyright notice(s) and this permission notice appear in all copies of
* the Software and that both the above copyright notice(s) and this
* permission notice appear in supporting documentation.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF
* THIRD PARTY RIGHTS. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR HOLDERS
* INCLUDED IN THIS NOTICE BE LIABLE FOR ANY CLAIM, OR ANY SPECIAL INDIRECT
* OR CONSEQUENTIAL DAMAGES, OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
* OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
* OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
* PERFORMANCE OF THIS SOFTWARE.
*/
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -