forked from eden-emu/eden
		
	Virtual Filesystem (#597)
* Add VfsFile and VfsDirectory classes * Finish abstract Vfs classes * Implement RealVfsFile (computer fs backend) * Finish RealVfsFile and RealVfsDirectory * Finished OffsetVfsFile * More changes * Fix import paths * Major refactor * Remove double const * Use experimental/filesystem or filesystem depending on compiler * Port partition_filesystem * More changes * More Overhaul * FSP_SRV fixes * Fixes and testing * Try to get filesystem to compile * Filesystem on linux * Remove std::filesystem and document/test * Compile fixes * Missing include * Bug fixes * Fixes * Rename v_file and v_dir * clang-format fix * Rename NGLOG_* to LOG_* * Most review changes * Fix TODO * Guess 'main' to be Directory by filename
This commit is contained in:
		
							parent
							
								
									0ea724e06b
								
							
						
					
					
						commit
						12e9522b32
					
				
					 45 changed files with 1796 additions and 1688 deletions
				
			
		|  | @ -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); | ||||
|     std::vector<u8> file_data = file->ReadBytes(metadata_size); | ||||
| 
 | ||||
|     if (!file.ReadBytes(file_data.data(), metadata_size)) | ||||
|         return Loader::ResultStatus::Error; | ||||
|     if (file_data.size() != metadata_size) { | ||||
|         status = Loader::ResultStatus::Error; | ||||
|         return; | ||||
|     } | ||||
| 
 | ||||
|     Loader::ResultStatus result = Load(file_data); | ||||
|     if (result != Loader::ResultStatus::Success) | ||||
|         LOG_ERROR(Service_FS, "Failed to load PFS from file {}!", file_path); | ||||
|     size_t total_size = file_data.size(); | ||||
|     if (total_size < sizeof(Header)) { | ||||
|         status = Loader::ResultStatus::Error; | ||||
|         return; | ||||
|     } | ||||
| 
 | ||||
|     return result; | ||||
| } | ||||
| 
 | ||||
| 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; | ||||
| 
 | ||||
|     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); | ||||
|     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<const char*>( | ||||
|             &file_data[strtab_offset + entry.fs_entry.strtab_offset])); | ||||
|         pfs_entries.push_back(std::move(entry)); | ||||
|     } | ||||
| 
 | ||||
|     content_offset = strtab_offset + pfs_header.strtab_size; | ||||
|     for (u16 i = 0; i < pfs_header.num_entries; i++) { | ||||
|         FSEntry entry; | ||||
| 
 | ||||
|     return Loader::ResultStatus::Success; | ||||
| } | ||||
|         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])); | ||||
| 
 | ||||
| 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; | ||||
|         pfs_files.emplace_back( | ||||
|             std::make_shared<OffsetVfsFile>(file, entry.size, content_offset + entry.offset, name)); | ||||
|     } | ||||
| 
 | ||||
|     return 0; | ||||
|     status = Loader::ResultStatus::Success; | ||||
| } | ||||
| 
 | ||||
| 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; | ||||
| Loader::ResultStatus PartitionFilesystem::GetStatus() const { | ||||
|     return status; | ||||
| } | ||||
| 
 | ||||
| void PartitionFilesystem::Print() const { | ||||
|     LOG_DEBUG(Service_FS, "Magic:                  {}", pfs_header.magic); | ||||
| std::vector<std::shared_ptr<VfsFile>> PartitionFilesystem::GetFiles() const { | ||||
|     return pfs_files; | ||||
| } | ||||
| 
 | ||||
| std::vector<std::shared_ptr<VfsDirectory>> PartitionFilesystem::GetSubdirectories() const { | ||||
|     return {}; | ||||
| } | ||||
| 
 | ||||
| std::string PartitionFilesystem::GetName() const { | ||||
|     return is_hfs ? "HFS0" : "PFS0"; | ||||
| } | ||||
| 
 | ||||
| std::shared_ptr<VfsDirectory> PartitionFilesystem::GetParentDirectory() const { | ||||
|     // TODO(DarkLordZach): Add support for nested containers.
 | ||||
|     return nullptr; | ||||
| } | ||||
| 
 | ||||
| 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++) { | ||||
|         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_files[i]->GetName(), pfs_files[i]->GetSize(), | ||||
|                   dynamic_cast<OffsetVfsFile*>(pfs_files[i].get())->GetOffset()); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| 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; | ||||
| 
 | ||||
|     pfs_files[iter - pfs_files.begin()] = pfs_files.back(); | ||||
|     pfs_files.pop_back(); | ||||
| 
 | ||||
|     pfs_dirs.emplace_back(dir); | ||||
| 
 | ||||
|     return true; | ||||
| } | ||||
| } // namespace FileSys
 | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Zach Hilman
						Zach Hilman