mirror of
				https://github.com/Atmosphere-NX/Atmosphere.git
				synced 2025-10-25 09:45:47 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			405 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			405 lines
		
	
	
		
			14 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 <cstdio>
 | |
| #include <cstring>
 | |
| #include <sys/stat.h>
 | |
| #include <sys/types.h>
 | |
| #include <switch.h>
 | |
| 
 | |
| #include "creport_crash_report.hpp"
 | |
| #include "creport_debug_types.hpp"
 | |
| 
 | |
| void CrashReport::BuildReport(u64 pid, bool has_extra_info) {
 | |
|     this->has_extra_info = has_extra_info;
 | |
|     if (OpenProcess(pid)) {
 | |
|         ProcessExceptions();
 | |
|         this->code_list.ReadCodeRegionsFromThreadInfo(this->debug_handle, &this->crashed_thread_info);
 | |
|         this->thread_list.ReadThreadsFromProcess(this->thread_tls_map, this->debug_handle, Is64Bit());
 | |
|         this->crashed_thread_info.SetCodeList(&this->code_list);
 | |
|         this->thread_list.SetCodeList(&this->code_list);
 | |
| 
 | |
|         if (IsApplication()) {
 | |
|             ProcessDyingMessage();
 | |
|         }
 | |
| 
 | |
|         /* Real creport only does this if application, but there's no reason not to do it all the time. */
 | |
|         for (u32 i = 0; i < this->thread_list.GetThreadCount(); i++) {
 | |
|             this->code_list.ReadCodeRegionsFromThreadInfo(this->debug_handle, this->thread_list.GetThreadInfo(i));
 | |
|         }
 | |
| 
 | |
|         /* Real creport builds the report here. We do it later. */
 | |
| 
 | |
|         Close();
 | |
|     }
 | |
| }
 | |
| 
 | |
| FatalContext *CrashReport::GetFatalContext() {
 | |
|     FatalContext *ctx = new FatalContext;
 | |
|     *ctx = (FatalContext){0};
 | |
| 
 | |
|     ctx->is_aarch32 = false;
 | |
|     ctx->type = static_cast<u32>(this->exception_info.type);
 | |
| 
 | |
|     for (size_t i = 0; i < 29; i++) {
 | |
|         ctx->aarch64_ctx.x[i] = this->crashed_thread_info.context.cpu_gprs[i].x;
 | |
|     }
 | |
|     ctx->aarch64_ctx.fp = this->crashed_thread_info.context.fp;
 | |
|     ctx->aarch64_ctx.lr = this->crashed_thread_info.context.lr;
 | |
|     ctx->aarch64_ctx.pc = this->crashed_thread_info.context.pc.x;
 | |
| 
 | |
|     ctx->aarch64_ctx.stack_trace_size = this->crashed_thread_info.stack_trace_size;
 | |
|     for (size_t i = 0; i < ctx->aarch64_ctx.stack_trace_size; i++) {
 | |
|         ctx->aarch64_ctx.stack_trace[i] = this->crashed_thread_info.stack_trace[i];
 | |
|     }
 | |
| 
 | |
|     if (this->code_list.code_count) {
 | |
|         ctx->aarch64_ctx.start_address = this->code_list.code_infos[0].start_address;
 | |
|     }
 | |
| 
 | |
|     /* For ams fatal... */
 | |
|     ctx->aarch64_ctx.afsr0 = this->process_info.title_id;
 | |
| 
 | |
|     return ctx;
 | |
| }
 | |
| 
 | |
