/* * Copyright (c) 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 . */ #include #include "diag_dump_stack_trace.hpp" namespace ams::diag::impl { #if defined(AMS_BUILD_FOR_DEBUGGING) || defined(AMS_BUILD_FOR_DEBUGGING) namespace { constexpr const char *ToString(AbortReason reason) { switch (reason) { case AbortReason_Audit: return "Auditing Assertion Failure"; case AbortReason_Assert: return "Assertion Failure"; case AbortReason_UnexpectedDefault: return "Unexpected Default"; case AbortReason_Abort: default: return "Abort"; } } void DefaultPrinter(const AbortInfo &info) { /* Get the thread name. */ const char *thread_name; if (auto *cur_thread = os::GetCurrentThread(); cur_thread != nullptr) { thread_name = os::GetThreadNamePointer(cur_thread); } else { thread_name = "unknown"; } #if defined(ATMOSPHERE_OS_HORIZON) { u64 process_id = 0; u64 thread_id = 0; svc::GetProcessId(std::addressof(process_id), svc::PseudoHandle::CurrentProcess); svc::GetThreadId(std::addressof(thread_id), svc::PseudoHandle::CurrentThread); AMS_SDK_LOG("%s: '%s' in %s, process=0x%02" PRIX64 ", thread=%" PRIu64 " (%s)\n%s:%d\n", ToString(info.reason), info.expr, info.func, process_id, thread_id, thread_name, info.file, info.line); } #elif defined(ATMOSPHERE_OS_WINDOWS) { DWORD process_id = ::GetCurrentProcessId(); DWORD thread_id = ::GetCurrentThreadId(); AMS_SDK_LOG("%s: '%s' in %s, process=0x%" PRIX64 ", thread=%" PRIu64 " (%s)\n%s:%d\n", ToString(info.reason), info.expr, info.func, static_cast(process_id), static_cast(thread_id), thread_name, info.file, info.line); } #else { AMS_SDK_LOG("%s: '%s' in %s, thread=%s\n%s:%d\n", ToString(info.reason), info.expr, info.func, thread_name, info.file, info.line); } #endif AMS_SDK_VLOG(info.message->fmt, *(info.message->vl)); AMS_SDK_LOG("\n"); TentativeDumpStackTrace(); } } #endif void DefaultAbortObserver(const AbortInfo &info) { #if defined(AMS_BUILD_FOR_DEBUGGING) || defined(AMS_BUILD_FOR_AUDITING) DefaultPrinter(info); #else AMS_UNUSED(info); #endif } }