| | |
| | |
| |
|
| | #include <ctime> |
| | #include <fstream> |
| | #include <iomanip> |
| |
|
| | #include <fmt/chrono.h> |
| | #include <fmt/format.h> |
| | #include <nlohmann/json.hpp> |
| |
|
| | #include "common/fs/file.h" |
| | #include "common/fs/fs.h" |
| | #include "common/fs/path_util.h" |
| | #include "common/hex_util.h" |
| | #include "common/scm_rev.h" |
| | #include "common/settings.h" |
| | #include "core/arm/arm_interface.h" |
| | #include "core/core.h" |
| | #include "core/hle/kernel/k_page_table.h" |
| | #include "core/hle/kernel/k_process.h" |
| | #include "core/hle/result.h" |
| | #include "core/hle/service/hle_ipc.h" |
| | #include "core/memory.h" |
| | #include "core/reporter.h" |
| |
|
| | namespace { |
| |
|
| | 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); |
| | } |
| |
|
| | std::string GetTimestamp() { |
| | const auto time = std::time(nullptr); |
| | return fmt::format("{:%FT%H-%M-%S}", *std::localtime(&time)); |
| | } |
| |
|
| | using namespace nlohmann; |
| |
|
| | void SaveToFile(const json& json, const std::filesystem::path& filename) { |
| | if (!Common::FS::CreateParentDirs(filename)) { |
| | LOG_ERROR(Core, "Failed to create path for '{}' to save report!", |
| | Common::FS::PathToUTF8String(filename)); |
| | return; |
| | } |
| |
|
| | std::ofstream file; |
| | Common::FS::OpenFileStream(file, filename, std::ios_base::out | std::ios_base::trunc); |
| |
|
| | 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)}, |
| | }; |
| | } |
| |
|
| | json GetReportCommonData(u64 title_id, Result result, const std::string& timestamp, |
| | 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.GetModule()))}, |
| | {"result_description", fmt::format("{:08X}", result.GetDescription())}, |
| | {"timestamp", timestamp}, |
| | }; |
| |
|
| | if (user_id.has_value()) { |
| | out["user_id"] = fmt::format("{:016X}{:016X}", (*user_id)[1], (*user_id)[0]); |
| | } |
| |
|
| | return out; |
| | } |
| |
|
| | json GetProcessorStateData(const std::string& architecture, u64 entry_point, u64 sp, u64 pc, |
| | u64 pstate, const std::array<u64, 31>& registers, |
| | const std::optional<std::array<u64, 32>>& backtrace = {}) { |
| | 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); |
| | } |
| |
|
| | return out; |
| | } |
| |
|
| | json GetFullDataAuto(const std::string& timestamp, u64 title_id, Core::System& system) { |
| | json out; |
| |
|
| | out["yuzu_version"] = GetYuzuVersionData(); |
| | out["report_common"] = GetReportCommonData(title_id, ResultSuccess, timestamp); |
| |
|
| | return out; |
| | } |
| |
|
| | template <bool read_value, typename DescriptorType> |
| | json GetHLEBufferDescriptorData(const std::vector<DescriptorType>& buffer, |
| | Core::Memory::Memory& memory) { |
| | 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()); |
| | memory.ReadBlock(desc.Address(), data.data(), desc.Size()); |
| | entry["data"] = Common::HexToString(data); |
| | } |
| |
|
| | buffer_out.push_back(std::move(entry)); |
| | } |
| |
|
| | return buffer_out; |
| | } |
| |
|
| | json GetHLERequestContextData(Service::HLERequestContext& ctx, Core::Memory::Memory& memory) { |
| | 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); |
| |
|
| | 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); |
| |
|
| | return out; |
| | } |
| |
|
| | } |
| |
|
| | namespace Core { |
| |
|
| | Reporter::Reporter(System& system_) : system(system_) { |
| | ClearFSAccessLog(); |
| | } |
| |
|
| | Reporter::~Reporter() = default; |
| |
|
| | 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, |
| | const std::array<u64, 31>& registers, |
| | const std::array<u64, 32>& backtrace, u32 backtrace_size, |
| | const std::string& arch, u32 unk10) const { |
| | if (!IsReportingEnabled()) { |
| | return; |
| | } |
| |
|
| | const auto timestamp = GetTimestamp(); |
| | 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); |
| |
|
| | SaveToFile(out, GetPath("crash_report", title_id, timestamp)); |
| | } |
| |
|
| | void Reporter::SaveSvcBreakReport(u32 type, bool signal_debugger, u64 info1, u64 info2, |
| | const std::optional<std::vector<u8>>& resolved_buffer) const { |
| | if (!IsReportingEnabled()) { |
| | return; |
| | } |
| |
|
| | const auto timestamp = GetTimestamp(); |
| | const auto title_id = system.GetApplicationProcessProgramID(); |
| | auto out = GetFullDataAuto(timestamp, title_id, system); |
| |
|
| | 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()) { |
| | break_out["debug_buffer"] = Common::HexToString(*resolved_buffer); |
| | } |
| |
|
| | out["svc_break"] = std::move(break_out); |
| |
|
| | SaveToFile(out, GetPath("svc_break_report", title_id, timestamp)); |
| | } |
| |
|
| | void Reporter::SaveUnimplementedFunctionReport(Service::HLERequestContext& ctx, u32 command_id, |
| | const std::string& name, |
| | const std::string& service_name) const { |
| | if (!IsReportingEnabled()) { |
| | return; |
| | } |
| |
|
| | const auto timestamp = GetTimestamp(); |
| | const auto title_id = system.GetApplicationProcessProgramID(); |
| | auto out = GetFullDataAuto(timestamp, title_id, system); |
| |
|
| | auto function_out = GetHLERequestContextData(ctx, system.ApplicationMemory()); |
| | function_out["command_id"] = command_id; |
| | function_out["function_name"] = name; |
| | function_out["service_name"] = service_name; |
| |
|
| | out["function"] = std::move(function_out); |
| |
|
| | SaveToFile(out, GetPath("unimpl_func_report", title_id, timestamp)); |
| | } |
| |
|
| | void Reporter::SaveUnimplementedAppletReport( |
| | u32 applet_id, u32 common_args_version, u32 library_version, u32 theme_color, |
| | bool startup_sound, u64 system_tick, const std::vector<std::vector<u8>>& normal_channel, |
| | const std::vector<std::vector<u8>>& interactive_channel) const { |
| | if (!IsReportingEnabled()) { |
| | return; |
| | } |
| |
|
| | const auto timestamp = GetTimestamp(); |
| | const auto title_id = system.GetApplicationProcessProgramID(); |
| | auto out = GetFullDataAuto(timestamp, title_id, system); |
| |
|
| | 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) { |
| | normal_out.push_back(Common::HexToString(data)); |
| | } |
| |
|
| | auto interactive_out = json::array(); |
| | for (const auto& data : interactive_channel) { |
| | interactive_out.push_back(Common::HexToString(data)); |
| | } |
| |
|
| | out["applet_normal_data"] = std::move(normal_out); |
| | out["applet_interactive_data"] = std::move(interactive_out); |
| |
|
| | SaveToFile(out, GetPath("unimpl_applet_report", title_id, timestamp)); |
| | } |
| |
|
| | void Reporter::SavePlayReport(PlayReportType type, u64 title_id, |
| | const std::vector<std::span<const u8>>& data, |
| | std::optional<u64> process_id, std::optional<u128> user_id) const { |
| | if (!IsReportingEnabled()) { |
| | return; |
| | } |
| |
|
| | const auto timestamp = GetTimestamp(); |
| | json out; |
| |
|
| | out["yuzu_version"] = GetYuzuVersionData(); |
| | out["report_common"] = GetReportCommonData(title_id, ResultSuccess, timestamp, user_id); |
| |
|
| | auto data_out = json::array(); |
| | for (const auto& d : data) { |
| | data_out.push_back(Common::HexToString(d)); |
| | } |
| |
|
| | 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)); |
| | out["play_report_data"] = std::move(data_out); |
| |
|
| | SaveToFile(out, GetPath("play_report", title_id, timestamp)); |
| | } |
| |
|
| | void Reporter::SaveErrorReport(u64 title_id, Result result, |
| | const std::optional<std::string>& custom_text_main, |
| | const std::optional<std::string>& custom_text_detail) const { |
| | if (!IsReportingEnabled()) { |
| | return; |
| | } |
| |
|
| | const auto timestamp = GetTimestamp(); |
| | json out; |
| |
|
| | out["yuzu_version"] = GetYuzuVersionData(); |
| | out["report_common"] = GetReportCommonData(title_id, result, timestamp); |
| |
|
| | out["error_custom_text"] = { |
| | {"main", custom_text_main.value_or("")}, |
| | {"detail", custom_text_detail.value_or("")}, |
| | }; |
| |
|
| | SaveToFile(out, GetPath("error_report", title_id, timestamp)); |
| | } |
| |
|
| | void Reporter::SaveFSAccessLog(std::string_view log_message) const { |
| | const auto access_log_path = |
| | Common::FS::GetYuzuPath(Common::FS::YuzuPath::SDMCDir) / "FsAccessLog.txt"; |
| |
|
| | void(Common::FS::AppendStringToFile(access_log_path, Common::FS::FileType::TextFile, |
| | log_message)); |
| | } |
| |
|
| | void Reporter::SaveUserReport() const { |
| | if (!IsReportingEnabled()) { |
| | return; |
| | } |
| |
|
| | const auto timestamp = GetTimestamp(); |
| | const auto title_id = system.GetApplicationProcessProgramID(); |
| |
|
| | SaveToFile(GetFullDataAuto(timestamp, title_id, system), |
| | GetPath("user_report", title_id, timestamp)); |
| | } |
| |
|
| | 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."); |
| | } |
| | } |
| |
|
| | bool Reporter::IsReportingEnabled() const { |
| | return Settings::values.reporting_services.GetValue(); |
| | } |
| |
|
| | } |
| |
|