| void CrashReport::ProcessExceptions() {
 | |
|     if (!IsOpen()) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     DebugEventInfo d;
 | |
|     while (R_SUCCEEDED(svcGetDebugEvent((u8 *)&d, this->debug_handle))) {
 | |
|         switch (d.type) {
 | |
|             case DebugEventType::AttachProcess:
 | |
|                 HandleAttachProcess(d);
 | |
|                 break;
 | |
|             case DebugEventType::Exception:
 | |
|                 HandleException(d);
 | |
|                 break;
 | |
|             case DebugEventType::AttachThread:
 | |
|                 HandleAttachThread(d);
 | |
|             case DebugEventType::ExitProcess:
 | |
|             case DebugEventType::ExitThread:
 | |
|             default:
 | |
|                 break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* Parse crashing thread info. */
 | |
|     this->crashed_thread_info.ReadFromProcess(this->thread_tls_map, this->debug_handle, this->crashed_thread_id, Is64Bit());
 | |
| }
 | |
| 
 | |
| void CrashReport::HandleAttachProcess(DebugEventInfo &d) {
 | |
|     this->process_info = d.info.attach_process;
 | |
|     if ((GetRuntimeFirmwareVersion() >= FirmwareVersion_500) && IsApplication()) {
 | |
|         /* Parse out user data. */
 | |
|         u64 address = this->process_info.user_exception_context_address;
 | |
|         u64 userdata_address = 0;
 | |
|         u64 userdata_size = 0;
 | |
| 
 | |
|         if (!IsAddressReadable(address, sizeof(userdata_address) + sizeof(userdata_size))) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Read userdata address. */
 | |
|         if (R_FAILED(svcReadDebugProcessMemory(&userdata_address, this->debug_handle, address, sizeof(userdata_address)))) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Validate userdata address. */
 | |
|         if (userdata_address == 0 || userdata_address & 0xFFF) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Read userdata size. */
 | |
|         if (R_FAILED(svcReadDebugProcessMemory(&userdata_size, this->debug_handle, address + sizeof(userdata_address), sizeof(userdata_size)))) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Cap userdata size. */
 | |
|         if (userdata_size > sizeof(this->dying_message)) {
 | |
|             userdata_size = sizeof(this->dying_message);
 | |
|         }
 | |
| 
 | |
|         /* Assign. */
 | |
|         this->dying_message_address = userdata_address;
 | |
|         this->dying_message_size = userdata_size;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CrashReport::HandleException(DebugEventInfo &d) {
 | |
|     switch (d.info.exception.type) {
 | |
|         case DebugExceptionType::UndefinedInstruction:
 | |
|             this->result = ResultCreportUndefinedInstruction;
 | |
|             break;
 | |
|         case DebugExceptionType::InstructionAbort:
 | |
|             this->result = ResultCreportInstructionAbort;
 | |
|             d.info.exception.specific.raw = 0;
 | |
|             break;
 | |
|         case DebugExceptionType::DataAbort:
 | |
|             this->result = ResultCreportDataAbort;
 | |
|             break;
 | |
|         case DebugExceptionType::AlignmentFault:
 | |
|             this->result = ResultCreportAlignmentFault;
 | |
|             break;
 | |
|         case DebugExceptionType::UserBreak:
 | |
|             this->result = ResultCreportUserBreak;
 | |
|             /* Try to parse out the user break result. */
 | |
|             if ((GetRuntimeFirmwareVersion() >= FirmwareVersion_500)) {
 | |
|                 Result user_result = 0;
 | |
|                 if (IsAddressReadable(d.info.exception.specific.user_break.address, sizeof(user_result))) {
 | |
|                     svcReadDebugProcessMemory(&user_result, this->debug_handle, d.info.exception.specific.user_break.address, sizeof(user_result));
 | |
|                 }
 | |
|                 /* Only copy over the user result if it gives us information (as by default nnSdk uses the success code, which is confusing). */
 | |
|                 if (R_FAILED(user_result)) {
 | |
|                     this->result = user_result;
 | |
|                 }
 | |
|             }
 | |
|             break;
 | |
|         case DebugExceptionType::BadSvc:
 | |
|             this->result = ResultCreportBadSvc;
 | |
|             break;
 | |
|         case DebugExceptionType::SystemMemoryError:
 | |
|             this->result = ResultCreportSystemMemoryError;
 | |
|             d.info.exception.specific.raw = 0;
 | |
|             break;
 | |
|         case DebugExceptionType::DebuggerAttached:
 | |
|         case DebugExceptionType::BreakPoint:
 | |
|         case DebugExceptionType::DebuggerBreak:
 | |
|         default:
 | |
|             return;
 | |
|     }
 | |
|     this->exception_info = d.info.exception;
 | |
|     this->crashed_thread_id = d.thread_id;
 | |
| }
 | |
| 
 | |
| void CrashReport::HandleAttachThread(DebugEventInfo &d) {
 | |
|     this->thread_tls_map[d.info.attach_thread.thread_id] = d.info.attach_thread.tls_address;
 | |
| }
 | |
| 
 | |
| void CrashReport::ProcessDyingMessage() {
 | |
|     /* Dying message is only stored starting in 5.0.0. */
 | |
|     if ((GetRuntimeFirmwareVersion() < FirmwareVersion_500)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Validate the message address/size. */
 | |
|     if (this->dying_message_address == 0 || this->dying_message_address & 0xFFF) {
 | |
|         return;
 | |
|     }
 | |
|     if (this->dying_message_size > sizeof(this->dying_message)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Validate that the report isn't garbage. */
 | |
|     if (!IsOpen() || !WasSuccessful()) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (!IsAddressReadable(this->dying_message_address, this->dying_message_size)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     svcReadDebugProcessMemory(this->dying_message, this->debug_handle, this->dying_message_address, this->dying_message_size);
 | |
| }
 | |
| 
 | |
| bool CrashReport::IsAddressReadable(u64 address, u64 size, MemoryInfo *o_mi) {
 | |
|     MemoryInfo mi;
 | |
|     u32 pi;
 | |
| 
 | |
|     if (o_mi == NULL) {
 | |
|         o_mi = &mi;
 | |
|     }
 | |
| 
 | |
|     if (R_FAILED(svcQueryDebugProcessMemory(o_mi, &pi, this->debug_handle, address))) {
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /* Must be read or read-write */
 | |
|     if ((o_mi->perm | Perm_W) != Perm_Rw) {
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /* Must have space for both userdata address and userdata size. */
 | |
|     if (address < o_mi->addr || o_mi->addr + o_mi->size < address + size) {
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| bool CrashReport::GetCurrentTime(u64 *out) {
 | |
|     *out = 0;
 | |
| 
 | |
|     /* Verify that pcv isn't dead. */
 | |
|     {
 | |
|         bool has_time_service;
 | |
|         DoWithSmSession([&]() {
 | |
|             Handle dummy;
 | |
|             if (R_SUCCEEDED(smRegisterService(&dummy, "time:s", false, 0x20))) {
 | |
|                 svcCloseHandle(dummy);
 | |
|                 has_time_service = false;
 | |
|             } else {
 | |
|                 has_time_service = true;
 | |
|             }
 | |
|         });
 | |
|         if (!has_time_service) {
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* Try to get the current time. */
 | |
|     bool success = true;
 | |
|     DoWithSmSession([&]() {
 | |
|         success &= R_SUCCEEDED(timeInitialize());
 | |
|     });
 | |
|     if (success) {
 | |
|         success &= R_SUCCEEDED(timeGetCurrentTime(TimeType_LocalSystemClock, out));
 | |
|         timeExit();
 | |
|     }
 | |
|     return success;
 | |
| }
 | |
| 
 | |
| void CrashReport::EnsureReportDirectories() {
 | |
|     char path[FS_MAX_PATH];
 | |
|     strcpy(path, "sdmc:/atmosphere");
 | |
|     mkdir(path, S_IRWXU);
 | |
|     strcat(path, "/crash_reports");
 | |
|     mkdir(path, S_IRWXU);
 | |
|     strcat(path, "/dumps");
 | |
|     mkdir(path, S_IRWXU);
 | |
| }
 | |
| 
 | |
| void CrashReport::SaveReport() {
 | |
|     /* Save the report to the SD card. */
 | |
|     char report_path[FS_MAX_PATH];
 | |
| 
 | |
|     /* Ensure path exists. */
 | |
|     EnsureReportDirectories();
 | |
| 
 | |
|     /* Get a timestamp. */
 | |
|     u64 timestamp;
 | |
|     if (!GetCurrentTime(×tamp)) {
 | |
|         timestamp = svcGetSystemTick();
 | |
|     }
 | |
| 
 | |
|     /* Open report file. */
 | |
|     snprintf(report_path, sizeof(report_path) - 1, "sdmc:/atmosphere/crash_reports/%011lu_%016lx.log", timestamp, process_info.title_id);
 | |
|     FILE *f_report = fopen(report_path, "w");
 | |
|     if (f_report == NULL) {
 | |
|         return;
 | |
|     }
 | |
|     this->SaveToFile(f_report);
 | |
|     fclose(f_report);
 | |
| 
 | |
|     /* Dump threads. */
 | |
|     snprintf(report_path, sizeof(report_path) - 1, "sdmc:/atmosphere/crash_reports/dumps/%011lu_%016lx_thread_info.bin", timestamp, process_info.title_id);
 | |
|     f_report = fopen(report_path, "wb");
 | |
|     this->thread_list.DumpBinary(f_report, this->crashed_thread_info.GetId());
 | |
|     fclose(f_report);
 | |
| }
 | |
| 
 | |
| void CrashReport::SaveToFile(FILE *f_report) {
 | |
|     char buf[0x10] = {0};
 | |
|     fprintf(f_report, "Atmosphère Crash Report (v1.3):\n");
 | |
|     fprintf(f_report, "Result:                          0x%X (2%03d-%04d)\n\n", this->result, R_MODULE(this->result), R_DESCRIPTION(this->result));
 | |
| 
 | |
|     /* Process Info. */
 | |
|     memcpy(buf, this->process_info.name, sizeof(this->process_info.name));
 | |
|     fprintf(f_report, "Process Info:\n");
 | |
|     fprintf(f_report, "    Process Name:                %s\n", buf);
 | |
|     fprintf(f_report, "    Title ID:                    %016lx\n", this->process_info.title_id);
 | |
|     fprintf(f_report, "    Process ID:                  %016lx\n", this->process_info.process_id);
 | |
|     fprintf(f_report, "    Process Flags:               %08x\n", this->process_info.flags);
 | |
|     if ((GetRuntimeFirmwareVersion() >= FirmwareVersion_500)) {
 | |
|         fprintf(f_report, "    User Exception Address:      %s\n", this->code_list.GetFormattedAddressString(this->process_info.user_exception_context_address));
 | |
|     }
 | |
| 
 | |
|     fprintf(f_report, "Exception Info:\n");
 | |
|     fprintf(f_report, "    Type:                        %s\n", GetDebugExceptionTypeStr(this->exception_info.type));
 | |
|     fprintf(f_report, "    Address:                     %s\n", this->code_list.GetFormattedAddressString(this->exception_info.address));
 | |
|     switch (this->exception_info.type) {
 | |
|         case DebugExceptionType::UndefinedInstruction:
 | |
|             fprintf(f_report, "    Opcode:                      %08x\n", this->exception_info.specific.undefined_instruction.insn);
 | |
|             break;
 | |
|         case DebugExceptionType::DataAbort:
 | |
|         case DebugExceptionType::AlignmentFault:
 | |
|             if (this->exception_info.specific.raw != this->exception_info.address) {
 | |
|                 fprintf(f_report, "    Fault Address:               %s\n", this->code_list.GetFormattedAddressString(this->exception_info.specific.raw));
 | |
|             }
 | |
|             break;
 | |
|         case DebugExceptionType::BadSvc:
 | |
|             fprintf(f_report, "    Svc Id:                      0x%02x\n", this->exception_info.specific.bad_svc.id);
 | |
|             break;
 | |
|         case DebugExceptionType::UserBreak:
 | |
|             fprintf(f_report, "    Break Reason:                0x%lx\n", this->exception_info.specific.user_break.break_reason);
 | |
|             fprintf(f_report, "    Break Address:               %s\n", this->code_list.GetFormattedAddressString(this->exception_info.specific.user_break.address));
 | |
|             fprintf(f_report, "    Break Size:                  0x%lx\n", this->exception_info.specific.user_break.size);
 | |
|             break;
 | |
|         default:
 | |
|             break;
 | |
|     }
 | |
| 
 | |
|     fprintf(f_report, "Crashed Thread Info:\n");
 | |
|     this->crashed_thread_info.SaveToFile(f_report);
 | |
| 
 | |
|     if ((GetRuntimeFirmwareVersion() >= FirmwareVersion_500)) {
 | |
|         if (this->dying_message_size) {
 | |
|             fprintf(f_report, "Dying Message Info:\n");
 | |
|             fprintf(f_report, "    Address:                     0x%s\n", this->code_list.GetFormattedAddressString(this->dying_message_address));
 | |
|             fprintf(f_report, "    Size:                        0x%016lx\n", this->dying_message_size);
 | |
|             CrashReport::Memdump(f_report, "    Dying Message:              ", this->dying_message, this->dying_message_size);
 | |
|         }
 | |
|     }
 | |
|     fprintf(f_report, "Code Region Info:\n");
 | |
|     this->code_list.SaveToFile(f_report);
 | |
|     fprintf(f_report, "\nThread Report:\n");
 | |
|     this->thread_list.SaveToFile(f_report);
 | |
| }
 | |
| 
 | |
| /* Lifted from hactool. */
 | |
| void CrashReport::Memdump(FILE *f, const char *prefix, const void *data, size_t size) {
 | |
|     uint8_t *p = (uint8_t *)data;
 | |
| 
 | |
|     unsigned int prefix_len = strlen(prefix);
 | |
|     size_t offset = 0;
 | |
|     int first = 1;
 | |
| 
 | |
|     while (size) {
 | |
|         unsigned int max = 32;
 | |
| 
 | |
|         if (max > size) {
 | |
|             max = size;
 | |
|         }
 | |
| 
 | |
|         if (first) {
 | |
|             fprintf(f, "%s", prefix);
 | |
|             first = 0;
 | |
|         } else {
 | |
|             fprintf(f, "%*s", prefix_len, "");
 | |
|         }
 | |
| 
 | |
|         for (unsigned int i = 0; i < max; i++) {
 | |
|             fprintf(f, "%02X", p[offset++]);
 | |
|         }
 | |
| 
 | |
|         fprintf(f, "\n");
 | |
| 
 | |
|         size -= max;
 | |
|     }
 | |
| }
 |