summaryrefslogtreecommitdiffstats
path: root/src/core/file_sys/partition_filesystem.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/core/file_sys/partition_filesystem.cpp')
-rw-r--r--src/core/file_sys/partition_filesystem.cpp136
1 files changed, 62 insertions, 74 deletions
diff --git a/src/core/file_sys/partition_filesystem.cpp b/src/core/file_sys/partition_filesystem.cpp
index 46d438aca..15b1fb946 100644
--- a/src/core/file_sys/partition_filesystem.cpp
+++ b/src/core/file_sys/partition_filesystem.cpp
@@ -6,29 +6,30 @@
#include "common/file_util.h"
#include "common/logging/log.h"
#include "core/file_sys/partition_filesystem.h"
+#include "core/file_sys/vfs_offset.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;
-
+PartitionFilesystem::PartitionFilesystem(std::shared_ptr<VfsFile> file) {
// At least be as large as the header
- if (file.GetSize() < sizeof(Header))
- return Loader::ResultStatus::Error;
+ if (file->GetSize() < sizeof(Header)) {
+ status = Loader::ResultStatus::Error;
+ return;
+ }
- 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 (sizeof(Header) != file->ReadObject(&pfs_header)) {
+ status = Loader::ResultStatus::Error;
+ return;
+ }
if (pfs_header.magic != Common::MakeMagic('H', 'F', 'S', '0') &&
pfs_header.magic != Common::MakeMagic('P', 'F', 'S', '0')) {
- return Loader::ResultStatus::ErrorInvalidFormat;
+ status = Loader::ResultStatus::ErrorInvalidFormat;
+ return;
}
bool is_hfs = pfs_header.magic == Common::MakeMagic('H', 'F', 'S', '0');
@@ -38,99 +39,86 @@ Loader::ResultStatus PartitionFilesystem::Load(const std::string& file_path, siz
sizeof(Header) + (pfs_header.num_entries * entry_size) + pfs_header.strtab_size;
// Actually read in now...
- file.Seek(offset, SEEK_SET);
- std::vector<u8> file_data(metadata_size);
-
- if (!file.ReadBytes(file_data.data(), metadata_size))
- return Loader::ResultStatus::Error;
+ std::vector<u8> file_data = file->ReadBytes(metadata_size);
- Loader::ResultStatus result = Load(file_data);
- if (result != Loader::ResultStatus::Success)
- LOG_ERROR(Service_FS, "Failed to load PFS from file {}!", file_path);
-
- return result;
-}
+ if (file_data.size() != metadata_size) {
+ status = Loader::ResultStatus::Error;
+ return;
+ }
-Loader::ResultStatus PartitionFilesystem::Load(const std::vector<u8>& file_data, size_t offset) {
- size_t total_size = file_data.size() - offset;
- if (total_size < sizeof(Header))
- return Loader::ResultStatus::Error;
+ size_t total_size = file_data.size();
+ if (total_size < sizeof(Header)) {
+ status = Loader::ResultStatus::Error;
+ return;
+ }
- memcpy(&pfs_header, &file_data[offset], sizeof(Header));
+ memcpy(&pfs_header, file_data.data(), 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;
+ status = Loader::ResultStatus::ErrorInvalidFormat;
+ return;
}
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 entries_offset = sizeof(Header);
size_t strtab_offset = entries_offset + (pfs_header.num_entries * entry_size);
+ content_offset = strtab_offset + pfs_header.strtab_size;
for (u16 i = 0; i < pfs_header.num_entries; i++) {
- FileEntry entry;
+ FSEntry entry;
- memcpy(&entry.fs_entry, &file_data[entries_offset + (i * entry_size)], sizeof(FSEntry));
- entry.name = std::string(reinterpret_cast<const char*>(
- &file_data[strtab_offset + entry.fs_entry.strtab_offset]));
- pfs_entries.push_back(std::move(entry));
- }
+ memcpy(&entry, &file_data[entries_offset + (i * entry_size)], sizeof(FSEntry));
+ std::string name(
+ reinterpret_cast<const char*>(&file_data[strtab_offset + entry.strtab_offset]));
- content_offset = strtab_offset + pfs_header.strtab_size;
+ pfs_files.emplace_back(
+ std::make_shared<OffsetVfsFile>(file, entry.size, content_offset + entry.offset, name));
+ }
- return Loader::ResultStatus::Success;
+ status = Loader::ResultStatus::Success;
}
-u32 PartitionFilesystem::GetNumEntries() const {
- return pfs_header.num_entries;
+Loader::ResultStatus PartitionFilesystem::GetStatus() const {
+ return status;
}
-u64 PartitionFilesystem::GetEntryOffset(u32 index) const {
- if (index > GetNumEntries())
- return 0;
-
- return content_offset + pfs_entries[index].fs_entry.offset;
+std::vector<std::shared_ptr<VfsFile>> PartitionFilesystem::GetFiles() const {
+ return pfs_files;
}
-u64 PartitionFilesystem::GetEntrySize(u32 index) const {
- if (index > GetNumEntries())
- return 0;
-
- return pfs_entries[index].fs_entry.size;
+std::vector<std::shared_ptr<VfsDirectory>> PartitionFilesystem::GetSubdirectories() const {
+ return {};
}
-std::string PartitionFilesystem::GetEntryName(u32 index) const {
- if (index > GetNumEntries())
- return "";
+std::string PartitionFilesystem::GetName() const {
+ return is_hfs ? "HFS0" : "PFS0";
+}
- return pfs_entries[index].name;
+std::shared_ptr<VfsDirectory> PartitionFilesystem::GetParentDirectory() const {
+ // TODO(DarkLordZach): Add support for nested containers.
+ return nullptr;
}
-u64 PartitionFilesystem::GetFileOffset(const std::string& name) const {
+void PartitionFilesystem::PrintDebugInfo() const {
+ LOG_DEBUG(Service_FS, "Magic: {:.4}", pfs_header.magic);
+ LOG_DEBUG(Service_FS, "Files: {}", pfs_header.num_entries);
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;
+ LOG_DEBUG(Service_FS, " > File {}: {} (0x{:X} bytes, at 0x{:X})", i,
+ pfs_files[i]->GetName(), pfs_files[i]->GetSize(),
+ dynamic_cast<OffsetVfsFile*>(pfs_files[i].get())->GetOffset());
}
-
- 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;
- }
+bool PartitionFilesystem::ReplaceFileWithSubdirectory(VirtualFile file, VirtualDir dir) {
+ auto iter = std::find(pfs_files.begin(), pfs_files.end(), file);
+ if (iter == pfs_files.end())
+ return false;
- return 0;
-}
+ pfs_files[iter - pfs_files.begin()] = pfs_files.back();
+ pfs_files.pop_back();
-void PartitionFilesystem::Print() const {
- LOG_DEBUG(Service_FS, "Magic: {}", pfs_header.magic);
- LOG_DEBUG(Service_FS, "Files: {}", pfs_header.num_entries);
- for (u32 i = 0; i < pfs_header.num_entries; i++) {
- LOG_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));
- }
+ pfs_dirs.emplace_back(dir);
+
+ return true;
}
} // namespace FileSys