// Copyright 2008 Dolphin Emulator Project // SPDX-License-Identifier: GPL-2.0-or-later #pragma once #include #include #include #include #include #include #include #include #include #include #include "Common/CommonTypes.h" #include "Common/FileUtil.h" #include "DiscIO/Blob.h" #include "DiscIO/Volume.h" #include "DiscIO/WiiEncryptionCache.h" namespace File { struct FSTEntry; class IOFile; } // namespace File namespace DiscIO { enum class PartitionType : u32; class DirectoryBlobReader; class VolumeDisc; // Returns true if the path is inside a DirectoryBlob and doesn't represent the DirectoryBlob itself bool ShouldHideFromGameList(const std::string& volume_path); // Content chunk that is loaded from a file in the host file system. struct ContentFile { // Path where the file can be found. std::string m_filename; // Offset from the start of the file where the first byte of this content chunk is. u64 m_offset; }; // Content chunk that loads data from a DirectoryBlobReader. // Intented for representing a partition within a disc. struct ContentPartition { // The reader to read data from. DirectoryBlobReader* m_reader; // Offset from the start of the partition for the first byte represented by this chunk. u64 m_offset; // The value passed as partition_data_offset to EncryptPartitionData(). u64 m_partition_data_offset; }; // Content chunk that loads data from a Volume. struct ContentVolume { // Offset from the start of the volume for the first byte represented by this chunk. u64 m_offset; // The volume to read data from. const Volume* m_volume; // The partition passed to the Volume's Read() method. Partition m_partition; }; // Content chunk representing a run of identical bytes. // Useful for padding between chunks within a file. struct ContentFixedByte { u8 m_byte; }; using ContentSource = std::variant; struct BuilderContentSource { u64 m_offset; u64 m_size; ContentSource m_source; }; struct FSTBuilderNode { std::string m_filename; u64 m_size; std::variant, std::vector> m_content; void* m_user_data = nullptr; bool IsFile() const { return std::holds_alternative>(m_content); } std::vector& GetFileContent() { return std::get>(m_content); } const std::vector& GetFileContent() const { return std::get>(m_content); } bool IsFolder() const { return std::holds_alternative>(m_content); } std::vector& GetFolderContent() { return std::get>(m_content); } const std::vector& GetFolderContent() const { return std::get>(m_content); } }; class DiscContent { public: DiscContent(u64 offset, u64 size, ContentSource source); // Provided because it's convenient when searching for DiscContent in an std::set explicit DiscContent(u64 offset); u64 GetOffset() const; u64 GetEndOffset() const; u64 GetSize() const; bool Read(u64* offset, u64* length, u8** buffer) const; bool operator==(const DiscContent& other) const { return GetEndOffset() == other.GetEndOffset(); } bool operator!=(const DiscContent& other) const { return !(*this == other); } bool operator<(const DiscContent& other) const { return GetEndOffset() < other.GetEndOffset(); } bool operator>(const DiscContent& other) const { return other < *this; } bool operator<=(const DiscContent& other) const { return !(*this > other); } bool operator>=(const DiscContent& other) const { return !(*this < other); } private: // Position of this content chunk within its parent DiscContentContainer. u64 m_offset; // Number of bytes this content chunk takes up. u64 m_size = 0; // Where and how to find the data for this content chunk. ContentSource m_content_source; }; class DiscContentContainer { public: template void AddReference(u64 offset, const std::vector& vector) { return Add(offset, vector.size() * sizeof(T), reinterpret_cast(vector.data())); } void Add(u64 offset, u64 size, ContentSource source); u64 CheckSizeAndAdd(u64 offset, const std::string& path); u64 CheckSizeAndAdd(u64 offset, u64 max_size, const std::string& path); bool Read(u64 offset, u64 length, u8* buffer) const; private: std::set m_contents; }; class DirectoryBlobPartition { public: DirectoryBlobPartition() = default; DirectoryBlobPartition(const std::string& root_directory, std::optional is_wii); DirectoryBlobPartition(DiscIO::VolumeDisc* volume, const DiscIO::Partition& partition, std::optional is_wii, const std::function* fst_nodes, FSTBuilderNode* dol_node)>& fst_callback); // We do not allow copying, because it might mess up the pointers inside DiscContents DirectoryBlobPartition(const DirectoryBlobPartition&) = delete; DirectoryBlobPartition& operator=(const DirectoryBlobPartition&) = delete; DirectoryBlobPartition(DirectoryBlobPartition&&) = default; DirectoryBlobPartition& operator=(DirectoryBlobPartition&&) = default; bool IsWii() const { return m_is_wii; } u64 GetDataSize() const { return m_data_size; } const std::string& GetRootDirectory() const { return m_root_directory; } const std::vector& GetHeader() const { return m_disc_header; } const DiscContentContainer& GetContents() const { return m_contents; } const std::optional& GetWrappedPartition() const { return m_wrapped_partition; } const std::array& GetKey() const { return m_key; } void SetKey(std::array key) { m_key = key; } private: void SetDiscHeaderFromFile(const std::string& boot_bin_path); void SetDiscHeader(std::vector boot_bin); void SetDiscType(std::optional is_wii); void SetBI2FromFile(const std::string& bi2_path); void SetBI2(std::vector bi2); // Returns DOL address u64 SetApploaderFromFile(const std::string& path); u64 SetApploader(std::vector apploader, const std::string& log_path); // Returns FST address u64 SetDOLFromFile(const std::string& path, u64 dol_address); u64 SetDOL(FSTBuilderNode dol_node, u64 dol_address); void BuildFSTFromFolder(const std::string& fst_root_path, u64 fst_address); void BuildFST(std::vector root_nodes, u64 fst_address); // FST creation void WriteEntryData(u32* entry_offset, u8 type, u32 name_offset, u64 data_offset, u64 length, u32 address_shift); void WriteEntryName(u32* name_offset, const std::string& name, u64 name_table_offset); void WriteDirectory(std::vector* parent_entries, u32* fst_offset, u32* name_offset, u64* data_offset, u32 parent_entry_index, u64 name_table_offset); DiscContentContainer m_contents; std::vector m_disc_header; std::vector m_bi2; std::vector m_apploader; std::vector m_fst_data; std::array m_key{}; std::string m_root_directory; bool m_is_wii = false; // GameCube has no shift, Wii has 2 bit shift u32 m_address_shift = 0; u64 m_data_size = 0; std::optional m_wrapped_partition = std::nullopt; }; class DirectoryBlobReader : public BlobReader { friend DiscContent; public: static std::unique_ptr Create(const std::string& dol_path); static std::unique_ptr Create( std::unique_ptr volume, const std::function* fst_nodes, FSTBuilderNode* dol_node)>& fst_callback); // We do not allow copying, because it might mess up the pointers inside DiscContents DirectoryBlobReader(const DirectoryBlobReader&) = delete; DirectoryBlobReader& operator=(const DirectoryBlobReader&) = delete; DirectoryBlobReader(DirectoryBlobReader&&) = default; DirectoryBlobReader& operator=(DirectoryBlobReader&&) = default; bool Read(u64 offset, u64 length, u8* buffer) override; bool SupportsReadWiiDecrypted(u64 offset, u64 size, u64 partition_data_offset) const override; bool ReadWiiDecrypted(u64 offset, u64 size, u8* buffer, u64 partition_data_offset) override; BlobType GetBlobType() const override; u64 GetRawSize() const override; u64 GetDataSize() const override; bool IsDataSizeAccurate() const override { return true; } u64 GetBlockSize() const override { return 0; } bool HasFastRandomAccessInBlock() const override { return true; } std::string GetCompressionMethod() const override { return {}; } std::optional GetCompressionLevel() const override { return std::nullopt; } private: struct PartitionWithType { PartitionWithType(DirectoryBlobPartition&& partition_, PartitionType type_) : partition(std::move(partition_)), type(type_) { } DirectoryBlobPartition partition; PartitionType type; }; explicit DirectoryBlobReader(const std::string& game_partition_root, const std::string& true_root); explicit DirectoryBlobReader(std::unique_ptr volume, const std::function* fst_nodes, FSTBuilderNode* dol_node)>& fst_callback); const DirectoryBlobPartition* GetPartition(u64 offset, u64 size, u64 partition_data_offset) const; bool EncryptPartitionData(u64 offset, u64 size, u8* buffer, u64 partition_data_offset, u64 partition_data_decrypted_size); void SetNonpartitionDiscHeaderFromFile(const std::vector& partition_header, const std::string& game_partition_root); void SetNonpartitionDiscHeader(const std::vector& partition_header, std::vector header_bin); void SetWiiRegionDataFromFile(const std::string& game_partition_root); void SetWiiRegionData(const std::vector& wii_region_data, const std::string& log_path); void SetPartitions(std::vector&& partitions); void SetPartitionHeader(DirectoryBlobPartition* partition, u64 partition_address); // For GameCube: DirectoryBlobPartition m_gamecube_pseudopartition; // For Wii: DiscContentContainer m_nonpartition_contents; std::map m_partitions; WiiEncryptionCache m_encryption_cache; bool m_is_wii; bool m_encrypted; std::vector m_disc_header_nonpartition; std::vector m_partition_table; std::vector m_wii_region_data; std::vector> m_extra_data; u64 m_data_size; std::unique_ptr m_wrapped_volume; }; } // namespace DiscIO