osb/source/core/StarSignalHandler_windows.cpp

217 lines
7.5 KiB
C++
Raw Normal View History

2023-06-20 14:33:09 +10:00
#include "StarSignalHandler.hpp"
#include "StarFormat.hpp"
#include "StarString.hpp"
#include "StarLogging.hpp"
#include <windows.h>
2024-10-18 14:18:40 +11:00
#include "minidumpapiset.h"
2023-06-20 14:33:09 +10:00
namespace Star {
String g_sehMessage;
2024-10-18 14:18:40 +11:00
static DWORD WINAPI writeMiniDump(void* ExceptionInfo) {
auto hFile = CreateFileA("starbound.dmp", GENERIC_WRITE, FILE_SHARE_READ, 0, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0);
if (hFile == INVALID_HANDLE_VALUE)
return 0;
MINIDUMP_EXCEPTION_INFORMATION dumpExceptionInfo{};
dumpExceptionInfo.ThreadId = GetCurrentThreadId();
dumpExceptionInfo.ExceptionPointers = (PEXCEPTION_POINTERS)ExceptionInfo;
dumpExceptionInfo.ClientPointers = FALSE;
MiniDumpWriteDump(
GetCurrentProcess(),
GetCurrentProcessId(),
hFile,
MiniDumpNormal,
&dumpExceptionInfo,
NULL,
NULL);
CloseHandle(hFile);
if (dumpExceptionInfo.ExceptionPointers->ExceptionRecord->ExceptionCode == EXCEPTION_STACK_OVERFLOW) {
MessageBoxA(NULL, "Stack overflow encountered\nA minidump has been generated", NULL, MB_OK | MB_ICONERROR | MB_SETFOREGROUND);
}
return 0;
};
2023-06-20 14:33:09 +10:00
struct SignalHandlerImpl {
bool handlingFatal;
bool handlingInterrupt;
bool interrupted;
PVOID handler;
SignalHandlerImpl() : handlingFatal(false), handlingInterrupt(false), interrupted(false) {}
~SignalHandlerImpl() {
setHandleFatal(false);
setHandleInterrupt(false);
}
void setHandleFatal(bool b) {
handlingFatal = b;
if (handler) {
RemoveVectoredExceptionHandler(handler);
handler = nullptr;
}
if (handlingFatal)
handler = AddVectoredExceptionHandler(1, vectoredExceptionHandler);
}
void setHandleInterrupt(bool b) {
handlingInterrupt = b;
SetConsoleCtrlHandler(nullptr, false);
if (handlingInterrupt)
SetConsoleCtrlHandler((PHANDLER_ROUTINE)consoleCtrlHandler, true);
}
static void sehTrampoline() {
fatalError(g_sehMessage.utf8Ptr(), true);
}
static void handleFatalError(String const& msg, PEXCEPTION_POINTERS ExceptionInfo) {
if (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_ACCESS_VIOLATION) {
String mode;
DWORD modeFlag = ExceptionInfo->ExceptionRecord->ExceptionInformation[0];
if (modeFlag == 0)
mode = "Read";
else if (modeFlag == 1)
mode = "Write";
else if (modeFlag == 8)
mode = "Execute";
else
2023-06-27 20:23:44 +10:00
mode = strf("Mode({})", modeFlag);
g_sehMessage = strf("Access violation detected at {} ({} of address {})",
2023-06-20 14:33:09 +10:00
ExceptionInfo->ExceptionRecord->ExceptionAddress,
mode,
(PVOID)ExceptionInfo->ExceptionRecord->ExceptionInformation[1]);
} else {
g_sehMessage = msg;
2023-06-27 20:23:44 +10:00
g_sehMessage = strf("{} (%p @ {})",
2023-06-20 14:33:09 +10:00
g_sehMessage,
ExceptionInfo->ExceptionRecord->ExceptionCode,
ExceptionInfo->ExceptionRecord->ExceptionAddress);
for (DWORD i = 0; i < ExceptionInfo->ExceptionRecord->NumberParameters; i++)
2023-06-27 20:23:44 +10:00
g_sehMessage = strf("{} [{}]", g_sehMessage, (PVOID)ExceptionInfo->ExceptionRecord->ExceptionInformation[i]);
2023-06-20 14:33:09 +10:00
}
// setup a hijack into our own trampoline as if the failure actually was a
// function call
#ifdef STAR_ARCHITECTURE_X86_64
DWORD64 rsp = ExceptionInfo->ContextRecord->Rsp - 8;
DWORD64 rip = ExceptionInfo->ContextRecord->Rip; // an offset avoid the issue of gdb thinking
// the error is one statement too early, but
// the offset is instruction dependent, and we
// don't know its size + 1;
*((DWORD64*)rsp) = rip;
ExceptionInfo->ContextRecord->Rsp = rsp;
ExceptionInfo->ContextRecord->Rip = (DWORD64)&sehTrampoline;
#else
DWORD esp = ExceptionInfo->ContextRecord->Esp - 4;
DWORD eip = ExceptionInfo->ContextRecord->Eip; // an offset avoid the issue of gdb thinking the
// error is one statement too early, but the
// offset is instruction dependent, and we don't
// know its size + 1;
*((DWORD*)esp) = eip;
ExceptionInfo->ContextRecord->Esp = esp;
ExceptionInfo->ContextRecord->Eip = (DWORD)&sehTrampoline;
#endif
}
static LONG CALLBACK vectoredExceptionHandler(PEXCEPTION_POINTERS ExceptionInfo) {
2024-10-18 14:18:40 +11:00
auto thread = CreateThread(NULL, 0, writeMiniDump, (void*)ExceptionInfo, 0, NULL);
2024-10-23 15:05:00 +11:00
LONG result = EXCEPTION_CONTINUE_SEARCH;
2023-06-20 14:33:09 +10:00
if (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_ACCESS_VIOLATION) {
handleFatalError("Access violation detected", ExceptionInfo);
2024-10-23 15:05:00 +11:00
result = EXCEPTION_CONTINUE_EXECUTION;
2023-06-20 14:33:09 +10:00
}
if ((ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_ILLEGAL_INSTRUCTION)
|| (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_PRIV_INSTRUCTION)) {
handleFatalError("Illegal instruction encountered", ExceptionInfo);
2024-10-23 15:05:00 +11:00
result = EXCEPTION_CONTINUE_EXECUTION;
2023-06-20 14:33:09 +10:00
}
if ((ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_FLT_DENORMAL_OPERAND)
|| (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_FLT_DIVIDE_BY_ZERO)
|| (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_FLT_INEXACT_RESULT)
|| (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_FLT_INVALID_OPERATION)
|| (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_FLT_OVERFLOW)
|| (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_FLT_STACK_CHECK)
|| (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_FLT_UNDERFLOW)
) {
handleFatalError("Floating point exception", ExceptionInfo);
2024-10-23 15:05:00 +11:00
result = EXCEPTION_CONTINUE_EXECUTION;
2023-06-20 14:33:09 +10:00
}
if (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_INT_DIVIDE_BY_ZERO) {
handleFatalError("Division by zero", ExceptionInfo);
2024-10-23 15:05:00 +11:00
result = EXCEPTION_CONTINUE_EXECUTION;
2023-06-20 14:33:09 +10:00
}
if (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_INT_OVERFLOW) {
handleFatalError("Integer overflow", ExceptionInfo);
2024-10-23 15:05:00 +11:00
result = EXCEPTION_CONTINUE_EXECUTION;
2023-06-20 14:33:09 +10:00
}
if ((ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_DATATYPE_MISALIGNMENT)
|| (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_ARRAY_BOUNDS_EXCEEDED)
|| (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_IN_PAGE_ERROR)
|| (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_NONCONTINUABLE_EXCEPTION)
|| (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_INVALID_DISPOSITION)
|| (ExceptionInfo->ExceptionRecord->ExceptionCode == EXCEPTION_INVALID_HANDLE)) {
handleFatalError("Error occured", ExceptionInfo);
2024-10-23 15:05:00 +11:00
result = EXCEPTION_CONTINUE_EXECUTION;
2023-06-20 14:33:09 +10:00
}
2024-10-18 14:18:40 +11:00
if (thread != NULL) {
WaitForSingleObject(thread, 10000);
CloseHandle(thread);
}
2024-10-23 15:05:00 +11:00
return result;
2023-06-20 14:33:09 +10:00
}
static BOOL WINAPI consoleCtrlHandler(DWORD) {
if (SignalHandler::s_singleton)
SignalHandler::s_singleton->interrupted = true;
return true;
}
};
SignalHandlerImplUPtr SignalHandler::s_singleton;
SignalHandler::SignalHandler() {
if (s_singleton)
throw StarException("Singleton SignalHandler has been constructed twice!");
s_singleton = make_unique<SignalHandlerImpl>();
}
SignalHandler::~SignalHandler() {
s_singleton.reset();
}
void SignalHandler::setHandleFatal(bool handleFatal) {
s_singleton->setHandleFatal(handleFatal);
}
bool SignalHandler::handlingFatal() const {
return s_singleton->handlingFatal;
}
void SignalHandler::setHandleInterrupt(bool handleInterrupt) {
s_singleton->setHandleInterrupt(handleInterrupt);
}
bool SignalHandler::handlingInterrupt() const {
return s_singleton->handlingInterrupt;
}
bool SignalHandler::interruptCaught() const {
return s_singleton->interrupted;
}
}