| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2019 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-05 17:19:43 -04:00
										 |  |  | #include <ctime>
 | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | #include <fstream>
 | 
					
						
							| 
									
										
										
										
											2020-05-08 15:09:29 -06:00
										 |  |  | #include <iomanip>
 | 
					
						
							| 
									
										
										
										
											2019-07-05 17:19:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-29 21:34:51 +02:00
										 |  |  | #include <fmt/chrono.h>
 | 
					
						
							| 
									
										
										
										
											2019-08-29 22:44:49 +02:00
										 |  |  | #include <fmt/format.h>
 | 
					
						
							| 
									
										
										
										
											2020-05-08 15:09:29 -06:00
										 |  |  | #include <nlohmann/json.hpp>
 | 
					
						
							| 
									
										
										
										
											2019-07-05 17:19:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  | #include "common/fs/file.h"
 | 
					
						
							|  |  |  | #include "common/fs/fs.h"
 | 
					
						
							|  |  |  | #include "common/fs/path_util.h"
 | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | #include "common/hex_util.h"
 | 
					
						
							|  |  |  | #include "common/scm_rev.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-14 16:07:40 -07:00
										 |  |  | #include "common/settings.h"
 | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | #include "core/arm/arm_interface.h"
 | 
					
						
							|  |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2021-02-12 17:58:31 -08:00
										 |  |  | #include "core/hle/kernel/k_page_table.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-23 22:04:28 -07:00
										 |  |  | #include "core/hle/kernel/k_process.h"
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  | #include "core/hle/result.h"
 | 
					
						
							| 
									
										
										
										
											2023-02-19 14:42:12 -05:00
										 |  |  | #include "core/hle/service/hle_ipc.h"
 | 
					
						
							| 
									
										
										
										
											2020-04-08 22:22:16 -04:00
										 |  |  | #include "core/memory.h"
 | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | #include "core/reporter.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  | std::filesystem::path GetPath(std::string_view type, u64 title_id, std::string_view timestamp) { | 
					
						
							|  |  |  |     return Common::FS::GetYuzuPath(Common::FS::YuzuPath::LogDir) / type / | 
					
						
							|  |  |  |            fmt::format("{:016X}_{}.json", title_id, timestamp); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string GetTimestamp() { | 
					
						
							|  |  |  |     const auto time = std::time(nullptr); | 
					
						
							|  |  |  |     return fmt::format("{:%FT%H-%M-%S}", *std::localtime(&time)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using namespace nlohmann; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-05 21:42:22 -05:00
										 |  |  | void SaveToFile(const json& json, const std::filesystem::path& filename) { | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  |     if (!Common::FS::CreateParentDirs(filename)) { | 
					
						
							|  |  |  |         LOG_ERROR(Core, "Failed to create path for '{}' to save report!", | 
					
						
							|  |  |  |                   Common::FS::PathToUTF8String(filename)); | 
					
						
							| 
									
										
										
										
											2019-07-05 17:02:29 -04:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  |     std::ofstream file; | 
					
						
							|  |  |  |     Common::FS::OpenFileStream(file, filename, std::ios_base::out | std::ios_base::trunc); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     file << std::setw(4) << json << std::endl; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | json GetYuzuVersionData() { | 
					
						
							|  |  |  |     return { | 
					
						
							|  |  |  |         {"scm_rev", std::string(Common::g_scm_rev)}, | 
					
						
							|  |  |  |         {"scm_branch", std::string(Common::g_scm_branch)}, | 
					
						
							|  |  |  |         {"scm_desc", std::string(Common::g_scm_desc)}, | 
					
						
							|  |  |  |         {"build_name", std::string(Common::g_build_name)}, | 
					
						
							|  |  |  |         {"build_date", std::string(Common::g_build_date)}, | 
					
						
							|  |  |  |         {"build_fullname", std::string(Common::g_build_fullname)}, | 
					
						
							|  |  |  |         {"build_version", std::string(Common::g_build_version)}, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | json GetReportCommonData(u64 title_id, Result result, const std::string& timestamp, | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |                          std::optional<u128> user_id = {}) { | 
					
						
							|  |  |  |     auto out = json{ | 
					
						
							|  |  |  |         {"title_id", fmt::format("{:016X}", title_id)}, | 
					
						
							|  |  |  |         {"result_raw", fmt::format("{:08X}", result.raw)}, | 
					
						
							|  |  |  |         {"result_module", fmt::format("{:08X}", static_cast<u32>(result.module.Value()))}, | 
					
						
							|  |  |  |         {"result_description", fmt::format("{:08X}", result.description.Value())}, | 
					
						
							|  |  |  |         {"timestamp", timestamp}, | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (user_id.has_value()) { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |         out["user_id"] = fmt::format("{:016X}{:016X}", (*user_id)[1], (*user_id)[0]); | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     return out; | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | json GetProcessorStateData(const std::string& architecture, u64 entry_point, u64 sp, u64 pc, | 
					
						
							| 
									
										
										
										
											2022-12-05 21:42:22 -05:00
										 |  |  |                            u64 pstate, const std::array<u64, 31>& registers, | 
					
						
							|  |  |  |                            const std::optional<std::array<u64, 32>>& backtrace = {}) { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     auto out = json{ | 
					
						
							|  |  |  |         {"entry_point", fmt::format("{:016X}", entry_point)}, | 
					
						
							|  |  |  |         {"sp", fmt::format("{:016X}", sp)}, | 
					
						
							|  |  |  |         {"pc", fmt::format("{:016X}", pc)}, | 
					
						
							|  |  |  |         {"pstate", fmt::format("{:016X}", pstate)}, | 
					
						
							|  |  |  |         {"architecture", architecture}, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto registers_out = json::object(); | 
					
						
							|  |  |  |     for (std::size_t i = 0; i < registers.size(); ++i) { | 
					
						
							|  |  |  |         registers_out[fmt::format("X{:02d}", i)] = fmt::format("{:016X}", registers[i]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     out["registers"] = std::move(registers_out); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (backtrace.has_value()) { | 
					
						
							|  |  |  |         auto backtrace_out = json::array(); | 
					
						
							|  |  |  |         for (const auto& entry : *backtrace) { | 
					
						
							|  |  |  |             backtrace_out.push_back(fmt::format("{:016X}", entry)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         out["backtrace"] = std::move(backtrace_out); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     return out; | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  | json GetProcessorStateDataAuto(Core::System& system) { | 
					
						
							| 
									
										
										
										
											2023-02-13 11:21:43 -05:00
										 |  |  |     const auto* process{system.ApplicationProcess()}; | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     auto& arm{system.CurrentArmInterface()}; | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-01 23:46:10 -05:00
										 |  |  |     Core::ARM_Interface::ThreadContext64 context{}; | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     arm.SaveContext(context); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return GetProcessorStateData(process->Is64BitProcess() ? "AArch64" : "AArch32", | 
					
						
							| 
									
										
										
										
											2023-03-17 21:26:04 -04:00
										 |  |  |                                  GetInteger(process->PageTable().GetCodeRegionStart()), context.sp, | 
					
						
							|  |  |  |                                  context.pc, context.pstate, context.cpu_registers); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  | json GetBacktraceData(Core::System& system) { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     auto out = json::array(); | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     const auto& backtrace{system.CurrentArmInterface().GetBacktrace()}; | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     for (const auto& entry : backtrace) { | 
					
						
							|  |  |  |         out.push_back({ | 
					
						
							|  |  |  |             {"module", entry.module}, | 
					
						
							|  |  |  |             {"address", fmt::format("{:016X}", entry.address)}, | 
					
						
							|  |  |  |             {"original_address", fmt::format("{:016X}", entry.original_address)}, | 
					
						
							|  |  |  |             {"offset", fmt::format("{:016X}", entry.offset)}, | 
					
						
							|  |  |  |             {"symbol_name", entry.name}, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     return out; | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  | json GetFullDataAuto(const std::string& timestamp, u64 title_id, Core::System& system) { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     json out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     out["yuzu_version"] = GetYuzuVersionData(); | 
					
						
							| 
									
										
										
										
											2021-05-21 01:05:04 -04:00
										 |  |  |     out["report_common"] = GetReportCommonData(title_id, ResultSuccess, timestamp); | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     out["processor_state"] = GetProcessorStateDataAuto(system); | 
					
						
							|  |  |  |     out["backtrace"] = GetBacktraceData(system); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     return out; | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <bool read_value, typename DescriptorType> | 
					
						
							| 
									
										
										
										
											2020-04-08 22:22:16 -04:00
										 |  |  | json GetHLEBufferDescriptorData(const std::vector<DescriptorType>& buffer, | 
					
						
							|  |  |  |                                 Core::Memory::Memory& memory) { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     auto buffer_out = json::array(); | 
					
						
							|  |  |  |     for (const auto& desc : buffer) { | 
					
						
							|  |  |  |         auto entry = json{ | 
					
						
							|  |  |  |             {"address", fmt::format("{:016X}", desc.Address())}, | 
					
						
							|  |  |  |             {"size", fmt::format("{:016X}", desc.Size())}, | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if constexpr (read_value) { | 
					
						
							|  |  |  |             std::vector<u8> data(desc.Size()); | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  |             memory.ReadBlock(desc.Address(), data.data(), desc.Size()); | 
					
						
							| 
									
										
										
										
											2019-06-22 14:39:52 +01:00
										 |  |  |             entry["data"] = Common::HexToString(data); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         buffer_out.push_back(std::move(entry)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     return buffer_out; | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-19 14:42:12 -05:00
										 |  |  | json GetHLERequestContextData(Service::HLERequestContext& ctx, Core::Memory::Memory& memory) { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     json out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto cmd_buf = json::array(); | 
					
						
							|  |  |  |     for (std::size_t i = 0; i < IPC::COMMAND_BUFFER_LENGTH; ++i) { | 
					
						
							|  |  |  |         cmd_buf.push_back(fmt::format("{:08X}", ctx.CommandBuffer()[i])); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     out["command_buffer"] = std::move(cmd_buf); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 14:10:49 -05:00
										 |  |  |     out["buffer_descriptor_a"] = GetHLEBufferDescriptorData<true>(ctx.BufferDescriptorA(), memory); | 
					
						
							|  |  |  |     out["buffer_descriptor_b"] = GetHLEBufferDescriptorData<false>(ctx.BufferDescriptorB(), memory); | 
					
						
							|  |  |  |     out["buffer_descriptor_c"] = GetHLEBufferDescriptorData<false>(ctx.BufferDescriptorC(), memory); | 
					
						
							|  |  |  |     out["buffer_descriptor_x"] = GetHLEBufferDescriptorData<true>(ctx.BufferDescriptorX(), memory); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-05 17:08:10 -04:00
										 |  |  |     return out; | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // Anonymous namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Core { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 04:30:54 -04:00
										 |  |  | Reporter::Reporter(System& system_) : system(system_) { | 
					
						
							|  |  |  |     ClearFSAccessLog(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | Reporter::~Reporter() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | void Reporter::SaveCrashReport(u64 title_id, Result result, u64 set_flags, u64 entry_point, u64 sp, | 
					
						
							|  |  |  |                                u64 pc, u64 pstate, u64 afsr0, u64 afsr1, u64 esr, u64 far, | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |                                const std::array<u64, 31>& registers, | 
					
						
							|  |  |  |                                const std::array<u64, 32>& backtrace, u32 backtrace_size, | 
					
						
							|  |  |  |                                const std::string& arch, u32 unk10) const { | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     if (!IsReportingEnabled()) { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     const auto timestamp = GetTimestamp(); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     json out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     out["yuzu_version"] = GetYuzuVersionData(); | 
					
						
							|  |  |  |     out["report_common"] = GetReportCommonData(title_id, result, timestamp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto proc_out = GetProcessorStateData(arch, entry_point, sp, pc, pstate, registers, backtrace); | 
					
						
							|  |  |  |     proc_out["set_flags"] = fmt::format("{:016X}", set_flags); | 
					
						
							|  |  |  |     proc_out["afsr0"] = fmt::format("{:016X}", afsr0); | 
					
						
							|  |  |  |     proc_out["afsr1"] = fmt::format("{:016X}", afsr1); | 
					
						
							|  |  |  |     proc_out["esr"] = fmt::format("{:016X}", esr); | 
					
						
							|  |  |  |     proc_out["far"] = fmt::format("{:016X}", far); | 
					
						
							|  |  |  |     proc_out["backtrace_size"] = fmt::format("{:08X}", backtrace_size); | 
					
						
							|  |  |  |     proc_out["unknown_10"] = fmt::format("{:08X}", unk10); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     out["processor_state"] = std::move(proc_out); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-05 21:42:22 -05:00
										 |  |  |     SaveToFile(out, GetPath("crash_report", title_id, timestamp)); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Reporter::SaveSvcBreakReport(u32 type, bool signal_debugger, u64 info1, u64 info2, | 
					
						
							| 
									
										
										
										
											2022-12-05 21:42:22 -05:00
										 |  |  |                                   const std::optional<std::vector<u8>>& resolved_buffer) const { | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     if (!IsReportingEnabled()) { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     const auto timestamp = GetTimestamp(); | 
					
						
							| 
									
										
										
										
											2023-02-13 11:21:43 -05:00
										 |  |  |     const auto title_id = system.GetApplicationProcessProgramID(); | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     auto out = GetFullDataAuto(timestamp, title_id, system); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     auto break_out = json{ | 
					
						
							|  |  |  |         {"type", fmt::format("{:08X}", type)}, | 
					
						
							|  |  |  |         {"signal_debugger", fmt::format("{}", signal_debugger)}, | 
					
						
							|  |  |  |         {"info1", fmt::format("{:016X}", info1)}, | 
					
						
							|  |  |  |         {"info2", fmt::format("{:016X}", info2)}, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (resolved_buffer.has_value()) { | 
					
						
							| 
									
										
										
										
											2019-06-22 14:39:52 +01:00
										 |  |  |         break_out["debug_buffer"] = Common::HexToString(*resolved_buffer); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     out["svc_break"] = std::move(break_out); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-05 21:42:22 -05:00
										 |  |  |     SaveToFile(out, GetPath("svc_break_report", title_id, timestamp)); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-19 14:42:12 -05:00
										 |  |  | void Reporter::SaveUnimplementedFunctionReport(Service::HLERequestContext& ctx, u32 command_id, | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |                                                const std::string& name, | 
					
						
							|  |  |  |                                                const std::string& service_name) const { | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     if (!IsReportingEnabled()) { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     const auto timestamp = GetTimestamp(); | 
					
						
							| 
									
										
										
										
											2023-02-13 11:21:43 -05:00
										 |  |  |     const auto title_id = system.GetApplicationProcessProgramID(); | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     auto out = GetFullDataAuto(timestamp, title_id, system); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-23 19:58:48 -04:00
										 |  |  |     auto function_out = GetHLERequestContextData(ctx, system.ApplicationMemory()); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     function_out["command_id"] = command_id; | 
					
						
							|  |  |  |     function_out["function_name"] = name; | 
					
						
							|  |  |  |     function_out["service_name"] = service_name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     out["function"] = std::move(function_out); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-05 21:42:22 -05:00
										 |  |  |     SaveToFile(out, GetPath("unimpl_func_report", title_id, timestamp)); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Reporter::SaveUnimplementedAppletReport( | 
					
						
							|  |  |  |     u32 applet_id, u32 common_args_version, u32 library_version, u32 theme_color, | 
					
						
							| 
									
										
										
										
											2022-12-05 21:42:22 -05:00
										 |  |  |     bool startup_sound, u64 system_tick, const std::vector<std::vector<u8>>& normal_channel, | 
					
						
							|  |  |  |     const std::vector<std::vector<u8>>& interactive_channel) const { | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     if (!IsReportingEnabled()) { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     const auto timestamp = GetTimestamp(); | 
					
						
							| 
									
										
										
										
											2023-02-13 11:21:43 -05:00
										 |  |  |     const auto title_id = system.GetApplicationProcessProgramID(); | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     auto out = GetFullDataAuto(timestamp, title_id, system); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     out["applet_common_args"] = { | 
					
						
							|  |  |  |         {"applet_id", fmt::format("{:02X}", applet_id)}, | 
					
						
							|  |  |  |         {"common_args_version", fmt::format("{:08X}", common_args_version)}, | 
					
						
							|  |  |  |         {"library_version", fmt::format("{:08X}", library_version)}, | 
					
						
							|  |  |  |         {"theme_color", fmt::format("{:08X}", theme_color)}, | 
					
						
							|  |  |  |         {"startup_sound", fmt::format("{}", startup_sound)}, | 
					
						
							|  |  |  |         {"system_tick", fmt::format("{:016X}", system_tick)}, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto normal_out = json::array(); | 
					
						
							|  |  |  |     for (const auto& data : normal_channel) { | 
					
						
							| 
									
										
										
										
											2019-06-22 14:39:52 +01:00
										 |  |  |         normal_out.push_back(Common::HexToString(data)); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto interactive_out = json::array(); | 
					
						
							|  |  |  |     for (const auto& data : interactive_channel) { | 
					
						
							| 
									
										
										
										
											2019-06-22 14:39:52 +01:00
										 |  |  |         interactive_out.push_back(Common::HexToString(data)); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     out["applet_normal_data"] = std::move(normal_out); | 
					
						
							|  |  |  |     out["applet_interactive_data"] = std::move(interactive_out); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-05 21:42:22 -05:00
										 |  |  |     SaveToFile(out, GetPath("unimpl_applet_report", title_id, timestamp)); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-05 21:42:22 -05:00
										 |  |  | void Reporter::SavePlayReport(PlayReportType type, u64 title_id, | 
					
						
							| 
									
										
										
										
											2023-02-03 00:08:45 -05:00
										 |  |  |                               const std::vector<std::span<const u8>>& data, | 
					
						
							| 
									
										
										
										
											2019-06-26 18:02:19 -04:00
										 |  |  |                               std::optional<u64> process_id, std::optional<u128> user_id) const { | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     if (!IsReportingEnabled()) { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     const auto timestamp = GetTimestamp(); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     json out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     out["yuzu_version"] = GetYuzuVersionData(); | 
					
						
							| 
									
										
										
										
											2021-05-21 01:05:04 -04:00
										 |  |  |     out["report_common"] = GetReportCommonData(title_id, ResultSuccess, timestamp, user_id); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     auto data_out = json::array(); | 
					
						
							|  |  |  |     for (const auto& d : data) { | 
					
						
							| 
									
										
										
										
											2019-06-22 14:39:52 +01:00
										 |  |  |         data_out.push_back(Common::HexToString(d)); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-26 18:02:19 -04:00
										 |  |  |     if (process_id.has_value()) { | 
					
						
							|  |  |  |         out["play_report_process_id"] = fmt::format("{:016X}", *process_id); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     out["play_report_type"] = fmt::format("{:02}", static_cast<u8>(type)); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     out["play_report_data"] = std::move(data_out); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-05 21:42:22 -05:00
										 |  |  |     SaveToFile(out, GetPath("play_report", title_id, timestamp)); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-25 22:44:19 -05:00
										 |  |  | void Reporter::SaveErrorReport(u64 title_id, Result result, | 
					
						
							| 
									
										
										
										
											2022-12-05 21:26:00 -05:00
										 |  |  |                                const std::optional<std::string>& custom_text_main, | 
					
						
							|  |  |  |                                const std::optional<std::string>& custom_text_detail) const { | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     if (!IsReportingEnabled()) { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     const auto timestamp = GetTimestamp(); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     json out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     out["yuzu_version"] = GetYuzuVersionData(); | 
					
						
							|  |  |  |     out["report_common"] = GetReportCommonData(title_id, result, timestamp); | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     out["processor_state"] = GetProcessorStateDataAuto(system); | 
					
						
							|  |  |  |     out["backtrace"] = GetBacktraceData(system); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     out["error_custom_text"] = { | 
					
						
							| 
									
										
										
										
											2022-12-05 21:26:00 -05:00
										 |  |  |         {"main", custom_text_main.value_or("")}, | 
					
						
							|  |  |  |         {"detail", custom_text_detail.value_or("")}, | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-05 21:42:22 -05:00
										 |  |  |     SaveToFile(out, GetPath("error_report", title_id, timestamp)); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 04:30:54 -04:00
										 |  |  | void Reporter::SaveFSAccessLog(std::string_view log_message) const { | 
					
						
							|  |  |  |     const auto access_log_path = | 
					
						
							|  |  |  |         Common::FS::GetYuzuPath(Common::FS::YuzuPath::SDMCDir) / "FsAccessLog.txt"; | 
					
						
							| 
									
										
										
										
											2019-06-28 21:02:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 04:30:54 -04:00
										 |  |  |     void(Common::FS::AppendStringToFile(access_log_path, Common::FS::FileType::TextFile, | 
					
						
							|  |  |  |                                         log_message)); | 
					
						
							| 
									
										
										
										
											2019-06-28 21:02:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | void Reporter::SaveUserReport() const { | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     if (!IsReportingEnabled()) { | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-07-05 17:05:24 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     const auto timestamp = GetTimestamp(); | 
					
						
							| 
									
										
										
										
											2023-02-13 11:21:43 -05:00
										 |  |  |     const auto title_id = system.GetApplicationProcessProgramID(); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 11:40:41 -04:00
										 |  |  |     SaveToFile(GetFullDataAuto(timestamp, title_id, system), | 
					
						
							|  |  |  |                GetPath("user_report", title_id, timestamp)); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 04:30:54 -04:00
										 |  |  | void Reporter::ClearFSAccessLog() const { | 
					
						
							|  |  |  |     const auto access_log_path = | 
					
						
							|  |  |  |         Common::FS::GetYuzuPath(Common::FS::YuzuPath::SDMCDir) / "FsAccessLog.txt"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Common::FS::IOFile access_log_file{access_log_path, Common::FS::FileAccessMode::Write, | 
					
						
							|  |  |  |                                        Common::FS::FileType::TextFile}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!access_log_file.IsOpen()) { | 
					
						
							|  |  |  |         LOG_ERROR(Common_Filesystem, "Failed to clear the filesystem access log."); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | bool Reporter::IsReportingEnabled() const { | 
					
						
							| 
									
										
										
										
											2021-06-28 15:58:16 -04:00
										 |  |  |     return Settings::values.reporting_services.GetValue(); | 
					
						
							| 
									
										
										
										
											2019-05-17 21:45:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Core
 |