// Copyright 2018 yuzu emulator team // Licensed under GPLv2 or any later version // Refer to the license.txt file included. #include #include "common/file_util.h" #include "common/logging/log.h" #include "core/file_sys/partition_filesystem.h" #include "core/loader/loader.h" namespace FileSys { Loader::ResultStatus PartitionFilesystem::Load(const std::string& file_path, size_t offset) { FileUtil::IOFile file(file_path, "rb"); if (!file.IsOpen()) return Loader::ResultStatus::Error; // At least be as large as the header if (file.GetSize() < sizeof(Header)) return Loader::ResultStatus::Error; file.Seek(offset, SEEK_SET); // For cartridges, HFSs can get very large, so we need to calculate the size up to // the actual content itself instead of just blindly reading in the entire file. Header pfs_header; if (!file.ReadBytes(&pfs_header, sizeof(Header))) return Loader::ResultStatus::Error; if (pfs_header.magic != Common::MakeMagic('H', 'F', 'S', '0') && pfs_header.magic != Common::MakeMagic('P', 'F', 'S', '0')) { return Loader::ResultStatus::ErrorInvalidFormat; } bool is_hfs = pfs_header.magic == Common::MakeMagic('H', 'F', 'S', '0'); size_t entry_size = is_hfs ? sizeof(HFSEntry) : sizeof(PFSEntry); size_t metadata_size = sizeof(Header) + (pfs_header.num_entries * entry_size) + pfs_header.strtab_size; // Actually read in now... file.Seek(offset, SEEK_SET); std::vector file_data(metadata_size); if (!file.ReadBytes(file_data.data(), metadata_size)) return Loader::ResultStatus::Error; Loader::ResultStatus result = Load(file_data); if (result != Loader::ResultStatus::Success) NGLOG_ERROR(Service_FS, "Failed to load PFS from file {}!", file_path); return result; } Loader::ResultStatus PartitionFilesystem::Load(const std::vector& file_data, size_t offset) { size_t total_size = file_data.size() - offset; if (total_size < sizeof(Header)) return Loader::ResultStatus::Error; memcpy(&pfs_header, &file_data[offset], sizeof(Header)); if (pfs_header.magic != Common::MakeMagic('H', 'F', 'S', '0') && pfs_header.magic != Common::MakeMagic('P', 'F', 'S', '0')) { return Loader::ResultStatus::ErrorInvalidFormat; } is_hfs = pfs_header.magic == Common::MakeMagic('H', 'F', 'S', '0'); size_t entries_offset = offset + sizeof(Header); size_t entry_size = is_hfs ? sizeof(HFSEntry) : sizeof(PFSEntry); size_t strtab_offset = entries_offset + (pfs_header.num_entries * entry_size); for (u16 i = 0; i < pfs_header.num_entries; i++) { FileEntry entry; memcpy(&entry.fs_entry, &file_data[entries_offset + (i * entry_size)], sizeof(FSEntry)); entry.name = std::string(reinterpret_cast( &file_data[strtab_offset + entry.fs_entry.strtab_offset])); pfs_entries.push_back(std::move(entry)); } content_offset = strtab_offset + pfs_header.strtab_size; return Loader::ResultStatus::Success; } u32 PartitionFilesystem::GetNumEntries() const { return pfs_header.num_entries; } u64 PartitionFilesystem::GetEntryOffset(u32 index) const { if (index > GetNumEntries()) return 0; return content_offset + pfs_entries[index].fs_entry.offset; } u64 PartitionFilesystem::GetEntrySize(u32 index) const { if (index > GetNumEntries()) return 0; return pfs_entries[index].fs_entry.size; } std::string PartitionFilesystem::GetEntryName(u32 index) const { if (index > GetNumEntries()) return ""; return pfs_entries[index].name; } u64 PartitionFilesystem::GetFileOffset(const std::string& name) const { for (u32 i = 0; i < pfs_header.num_entries; i++) { if (pfs_entries[i].name == name) return content_offset + pfs_entries[i].fs_entry.offset; } return 0; } u64 PartitionFilesystem::GetFileSize(const std::string& name) const { for (u32 i = 0; i < pfs_header.num_entries; i++) { if (pfs_entries[i].name == name) return pfs_entries[i].fs_entry.size; } return 0; } void PartitionFilesystem::Print() const { NGLOG_DEBUG(Service_FS, "Magic: {}", pfs_header.magic); NGLOG_DEBUG(Service_FS, "Files: {}", pfs_header.num_entries); for (u32 i = 0; i < pfs_header.num_entries; i++) { NGLOG_DEBUG(Service_FS, " > File {}: {} (0x{:X} bytes, at 0x{:X})", i, pfs_entries[i].name.c_str(), pfs_entries[i].fs_entry.size, GetFileOffset(pfs_entries[i].name)); } } } // namespace FileSys