| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2018-12-29 12:55:19 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 12:54:24 -04:00
										 |  |  | #ifndef _MSC_VER
 | 
					
						
							|  |  |  | #include <cxxabi.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  | #include <map>
 | 
					
						
							|  |  |  | #include <optional>
 | 
					
						
							|  |  |  | #include "common/bit_field.h"
 | 
					
						
							| 
									
										
										
										
											2018-12-29 12:55:19 +11:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2018-12-30 20:46:27 -05:00
										 |  |  | #include "core/arm/arm_interface.h"
 | 
					
						
							| 
									
										
										
										
											2022-04-08 23:53:42 -04:00
										 |  |  | #include "core/arm/symbols.h"
 | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2022-05-30 19:35:01 -04:00
										 |  |  | #include "core/debugger/debugger.h"
 | 
					
						
							| 
									
										
										
										
											2022-04-08 23:53:42 -04:00
										 |  |  | #include "core/hle/kernel/k_process.h"
 | 
					
						
							| 
									
										
										
										
											2022-05-31 14:37:37 -04:00
										 |  |  | #include "core/hle/kernel/svc.h"
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  | #include "core/loader/loader.h"
 | 
					
						
							| 
									
										
										
										
											2018-12-29 12:55:19 +11:00
										 |  |  | #include "core/memory.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-20 20:17:48 -04:00
										 |  |  | #include "core/arm/dynarmic/arm_dynarmic_32.h"
 | 
					
						
							|  |  |  | #include "core/arm/dynarmic/arm_dynarmic_64.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-29 12:55:19 +11:00
										 |  |  | namespace Core { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | constexpr u64 SEGMENT_BASE = 0x7100000000ull; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 14:05:47 -04:00
										 |  |  | std::vector<ARM_Interface::BacktraceEntry> ARM_Interface::GetBacktraceFromContext( | 
					
						
							| 
									
										
										
										
											2022-04-20 20:17:48 -04:00
										 |  |  |     Core::System& system, const ARM_Interface::ThreadContext32& ctx) { | 
					
						
							|  |  |  |     return ARM_Dynarmic_32::GetBacktraceFromContext(system, ctx); | 
					
						
							| 
									
										
										
										
											2020-03-20 14:05:47 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-20 20:17:48 -04:00
										 |  |  | std::vector<ARM_Interface::BacktraceEntry> ARM_Interface::GetBacktraceFromContext( | 
					
						
							|  |  |  |     Core::System& system, const ARM_Interface::ThreadContext64& ctx) { | 
					
						
							|  |  |  |     return ARM_Dynarmic_64::GetBacktraceFromContext(system, ctx); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-20 20:17:48 -04:00
										 |  |  | void ARM_Interface::SymbolicateBacktrace(Core::System& system, std::vector<BacktraceEntry>& out) { | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     std::map<VAddr, std::string> modules; | 
					
						
							| 
									
										
										
										
											2019-11-26 14:10:49 -05:00
										 |  |  |     auto& loader{system.GetAppLoader()}; | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     if (loader.ReadNSOModules(modules) != Loader::ResultStatus::Success) { | 
					
						
							| 
									
										
										
										
											2022-04-20 20:17:48 -04:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-08 23:53:42 -04:00
										 |  |  |     std::map<std::string, Symbols::Symbols> symbols; | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  |     for (const auto& module : modules) { | 
					
						
							| 
									
										
										
										
											2022-04-08 23:53:42 -04:00
										 |  |  |         symbols.insert_or_assign(module.second, | 
					
						
							|  |  |  |                                  Symbols::GetSymbols(module.first, system.Memory(), | 
					
						
							|  |  |  |                                                      system.CurrentProcess()->Is64BitProcess())); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (auto& entry : out) { | 
					
						
							|  |  |  |         VAddr base = 0; | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |         for (auto iter = modules.rbegin(); iter != modules.rend(); ++iter) { | 
					
						
							|  |  |  |             const auto& module{*iter}; | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  |             if (entry.original_address >= module.first) { | 
					
						
							|  |  |  |                 entry.module = module.second; | 
					
						
							|  |  |  |                 base = module.first; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         entry.offset = entry.original_address - base; | 
					
						
							|  |  |  |         entry.address = SEGMENT_BASE + entry.offset; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-20 20:17:48 -04:00
										 |  |  |         if (entry.module.empty()) { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  |             entry.module = "unknown"; | 
					
						
							| 
									
										
										
										
											2022-04-20 20:17:48 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         const auto symbol_set = symbols.find(entry.module); | 
					
						
							|  |  |  |         if (symbol_set != symbols.end()) { | 
					
						
							| 
									
										
										
										
											2022-04-08 23:53:42 -04:00
										 |  |  |             const auto symbol = Symbols::GetSymbolName(symbol_set->second, entry.offset); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  |             if (symbol.has_value()) { | 
					
						
							| 
									
										
										
										
											2022-06-25 12:54:24 -04:00
										 |  |  | #ifdef _MSC_VER
 | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  |                 // TODO(DarkLordZach): Add demangling of symbol names.
 | 
					
						
							|  |  |  |                 entry.name = *symbol; | 
					
						
							| 
									
										
										
										
											2022-06-25 12:54:24 -04:00
										 |  |  | #else
 | 
					
						
							|  |  |  |                 int status{-1}; | 
					
						
							|  |  |  |                 char* demangled{abi::__cxa_demangle(symbol->c_str(), nullptr, nullptr, &status)}; | 
					
						
							|  |  |  |                 if (status == 0 && demangled != nullptr) { | 
					
						
							|  |  |  |                     entry.name = demangled; | 
					
						
							|  |  |  |                     std::free(demangled); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     entry.name = *symbol; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-12-29 12:55:19 +11:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | void ARM_Interface::LogBacktrace() const { | 
					
						
							| 
									
										
										
										
											2022-04-20 20:17:48 -04:00
										 |  |  |     const VAddr sp = GetSP(); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  |     const VAddr pc = GetPC(); | 
					
						
							|  |  |  |     LOG_ERROR(Core_ARM, "Backtrace, sp={:016X}, pc={:016X}", sp, pc); | 
					
						
							|  |  |  |     LOG_ERROR(Core_ARM, "{:20}{:20}{:20}{:20}{}", "Module Name", "Address", "Original Address", | 
					
						
							|  |  |  |               "Offset", "Symbol"); | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     LOG_ERROR(Core_ARM, ""); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:43:26 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const auto backtrace = GetBacktrace(); | 
					
						
							|  |  |  |     for (const auto& entry : backtrace) { | 
					
						
							|  |  |  |         LOG_ERROR(Core_ARM, "{:20}{:016X}    {:016X}    {:016X}    {}", entry.module, entry.address, | 
					
						
							|  |  |  |                   entry.original_address, entry.offset, entry.name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-31 14:37:37 -04:00
										 |  |  | void ARM_Interface::Run() { | 
					
						
							|  |  |  |     using Kernel::StepState; | 
					
						
							|  |  |  |     using Kernel::SuspendType; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (true) { | 
					
						
							| 
									
										
										
										
											2022-06-16 10:35:52 -04:00
										 |  |  |         Kernel::KThread* current_thread{Kernel::GetCurrentThreadPointer(system.Kernel())}; | 
					
						
							| 
									
										
										
										
											2022-05-31 14:37:37 -04:00
										 |  |  |         Dynarmic::HaltReason hr{}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Notify the debugger and go to sleep if a step was performed
 | 
					
						
							|  |  |  |         // and this thread has been scheduled again.
 | 
					
						
							|  |  |  |         if (current_thread->GetStepState() == StepState::StepPerformed) { | 
					
						
							|  |  |  |             system.GetDebugger().NotifyThreadStopped(current_thread); | 
					
						
							|  |  |  |             current_thread->RequestSuspend(SuspendType::Debug); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Otherwise, run the thread.
 | 
					
						
							| 
									
										
										
										
											2022-06-14 18:19:04 -04:00
										 |  |  |         system.EnterDynarmicProfile(); | 
					
						
							| 
									
										
										
										
											2022-05-31 14:37:37 -04:00
										 |  |  |         if (current_thread->GetStepState() == StepState::StepPending) { | 
					
						
							|  |  |  |             hr = StepJit(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (Has(hr, step_thread)) { | 
					
						
							|  |  |  |                 current_thread->SetStepState(StepState::StepPerformed); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             hr = RunJit(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-06-14 18:19:04 -04:00
										 |  |  |         system.ExitDynarmicProfile(); | 
					
						
							| 
									
										
										
										
											2022-05-31 14:37:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-05 18:19:30 -07:00
										 |  |  |         // If the thread is scheduled for termination, exit the thread.
 | 
					
						
							|  |  |  |         if (current_thread->HasDpc()) { | 
					
						
							|  |  |  |             if (current_thread->IsTerminationRequested()) { | 
					
						
							|  |  |  |                 current_thread->Exit(); | 
					
						
							|  |  |  |                 UNREACHABLE(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-20 20:39:16 -04:00
										 |  |  |         // Notify the debugger and go to sleep if a breakpoint was hit,
 | 
					
						
							|  |  |  |         // or if the thread is unable to continue for any reason.
 | 
					
						
							|  |  |  |         if (Has(hr, breakpoint) || Has(hr, no_execute)) { | 
					
						
							| 
									
										
										
										
											2022-12-01 22:48:43 -05:00
										 |  |  |             if (!Has(hr, no_execute)) { | 
					
						
							|  |  |  |                 RewindBreakpointInstruction(); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-06-20 20:39:16 -04:00
										 |  |  |             if (system.DebuggerEnabled()) { | 
					
						
							|  |  |  |                 system.GetDebugger().NotifyThreadStopped(current_thread); | 
					
						
							| 
									
										
										
										
											2022-12-01 22:48:43 -05:00
										 |  |  |             } else { | 
					
						
							|  |  |  |                 LogBacktrace(); | 
					
						
							| 
									
										
										
										
											2022-06-20 20:39:16 -04:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-12-01 22:48:43 -05:00
										 |  |  |             current_thread->RequestSuspend(SuspendType::Debug); | 
					
						
							| 
									
										
										
										
											2022-06-06 12:56:01 -04:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-06-20 20:39:16 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Notify the debugger and go to sleep if a watchpoint was hit.
 | 
					
						
							| 
									
										
										
										
											2022-06-06 12:56:01 -04:00
										 |  |  |         if (Has(hr, watchpoint)) { | 
					
						
							| 
									
										
										
										
											2022-06-20 20:39:16 -04:00
										 |  |  |             if (system.DebuggerEnabled()) { | 
					
						
							|  |  |  |                 system.GetDebugger().NotifyThreadWatchpoint(current_thread, *HaltedWatchpoint()); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-06-06 12:56:01 -04:00
										 |  |  |             current_thread->RequestSuspend(SuspendType::Debug); | 
					
						
							| 
									
										
										
										
											2022-05-31 14:37:37 -04:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-26 18:52:16 -04:00
										 |  |  |         // Handle syscalls and scheduling (this may change the current thread/core)
 | 
					
						
							| 
									
										
										
										
											2022-05-31 14:37:37 -04:00
										 |  |  |         if (Has(hr, svc_call)) { | 
					
						
							|  |  |  |             Kernel::Svc::Call(system, GetSvcNumber()); | 
					
						
							| 
									
										
										
										
											2022-06-26 18:52:16 -04:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2022-05-31 14:37:37 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (Has(hr, break_loop) || !uses_wall_clock) { | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-05-30 19:35:01 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-06 12:56:01 -04:00
										 |  |  | void ARM_Interface::LoadWatchpointArray(const WatchpointArray& wp) { | 
					
						
							|  |  |  |     watchpoints = ℘ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const Kernel::DebugWatchpoint* ARM_Interface::MatchingWatchpoint( | 
					
						
							|  |  |  |     VAddr addr, u64 size, Kernel::DebugWatchpointType access_type) const { | 
					
						
							|  |  |  |     if (!watchpoints) { | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const VAddr start_address{addr}; | 
					
						
							|  |  |  |     const VAddr end_address{addr + size}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (size_t i = 0; i < Core::Hardware::NUM_WATCHPOINTS; i++) { | 
					
						
							|  |  |  |         const auto& watch{(*watchpoints)[i]}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (end_address <= watch.start_address) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (start_address >= watch.end_address) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if ((access_type & watch.type) == Kernel::DebugWatchpointType::None) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return &watch; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-30 20:41:30 -05:00
										 |  |  | } // namespace Core
 |