warzone2100/lib/framework/exceptionhandler.c

565 lines
16 KiB
C

/*
This file is part of Warzone 2100.
Copyright (C) 1999-2004 Eidos Interactive
Copyright (C) 2005-2007 Warzone Resurrection Project
Warzone 2100 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.
Warzone 2100 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 Warzone 2100; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include "frame.h"
#if defined(WZ_OS_WIN)
# include "dbghelp.h"
/**
* Exception handling on Windows.
* Ask the user whether he wants to safe a Minidump and then dump it into the temp directory.
*
* \param pExceptionInfo Information on the exception, passed from Windows
* \return whether further exception handlers (i.e. the Windows internal one) should be invoked
*/
static LONG WINAPI windowsExceptionHandler(PEXCEPTION_POINTERS pExceptionInfo)
{
LPCSTR applicationName = "Warzone 2100";
char miniDumpPath[MAX_PATH] = {'\0'}, resultMessage[MAX_PATH] = {'\0'};
// Write to temp dir, to support unprivileged users
if (!GetTempPathA( MAX_PATH, miniDumpPath ))
strcpy( miniDumpPath, "c:\\temp\\" );
strcat( miniDumpPath, "warzone2100.mdmp" );
/*
Alternative:
GetModuleFileName( NULL, miniDumpPath, MAX_PATH );
strcat( miniDumpPath, ".mdmp" );
*/
if ( MessageBoxA( NULL, "Warzone crashed unexpectedly, would you like to save a diagnostic file?", applicationName, MB_YESNO ) == IDYES )
{
HANDLE miniDumpFile = CreateFileA( miniDumpPath, GENERIC_WRITE, FILE_SHARE_WRITE, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL );
if (miniDumpFile != INVALID_HANDLE_VALUE)
{
MINIDUMP_USER_STREAM uStream = { LastReservedStream+1, strlen(VERSION), VERSION };
MINIDUMP_USER_STREAM_INFORMATION uInfo = { 1, &uStream };
MINIDUMP_EXCEPTION_INFORMATION eInfo = { GetCurrentThreadId(), pExceptionInfo, FALSE };
if ( MiniDumpWriteDump(
GetCurrentProcess(),
GetCurrentProcessId(),
miniDumpFile,
MiniDumpNormal,
pExceptionInfo ? &eInfo : NULL,
&uInfo,
NULL ) )
{
sprintf( resultMessage, "Saved dump file to '%s'", miniDumpPath );
MessageBoxA( NULL, resultMessage, applicationName, MB_OK );
}
else
{
sprintf( resultMessage, "Failed to save dump file to '%s' (error %d)", miniDumpPath, (int)GetLastError() );
MessageBoxA( NULL, resultMessage, applicationName, MB_OK );
}
CloseHandle(miniDumpFile);
}
else
{
sprintf( resultMessage, "Failed to create dump file '%s' (error %d)", miniDumpPath, (int)GetLastError() );
MessageBoxA( NULL, resultMessage, applicationName, MB_OK );
}
}
return EXCEPTION_CONTINUE_SEARCH;
}
#elif defined(WZ_OS_POSIX)
// C99 headers:
# include <stdint.h>
# include <signal.h>
# include <string.h>
// POSIX headers:
# include <unistd.h>
# include <fcntl.h>
# include <sys/types.h>
# include <sys/stat.h>
# include <sys/wait.h>
# include <sys/utsname.h>
// GNU extension for backtrace():
# if defined(__GLIBC__)
# include <execinfo.h>
# define MAX_BACKTRACE 20
# endif
# define MAX_PID_STRING 16
typedef void(*SigActionHandler)(int, siginfo_t *, void *);
static struct sigaction oldAction[NSIG];
static struct utsname sysInfo;
static BOOL gdbIsAvailable = FALSE, sysInfoValid = FALSE;
static char programPID[MAX_PID_STRING] = {'\0'}, gdbPath[MAX_PATH] = {'\0'};
static const char * gdmpPath, * programCommand;
/**
* Signal number to string mapper.
* Also takes into account the signal code with details about the signal.
*
* \param signum Signal number
* \param sigcode Signal code
* \return String with the description of the signal. "Unknown signal" when no description is available.
*/
static const char * wz_strsignal(int signum, int sigcode)
{
switch (signum)
{
case SIGABRT:
return "SIGABRT: Process abort signal";
case SIGALRM:
return "SIGALRM: Alarm clock";
case SIGBUS:
switch (sigcode)
{
case BUS_ADRALN:
return "SIGBUS: Access to an undefined portion of a memory object: Invalid address alignment";
case BUS_ADRERR:
return "SIGBUS: Access to an undefined portion of a memory object: Nonexistent physical address";
case BUS_OBJERR:
return "SIGBUS: Access to an undefined portion of a memory object: Object-specific hardware error";
default:
return "SIGBUS: Access to an undefined portion of a memory object";
}
case SIGCHLD:
switch (sigcode)
{
case CLD_EXITED:
return "SIGCHLD: Child process terminated, stopped, or continued: Child has exited";
case CLD_KILLED:
return "SIGCHLD: Child process terminated, stopped, or continued: Child has terminated abnormally and did not create a core file";
case CLD_DUMPED:
return "SIGCHLD: Child process terminated, stopped, or continued: Child has terminated abnormally and created a core file";
case CLD_TRAPPED:
return "SIGCHLD: Child process terminated, stopped, or continued: Traced child has trapped";
case CLD_STOPPED:
return "SIGCHLD: Child process terminated, stopped, or continued: Child has stopped";
case CLD_CONTINUED:
return "SIGCHLD: Child process terminated, stopped, or continued: Stopped child has continued";
}
case SIGCONT:
return "SIGCONT: Continue executing, if stopped";
case SIGFPE:
switch (sigcode)
{
case FPE_INTDIV:
return "SIGFPE: Erroneous arithmetic operation: Integer divide by zero";
case FPE_INTOVF:
return "SIGFPE: Erroneous arithmetic operation: Integer overflow";
case FPE_FLTDIV:
return "SIGFPE: Erroneous arithmetic operation: Floating-point divide by zero";
case FPE_FLTOVF:
return "SIGFPE: Erroneous arithmetic operation: Floating-point overflow";
case FPE_FLTUND:
return "SIGFPE: Erroneous arithmetic operation: Floating-point underflow";
case FPE_FLTRES:
return "SIGFPE: Erroneous arithmetic operation: Floating-point inexact result";
case FPE_FLTINV:
return "SIGFPE: Erroneous arithmetic operation: Invalid floating-point operation";
case FPE_FLTSUB:
return "SIGFPE: Erroneous arithmetic operation: Subscript out of range";
default:
return "SIGFPE: Erroneous arithmetic operation";
};
case SIGHUP:
return "SIGHUP: Hangup";
case SIGILL:
switch (sigcode)
{
case ILL_ILLOPC:
return "SIGILL: Illegal instruction: Illegal opcode";
case ILL_ILLOPN:
return "SIGILL: Illegal instruction: Illegal operand";
case ILL_ILLADR:
return "SIGILL: Illegal instruction: Illegal addressing mode";
case ILL_ILLTRP:
return "SIGILL: Illegal instruction: Illegal trap";
case ILL_PRVOPC:
return "SIGILL: Illegal instruction: Privileged opcode";
case ILL_PRVREG:
return "SIGILL: Illegal instruction: Privileged register";
case ILL_COPROC:
return "SIGILL: Illegal instruction: Coprocessor error";
case ILL_BADSTK:
return "SIGILL: Illegal instruction: Internal stack error";
default:
return "SIGILL: Illegal instruction";
}
case SIGINT:
return "SIGINT: Terminal interrupt signal";
case SIGKILL:
return "SIGKILL: Kill";
case SIGPIPE:
return "SIGPIPE: Write on a pipe with no one to read it";
case SIGQUIT:
return "SIGQUIT: Terminal quit signal";
case SIGSEGV:
switch (sigcode)
{
case SEGV_MAPERR:
return "SIGSEGV: Invalid memory reference: Address not mapped to object";
case SEGV_ACCERR:
return "SIGSEGV: Invalid memory reference: Invalid permissions for mapped object";
default:
return "SIGSEGV: Invalid memory reference";
}
case SIGSTOP:
return "SIGSTOP: Stop executing";
case SIGTERM:
return "SIGTERM: Termination signal";
case SIGTSTP:
return "SIGTSTP: Terminal stop signal";
case SIGTTIN:
return "SIGTTIN: Background process attempting read";
case SIGTTOU:
return "SIGTTOU: Background process attempting write";
case SIGUSR1:
return "SIGUSR1: User-defined signal 1";
case SIGUSR2:
return "SIGUSR2: User-defined signal 2";
#if _XOPEN_UNIX
case SIGPOLL:
switch (sigcode)
{
case POLL_IN:
return "SIGPOLL: Pollable event: Data input available";
case POLL_OUT:
return "SIGPOLL: Pollable event: Output buffers available";
case POLL_MSG:
return "SIGPOLL: Pollable event: Input message available";
case POLL_ERR:
return "SIGPOLL: Pollable event: I/O error";
case POLL_PRI:
return "SIGPOLL: Pollable event: High priority input available";
case POLL_HUP:
return "SIGPOLL: Pollable event: Device disconnected.";
default:
return "SIGPOLL: Pollable event";
}
case SIGPROF:
return "SIGPROF: Profiling timer expired";
case SIGSYS:
return "SIGSYS: Bad system call";
case SIGTRAP:
switch (sigcode)
{
case TRAP_BRKPT:
return "SIGTRAP: Trace/breakpoint trap: Process breakpoint";
case TRAP_TRACE:
return "SIGTRAP: Trace/breakpoint trap: Process trace trap";
default:
return "SIGTRAP: Trace/breakpoint trap";
}
#endif // _XOPEN_UNIX
case SIGURG:
return "SIGURG: High bandwidth data is available at a socket";
#if _XOPEN_UNIX
case SIGVTALRM:
return "SIGVTALRM: Virtual timer expired";
case SIGXCPU:
return "SIGXCPU: CPU time limit exceeded";
case SIGXFSZ:
return "SIGXFSZ: File size limit exceeded";
#endif // _XOPEN_UNIX
default:
return "Unknown signal";
}
}
/**
* Set signal handlers for fatal signals on POSIX systems
*
* \param signalHandler Pointer to the signal handler function
*/
static void setFatalSignalHandler(SigActionHandler signalHandler)
{
struct sigaction new_handler;
new_handler.sa_sigaction = signalHandler;
sigemptyset(&new_handler.sa_mask);
new_handler.sa_flags = SA_SIGINFO;
sigaction(SIGABRT, NULL, &oldAction[SIGABRT]);
if (oldAction[SIGABRT].sa_handler != SIG_IGN)
sigaction(SIGABRT, &new_handler, NULL);
sigaction(SIGBUS, NULL, &oldAction[SIGBUS]);
if (oldAction[SIGBUS].sa_handler != SIG_IGN)
sigaction(SIGBUS, &new_handler, NULL);
sigaction(SIGFPE, NULL, &oldAction[SIGFPE]);
if (oldAction[SIGFPE].sa_handler != SIG_IGN)
sigaction(SIGFPE, &new_handler, NULL);
sigaction(SIGILL, NULL, &oldAction[SIGILL]);
if (oldAction[SIGILL].sa_handler != SIG_IGN)
sigaction(SIGILL, &new_handler, NULL);
sigaction(SIGQUIT, NULL, &oldAction[SIGQUIT]);
if (oldAction[SIGQUIT].sa_handler != SIG_IGN)
sigaction(SIGQUIT, &new_handler, NULL);
sigaction(SIGSEGV, NULL, &oldAction[SIGSEGV]);
if (oldAction[SIGSEGV].sa_handler != SIG_IGN)
sigaction(SIGSEGV, &new_handler, NULL);
#if _XOPEN_UNIX
sigaction(SIGSYS, NULL, &oldAction[SIGSYS]);
if (oldAction[SIGSYS].sa_handler != SIG_IGN)
sigaction(SIGSYS, &new_handler, NULL);
sigaction(SIGTRAP, NULL, &oldAction[SIGTRAP]);
if (oldAction[SIGTRAP].sa_handler != SIG_IGN)
sigaction(SIGTRAP, &new_handler, NULL);
sigaction(SIGXCPU, NULL, &oldAction[SIGXCPU]);
if (oldAction[SIGXCPU].sa_handler != SIG_IGN)
sigaction(SIGXCPU, &new_handler, NULL);
sigaction(SIGXFSZ, NULL, &oldAction[SIGXFSZ]);
if (oldAction[SIGXFSZ].sa_handler != SIG_IGN)
sigaction(SIGXFSZ, &new_handler, NULL);
#endif // _XOPEN_UNIX
}
/**
* Exception (signal) handling on POSIX systems.
* Dumps info about the system incl. backtrace (when GLibC or GDB is present) to /tmp/warzone2100.gdmp
*
* \param signum Signal number
* \param siginfo Signal info
* \param sigcontext Signal context
*/
static void posixExceptionHandler(int signum, siginfo_t * siginfo, WZ_DECL_UNUSED void * sigcontext)
{
static sig_atomic_t allreadyRunning = 0;
if (allreadyRunning)
raise(signum);
allreadyRunning = 1;
# if defined(__GLIBC__)
void * btBuffer[MAX_BACKTRACE] = {NULL};
uint32_t btSize = backtrace(btBuffer, MAX_BACKTRACE);
# endif
pid_t pid = 0;
int gdbPipe[2] = {0}, dumpFile = open(gdmpPath, O_WRONLY|O_CREAT|O_TRUNC, S_IRUSR|S_IWUSR);
if (!dumpFile)
{
printf("Failed to create dump file '%s'", gdmpPath);
return;
}
write(dumpFile, "Program command: ", strlen("Program command: "));
write(dumpFile, programCommand, strlen(programCommand));
write(dumpFile, "\n", 1);
write(dumpFile, "Version: ", strlen("Version: "));
write(dumpFile, VERSION, strlen(VERSION));
write(dumpFile, "\n", 1);
# if defined(DEBUG)
write(dumpFile, "Type: Debug\n", strlen("Type: Debug\n"));
# else
write(dumpFile, "Type: Release\n", strlen("Type: Release\n"));
# endif
write(dumpFile, "Compiled on: ", strlen("Compiled on: "));
write(dumpFile, __DATE__, strlen(__DATE__));
write(dumpFile, "\n\n", 2);
if (!sysInfoValid)
write(dumpFile, "System information may be invalid!\n",
strlen("System information may be invalid!\n\n"));
write(dumpFile, "Operating system: ", strlen("Operating system: "));
write(dumpFile, sysInfo.sysname, strlen(sysInfo.sysname));
write(dumpFile, "\n", 1);
write(dumpFile, "Node name: ", strlen("Node name: "));
write(dumpFile, sysInfo.nodename, strlen(sysInfo.nodename));
write(dumpFile, "\n", 1);
write(dumpFile, "Release: ", strlen("Release: "));
write(dumpFile, sysInfo.release, strlen(sysInfo.release));
write(dumpFile, "\n", 1);
write(dumpFile, "Version: ", strlen("Version: "));
write(dumpFile, sysInfo.version, strlen(sysInfo.version));
write(dumpFile, "\n", 1);
write(dumpFile, "Machine: ", strlen("Machine: "));
write(dumpFile, sysInfo.machine, strlen(sysInfo.machine));
write(dumpFile, "\n\n", 2);
if (sizeof(void*) == 4)
write(dumpFile, "Pointers: 32bit\n\n", strlen("Pointers: 32bit\n\n"));
else if (sizeof(void*) == 8)
write(dumpFile, "Pointers: 64bit\n\n", strlen("Pointers: 64bit\n\n"));
else
write(dumpFile, "Pointers: Unknown\n\n", strlen("Pointers: Unknown\n\n"));
write(dumpFile, "Dump caused by signal: ",
strlen("Dump caused by signal: "));
write(dumpFile, wz_strsignal(siginfo->si_signo, siginfo->si_code),
strlen(wz_strsignal(siginfo->si_signo, siginfo->si_code)));
write(dumpFile, "\n\n", 2);
# if defined(__GLIBC__)
// Dump raw backtrace in case GDB is not available or fails
write(dumpFile, "GLIBC raw backtrace:\n", strlen("GLIBC raw backtrace:\n"));
backtrace_symbols_fd(btBuffer, btSize, dumpFile);
write(dumpFile, "\n", 1);
# else
write(dumpFile, "GLIBC not available, no raw backtrace dumped\n\n",
strlen("GLIBC not available, no raw backtrace dumped\n\n"));
# endif
// Make sure everything is written before letting GDB write to it
fsync(dumpFile);
if (gdbIsAvailable)
{
if (pipe(gdbPipe) == 0)
{
pid = fork();
if (pid == (pid_t)0)
{
const char * gdbArgv[] = { gdbPath, programCommand, programPID, NULL },
* gdbEnv[] = {NULL};
close(gdbPipe[1]); // No output to pipe
dup2(gdbPipe[0], STDIN_FILENO); // STDIN from pipe
dup2(dumpFile, STDOUT_FILENO); // STDOUT to dumpFile
write(dumpFile, "GDB extended backtrace:\n",
strlen("GDB extended backtrace:\n"));
execve(gdbPath, (char**)gdbArgv, (char**)gdbEnv);
}
else if (pid > (pid_t)0)
{
close(gdbPipe[0]); // No input from pipe
write(gdbPipe[1], "backtrace full\n" "quit\n",
strlen("backtrace full\n" "quit\n"));
if (waitpid(pid, NULL, 0) < 0)
{
printf("GDB failed\n");
}
close(gdbPipe[1]);
}
else
{
printf("Fork failed\n");
}
}
else
{
printf("Pipe failed\n");
}
}
else
{
write(dumpFile, "GDB not available, no extended backtrace dumped\n",
strlen("GDB not available, no extended backtrace dumped\n"));
}
printf("Saved dump file to '%s'\n", gdmpPath);
close(dumpFile);
sigaction(signum, &oldAction[signum], NULL);
raise(signum);
}
#endif // WZ_OS_*
/**
* Setup the exception handler responsible for target OS.
*
* \param programCommand_x Command used to launch this program. Only used for POSIX handler.
*/
void setupExceptionHandler(const char * programCommand_x)
{
#if defined(WZ_OS_WIN)
SetUnhandledExceptionFilter(windowsExceptionHandler);
#elif defined(WZ_OS_POSIX)
// Get full path to 'gdb'
FILE * whichStream = popen("which gdb", "r");
fread(gdbPath, 1, MAX_PATH, whichStream);
pclose(whichStream);
// Did we find GDB?
if (strlen(gdbPath) > 0)
{
gdbIsAvailable = TRUE;
*(strrchr(gdbPath, '\n')) = '\0'; // `which' adds a \n which confuses exec()
}
else
{
debug(LOG_WARNING, "GDB not available, will not create extended backtrace\n");
}
sysInfoValid = (uname(&sysInfo) == 0);
snprintf( programPID, MAX_PID_STRING, "%i", getpid() );
programCommand = programCommand_x;
gdmpPath = "/tmp/warzone2100.gdmp";
setFatalSignalHandler(posixExceptionHandler);
#endif // WZ_OS_*
}