Merge pull request #1396 from DarkLordZach/packed-updates
loader: Add support for packed updates
This commit is contained in:
commit
6e4d2e672d
|
@ -184,8 +184,8 @@ static void ApplyLayeredFS(VirtualFile& romfs, u64 title_id, ContentRecordType t
|
||||||
romfs = std::move(packed);
|
romfs = std::move(packed);
|
||||||
}
|
}
|
||||||
|
|
||||||
VirtualFile PatchManager::PatchRomFS(VirtualFile romfs, u64 ivfc_offset,
|
VirtualFile PatchManager::PatchRomFS(VirtualFile romfs, u64 ivfc_offset, ContentRecordType type,
|
||||||
ContentRecordType type) const {
|
VirtualFile update_raw) const {
|
||||||
LOG_INFO(Loader, "Patching RomFS for title_id={:016X}, type={:02X}", title_id,
|
LOG_INFO(Loader, "Patching RomFS for title_id={:016X}, type={:02X}", title_id,
|
||||||
static_cast<u8>(type));
|
static_cast<u8>(type));
|
||||||
|
|
||||||
|
@ -205,6 +205,13 @@ VirtualFile PatchManager::PatchRomFS(VirtualFile romfs, u64 ivfc_offset,
|
||||||
FormatTitleVersion(installed->GetEntryVersion(update_tid).get_value_or(0)));
|
FormatTitleVersion(installed->GetEntryVersion(update_tid).get_value_or(0)));
|
||||||
romfs = new_nca->GetRomFS();
|
romfs = new_nca->GetRomFS();
|
||||||
}
|
}
|
||||||
|
} else if (update_raw != nullptr) {
|
||||||
|
const auto new_nca = std::make_shared<NCA>(update_raw, romfs, ivfc_offset);
|
||||||
|
if (new_nca->GetStatus() == Loader::ResultStatus::Success &&
|
||||||
|
new_nca->GetRomFS() != nullptr) {
|
||||||
|
LOG_INFO(Loader, " RomFS: Update (PACKED) applied successfully");
|
||||||
|
romfs = new_nca->GetRomFS();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// LayeredFS
|
// LayeredFS
|
||||||
|
@ -224,7 +231,8 @@ static bool IsDirValidAndNonEmpty(const VirtualDir& dir) {
|
||||||
return dir != nullptr && (!dir->GetFiles().empty() || !dir->GetSubdirectories().empty());
|
return dir != nullptr && (!dir->GetFiles().empty() || !dir->GetSubdirectories().empty());
|
||||||
}
|
}
|
||||||
|
|
||||||
std::map<std::string, std::string, std::less<>> PatchManager::GetPatchVersionNames() const {
|
std::map<std::string, std::string, std::less<>> PatchManager::GetPatchVersionNames(
|
||||||
|
VirtualFile update_raw) const {
|
||||||
std::map<std::string, std::string, std::less<>> out;
|
std::map<std::string, std::string, std::less<>> out;
|
||||||
const auto installed = Service::FileSystem::GetUnionContents();
|
const auto installed = Service::FileSystem::GetUnionContents();
|
||||||
|
|
||||||
|
@ -245,6 +253,8 @@ std::map<std::string, std::string, std::less<>> PatchManager::GetPatchVersionNam
|
||||||
"Update",
|
"Update",
|
||||||
FormatTitleVersion(meta_ver.get(), TitleVersionFormat::ThreeElements));
|
FormatTitleVersion(meta_ver.get(), TitleVersionFormat::ThreeElements));
|
||||||
}
|
}
|
||||||
|
} else if (update_raw != nullptr) {
|
||||||
|
out.insert_or_assign("Update", "PACKED");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -46,11 +46,13 @@ public:
|
||||||
// - Game Updates
|
// - Game Updates
|
||||||
// - LayeredFS
|
// - LayeredFS
|
||||||
VirtualFile PatchRomFS(VirtualFile base, u64 ivfc_offset,
|
VirtualFile PatchRomFS(VirtualFile base, u64 ivfc_offset,
|
||||||
ContentRecordType type = ContentRecordType::Program) const;
|
ContentRecordType type = ContentRecordType::Program,
|
||||||
|
VirtualFile update_raw = nullptr) const;
|
||||||
|
|
||||||
// Returns a vector of pairs between patch names and patch versions.
|
// Returns a vector of pairs between patch names and patch versions.
|
||||||
// i.e. Update 3.2.2 will return {"Update", "3.2.2"}
|
// i.e. Update 3.2.2 will return {"Update", "3.2.2"}
|
||||||
std::map<std::string, std::string, std::less<>> GetPatchVersionNames() const;
|
std::map<std::string, std::string, std::less<>> GetPatchVersionNames(
|
||||||
|
VirtualFile update_raw = nullptr) const;
|
||||||
|
|
||||||
// Given title_id of the program, attempts to get the control data of the update and parse it,
|
// Given title_id of the program, attempts to get the control data of the update and parse it,
|
||||||
// falling back to the base control data.
|
// falling back to the base control data.
|
||||||
|
|
|
@ -30,12 +30,17 @@ RomFSFactory::RomFSFactory(Loader::AppLoader& app_loader) {
|
||||||
|
|
||||||
RomFSFactory::~RomFSFactory() = default;
|
RomFSFactory::~RomFSFactory() = default;
|
||||||
|
|
||||||
|
void RomFSFactory::SetPackedUpdate(VirtualFile update_raw) {
|
||||||
|
this->update_raw = std::move(update_raw);
|
||||||
|
}
|
||||||
|
|
||||||
ResultVal<VirtualFile> RomFSFactory::OpenCurrentProcess() {
|
ResultVal<VirtualFile> RomFSFactory::OpenCurrentProcess() {
|
||||||
if (!updatable)
|
if (!updatable)
|
||||||
return MakeResult<VirtualFile>(file);
|
return MakeResult<VirtualFile>(file);
|
||||||
|
|
||||||
const PatchManager patch_manager(Core::CurrentProcess()->GetTitleID());
|
const PatchManager patch_manager(Core::CurrentProcess()->GetTitleID());
|
||||||
return MakeResult<VirtualFile>(patch_manager.PatchRomFS(file, ivfc_offset));
|
return MakeResult<VirtualFile>(
|
||||||
|
patch_manager.PatchRomFS(file, ivfc_offset, ContentRecordType::Program, update_raw));
|
||||||
}
|
}
|
||||||
|
|
||||||
ResultVal<VirtualFile> RomFSFactory::Open(u64 title_id, StorageId storage, ContentRecordType type) {
|
ResultVal<VirtualFile> RomFSFactory::Open(u64 title_id, StorageId storage, ContentRecordType type) {
|
||||||
|
|
|
@ -32,11 +32,13 @@ public:
|
||||||
explicit RomFSFactory(Loader::AppLoader& app_loader);
|
explicit RomFSFactory(Loader::AppLoader& app_loader);
|
||||||
~RomFSFactory();
|
~RomFSFactory();
|
||||||
|
|
||||||
|
void SetPackedUpdate(VirtualFile update_raw);
|
||||||
ResultVal<VirtualFile> OpenCurrentProcess();
|
ResultVal<VirtualFile> OpenCurrentProcess();
|
||||||
ResultVal<VirtualFile> Open(u64 title_id, StorageId storage, ContentRecordType type);
|
ResultVal<VirtualFile> Open(u64 title_id, StorageId storage, ContentRecordType type);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
VirtualFile file;
|
VirtualFile file;
|
||||||
|
VirtualFile update_raw;
|
||||||
bool updatable;
|
bool updatable;
|
||||||
u64 ivfc_offset;
|
u64 ivfc_offset;
|
||||||
};
|
};
|
||||||
|
|
|
@ -259,9 +259,12 @@ void NSP::ReadNCAs(const std::vector<VirtualFile>& files) {
|
||||||
auto next_nca = std::make_shared<NCA>(next_file);
|
auto next_nca = std::make_shared<NCA>(next_file);
|
||||||
if (next_nca->GetType() == NCAContentType::Program)
|
if (next_nca->GetType() == NCAContentType::Program)
|
||||||
program_status[cnmt.GetTitleID()] = next_nca->GetStatus();
|
program_status[cnmt.GetTitleID()] = next_nca->GetStatus();
|
||||||
if (next_nca->GetStatus() == Loader::ResultStatus::Success)
|
if (next_nca->GetStatus() == Loader::ResultStatus::Success ||
|
||||||
|
(next_nca->GetStatus() == Loader::ResultStatus::ErrorMissingBKTRBaseRomFS &&
|
||||||
|
(cnmt.GetTitleID() & 0x800) != 0)) {
|
||||||
ncas_title[rec.type] = std::move(next_nca);
|
ncas_title[rec.type] = std::move(next_nca);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
@ -264,6 +264,15 @@ ResultCode RegisterBIS(std::unique_ptr<FileSys::BISFactory>&& factory) {
|
||||||
return RESULT_SUCCESS;
|
return RESULT_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SetPackedUpdate(FileSys::VirtualFile update_raw) {
|
||||||
|
LOG_TRACE(Service_FS, "Setting packed update for romfs");
|
||||||
|
|
||||||
|
if (romfs_factory == nullptr)
|
||||||
|
return;
|
||||||
|
|
||||||
|
romfs_factory->SetPackedUpdate(std::move(update_raw));
|
||||||
|
}
|
||||||
|
|
||||||
ResultVal<FileSys::VirtualFile> OpenRomFSCurrentProcess() {
|
ResultVal<FileSys::VirtualFile> OpenRomFSCurrentProcess() {
|
||||||
LOG_TRACE(Service_FS, "Opening RomFS for current process");
|
LOG_TRACE(Service_FS, "Opening RomFS for current process");
|
||||||
|
|
||||||
|
|
|
@ -39,6 +39,7 @@ ResultCode RegisterSaveData(std::unique_ptr<FileSys::SaveDataFactory>&& factory)
|
||||||
ResultCode RegisterSDMC(std::unique_ptr<FileSys::SDMCFactory>&& factory);
|
ResultCode RegisterSDMC(std::unique_ptr<FileSys::SDMCFactory>&& factory);
|
||||||
ResultCode RegisterBIS(std::unique_ptr<FileSys::BISFactory>&& factory);
|
ResultCode RegisterBIS(std::unique_ptr<FileSys::BISFactory>&& factory);
|
||||||
|
|
||||||
|
void SetPackedUpdate(FileSys::VirtualFile update_raw);
|
||||||
ResultVal<FileSys::VirtualFile> OpenRomFSCurrentProcess();
|
ResultVal<FileSys::VirtualFile> OpenRomFSCurrentProcess();
|
||||||
ResultVal<FileSys::VirtualFile> OpenRomFS(u64 title_id, FileSys::StorageId storage_id,
|
ResultVal<FileSys::VirtualFile> OpenRomFS(u64 title_id, FileSys::StorageId storage_id,
|
||||||
FileSys::ContentRecordType type);
|
FileSys::ContentRecordType type);
|
||||||
|
|
|
@ -93,7 +93,7 @@ std::string GetFileTypeString(FileType type) {
|
||||||
return "unknown";
|
return "unknown";
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr std::array<const char*, 58> RESULT_MESSAGES{
|
constexpr std::array<const char*, 59> RESULT_MESSAGES{
|
||||||
"The operation completed successfully.",
|
"The operation completed successfully.",
|
||||||
"The loader requested to load is already loaded.",
|
"The loader requested to load is already loaded.",
|
||||||
"The operation is not implemented.",
|
"The operation is not implemented.",
|
||||||
|
@ -152,6 +152,7 @@ constexpr std::array<const char*, 58> RESULT_MESSAGES{
|
||||||
"The BKTR-type NCA has a bad Relocation bucket.",
|
"The BKTR-type NCA has a bad Relocation bucket.",
|
||||||
"The BKTR-type NCA has a bad Subsection bucket.",
|
"The BKTR-type NCA has a bad Subsection bucket.",
|
||||||
"The BKTR-type NCA is missing the base RomFS.",
|
"The BKTR-type NCA is missing the base RomFS.",
|
||||||
|
"The NSP or XCI does not contain an update in addition to the base game.",
|
||||||
};
|
};
|
||||||
|
|
||||||
std::ostream& operator<<(std::ostream& os, ResultStatus status) {
|
std::ostream& operator<<(std::ostream& os, ResultStatus status) {
|
||||||
|
|
|
@ -114,6 +114,7 @@ enum class ResultStatus : u16 {
|
||||||
ErrorBadRelocationBuckets,
|
ErrorBadRelocationBuckets,
|
||||||
ErrorBadSubsectionBuckets,
|
ErrorBadSubsectionBuckets,
|
||||||
ErrorMissingBKTRBaseRomFS,
|
ErrorMissingBKTRBaseRomFS,
|
||||||
|
ErrorNoPackedUpdate,
|
||||||
};
|
};
|
||||||
|
|
||||||
std::ostream& operator<<(std::ostream& os, ResultStatus status);
|
std::ostream& operator<<(std::ostream& os, ResultStatus status);
|
||||||
|
@ -196,10 +197,19 @@ public:
|
||||||
/**
|
/**
|
||||||
* Get the RomFS of the application
|
* Get the RomFS of the application
|
||||||
* Since the RomFS can be huge, we return a file reference instead of copying to a buffer
|
* Since the RomFS can be huge, we return a file reference instead of copying to a buffer
|
||||||
* @param dir The directory containing the RomFS
|
* @param file The directory containing the RomFS
|
||||||
* @return ResultStatus result of function
|
* @return ResultStatus result of function
|
||||||
*/
|
*/
|
||||||
virtual ResultStatus ReadRomFS(FileSys::VirtualFile& dir) {
|
virtual ResultStatus ReadRomFS(FileSys::VirtualFile& file) {
|
||||||
|
return ResultStatus::ErrorNotImplemented;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the raw update of the application, should it come packed with one
|
||||||
|
* @param file The raw update NCA file (Program-type
|
||||||
|
* @return ResultStatus result of function
|
||||||
|
*/
|
||||||
|
virtual ResultStatus ReadUpdateRaw(FileSys::VirtualFile& file) {
|
||||||
return ResultStatus::ErrorNotImplemented;
|
return ResultStatus::ErrorNotImplemented;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -72,6 +72,10 @@ ResultStatus AppLoader_NAX::ReadRomFS(FileSys::VirtualFile& dir) {
|
||||||
return nca_loader->ReadRomFS(dir);
|
return nca_loader->ReadRomFS(dir);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
u64 AppLoader_NAX::ReadRomFSIVFCOffset() const {
|
||||||
|
return nca_loader->ReadRomFSIVFCOffset();
|
||||||
|
}
|
||||||
|
|
||||||
ResultStatus AppLoader_NAX::ReadProgramId(u64& out_program_id) {
|
ResultStatus AppLoader_NAX::ReadProgramId(u64& out_program_id) {
|
||||||
return nca_loader->ReadProgramId(out_program_id);
|
return nca_loader->ReadProgramId(out_program_id);
|
||||||
}
|
}
|
||||||
|
|
|
@ -36,6 +36,7 @@ public:
|
||||||
ResultStatus Load(Kernel::Process& process) override;
|
ResultStatus Load(Kernel::Process& process) override;
|
||||||
|
|
||||||
ResultStatus ReadRomFS(FileSys::VirtualFile& dir) override;
|
ResultStatus ReadRomFS(FileSys::VirtualFile& dir) override;
|
||||||
|
u64 ReadRomFSIVFCOffset() const override;
|
||||||
ResultStatus ReadProgramId(u64& out_program_id) override;
|
ResultStatus ReadProgramId(u64& out_program_id) override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
|
@ -10,8 +10,10 @@
|
||||||
#include "core/file_sys/control_metadata.h"
|
#include "core/file_sys/control_metadata.h"
|
||||||
#include "core/file_sys/nca_metadata.h"
|
#include "core/file_sys/nca_metadata.h"
|
||||||
#include "core/file_sys/patch_manager.h"
|
#include "core/file_sys/patch_manager.h"
|
||||||
|
#include "core/file_sys/registered_cache.h"
|
||||||
#include "core/file_sys/submission_package.h"
|
#include "core/file_sys/submission_package.h"
|
||||||
#include "core/hle/kernel/process.h"
|
#include "core/hle/kernel/process.h"
|
||||||
|
#include "core/hle/service/filesystem/filesystem.h"
|
||||||
#include "core/loader/deconstructed_rom_directory.h"
|
#include "core/loader/deconstructed_rom_directory.h"
|
||||||
#include "core/loader/nca.h"
|
#include "core/loader/nca.h"
|
||||||
#include "core/loader/nsp.h"
|
#include "core/loader/nsp.h"
|
||||||
|
@ -91,13 +93,39 @@ ResultStatus AppLoader_NSP::Load(Kernel::Process& process) {
|
||||||
if (result != ResultStatus::Success)
|
if (result != ResultStatus::Success)
|
||||||
return result;
|
return result;
|
||||||
|
|
||||||
|
FileSys::VirtualFile update_raw;
|
||||||
|
if (ReadUpdateRaw(update_raw) == ResultStatus::Success && update_raw != nullptr)
|
||||||
|
Service::FileSystem::SetPackedUpdate(std::move(update_raw));
|
||||||
|
|
||||||
is_loaded = true;
|
is_loaded = true;
|
||||||
|
|
||||||
return ResultStatus::Success;
|
return ResultStatus::Success;
|
||||||
}
|
}
|
||||||
|
|
||||||
ResultStatus AppLoader_NSP::ReadRomFS(FileSys::VirtualFile& dir) {
|
ResultStatus AppLoader_NSP::ReadRomFS(FileSys::VirtualFile& file) {
|
||||||
return secondary_loader->ReadRomFS(dir);
|
return secondary_loader->ReadRomFS(file);
|
||||||
|
}
|
||||||
|
|
||||||
|
u64 AppLoader_NSP::ReadRomFSIVFCOffset() const {
|
||||||
|
return secondary_loader->ReadRomFSIVFCOffset();
|
||||||
|
}
|
||||||
|
|
||||||
|
ResultStatus AppLoader_NSP::ReadUpdateRaw(FileSys::VirtualFile& file) {
|
||||||
|
if (nsp->IsExtractedType())
|
||||||
|
return ResultStatus::ErrorNoPackedUpdate;
|
||||||
|
|
||||||
|
const auto read =
|
||||||
|
nsp->GetNCAFile(FileSys::GetUpdateTitleID(title_id), FileSys::ContentRecordType::Program);
|
||||||
|
|
||||||
|
if (read == nullptr)
|
||||||
|
return ResultStatus::ErrorNoPackedUpdate;
|
||||||
|
const auto nca_test = std::make_shared<FileSys::NCA>(read);
|
||||||
|
|
||||||
|
if (nca_test->GetStatus() != ResultStatus::ErrorMissingBKTRBaseRomFS)
|
||||||
|
return nca_test->GetStatus();
|
||||||
|
|
||||||
|
file = read;
|
||||||
|
return ResultStatus::Success;
|
||||||
}
|
}
|
||||||
|
|
||||||
ResultStatus AppLoader_NSP::ReadProgramId(u64& out_program_id) {
|
ResultStatus AppLoader_NSP::ReadProgramId(u64& out_program_id) {
|
||||||
|
|
|
@ -37,7 +37,9 @@ public:
|
||||||
|
|
||||||
ResultStatus Load(Kernel::Process& process) override;
|
ResultStatus Load(Kernel::Process& process) override;
|
||||||
|
|
||||||
ResultStatus ReadRomFS(FileSys::VirtualFile& dir) override;
|
ResultStatus ReadRomFS(FileSys::VirtualFile& file) override;
|
||||||
|
u64 ReadRomFSIVFCOffset() const override;
|
||||||
|
ResultStatus ReadUpdateRaw(FileSys::VirtualFile& file) override;
|
||||||
ResultStatus ReadProgramId(u64& out_program_id) override;
|
ResultStatus ReadProgramId(u64& out_program_id) override;
|
||||||
ResultStatus ReadIcon(std::vector<u8>& buffer) override;
|
ResultStatus ReadIcon(std::vector<u8>& buffer) override;
|
||||||
ResultStatus ReadTitle(std::string& title) override;
|
ResultStatus ReadTitle(std::string& title) override;
|
||||||
|
|
|
@ -9,7 +9,11 @@
|
||||||
#include "core/file_sys/content_archive.h"
|
#include "core/file_sys/content_archive.h"
|
||||||
#include "core/file_sys/control_metadata.h"
|
#include "core/file_sys/control_metadata.h"
|
||||||
#include "core/file_sys/patch_manager.h"
|
#include "core/file_sys/patch_manager.h"
|
||||||
|
#include "core/file_sys/registered_cache.h"
|
||||||
|
#include "core/file_sys/romfs.h"
|
||||||
|
#include "core/file_sys/submission_package.h"
|
||||||
#include "core/hle/kernel/process.h"
|
#include "core/hle/kernel/process.h"
|
||||||
|
#include "core/hle/service/filesystem/filesystem.h"
|
||||||
#include "core/loader/nca.h"
|
#include "core/loader/nca.h"
|
||||||
#include "core/loader/xci.h"
|
#include "core/loader/xci.h"
|
||||||
|
|
||||||
|
@ -63,13 +67,41 @@ ResultStatus AppLoader_XCI::Load(Kernel::Process& process) {
|
||||||
if (result != ResultStatus::Success)
|
if (result != ResultStatus::Success)
|
||||||
return result;
|
return result;
|
||||||
|
|
||||||
|
FileSys::VirtualFile update_raw;
|
||||||
|
if (ReadUpdateRaw(update_raw) == ResultStatus::Success && update_raw != nullptr)
|
||||||
|
Service::FileSystem::SetPackedUpdate(std::move(update_raw));
|
||||||
|
|
||||||
is_loaded = true;
|
is_loaded = true;
|
||||||
|
|
||||||
return ResultStatus::Success;
|
return ResultStatus::Success;
|
||||||
}
|
}
|
||||||
|
|
||||||
ResultStatus AppLoader_XCI::ReadRomFS(FileSys::VirtualFile& dir) {
|
ResultStatus AppLoader_XCI::ReadRomFS(FileSys::VirtualFile& file) {
|
||||||
return nca_loader->ReadRomFS(dir);
|
return nca_loader->ReadRomFS(file);
|
||||||
|
}
|
||||||
|
|
||||||
|
u64 AppLoader_XCI::ReadRomFSIVFCOffset() const {
|
||||||
|
return nca_loader->ReadRomFSIVFCOffset();
|
||||||
|
}
|
||||||
|
|
||||||
|
ResultStatus AppLoader_XCI::ReadUpdateRaw(FileSys::VirtualFile& file) {
|
||||||
|
u64 program_id{};
|
||||||
|
nca_loader->ReadProgramId(program_id);
|
||||||
|
if (program_id == 0)
|
||||||
|
return ResultStatus::ErrorXCIMissingProgramNCA;
|
||||||
|
|
||||||
|
const auto read = xci->GetSecurePartitionNSP()->GetNCAFile(
|
||||||
|
FileSys::GetUpdateTitleID(program_id), FileSys::ContentRecordType::Program);
|
||||||
|
|
||||||
|
if (read == nullptr)
|
||||||
|
return ResultStatus::ErrorNoPackedUpdate;
|
||||||
|
const auto nca_test = std::make_shared<FileSys::NCA>(read);
|
||||||
|
|
||||||
|
if (nca_test->GetStatus() != ResultStatus::ErrorMissingBKTRBaseRomFS)
|
||||||
|
return nca_test->GetStatus();
|
||||||
|
|
||||||
|
file = read;
|
||||||
|
return ResultStatus::Success;
|
||||||
}
|
}
|
||||||
|
|
||||||
ResultStatus AppLoader_XCI::ReadProgramId(u64& out_program_id) {
|
ResultStatus AppLoader_XCI::ReadProgramId(u64& out_program_id) {
|
||||||
|
|
|
@ -37,7 +37,9 @@ public:
|
||||||
|
|
||||||
ResultStatus Load(Kernel::Process& process) override;
|
ResultStatus Load(Kernel::Process& process) override;
|
||||||
|
|
||||||
ResultStatus ReadRomFS(FileSys::VirtualFile& dir) override;
|
ResultStatus ReadRomFS(FileSys::VirtualFile& file) override;
|
||||||
|
u64 ReadRomFSIVFCOffset() const override;
|
||||||
|
ResultStatus ReadUpdateRaw(FileSys::VirtualFile& file) override;
|
||||||
ResultStatus ReadProgramId(u64& out_program_id) override;
|
ResultStatus ReadProgramId(u64& out_program_id) override;
|
||||||
ResultStatus ReadIcon(std::vector<u8>& buffer) override;
|
ResultStatus ReadIcon(std::vector<u8>& buffer) override;
|
||||||
ResultStatus ReadTitle(std::string& title) override;
|
ResultStatus ReadTitle(std::string& title) override;
|
||||||
|
|
|
@ -57,16 +57,25 @@ QString FormatGameName(const std::string& physical_name) {
|
||||||
return physical_name_as_qstring;
|
return physical_name_as_qstring;
|
||||||
}
|
}
|
||||||
|
|
||||||
QString FormatPatchNameVersions(const FileSys::PatchManager& patch_manager, bool updatable = true) {
|
QString FormatPatchNameVersions(const FileSys::PatchManager& patch_manager,
|
||||||
|
Loader::AppLoader& loader, bool updatable = true) {
|
||||||
QString out;
|
QString out;
|
||||||
for (const auto& kv : patch_manager.GetPatchVersionNames()) {
|
FileSys::VirtualFile update_raw;
|
||||||
|
loader.ReadUpdateRaw(update_raw);
|
||||||
|
for (const auto& kv : patch_manager.GetPatchVersionNames(update_raw)) {
|
||||||
if (!updatable && kv.first == "Update")
|
if (!updatable && kv.first == "Update")
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (kv.second.empty()) {
|
if (kv.second.empty()) {
|
||||||
out.append(fmt::format("{}\n", kv.first).c_str());
|
out.append(fmt::format("{}\n", kv.first).c_str());
|
||||||
} else {
|
} else {
|
||||||
out.append(fmt::format("{} ({})\n", kv.first, kv.second).c_str());
|
auto ver = kv.second;
|
||||||
|
|
||||||
|
// Display container name for packed updates
|
||||||
|
if (ver == "PACKED" && kv.first == "Update")
|
||||||
|
ver = Loader::GetFileTypeString(loader.GetFileType());
|
||||||
|
|
||||||
|
out.append(fmt::format("{} ({})\n", kv.first, ver).c_str());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -116,7 +125,7 @@ void GameListWorker::AddInstalledTitlesToGameList() {
|
||||||
QString::fromStdString(Loader::GetFileTypeString(loader->GetFileType())),
|
QString::fromStdString(Loader::GetFileTypeString(loader->GetFileType())),
|
||||||
program_id),
|
program_id),
|
||||||
new GameListItemCompat(compatibility),
|
new GameListItemCompat(compatibility),
|
||||||
new GameListItem(FormatPatchNameVersions(patch)),
|
new GameListItem(FormatPatchNameVersions(patch, *loader)),
|
||||||
new GameListItem(
|
new GameListItem(
|
||||||
QString::fromStdString(Loader::GetFileTypeString(loader->GetFileType()))),
|
QString::fromStdString(Loader::GetFileTypeString(loader->GetFileType()))),
|
||||||
new GameListItemSize(file->GetSize()),
|
new GameListItemSize(file->GetSize()),
|
||||||
|
@ -206,7 +215,8 @@ void GameListWorker::AddFstEntriesToGameList(const std::string& dir_path, unsign
|
||||||
QString::fromStdString(Loader::GetFileTypeString(loader->GetFileType())),
|
QString::fromStdString(Loader::GetFileTypeString(loader->GetFileType())),
|
||||||
program_id),
|
program_id),
|
||||||
new GameListItemCompat(compatibility),
|
new GameListItemCompat(compatibility),
|
||||||
new GameListItem(FormatPatchNameVersions(patch, loader->IsRomFSUpdatable())),
|
new GameListItem(
|
||||||
|
FormatPatchNameVersions(patch, *loader, loader->IsRomFSUpdatable())),
|
||||||
new GameListItem(
|
new GameListItem(
|
||||||
QString::fromStdString(Loader::GetFileTypeString(loader->GetFileType()))),
|
QString::fromStdString(Loader::GetFileTypeString(loader->GetFileType()))),
|
||||||
new GameListItemSize(FileUtil::GetSize(physical_name)),
|
new GameListItemSize(FileUtil::GetSize(physical_name)),
|
||||||
|
|
Loading…
Reference in New Issue