From 4714a2fccb799105ef7b5d3c030a715dbf2a84e7 Mon Sep 17 00:00:00 2001 From: Stenzek Date: Mon, 6 Nov 2023 23:52:26 +1000 Subject: [PATCH] Achievements: Identify using running ELF instead of disc ELF --- pcsx2/Achievements.cpp | 38 +++++++++++++++++++------------------- pcsx2/VMManager.cpp | 6 +++++- pcsx2/VMManager.h | 3 +++ 3 files changed, 27 insertions(+), 20 deletions(-) diff --git a/pcsx2/Achievements.cpp b/pcsx2/Achievements.cpp index c1c66b4eeea5d..e3879dbc9c9f1 100644 --- a/pcsx2/Achievements.cpp +++ b/pcsx2/Achievements.cpp @@ -139,12 +139,12 @@ namespace Achievements static void BeginLoadingScreen(const char* text, bool* was_running_idle); static void EndLoadingScreen(bool was_running_idle); static std::string_view GetELFNameForHash(const std::string& elf_path); - static std::string GetGameHash(); + static std::string GetGameHash(const std::string& elf_path); static void SetHardcoreMode(bool enabled, bool force_display_message); static bool IsLoggedInOrLoggingIn(); static bool IsUnknownGame(); static void ShowLoginSuccess(const rc_client_t* client); - static void IdentifyGame(u32 disc_crc); + static void IdentifyGame(u32 disc_crc, u32 crc); static void BeginLoadGame(); static void UpdateGameSummary(); static void DownloadImage(std::string url, std::string cache_filename); @@ -211,10 +211,10 @@ namespace Achievements static std::string s_image_directory; static std::unique_ptr s_http_downloader; - static u32 s_game_disc_crc; static std::string s_game_hash; static std::string s_game_title; static std::string s_game_icon; + static u32 s_game_crc; static rc_client_user_game_summary_t s_game_summary; static u32 s_game_id = 0; @@ -314,12 +314,8 @@ std::string_view Achievements::GetELFNameForHash(const std::string& elf_path) return std::string_view(elf_path).substr(start, end - start); } -std::string Achievements::GetGameHash() +std::string Achievements::GetGameHash(const std::string& elf_path) { - const std::string elf_path = VMManager::GetDiscELF(); - if (elf_path.empty()) - return {}; - // this.. really shouldn't be invalid const std::string_view name_for_hash = GetELFNameForHash(elf_path); if (name_for_hash.empty()) @@ -460,7 +456,7 @@ bool Achievements::Initialize() // Begin disc identification early, before the login finishes. if (VMManager::HasValidVM()) - IdentifyGame(VMManager::GetDiscCRC()); + IdentifyGame(VMManager::GetDiscCRC(), VMManager::GetCurrentCRC()); const std::string username = Host::GetBaseStringSettingValue("Achievements", "Username"); const std::string api_token = Host::GetBaseStringSettingValue("Achievements", "Token"); @@ -845,21 +841,26 @@ void Achievements::GameChanged(u32 disc_crc, u32 crc) if (!IsActive()) return; - IdentifyGame(disc_crc); + IdentifyGame(disc_crc, crc); } -void Achievements::IdentifyGame(u32 disc_crc) +void Achievements::IdentifyGame(u32 disc_crc, u32 crc) { - // avoid reading+hashing the executable if the crc hasn't changed - if (s_game_disc_crc == disc_crc) + // If we're currently loading the ELF, assume that we're going to load the default ELF. + // That way we can download achievement data while the PS2 logo runs. Pretty safe assumption. + const bool booted_elf = VMManager::Internal::HasBootedELF(); + const u32 crc_to_use = booted_elf ? crc : disc_crc; + + // Avoid reading+hashing the executable if the crc hasn't changed. + if (s_game_crc == crc_to_use) return; - std::string game_hash = GetGameHash(); + const std::string game_hash = GetGameHash(booted_elf ? VMManager::GetCurrentELF() : VMManager::GetDiscELF()); if (s_game_hash == game_hash) return; ClearGameHash(); - s_game_disc_crc = disc_crc; + s_game_crc = crc_to_use; s_game_hash = std::move(game_hash); #ifdef ENABLE_RAINTEGRATION @@ -898,7 +899,7 @@ void Achievements::BeginLoadGame() if (s_game_hash.empty()) { // when we're booting the bios, or shutting down, this will fail - if (s_game_disc_crc != 0) + if (s_game_crc != 0) { Host::AddKeyedOSDMessage("retroachievements_disc_read_failed", TRANSLATE_STR("Achievements", "Failed to read executable from disc. Achievements disabled."), @@ -1019,7 +1020,7 @@ void Achievements::ClearGameInfo() void Achievements::ClearGameHash() { - s_game_disc_crc = 0; + s_game_crc = 0; std::string().swap(s_game_hash); } @@ -1492,8 +1493,7 @@ void Achievements::LoadState(const u8* state_data, u32 state_data_size) return; // this assumes that the CRC and ELF name has been loaded prior to the cheevos state (it should be). - if (VMManager::GetDiscCRC() != s_game_disc_crc) - GameChanged(VMManager::GetDiscCRC(), VMManager::GetCurrentCRC()); + GameChanged(VMManager::GetDiscCRC(), VMManager::GetCurrentCRC()); #ifdef ENABLE_RAINTEGRATION if (IsUsingRAIntegration()) diff --git a/pcsx2/VMManager.cpp b/pcsx2/VMManager.cpp index f03c9d3caa0d0..9e07d60a21f52 100644 --- a/pcsx2/VMManager.cpp +++ b/pcsx2/VMManager.cpp @@ -331,10 +331,14 @@ std::string VMManager::GetDiscVersion() u32 VMManager::GetCurrentCRC() { - std::unique_lock lock(s_info_mutex); return s_current_crc; } +const std::string& VMManager::GetCurrentELF() +{ + return s_elf_path; +} + bool VMManager::Internal::CPUThreadInitialize() { Threading::SetNameOfCurrentThread("CPU Thread"); diff --git a/pcsx2/VMManager.h b/pcsx2/VMManager.h index f93788db466ba..db724d4553d46 100644 --- a/pcsx2/VMManager.h +++ b/pcsx2/VMManager.h @@ -93,6 +93,9 @@ namespace VMManager /// Returns the crc of the executable currently running. u32 GetCurrentCRC(); + /// Returns the path to the ELF which is currently running. Only safe to read on the EE thread. + const std::string& GetCurrentELF(); + /// Initializes all system components. bool Initialize(VMBootParameters boot_params);