/* Copyright (C) 2022 nillerusr 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 3 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. */ #include #include #include #include #include #include "tier0/dbg.h" #include #include #include "libunwind/libunwind.h" struct sigaction old_sa; #define IN_LIBGCC2 1 // means we want to define __cxxabiv1::__cxa_demangle namespace __cxxabiv1 { extern "C" { #include "demangle/cp-demangle.c" } } #define MAX_FRAMES 2048 struct backtrace_t { int count; uintptr_t frames[MAX_FRAMES]; }; #define Log(msg) __android_log_print(ANDROID_LOG_DEBUG, "SRCENG", "%s", msg); DebugLogger()->Write(msg); void printPC(void *pc) { char message[4096]; const char* symbol = "unknown"; Dl_info info = { 0 }; const char *fname = "unknown"; if( dladdr(pc, &info) <= 0 ) { snprintf( message, sizeof(message), "0x%" PRIXPTR "\n", (uintptr_t)pc ); Log(message); return; } if( info.dli_fname ) fname = info.dli_fname; if( info.dli_sname ) symbol = info.dli_sname; int status = 0; char *demangled = __cxxabiv1::__cxa_demangle(symbol, 0, 0, &status); if( NULL != demangled && 0 == status ) symbol = demangled; uintptr_t relative_addr = (uintptr_t)pc - (uintptr_t)info.dli_fbase; snprintf( message, sizeof(message), "0x%" PRIXPTR ":\t%s (base=0x%" PRIXPTR ", %s)\n", relative_addr, symbol, (uintptr_t)info.dli_fbase, fname ); Log(message); } _Unwind_Reason_Code UnwindBacktraceCallback(struct _Unwind_Context* unwind_context, void* state_voidp) { uintptr_t pc = _Unwind_GetIP(unwind_context); backtrace_t *bt = (backtrace_t*)state_voidp; if( bt->count < MAX_FRAMES ) bt->frames[bt->count++] = pc; else return _URC_END_OF_STACK; return _URC_NO_REASON; } static void CrashHandler( int sig, siginfo_t *si, void *uc) { static char message[4096], symbol[256]; int len, line, logfd, i = 0; const ucontext_t* signal_ucontext = (ucontext_t*)uc; const mcontext_t* signal_mcontext = &(signal_ucontext->uc_mcontext); #ifdef __aarch64__ // Doesn't work good on armv7a static backtrace_t bt; bt.count = 0; _Unwind_Backtrace(UnwindBacktraceCallback, &bt); Log(">>> crash report begin\n"); snprintf(message, sizeof(message), "Signal=%d, errno=%d, code=%d, addr=0x%" PRIXPTR "\n", sig, si->si_errno, si->si_code, (uintptr_t)si->si_addr); Log(message); for( int i = 0; i < bt.count; i++ ) { printPC( (void*)bt.frames[i] ); } #else Log(">>> crash report begin\n"); snprintf(message, sizeof(message), "Signal=%d, errno=%d, code=%d, addr=0x%" PRIXPTR "\n", sig, si->si_errno, si->si_code, (uintptr_t)si->si_addr); Log(message); // Initialize unw_context and unw_cursor. unw_context_t unw_context = {}; unw_getcontext(&unw_context); unw_cursor_t unw_cursor = {}; unw_init_local(&unw_cursor, &unw_context); while (unw_step(&unw_cursor) > 0) { unw_word_t ip = 0; unw_get_reg(&unw_cursor, UNW_REG_IP, &ip); printPC( (void*)ip ); } #endif Log(">>> crash report end\n"); if (old_sa.sa_sigaction) (*old_sa.sa_sigaction)(sig, si, uc); else if(old_sa.sa_handler) (*old_sa.sa_handler)(sig); } void InitCrashHandler() { struct sigaction act; act.sa_sigaction = CrashHandler; act.sa_flags = SA_SIGINFO | SA_ONSTACK; sigaction(SIGSEGV, &act, &old_sa); sigaction(SIGABRT, &act, &old_sa); sigaction(SIGBUS, &act, &old_sa); sigaction(SIGFPE, &act, &old_sa); sigaction(SIGTRAP, &act, &old_sa); }