mirror of
				https://github.com/Atmosphere-NX/Atmosphere.git
				synced 2025-10-23 00:35:48 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			165 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			165 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2019 Atmosphère-NX
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or modify it
 | |
|  * under the terms and conditions of the GNU General Public License,
 | |
|  * version 2, as published by the Free Software Foundation.
 | |
|  *
 | |
|  * This program is distributed in the hope 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 this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| 
 | |
| #include <stratosphere.hpp>
 | |
| #include "ams_bpc.h"
 | |
| 
 | |
| namespace ams {
 | |
| 
 | |
|     namespace {
 | |
| 
 | |
|         inline u64 GetPc() {
 | |
|             u64 pc;
 | |
|             __asm__ __volatile__ ("adr %[pc], ." : [pc]"=&r"(pc) :: );
 | |
|             return pc;
 | |
|         }
 | |
| 
 | |
|         struct StackFrame {
 | |
|             u64 fp;
 | |
|             u64 lr;
 | |
|         };
 | |
| 
 | |
|     }
 | |
| 
 | |
|     extern ncm::ProgramId CurrentProgramId;
 | |
| 
 | |
|     void WEAK ExceptionHandler(FatalErrorContext *ctx) {
 | |
|         R_ASSERT(amsBpcInitialize());
 | |
|         R_ASSERT(amsBpcRebootToFatalError(ctx));
 | |
|         while (1) { /* ... */ }
 | |
|     }
 | |
| 
 | |
|     void CrashHandler(ThreadExceptionDump *ctx) {
 | |
|         FatalErrorContext ams_ctx;
 | |
| 
 | |
|         /* Convert thread dump to atmosphere dump. */
 | |
|         {
 | |
|             ams_ctx.magic = FatalErrorContext::Magic;
 | |
|             ams_ctx.error_desc = ctx->error_desc;
 | |
|             ams_ctx.program_id = static_cast<u64>(CurrentProgramId);
 | |
|             for (size_t i = 0; i < FatalErrorContext::NumGprs; i++) {
 | |
|                 ams_ctx.gprs[i] = ctx->cpu_gprs[i].x;
 | |
|             }
 | |
|             if (ams_ctx.error_desc == FatalErrorContext::DataAbortErrorDesc &&
 | |
|                 ams_ctx.gprs[27] == FatalErrorContext::StdAbortMagicAddress &&
 | |
|                 ams_ctx.gprs[28] == FatalErrorContext::StdAbortMagicValue)
 | |
|             {
 | |
|                 /* Detect std::abort(). */
 | |
|                 ams_ctx.error_desc = FatalErrorContext::StdAbortErrorDesc;
 | |
|             }
 | |
| 
 | |
|             ams_ctx.fp = ctx->fp.x;
 | |
|             ams_ctx.lr = ctx->lr.x;
 | |
|             ams_ctx.sp = ctx->sp.x;
 | |
|             ams_ctx.pc = ctx->pc.x;
 | |
|             ams_ctx.pstate = ctx->pstate;
 | |
|             ams_ctx.afsr0 = ctx->afsr0;
 | |
|             ams_ctx.afsr1 = ctx->afsr1;
 | |
|             ams_ctx.far = ctx->far.x;
 | |
|             ams_ctx.report_identifier = armGetSystemTick();
 | |
|             /* Grab module base. */
 | |
|             {
 | |
|                 MemoryInfo mem_info;
 | |
|                 u32 page_info;
 | |
|                 if (R_SUCCEEDED(svcQueryMemory(&mem_info, &page_info, GetPc()))) {
 | |
|                     ams_ctx.module_base = mem_info.addr;
 | |
|                 } else {
 | |
|                     ams_ctx.module_base = 0;
 | |
|                 }
 | |
|             }
 | |
|             ams_ctx.stack_trace_size = 0;
 | |
|             u64 cur_fp = ams_ctx.fp;
 | |
|             for (size_t i = 0; i < FatalErrorContext::MaxStackTrace; i++) {
 | |
|                 /* Validate current frame. */
 | |
|                 if (cur_fp == 0 || (cur_fp & 0xF)) {
 | |
|                     break;
 | |
|                 }
 | |
| 
 | |
|                 /* Read a new frame. */
 | |
|                 StackFrame cur_frame;
 | |
|                 MemoryInfo mem_info;
 | |
|                 u32 page_info;
 | |
|                 if (R_SUCCEEDED(svcQueryMemory(&mem_info, &page_info, cur_fp)) && (mem_info.perm & Perm_R) == Perm_R) {
 | |
|                     std::memcpy(&cur_frame, reinterpret_cast<void *>(cur_fp), sizeof(cur_frame));
 | |
|                 } else {
 | |
|                     break;
 | |
|                 }
 | |
| 
 | |
|                 /* Advance to the next frame. */
 | |
|                 ams_ctx.stack_trace[ams_ctx.stack_trace_size++] = cur_frame.lr;
 | |
|                 cur_fp = cur_frame.fp;
 | |
|             }
 | |
|             /* Clear unused parts of stack trace. */
 | |
|             for (size_t i = ams_ctx.stack_trace_size; i < FatalErrorContext::MaxStackTrace; i++) {
 | |
|                 ams_ctx.stack_trace[i] = 0;
 | |
|             }
 | |
| 
 | |
|             /* Grab up to 0x100 of stack. */
 | |
|             {
 | |
|                 MemoryInfo mem_info;
 | |
|                 u32 page_info;
 | |
|                 if (R_SUCCEEDED(svcQueryMemory(&mem_info, &page_info, ams_ctx.sp)) && (mem_info.perm & Perm_R) == Perm_R) {
 | |
|                     size_t copy_size = std::min(FatalErrorContext::MaxStackDumpSize, static_cast<size_t>(mem_info.addr + mem_info.size - ams_ctx.sp));
 | |
|                     ams_ctx.stack_dump_size = copy_size;
 | |
|                     std::memcpy(ams_ctx.stack_dump, reinterpret_cast<void *>(ams_ctx.sp), copy_size);
 | |
|                 } else {
 | |
|                     ams_ctx.stack_dump_size = 0;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /* Just call the user exception handler. */
 | |
|         ::ams::ExceptionHandler(&ams_ctx);
 | |
|     }
 | |
| 
 | |
|     inline NORETURN void AbortImpl() {
 | |
|         /* Just perform a data abort. */
 | |
|         register u64 addr __asm__("x27") = FatalErrorContext::StdAbortMagicAddress;
 | |
|         register u64 val __asm__("x28")  = FatalErrorContext::StdAbortMagicValue;
 | |
|         while (true) {
 | |
|             __asm__ __volatile__ (
 | |
|                 "str %[val], [%[addr]]"
 | |
|                 :
 | |
|                 : [val]"r"(val), [addr]"r"(addr)
 | |
|             );
 | |
|         }
 | |
|     }
 | |
| 
 | |
| }
 | |
| 
 | |
| extern "C" {
 | |
| 
 | |
|     /* Redefine abort to trigger these handlers. */
 | |
|     void abort();
 | |
| 
 | |
|     /* Redefine C++ exception handlers. Requires wrap linker flag. */
 | |
|     void __wrap___cxa_pure_virtual(void) { abort(); }
 | |
|     void __wrap___cxa_throw(void) { abort(); }
 | |
|     void __wrap___cxa_rethrow(void) { abort(); }
 | |
|     void __wrap___cxa_allocate_exception(void) { abort(); }
 | |
|     void __wrap___cxa_begin_catch(void) { abort(); }
 | |
|     void __wrap___cxa_end_catch(void) { abort(); }
 | |
|     void __wrap___cxa_call_unexpected(void) { abort(); }
 | |
|     void __wrap___cxa_call_terminate(void) { abort(); }
 | |
|     void __wrap___gxx_personality_v0(void) { abort(); }
 | |
| 
 | |
| }
 | |
| 
 | |
| /* Custom abort handler, so that std::abort will trigger these. */
 | |
| void abort() {
 | |
|     ams::AbortImpl();
 | |
| }
 |