From b31b2db5fa75d2397d6ef1551398618d33da7bbe Mon Sep 17 00:00:00 2001 From: "Admiral H. Curtiss" Date: Sun, 30 Oct 2022 02:00:28 +0200 Subject: [PATCH 1/4] GCMemcard: Remove unused methods. --- Source/Core/Core/HW/GCMemcard/GCMemcard.cpp | 120 -------------------- Source/Core/Core/HW/GCMemcard/GCMemcard.h | 11 -- 2 files changed, 131 deletions(-) diff --git a/Source/Core/Core/HW/GCMemcard/GCMemcard.cpp b/Source/Core/Core/HW/GCMemcard/GCMemcard.cpp index e6b6383de5..15173e0e48 100644 --- a/Source/Core/Core/HW/GCMemcard/GCMemcard.cpp +++ b/Source/Core/Core/HW/GCMemcard/GCMemcard.cpp @@ -420,51 +420,6 @@ std::optional GCMemcard::TitlePresent(const DEntry& d) const return std::nullopt; } -bool GCMemcard::GCI_FileName(u8 index, std::string& filename) const -{ - if (!m_valid || index >= DIRLEN || - GetActiveDirectory().m_dir_entries[index].m_gamecode == DEntry::UNINITIALIZED_GAMECODE) - return false; - - filename = GetActiveDirectory().m_dir_entries[index].GCI_FileName(); - return true; -} - -std::string GCMemcard::DEntry_GameCode(u8 index) const -{ - if (!m_valid || index >= DIRLEN) - return ""; - - return std::string( - reinterpret_cast(GetActiveDirectory().m_dir_entries[index].m_gamecode.data()), - GetActiveDirectory().m_dir_entries[index].m_gamecode.size()); -} - -std::string GCMemcard::DEntry_Makercode(u8 index) const -{ - if (!m_valid || index >= DIRLEN) - return ""; - - return std::string( - reinterpret_cast(GetActiveDirectory().m_dir_entries[index].m_makercode.data()), - GetActiveDirectory().m_dir_entries[index].m_makercode.size()); -} - -std::string GCMemcard::DEntry_BIFlags(u8 index) const -{ - if (!m_valid || index >= DIRLEN) - return ""; - - std::string flags; - int x = GetActiveDirectory().m_dir_entries[index].m_banner_and_icon_flags; - for (int i = 0; i < 8; i++) - { - flags.push_back((x & 0x80) ? '1' : '0'); - x = x << 1; - } - return flags; -} - bool GCMemcard::DEntry_IsPingPong(u8 index) const { if (!m_valid || index >= DIRLEN) @@ -474,81 +429,6 @@ bool GCMemcard::DEntry_IsPingPong(u8 index) const return (flags & 0b0000'0100) != 0; } -std::string GCMemcard::DEntry_FileName(u8 index) const -{ - if (!m_valid || index >= DIRLEN) - return ""; - - return std::string( - reinterpret_cast(GetActiveDirectory().m_dir_entries[index].m_filename.data()), - GetActiveDirectory().m_dir_entries[index].m_filename.size()); -} - -u32 GCMemcard::DEntry_ModTime(u8 index) const -{ - if (!m_valid || index >= DIRLEN) - return 0xFFFFFFFF; - - return GetActiveDirectory().m_dir_entries[index].m_modification_time; -} - -u32 GCMemcard::DEntry_ImageOffset(u8 index) const -{ - if (!m_valid || index >= DIRLEN) - return 0xFFFFFFFF; - - return GetActiveDirectory().m_dir_entries[index].m_image_offset; -} - -std::string GCMemcard::DEntry_IconFmt(u8 index) const -{ - if (!m_valid || index >= DIRLEN) - return ""; - - u16 x = GetActiveDirectory().m_dir_entries[index].m_icon_format; - std::string format; - for (size_t i = 0; i < 16; ++i) - { - format.push_back(Common::ExtractBit(x, 15 - i) ? '1' : '0'); - } - return format; -} - -std::string GCMemcard::DEntry_AnimSpeed(u8 index) const -{ - if (!m_valid || index >= DIRLEN) - return ""; - - u16 x = GetActiveDirectory().m_dir_entries[index].m_animation_speed; - std::string speed; - for (size_t i = 0; i < 16; ++i) - { - speed.push_back(Common::ExtractBit(x, 15 - i) ? '1' : '0'); - } - return speed; -} - -std::string GCMemcard::DEntry_Permissions(u8 index) const -{ - if (!m_valid || index >= DIRLEN) - return ""; - - u8 Permissions = GetActiveDirectory().m_dir_entries[index].m_file_permissions; - std::string permissionsString; - permissionsString.push_back((Permissions & 16) ? 'x' : 'M'); - permissionsString.push_back((Permissions & 8) ? 'x' : 'C'); - permissionsString.push_back((Permissions & 4) ? 'P' : 'x'); - return permissionsString; -} - -u8 GCMemcard::DEntry_CopyCounter(u8 index) const -{ - if (!m_valid || index >= DIRLEN) - return 0xFF; - - return GetActiveDirectory().m_dir_entries[index].m_copy_counter; -} - u16 GCMemcard::DEntry_FirstBlock(u8 index) const { if (!m_valid || index >= DIRLEN) diff --git a/Source/Core/Core/HW/GCMemcard/GCMemcard.h b/Source/Core/Core/HW/GCMemcard/GCMemcard.h index 6b82d162d2..ed4c36ba20 100644 --- a/Source/Core/Core/HW/GCMemcard/GCMemcard.h +++ b/Source/Core/Core/HW/GCMemcard/GCMemcard.h @@ -453,19 +453,8 @@ public: // with that identity exists in this card. std::optional TitlePresent(const DEntry& d) const; - bool GCI_FileName(u8 index, std::string& filename) const; // DEntry functions, all take u8 index < DIRLEN (127) - std::string DEntry_GameCode(u8 index) const; - std::string DEntry_Makercode(u8 index) const; - std::string DEntry_BIFlags(u8 index) const; bool DEntry_IsPingPong(u8 index) const; - std::string DEntry_FileName(u8 index) const; - u32 DEntry_ModTime(u8 index) const; - u32 DEntry_ImageOffset(u8 index) const; - std::string DEntry_IconFmt(u8 index) const; - std::string DEntry_AnimSpeed(u8 index) const; - std::string DEntry_Permissions(u8 index) const; - u8 DEntry_CopyCounter(u8 index) const; // get first block for file u16 DEntry_FirstBlock(u8 index) const; // get file length in blocks From c517e92719c352eebcdf250c2a4cdec4e657eaaf Mon Sep 17 00:00:00 2001 From: "Admiral H. Curtiss" Date: Sun, 30 Oct 2022 02:08:00 +0200 Subject: [PATCH 2/4] GCMemcardDirectory: Use HasSameIdentity() in LoadGCI(). This is cheaper and more accurate than comparing default GCI filenames. --- Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.cpp b/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.cpp index cbfbc200f3..3f02ef16b1 100644 --- a/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.cpp +++ b/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.cpp @@ -44,7 +44,7 @@ bool GCMemcardDirectory::LoadGCI(Memcard::GCIFile gci) // check if any already loaded file has the same internal name as the new file for (const Memcard::GCIFile& already_loaded_gci : m_saves) { - if (gci.m_gci_header.GCI_FileName() == already_loaded_gci.m_gci_header.GCI_FileName()) + if (HasSameIdentity(gci.m_gci_header, already_loaded_gci.m_gci_header)) { ERROR_LOG_FMT(EXPANSIONINTERFACE, "{}\nwas not loaded because it has the same internal filename as previously " From 4b0312ecf8f22514bc92e5bb30d618c6d53db93c Mon Sep 17 00:00:00 2001 From: "Admiral H. Curtiss" Date: Sun, 30 Oct 2022 02:40:54 +0100 Subject: [PATCH 3/4] GCMemcardDirectory: Decode and strip strings for GCI filenames. --- Source/Core/Core/HW/GCMemcard/GCMemcard.cpp | 16 ++++------ Source/Core/Core/HW/GCMemcard/GCMemcard.h | 5 ++- .../Core/HW/GCMemcard/GCMemcardDirectory.cpp | 32 +++++++++++++++++-- .../Core/HW/GCMemcard/GCMemcardDirectory.h | 3 +- Source/Core/Core/NetPlayServer.cpp | 7 ++-- 5 files changed, 43 insertions(+), 20 deletions(-) diff --git a/Source/Core/Core/HW/GCMemcard/GCMemcard.cpp b/Source/Core/Core/HW/GCMemcard/GCMemcard.cpp index 15173e0e48..d603e1075e 100644 --- a/Source/Core/Core/HW/GCMemcard/GCMemcard.cpp +++ b/Source/Core/Core/HW/GCMemcard/GCMemcard.cpp @@ -304,7 +304,7 @@ void GCMemcard::UpdateBat(const BlockAlloc& bat) bool GCMemcard::IsShiftJIS() const { - return m_header_block.m_data.m_encoding != 0; + return m_header_block.IsShiftJIS(); } bool GCMemcard::Save() @@ -1274,15 +1274,6 @@ DEntry::DEntry() memset(reinterpret_cast(this), 0xFF, DENTRY_SIZE); } -std::string DEntry::GCI_FileName() const -{ - std::string filename = - std::string(reinterpret_cast(m_makercode.data()), m_makercode.size()) + '-' + - std::string(reinterpret_cast(m_gamecode.data()), m_gamecode.size()) + '-' + - reinterpret_cast(m_filename.data()) + ".gci"; - return Common::EscapeFileName(filename); -} - void Header::FixChecksums() { std::tie(m_checksum, m_checksum_inv) = CalculateChecksums(); @@ -1324,6 +1315,11 @@ GCMemcardErrorCode Header::CheckForErrors(u16 card_size_mbits) const return error_code; } +bool Header::IsShiftJIS() const +{ + return m_data.m_encoding != 0; +} + Directory::Directory() { memset(reinterpret_cast(this), 0xFF, BLOCK_SIZE); diff --git a/Source/Core/Core/HW/GCMemcard/GCMemcard.h b/Source/Core/Core/HW/GCMemcard/GCMemcard.h index ed4c36ba20..80a2b3eea4 100644 --- a/Source/Core/Core/HW/GCMemcard/GCMemcard.h +++ b/Source/Core/Core/HW/GCMemcard/GCMemcard.h @@ -231,6 +231,8 @@ struct Header std::pair CalculateChecksums() const; GCMemcardErrorCode CheckForErrors(u16 card_size_mbits) const; + + bool IsShiftJIS() const; }; static_assert(sizeof(Header) == BLOCK_SIZE); static_assert(std::is_trivially_copyable_v
); @@ -239,9 +241,6 @@ struct DEntry { DEntry(); - // TODO: This probably shouldn't be here at all? - std::string GCI_FileName() const; - static constexpr std::array UNINITIALIZED_GAMECODE{{0xFF, 0xFF, 0xFF, 0xFF}}; // 4 bytes at 0x00: Gamecode diff --git a/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.cpp b/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.cpp index 3f02ef16b1..f7b318a6eb 100644 --- a/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.cpp +++ b/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.cpp @@ -9,6 +9,7 @@ #include #include #include +#include #include #include @@ -39,6 +40,28 @@ static const char* MC_HDR = "MC_SYSTEM_AREA"; +static std::string GenerateDefaultGCIFilename(const Memcard::DEntry& entry, + bool card_encoding_is_shift_jis) +{ + const auto string_decoder = card_encoding_is_shift_jis ? SHIFTJISToUTF8 : CP1252ToUTF8; + const auto strip_null = [](const std::string_view& s) { + auto offset = s.find('\0'); + if (offset == std::string_view::npos) + return s; + return s.substr(0, offset); + }; + + const std::string_view makercode(reinterpret_cast(entry.m_makercode.data()), + entry.m_makercode.size()); + const std::string_view gamecode(reinterpret_cast(entry.m_gamecode.data()), + entry.m_gamecode.size()); + const std::string_view filename(reinterpret_cast(entry.m_filename.data()), + entry.m_filename.size()); + return Common::EscapeFileName(fmt::format("{}-{}-{}.gci", strip_null(string_decoder(makercode)), + strip_null(string_decoder(gamecode)), + strip_null(string_decoder(filename)))); +} + bool GCMemcardDirectory::LoadGCI(Memcard::GCIFile gci) { // check if any already loaded file has the same internal name as the new file @@ -102,7 +125,8 @@ bool GCMemcardDirectory::LoadGCI(Memcard::GCIFile gci) // This is only used by NetPlay but it made sense to put it here to keep the relevant code together std::vector GCMemcardDirectory::GetFileNamesForGameID(const std::string& directory, - const std::string& game_id) + const std::string& game_id, + bool card_encoding_is_shift_jis) { std::vector filenames; @@ -123,7 +147,8 @@ std::vector GCMemcardDirectory::GetFileNamesForGameID(const std::st if (!gci_file.ReadBytes(&gci.m_gci_header, Memcard::DENTRY_SIZE)) continue; - const std::string gci_filename = gci.m_gci_header.GCI_FileName(); + const std::string gci_filename = + GenerateDefaultGCIFilename(gci.m_gci_header, card_encoding_is_shift_jis); if (std::find(loaded_saves.begin(), loaded_saves.end(), gci_filename) != loaded_saves.end()) continue; @@ -614,7 +639,8 @@ void GCMemcardDirectory::FlushToFile() } if (save.m_filename.empty()) { - std::string default_save_name = m_save_directory + save.m_gci_header.GCI_FileName(); + std::string default_save_name = + m_save_directory + GenerateDefaultGCIFilename(save.m_gci_header, m_hdr.IsShiftJIS()); // Check to see if another file is using the same name // This seems unlikely except in the case of file corruption diff --git a/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.h b/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.h index 5b02700a61..b61ad360b6 100644 --- a/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.h +++ b/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.h @@ -32,7 +32,8 @@ public: GCMemcardDirectory& operator=(GCMemcardDirectory&&) = delete; static std::vector GetFileNamesForGameID(const std::string& directory, - const std::string& game_id); + const std::string& game_id, + bool card_encoding_is_shift_jis); void FlushToFile(); void FlushThread(); s32 Read(u32 src_address, s32 length, u8* dest_address) override; diff --git a/Source/Core/Core/NetPlayServer.cpp b/Source/Core/Core/NetPlayServer.cpp index 18076ada29..9c0ba0a53d 100644 --- a/Source/Core/Core/NetPlayServer.cpp +++ b/Source/Core/Core/NetPlayServer.cpp @@ -1686,7 +1686,8 @@ bool NetPlayServer::SyncSaveData(const SaveSyncInfo& sync_info) return true; const auto game_region = sync_info.game->GetRegion(); - const std::string region = Config::GetDirectoryForRegion(Config::ToGameCubeRegion(game_region)); + const auto gamecube_region = Config::ToGameCubeRegion(game_region); + const std::string region = Config::GetDirectoryForRegion(gamecube_region); for (ExpansionInterface::Slot slot : ExpansionInterface::MEMCARD_SLOTS) { @@ -1733,8 +1734,8 @@ bool NetPlayServer::SyncSaveData(const SaveSyncInfo& sync_info) if (File::IsDirectory(path)) { - std::vector files = - GCMemcardDirectory::GetFileNamesForGameID(path + DIR_SEP, sync_info.game->GetGameID()); + std::vector files = GCMemcardDirectory::GetFileNamesForGameID( + path + DIR_SEP, sync_info.game->GetGameID(), gamecube_region == DiscIO::Region::NTSC_J); pac << static_cast(files.size()); From 1089d3cab6c9a40732fca47b164174bd6795614b Mon Sep 17 00:00:00 2001 From: "Admiral H. Curtiss" Date: Mon, 7 Nov 2022 06:16:34 +0100 Subject: [PATCH 4/4] GCMemcardDirectory: Compare GCI files in GetFileNamesForGameID() by their identity instead of their default filename. --- .../Core/Core/HW/GCMemcard/GCMemcardDirectory.cpp | 15 ++++++++------- .../Core/Core/HW/GCMemcard/GCMemcardDirectory.h | 3 +-- Source/Core/Core/NetPlayServer.cpp | 4 ++-- 3 files changed, 11 insertions(+), 11 deletions(-) diff --git a/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.cpp b/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.cpp index f7b318a6eb..304c7922d9 100644 --- a/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.cpp +++ b/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.cpp @@ -125,8 +125,7 @@ bool GCMemcardDirectory::LoadGCI(Memcard::GCIFile gci) // This is only used by NetPlay but it made sense to put it here to keep the relevant code together std::vector GCMemcardDirectory::GetFileNamesForGameID(const std::string& directory, - const std::string& game_id, - bool card_encoding_is_shift_jis) + const std::string& game_id) { std::vector filenames; @@ -134,7 +133,7 @@ std::vector GCMemcardDirectory::GetFileNamesForGameID(const std::st if (game_id.length() >= 4 && game_id != "00000000") game_code = Common::swap32(reinterpret_cast(game_id.c_str())); - std::vector loaded_saves; + std::vector loaded_saves; for (const std::string& file_name : Common::DoFileSearch({directory}, {".gci"})) { File::IOFile gci_file(file_name, "rb"); @@ -147,9 +146,11 @@ std::vector GCMemcardDirectory::GetFileNamesForGameID(const std::st if (!gci_file.ReadBytes(&gci.m_gci_header, Memcard::DENTRY_SIZE)) continue; - const std::string gci_filename = - GenerateDefaultGCIFilename(gci.m_gci_header, card_encoding_is_shift_jis); - if (std::find(loaded_saves.begin(), loaded_saves.end(), gci_filename) != loaded_saves.end()) + const auto same_identity_save_it = std::find_if( + loaded_saves.begin(), loaded_saves.end(), [&gci](const Memcard::DEntry& entry) { + return Memcard::HasSameIdentity(gci.m_gci_header, entry); + }); + if (same_identity_save_it != loaded_saves.end()) continue; const u16 num_blocks = gci.m_gci_header.m_block_count; @@ -169,7 +170,7 @@ std::vector GCMemcardDirectory::GetFileNamesForGameID(const std::st if (game_code == Common::swap32(gci.m_gci_header.m_gamecode.data())) { - loaded_saves.push_back(gci_filename); + loaded_saves.push_back(gci.m_gci_header); filenames.push_back(file_name); } } diff --git a/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.h b/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.h index b61ad360b6..5b02700a61 100644 --- a/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.h +++ b/Source/Core/Core/HW/GCMemcard/GCMemcardDirectory.h @@ -32,8 +32,7 @@ public: GCMemcardDirectory& operator=(GCMemcardDirectory&&) = delete; static std::vector GetFileNamesForGameID(const std::string& directory, - const std::string& game_id, - bool card_encoding_is_shift_jis); + const std::string& game_id); void FlushToFile(); void FlushThread(); s32 Read(u32 src_address, s32 length, u8* dest_address) override; diff --git a/Source/Core/Core/NetPlayServer.cpp b/Source/Core/Core/NetPlayServer.cpp index 9c0ba0a53d..f2fd713049 100644 --- a/Source/Core/Core/NetPlayServer.cpp +++ b/Source/Core/Core/NetPlayServer.cpp @@ -1734,8 +1734,8 @@ bool NetPlayServer::SyncSaveData(const SaveSyncInfo& sync_info) if (File::IsDirectory(path)) { - std::vector files = GCMemcardDirectory::GetFileNamesForGameID( - path + DIR_SEP, sync_info.game->GetGameID(), gamecube_region == DiscIO::Region::NTSC_J); + std::vector files = + GCMemcardDirectory::GetFileNamesForGameID(path + DIR_SEP, sync_info.game->GetGameID()); pac << static_cast(files.size());