12 changed files with 351 additions and 20 deletions
-
4src/core/CMakeLists.txt
-
19src/core/file_sys/content_archive.cpp
-
124src/core/file_sys/romfs.cpp
-
35src/core/file_sys/romfs.h
-
23src/core/file_sys/vfs.cpp
-
20src/core/file_sys/vfs.h
-
7src/core/file_sys/vfs_offset.cpp
-
3src/core/file_sys/vfs_offset.h
-
6src/core/file_sys/vfs_real.cpp
-
3src/core/file_sys/vfs_real.h
-
83src/core/file_sys/vfs_vector.cpp
-
44src/core/file_sys/vfs_vector.h
@ -0,0 +1,124 @@ |
|||||
|
// Copyright 2018 yuzu emulator team
|
||||
|
// Licensed under GPLv2 or any later version
|
||||
|
// Refer to the license.txt file included.
|
||||
|
|
||||
|
#include "common/common_types.h"
|
||||
|
#include "common/swap.h"
|
||||
|
#include "core/file_sys/romfs.h"
|
||||
|
#include "core/file_sys/vfs.h"
|
||||
|
#include "core/file_sys/vfs_offset.h"
|
||||
|
#include "core/file_sys/vfs_vector.h"
|
||||
|
|
||||
|
namespace FileSys { |
||||
|
|
||||
|
constexpr u32 ROMFS_ENTRY_EMPTY = 0xFFFFFFFF; |
||||
|
|
||||
|
struct TableLocation { |
||||
|
u64_le offset; |
||||
|
u64_le size; |
||||
|
}; |
||||
|
static_assert(sizeof(TableLocation) == 0x10, "TableLocation has incorrect size."); |
||||
|
|
||||
|
struct RomFSHeader { |
||||
|
u64_le header_size; |
||||
|
TableLocation directory_hash; |
||||
|
TableLocation directory_meta; |
||||
|
TableLocation file_hash; |
||||
|
TableLocation file_meta; |
||||
|
u64_le data_offset; |
||||
|
}; |
||||
|
static_assert(sizeof(RomFSHeader) == 0x50, "RomFSHeader has incorrect size."); |
||||
|
|
||||
|
struct DirectoryEntry { |
||||
|
u32_le sibling; |
||||
|
u32_le child_dir; |
||||
|
u32_le child_file; |
||||
|
u32_le hash; |
||||
|
u32_le name_length; |
||||
|
}; |
||||
|
static_assert(sizeof(DirectoryEntry) == 0x14, "DirectoryEntry has incorrect size."); |
||||
|
|
||||
|
struct FileEntry { |
||||
|
u32_le parent; |
||||
|
u32_le sibling; |
||||
|
u64_le offset; |
||||
|
u64_le size; |
||||
|
u32_le hash; |
||||
|
u32_le name_length; |
||||
|
}; |
||||
|
static_assert(sizeof(FileEntry) == 0x20, "FileEntry has incorrect size."); |
||||
|
|
||||
|
template <typename Entry> |
||||
|
static std::pair<Entry, std::string> GetEntry(const VirtualFile& file, size_t offset) { |
||||
|
Entry entry{}; |
||||
|
if (file->ReadObject(&entry, offset) != sizeof(Entry)) |
||||
|
return {}; |
||||
|
std::string string(entry.name_length, '\0'); |
||||
|
if (file->ReadArray(&string[0], string.size(), offset + sizeof(Entry)) != string.size()) |
||||
|
return {}; |
||||
|
return {entry, string}; |
||||
|
} |
||||
|
|
||||
|
void ProcessFile(VirtualFile file, size_t file_offset, size_t data_offset, u32 this_file_offset, |
||||
|
std::shared_ptr<VectorVfsDirectory> parent) { |
||||
|
while (true) { |
||||
|
auto entry = GetEntry<FileEntry>(file, file_offset + this_file_offset); |
||||
|
|
||||
|
parent->AddFile(std::make_shared<OffsetVfsFile>( |
||||
|
file, entry.first.size, entry.first.offset + data_offset, entry.second, parent)); |
||||
|
|
||||
|
if (entry.first.sibling == ROMFS_ENTRY_EMPTY) |
||||
|
break; |
||||
|
|
||||
|
this_file_offset = entry.first.sibling; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
void ProcessDirectory(VirtualFile file, size_t dir_offset, size_t file_offset, size_t data_offset, |
||||
|
u32 this_dir_offset, std::shared_ptr<VectorVfsDirectory> parent) { |
||||
|
while (true) { |
||||
|
auto entry = GetEntry<DirectoryEntry>(file, dir_offset + this_dir_offset); |
||||
|
auto current = std::make_shared<VectorVfsDirectory>( |
||||
|
std::vector<VirtualFile>{}, std::vector<VirtualDir>{}, parent, entry.second); |
||||
|
|
||||
|
if (entry.first.child_file != ROMFS_ENTRY_EMPTY) { |
||||
|
ProcessFile(file, file_offset, data_offset, entry.first.child_file, current); |
||||
|
} |
||||
|
|
||||
|
if (entry.first.child_dir != ROMFS_ENTRY_EMPTY) { |
||||
|
ProcessDirectory(file, dir_offset, file_offset, data_offset, entry.first.child_dir, |
||||
|
current); |
||||
|
} |
||||
|
|
||||
|
parent->AddDirectory(current); |
||||
|
if (entry.first.sibling == ROMFS_ENTRY_EMPTY) |
||||
|
break; |
||||
|
this_dir_offset = entry.first.sibling; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
VirtualDir ExtractRomFS(VirtualFile file) { |
||||
|
RomFSHeader header{}; |
||||
|
if (file->ReadObject(&header) != sizeof(RomFSHeader)) |
||||
|
return nullptr; |
||||
|
|
||||
|
if (header.header_size != sizeof(RomFSHeader)) |
||||
|
return nullptr; |
||||
|
|
||||
|
const u64 file_offset = header.file_meta.offset; |
||||
|
const u64 dir_offset = header.directory_meta.offset + 4; |
||||
|
|
||||
|
const auto root = |
||||
|
std::make_shared<VectorVfsDirectory>(std::vector<VirtualFile>{}, std::vector<VirtualDir>{}, |
||||
|
file->GetContainingDirectory(), file->GetName()); |
||||
|
|
||||
|
ProcessDirectory(file, dir_offset, file_offset, header.data_offset, 0, root); |
||||
|
|
||||
|
VirtualDir out = std::move(root); |
||||
|
|
||||
|
while (out->GetSubdirectory("") != nullptr) |
||||
|
out = out->GetSubdirectory(""); |
||||
|
|
||||
|
return out; |
||||
|
} |
||||
|
} // namespace FileSys
|
||||
@ -0,0 +1,35 @@ |
|||||
|
// Copyright 2018 yuzu emulator team |
||||
|
// Licensed under GPLv2 or any later version |
||||
|
// Refer to the license.txt file included. |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include <array> |
||||
|
#include "common/common_funcs.h" |
||||
|
#include "common/swap.h" |
||||
|
#include "core/file_sys/vfs.h" |
||||
|
|
||||
|
namespace FileSys { |
||||
|
|
||||
|
struct IVFCLevel { |
||||
|
u64_le offset; |
||||
|
u64_le size; |
||||
|
u32_le block_size; |
||||
|
u32_le reserved; |
||||
|
}; |
||||
|
static_assert(sizeof(IVFCLevel) == 0x18, "IVFCLevel has incorrect size."); |
||||
|
|
||||
|
struct IVFCHeader { |
||||
|
u32_le magic; |
||||
|
u32_le magic_number; |
||||
|
INSERT_PADDING_BYTES(8); |
||||
|
std::array<IVFCLevel, 6> levels; |
||||
|
INSERT_PADDING_BYTES(64); |
||||
|
}; |
||||
|
static_assert(sizeof(IVFCHeader) == 0xE0, "IVFCHeader has incorrect size."); |
||||
|
|
||||
|
// Converts a RomFS binary blob to VFS Filesystem |
||||
|
// Returns nullptr on failure |
||||
|
VirtualDir ExtractRomFS(VirtualFile file); |
||||
|
|
||||
|
} // namespace FileSys |
||||
@ -0,0 +1,83 @@ |
|||||
|
// Copyright 2018 yuzu emulator team
|
||||
|
// Licensed under GPLv2 or any later version
|
||||
|
// Refer to the license.txt file included.
|
||||
|
|
||||
|
#include <algorithm>
|
||||
|
#include "core/file_sys/vfs_vector.h"
|
||||
|
|
||||
|
namespace FileSys { |
||||
|
VectorVfsDirectory::VectorVfsDirectory(std::vector<VirtualFile> files_, |
||||
|
std::vector<VirtualDir> dirs_, VirtualDir parent_, |
||||
|
std::string name_) |
||||
|
: files(std::move(files_)), dirs(std::move(dirs_)), parent(std::move(parent_)), |
||||
|
name(std::move(name_)) {} |
||||
|
|
||||
|
std::vector<std::shared_ptr<VfsFile>> VectorVfsDirectory::GetFiles() const { |
||||
|
return files; |
||||
|
} |
||||
|
|
||||
|
std::vector<std::shared_ptr<VfsDirectory>> VectorVfsDirectory::GetSubdirectories() const { |
||||
|
return dirs; |
||||
|
} |
||||
|
|
||||
|
bool VectorVfsDirectory::IsWritable() const { |
||||
|
return false; |
||||
|
} |
||||
|
|
||||
|
bool VectorVfsDirectory::IsReadable() const { |
||||
|
return true; |
||||
|
} |
||||
|
|
||||
|
std::string VectorVfsDirectory::GetName() const { |
||||
|
return name; |
||||
|
} |
||||
|
std::shared_ptr<VfsDirectory> VectorVfsDirectory::GetParentDirectory() const { |
||||
|
return parent; |
||||
|
} |
||||
|
|
||||
|
template <typename T> |
||||
|
static bool FindAndRemoveVectorElement(std::vector<T>& vec, std::string_view name) { |
||||
|
auto iter = std::find_if(vec.begin(), vec.end(), [name](T e) { return e->GetName() == name; }); |
||||
|
if (iter == vec.end()) |
||||
|
return false; |
||||
|
auto old_size = vec.size(); |
||||
|
vec.erase(iter); |
||||
|
return true; |
||||
|
} |
||||
|
|
||||
|
bool VectorVfsDirectory::DeleteSubdirectory(std::string_view name) { |
||||
|
return FindAndRemoveVectorElement(dirs, name); |
||||
|
} |
||||
|
|
||||
|
bool VectorVfsDirectory::DeleteFile(std::string_view name) { |
||||
|
return FindAndRemoveVectorElement(files, name); |
||||
|
} |
||||
|
|
||||
|
bool VectorVfsDirectory::Rename(std::string_view name_) { |
||||
|
name = name_; |
||||
|
return true; |
||||
|
} |
||||
|
|
||||
|
std::shared_ptr<VfsDirectory> VectorVfsDirectory::CreateSubdirectory(std::string_view name) { |
||||
|
return nullptr; |
||||
|
} |
||||
|
|
||||
|
std::shared_ptr<VfsFile> VectorVfsDirectory::CreateFile(std::string_view name) { |
||||
|
return nullptr; |
||||
|
} |
||||
|
|
||||
|
void VectorVfsDirectory::AddFile(VirtualFile file) { |
||||
|
files.push_back(std::move(file)); |
||||
|
} |
||||
|
|
||||
|
void VectorVfsDirectory::AddDirectory(VirtualDir dir) { |
||||
|
dirs.push_back(std::move(dir)); |
||||
|
} |
||||
|
|
||||
|
bool VectorVfsDirectory::ReplaceFileWithSubdirectory(VirtualFile file, VirtualDir dir) { |
||||
|
if (!DeleteFile(file->GetName())) |
||||
|
return false; |
||||
|
dirs.emplace_back(dir); |
||||
|
return true; |
||||
|
} |
||||
|
} // namespace FileSys
|
||||
@ -0,0 +1,44 @@ |
|||||
|
// Copyright 2018 yuzu emulator team |
||||
|
// Licensed under GPLv2 or any later version |
||||
|
// Refer to the license.txt file included. |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include "core/file_sys/vfs.h" |
||||
|
|
||||
|
namespace FileSys { |
||||
|
|
||||
|
// An implementation of VfsDirectory that maintains two vectors for subdirectories and files. |
||||
|
// Vector data is supplied upon construction. |
||||
|
struct VectorVfsDirectory : public VfsDirectory { |
||||
|
explicit VectorVfsDirectory(std::vector<VirtualFile> files = {}, |
||||
|
std::vector<VirtualDir> dirs = {}, VirtualDir parent = nullptr, |
||||
|
std::string name = ""); |
||||
|
|
||||
|
std::vector<std::shared_ptr<VfsFile>> GetFiles() const override; |
||||
|
std::vector<std::shared_ptr<VfsDirectory>> GetSubdirectories() const override; |
||||
|
bool IsWritable() const override; |
||||
|
bool IsReadable() const override; |
||||
|
std::string GetName() const override; |
||||
|
std::shared_ptr<VfsDirectory> GetParentDirectory() const override; |
||||
|
bool DeleteSubdirectory(std::string_view name) override; |
||||
|
bool DeleteFile(std::string_view name) override; |
||||
|
bool Rename(std::string_view name) override; |
||||
|
std::shared_ptr<VfsDirectory> CreateSubdirectory(std::string_view name) override; |
||||
|
std::shared_ptr<VfsFile> CreateFile(std::string_view name) override; |
||||
|
|
||||
|
virtual void AddFile(VirtualFile file); |
||||
|
virtual void AddDirectory(VirtualDir dir); |
||||
|
|
||||
|
protected: |
||||
|
bool ReplaceFileWithSubdirectory(VirtualFile file, VirtualDir dir) override; |
||||
|
|
||||
|
private: |
||||
|
std::vector<VirtualFile> files; |
||||
|
std::vector<VirtualDir> dirs; |
||||
|
|
||||
|
VirtualDir parent; |
||||
|
std::string name; |
||||
|
}; |
||||
|
|
||||
|
} // namespace FileSys |
||||
Write
Preview
Loading…
Cancel
Save
Reference in new issue