mirror of
https://github.com/hrydgard/ppsspp.git
synced 2024-11-23 21:39:52 +00:00
cee71cbfc3
Motivated by some crashes seen in Google Play reports.
350 lines
9.9 KiB
C++
350 lines
9.9 KiB
C++
// Copyright 2008 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
// The corresponding file is called MemTools in the Dolphin project.
|
|
|
|
#include "ppsspp_config.h"
|
|
|
|
#include <cstdio>
|
|
#include <cstdlib>
|
|
#include <cstring>
|
|
#include <vector>
|
|
#include <thread>
|
|
|
|
#include "Common/CommonFuncs.h"
|
|
#include "Common/CommonTypes.h"
|
|
#include "Common/Log.h"
|
|
#include "Common/Thread/ThreadUtil.h"
|
|
#include "Common/MachineContext.h"
|
|
#include "Common/ExceptionHandlerSetup.h"
|
|
|
|
static BadAccessHandler g_badAccessHandler;
|
|
static void *altStack = nullptr;
|
|
|
|
#ifdef MACHINE_CONTEXT_SUPPORTED
|
|
|
|
// We cannot handle exceptions in UWP builds. Bleh.
|
|
#if PPSSPP_PLATFORM(WINDOWS) && !PPSSPP_PLATFORM(UWP)
|
|
|
|
static PVOID g_vectoredExceptionHandle;
|
|
|
|
static LONG NTAPI GlobalExceptionHandler(PEXCEPTION_POINTERS pPtrs) {
|
|
switch (pPtrs->ExceptionRecord->ExceptionCode) {
|
|
case EXCEPTION_ACCESS_VIOLATION:
|
|
{
|
|
int accessType = (int)pPtrs->ExceptionRecord->ExceptionInformation[0];
|
|
if (accessType == 8) { // Rule out DEP
|
|
return (DWORD)EXCEPTION_CONTINUE_SEARCH;
|
|
}
|
|
|
|
// virtual address of the inaccessible data
|
|
uintptr_t badAddress = (uintptr_t)pPtrs->ExceptionRecord->ExceptionInformation[1];
|
|
CONTEXT* ctx = pPtrs->ContextRecord;
|
|
|
|
if (g_badAccessHandler(badAddress, ctx)) {
|
|
return (DWORD)EXCEPTION_CONTINUE_EXECUTION;
|
|
} else {
|
|
// Let's not prevent debugging.
|
|
return (DWORD)EXCEPTION_CONTINUE_SEARCH;
|
|
}
|
|
}
|
|
|
|
case EXCEPTION_STACK_OVERFLOW:
|
|
// Dolphin has some handling of this for the RET optimization emulation.
|
|
return EXCEPTION_CONTINUE_SEARCH;
|
|
|
|
case EXCEPTION_ILLEGAL_INSTRUCTION:
|
|
// No SSE support? Or simply bad codegen?
|
|
return EXCEPTION_CONTINUE_SEARCH;
|
|
|
|
case EXCEPTION_PRIV_INSTRUCTION:
|
|
// okay, dynarec codegen is obviously broken.
|
|
return EXCEPTION_CONTINUE_SEARCH;
|
|
|
|
case EXCEPTION_IN_PAGE_ERROR:
|
|
// okay, something went seriously wrong, out of memory?
|
|
return EXCEPTION_CONTINUE_SEARCH;
|
|
|
|
case EXCEPTION_BREAKPOINT:
|
|
// might want to do something fun with this one day?
|
|
return EXCEPTION_CONTINUE_SEARCH;
|
|
|
|
default:
|
|
return EXCEPTION_CONTINUE_SEARCH;
|
|
}
|
|
}
|
|
|
|
void InstallExceptionHandler(BadAccessHandler badAccessHandler) {
|
|
if (g_vectoredExceptionHandle) {
|
|
g_badAccessHandler = badAccessHandler;
|
|
return;
|
|
}
|
|
|
|
INFO_LOG(SYSTEM, "Installing exception handler");
|
|
g_badAccessHandler = badAccessHandler;
|
|
g_vectoredExceptionHandle = AddVectoredExceptionHandler(TRUE, GlobalExceptionHandler);
|
|
}
|
|
|
|
void UninstallExceptionHandler() {
|
|
if (g_vectoredExceptionHandle) {
|
|
RemoveVectoredExceptionHandler(g_vectoredExceptionHandle);
|
|
INFO_LOG(SYSTEM, "Removed exception handler");
|
|
g_vectoredExceptionHandle = nullptr;
|
|
}
|
|
g_badAccessHandler = nullptr;
|
|
}
|
|
|
|
#elif defined(__APPLE__)
|
|
|
|
static void CheckKR(const char* name, kern_return_t kr) {
|
|
_assert_msg_(kr == 0, "%s failed: kr=%x", name, kr);
|
|
}
|
|
|
|
static void ExceptionThread(mach_port_t port) {
|
|
SetCurrentThreadName("Mach exception thread");
|
|
#pragma pack(4)
|
|
struct {
|
|
mach_msg_header_t Head;
|
|
NDR_record_t NDR;
|
|
exception_type_t exception;
|
|
mach_msg_type_number_t codeCnt;
|
|
int64_t code[2];
|
|
int flavor;
|
|
mach_msg_type_number_t old_stateCnt;
|
|
natural_t old_state[x86_THREAD_STATE64_COUNT];
|
|
mach_msg_trailer_t trailer;
|
|
} msg_in;
|
|
|
|
struct {
|
|
mach_msg_header_t Head;
|
|
NDR_record_t NDR;
|
|
kern_return_t RetCode;
|
|
int flavor;
|
|
mach_msg_type_number_t new_stateCnt;
|
|
natural_t new_state[x86_THREAD_STATE64_COUNT];
|
|
} msg_out;
|
|
#pragma pack()
|
|
memset(&msg_in, 0xee, sizeof(msg_in));
|
|
memset(&msg_out, 0xee, sizeof(msg_out));
|
|
mach_msg_header_t* send_msg = &msg_out.Head;
|
|
mach_msg_size_t send_size = 0;
|
|
mach_msg_option_t option = MACH_RCV_MSG;
|
|
while (true) {
|
|
// If this isn't the first run, send the reply message. Then, receive
|
|
// a message: either a mach_exception_raise_state RPC due to
|
|
// thread_set_exception_ports, or MACH_NOTIFY_NO_SENDERS due to
|
|
// mach_port_request_notification.
|
|
CheckKR("mach_msg_overwrite",
|
|
mach_msg_overwrite(send_msg, option, send_size, sizeof(msg_in), port,
|
|
MACH_MSG_TIMEOUT_NONE, MACH_PORT_NULL, &msg_in.Head, 0));
|
|
|
|
if (msg_in.Head.msgh_id == MACH_NOTIFY_NO_SENDERS) {
|
|
// the other thread exited
|
|
mach_port_destroy(mach_task_self(), port);
|
|
return;
|
|
}
|
|
|
|
_assert_msg_(msg_in.Head.msgh_id == 2406, "unknown message received");
|
|
_assert_msg_(msg_in.flavor == x86_THREAD_STATE64, "unknown flavor %d (expected %d)", msg_in.flavor, x86_THREAD_STATE64);
|
|
|
|
x86_thread_state64_t* state = (x86_thread_state64_t*)msg_in.old_state;
|
|
|
|
bool ok = g_badAccessHandler((uintptr_t)msg_in.code[1], state);
|
|
|
|
// Set up the reply.
|
|
msg_out.Head.msgh_bits = MACH_MSGH_BITS(MACH_MSGH_BITS_REMOTE(msg_in.Head.msgh_bits), 0);
|
|
msg_out.Head.msgh_remote_port = msg_in.Head.msgh_remote_port;
|
|
msg_out.Head.msgh_local_port = MACH_PORT_NULL;
|
|
msg_out.Head.msgh_id = msg_in.Head.msgh_id + 100;
|
|
msg_out.NDR = msg_in.NDR;
|
|
if (ok) {
|
|
msg_out.RetCode = KERN_SUCCESS;
|
|
msg_out.flavor = x86_THREAD_STATE64;
|
|
msg_out.new_stateCnt = x86_THREAD_STATE64_COUNT;
|
|
memcpy(msg_out.new_state, msg_in.old_state, x86_THREAD_STATE64_COUNT * sizeof(natural_t));
|
|
} else {
|
|
// Pass the exception to the next handler (debugger or crash).
|
|
msg_out.RetCode = KERN_FAILURE;
|
|
msg_out.flavor = 0;
|
|
msg_out.new_stateCnt = 0;
|
|
}
|
|
msg_out.Head.msgh_size =
|
|
offsetof(__typeof__(msg_out), new_state) + msg_out.new_stateCnt * sizeof(natural_t);
|
|
|
|
send_msg = &msg_out.Head;
|
|
send_size = msg_out.Head.msgh_size;
|
|
option |= MACH_SEND_MSG;
|
|
}
|
|
}
|
|
|
|
void InstallExceptionHandler(BadAccessHandler badAccessHandler) {
|
|
if (g_badAccessHandler) {
|
|
// The rest of the setup we don't need to do again.
|
|
g_badAccessHandler = badAccessHandler;
|
|
return;
|
|
}
|
|
g_badAccessHandler = badAccessHandler;
|
|
|
|
INFO_LOG(SYSTEM, "Installing exception handler");
|
|
mach_port_t port;
|
|
CheckKR("mach_port_allocate",
|
|
mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE, &port));
|
|
std::thread exc_thread(ExceptionThread, port);
|
|
exc_thread.detach();
|
|
// Obtain a send right for thread_set_exception_ports to copy...
|
|
CheckKR("mach_port_insert_right",
|
|
mach_port_insert_right(mach_task_self(), port, port, MACH_MSG_TYPE_MAKE_SEND));
|
|
// Mach tries the following exception ports in order: thread, task, host.
|
|
// Debuggers set the task port, so we grab the thread port.
|
|
CheckKR("thread_set_exception_ports",
|
|
thread_set_exception_ports(mach_thread_self(), EXC_MASK_BAD_ACCESS, port,
|
|
EXCEPTION_STATE | MACH_EXCEPTION_CODES, x86_THREAD_STATE64));
|
|
// ...and get rid of our copy so that MACH_NOTIFY_NO_SENDERS works.
|
|
CheckKR("mach_port_mod_refs",
|
|
mach_port_mod_refs(mach_task_self(), port, MACH_PORT_RIGHT_SEND, -1));
|
|
mach_port_t previous;
|
|
CheckKR("mach_port_request_notification",
|
|
mach_port_request_notification(mach_task_self(), port, MACH_NOTIFY_NO_SENDERS, 0, port,
|
|
MACH_MSG_TYPE_MAKE_SEND_ONCE, &previous));
|
|
}
|
|
|
|
void UninstallExceptionHandler() {
|
|
}
|
|
|
|
#else
|
|
|
|
#include <signal.h>
|
|
|
|
static struct sigaction old_sa_segv;
|
|
static struct sigaction old_sa_bus;
|
|
|
|
static void sigsegv_handler(int sig, siginfo_t* info, void* raw_context) {
|
|
if (sig != SIGSEGV && sig != SIGBUS) {
|
|
// We are not interested in other signals - handle it as usual.
|
|
return;
|
|
}
|
|
ucontext_t* context = (ucontext_t*)raw_context;
|
|
int sicode = info->si_code;
|
|
if (sicode != SEGV_MAPERR && sicode != SEGV_ACCERR) {
|
|
// Huh? Return.
|
|
return;
|
|
}
|
|
uintptr_t bad_address = (uintptr_t)info->si_addr;
|
|
|
|
// Get all the information we can out of the context.
|
|
#ifdef __OpenBSD__
|
|
ucontext_t* ctx = context;
|
|
#else
|
|
mcontext_t* ctx = &context->uc_mcontext;
|
|
#endif
|
|
// assume it's not a write
|
|
if (!g_badAccessHandler(bad_address,
|
|
#ifdef __APPLE__
|
|
*ctx
|
|
#else
|
|
ctx
|
|
#endif
|
|
)) {
|
|
// retry and crash
|
|
// According to the sigaction man page, if sa_flags "SA_SIGINFO" is set to the sigaction
|
|
// function pointer, otherwise sa_handler contains one of:
|
|
// SIG_DEF: The 'default' action is performed
|
|
// SIG_IGN: The signal is ignored
|
|
// Any other value is a function pointer to a signal handler
|
|
|
|
struct sigaction* old_sa;
|
|
if (sig == SIGSEGV) {
|
|
old_sa = &old_sa_segv;
|
|
} else {
|
|
old_sa = &old_sa_bus;
|
|
}
|
|
|
|
if (old_sa->sa_flags & SA_SIGINFO) {
|
|
old_sa->sa_sigaction(sig, info, raw_context);
|
|
return;
|
|
}
|
|
if (old_sa->sa_handler == SIG_DFL) {
|
|
signal(sig, SIG_DFL);
|
|
return;
|
|
}
|
|
if (old_sa->sa_handler == SIG_IGN) {
|
|
// Ignore signal
|
|
return;
|
|
}
|
|
old_sa->sa_handler(sig);
|
|
}
|
|
}
|
|
|
|
void InstallExceptionHandler(BadAccessHandler badAccessHandler) {
|
|
if (!badAccessHandler) {
|
|
return;
|
|
}
|
|
if (g_badAccessHandler) {
|
|
g_badAccessHandler = badAccessHandler;
|
|
return;
|
|
}
|
|
|
|
size_t altStackSize = SIGSTKSZ;
|
|
|
|
// Add some extra room.
|
|
altStackSize += 65536;
|
|
|
|
INFO_LOG(SYSTEM, "Installed exception handler. stack size: %d", (int)altStackSize);
|
|
g_badAccessHandler = badAccessHandler;
|
|
|
|
stack_t signal_stack{};
|
|
altStack = malloc(altStackSize);
|
|
#ifdef __FreeBSD__
|
|
signal_stack.ss_sp = (char*)altStack;
|
|
#else
|
|
signal_stack.ss_sp = altStack;
|
|
#endif
|
|
signal_stack.ss_size = altStackSize;
|
|
signal_stack.ss_flags = 0;
|
|
if (sigaltstack(&signal_stack, nullptr)) {
|
|
_assert_msg_(false, "sigaltstack failed");
|
|
}
|
|
struct sigaction sa{};
|
|
sa.sa_handler = nullptr;
|
|
sa.sa_sigaction = &sigsegv_handler;
|
|
sa.sa_flags = SA_SIGINFO | SA_ONSTACK;
|
|
sigemptyset(&sa.sa_mask);
|
|
sigaction(SIGSEGV, &sa, &old_sa_segv);
|
|
#ifdef __APPLE__
|
|
sigaction(SIGBUS, &sa, &old_sa_bus);
|
|
#endif
|
|
}
|
|
|
|
void UninstallExceptionHandler() {
|
|
if (!g_badAccessHandler) {
|
|
return;
|
|
}
|
|
stack_t signal_stack{};
|
|
signal_stack.ss_flags = SS_DISABLE;
|
|
if (0 != sigaltstack(&signal_stack, nullptr)) {
|
|
ERROR_LOG(SYSTEM, "Could not remove signal altstack");
|
|
}
|
|
if (altStack) {
|
|
free(altStack);
|
|
altStack = nullptr;
|
|
}
|
|
sigaction(SIGSEGV, &old_sa_segv, nullptr);
|
|
#ifdef __APPLE__
|
|
sigaction(SIGBUS, &old_sa_bus, nullptr);
|
|
#endif
|
|
INFO_LOG(SYSTEM, "Uninstalled exception handler");
|
|
g_badAccessHandler = nullptr;
|
|
}
|
|
|
|
#endif
|
|
|
|
#else // !MACHINE_CONTEXT_SUPPORTED
|
|
|
|
void InstallExceptionHandler(BadAccessHandler badAccessHandler) {
|
|
ERROR_LOG(SYSTEM, "Exception handler not implemented on this platform, can't install");
|
|
}
|
|
void UninstallExceptionHandler() { }
|
|
|
|
#endif // MACHINE_CONTEXT_SUPPORTED
|