// Copyright 2018 Dolphin Emulator Project // Licensed under GPLv2+ // Refer to the license.txt file included. #pragma once #include #include #include #include #include #include #include #include #include #include #include #include "Common/CommonTypes.h" #include "Common/File.h" #include "Common/Swap.h" #include "DiscIO/Blob.h" #include "DiscIO/MultithreadedCompressor.h" #include "DiscIO/WiiEncryptionCache.h" namespace DiscIO { class VolumeDisc; enum class WIACompressionType : u32 { None = 0, Purge = 1, Bzip2 = 2, LZMA = 3, LZMA2 = 4, Zstd = 5, }; std::pair GetAllowedCompressionLevels(WIACompressionType compression_type); constexpr u32 WIA_MAGIC = 0x01414957; // "WIA\x1" (byteswapped to little endian) constexpr u32 RVZ_MAGIC = 0x015A5652; // "RVZ\x1" (byteswapped to little endian) class WIAFileReader : public BlobReader { public: ~WIAFileReader(); static std::unique_ptr Create(File::IOFile file, const std::string& path); BlobType GetBlobType() const override; u64 GetRawSize() const override { return Common::swap64(m_header_1.wia_file_size); } u64 GetDataSize() const override { return Common::swap64(m_header_1.iso_file_size); } bool IsDataSizeAccurate() const override { return true; } u64 GetBlockSize() const override { return Common::swap32(m_header_2.chunk_size); } bool HasFastRandomAccessInBlock() const override { return false; } bool Read(u64 offset, u64 size, u8* out_ptr) override; bool SupportsReadWiiDecrypted() const override; bool ReadWiiDecrypted(u64 offset, u64 size, u8* out_ptr, u64 partition_data_offset) override; static ConversionResultCode ConvertToWIA(BlobReader* infile, const VolumeDisc* infile_volume, File::IOFile* outfile, bool rvz, WIACompressionType compression_type, int compression_level, int chunk_size, CompressCB callback, void* arg); private: using SHA1 = std::array; using WiiKey = std::array; // See docs/WIA.md for details about the format #pragma pack(push, 1) struct WIAHeader1 { u32 magic; u32 version; u32 version_compatible; u32 header_2_size; SHA1 header_2_hash; u64 iso_file_size; u64 wia_file_size; SHA1 header_1_hash; }; static_assert(sizeof(WIAHeader1) == 0x48, "Wrong size for WIA header 1"); struct WIAHeader2 { u32 disc_type; u32 compression_type; u32 compression_level; // Informative only u32 chunk_size; std::array disc_header; u32 number_of_partition_entries; u32 partition_entry_size; u64 partition_entries_offset; SHA1 partition_entries_hash; u32 number_of_raw_data_entries; u64 raw_data_entries_offset; u32 raw_data_entries_size; u32 number_of_group_entries; u64 group_entries_offset; u32 group_entries_size; u8 compressor_data_size; u8 compressor_data[7]; }; static_assert(sizeof(WIAHeader2) == 0xdc, "Wrong size for WIA header 2"); struct PartitionDataEntry { u32 first_sector; u32 number_of_sectors; u32 group_index; u32 number_of_groups; }; static_assert(sizeof(PartitionDataEntry) == 0x10, "Wrong size for WIA partition data entry"); struct PartitionEntry { WiiKey partition_key; std::array data_entries; }; static_assert(sizeof(PartitionEntry) == 0x30, "Wrong size for WIA partition entry"); struct RawDataEntry { u64 data_offset; u64 data_size; u32 group_index; u32 number_of_groups; }; static_assert(sizeof(RawDataEntry) == 0x18, "Wrong size for WIA raw data entry"); struct GroupEntry { u32 data_offset; // >> 2 u32 data_size; }; static_assert(sizeof(GroupEntry) == 0x08, "Wrong size for WIA group entry"); struct HashExceptionEntry { u16 offset; SHA1 hash; }; static_assert(sizeof(HashExceptionEntry) == 0x16, "Wrong size for WIA hash exception entry"); struct PurgeSegment { u32 offset; u32 size; }; static_assert(sizeof(PurgeSegment) == 0x08, "Wrong size for WIA purge segment"); #pragma pack(pop) struct DataEntry { u32 index; bool is_partition; u8 partition_data_index; DataEntry(size_t index_) : index(static_cast(index_)), is_partition(false) {} DataEntry(size_t index_, size_t partition_data_index_) : index(static_cast(index_)), is_partition(true), partition_data_index(static_cast(partition_data_index_)) { } }; struct DecompressionBuffer { std::vector data; size_t bytes_written = 0; }; class Decompressor { public: virtual ~Decompressor(); virtual bool Decompress(const DecompressionBuffer& in, DecompressionBuffer* out, size_t* in_bytes_read) = 0; virtual bool Done() const { return m_done; }; protected: bool m_done = false; }; class NoneDecompressor final : public Decompressor { public: bool Decompress(const DecompressionBuffer& in, DecompressionBuffer* out, size_t* in_bytes_read) override; }; // This class assumes that more bytes won't be added to in once in.bytes_written == in.data.size() // and that *in_bytes_read initially will be equal to the size of the exception lists class PurgeDecompressor final : public Decompressor { public: PurgeDecompressor(u64 decompressed_size); bool Decompress(const DecompressionBuffer& in, DecompressionBuffer* out, size_t* in_bytes_read) override; private: const u64 m_decompressed_size; PurgeSegment m_segment = {}; size_t m_bytes_read = 0; size_t m_segment_bytes_written = 0; size_t m_out_bytes_written = 0; bool m_started = false; mbedtls_sha1_context m_sha1_context; }; class Bzip2Decompressor final : public Decompressor { public: ~Bzip2Decompressor(); bool Decompress(const DecompressionBuffer& in, DecompressionBuffer* out, size_t* in_bytes_read) override; private: bz_stream m_stream = {}; bool m_started = false; }; class LZMADecompressor final : public Decompressor { public: LZMADecompressor(bool lzma2, const u8* filter_options, size_t filter_options_size); ~LZMADecompressor(); bool Decompress(const DecompressionBuffer& in, DecompressionBuffer* out, size_t* in_bytes_read) override; private: lzma_stream m_stream = LZMA_STREAM_INIT; lzma_options_lzma m_options = {}; lzma_filter m_filters[2]; bool m_started = false; bool m_error_occurred = false; }; class ZstdDecompressor final : public Decompressor { public: ZstdDecompressor(); ~ZstdDecompressor(); bool Decompress(const DecompressionBuffer& in, DecompressionBuffer* out, size_t* in_bytes_read) override; private: ZSTD_DStream* m_stream; }; class Compressor { public: virtual ~Compressor(); // First call Start, then AddDataOnlyForPurgeHashing/Compress any number of times, // then End, then GetData/GetSize any number of times. virtual bool Start() = 0; virtual bool AddPrecedingDataOnlyForPurgeHashing(const u8* data, size_t size) { return true; } virtual bool Compress(const u8* data, size_t size) = 0; virtual bool End() = 0; virtual const u8* GetData() const = 0; virtual size_t GetSize() const = 0; }; class PurgeCompressor final : public Compressor { public: PurgeCompressor(); ~PurgeCompressor(); bool Start() override; bool AddPrecedingDataOnlyForPurgeHashing(const u8* data, size_t size) override; bool Compress(const u8* data, size_t size) override; bool End() override; const u8* GetData() const override; size_t GetSize() const override; private: std::vector m_buffer; size_t m_bytes_written; mbedtls_sha1_context m_sha1_context; }; class Bzip2Compressor final : public Compressor { public: Bzip2Compressor(int compression_level); ~Bzip2Compressor(); bool Start() override; bool Compress(const u8* data, size_t size) override; bool End() override; const u8* GetData() const override; size_t GetSize() const override; private: void ExpandBuffer(size_t bytes_to_add); bz_stream m_stream = {}; std::vector m_buffer; int m_compression_level; }; class LZMACompressor final : public Compressor { public: LZMACompressor(bool lzma2, int compression_level, u8 compressor_data_out[7], u8* compressor_data_size_out); ~LZMACompressor(); bool Start() override; bool Compress(const u8* data, size_t size) override; bool End() override; const u8* GetData() const override; size_t GetSize() const override; private: void ExpandBuffer(size_t bytes_to_add); lzma_stream m_stream = LZMA_STREAM_INIT; lzma_options_lzma m_options = {}; lzma_filter m_filters[2]; std::vector m_buffer; bool m_initialization_failed = false; }; class ZstdCompressor final : public Compressor { public: ZstdCompressor(int compression_level); ~ZstdCompressor(); bool Start() override; bool Compress(const u8* data, size_t size) override; bool End() override; const u8* GetData() const override { return m_buffer.data(); } size_t GetSize() const override { return m_out_buffer.pos; } private: void ExpandBuffer(size_t bytes_to_add); ZSTD_CStream* m_stream; ZSTD_outBuffer m_out_buffer; std::vector m_buffer; }; class Chunk { public: Chunk(); Chunk(File::IOFile* file, u64 offset_in_file, u64 compressed_size, u64 decompressed_size, u32 exception_lists, bool compressed_exception_lists, std::unique_ptr decompressor); bool Read(u64 offset, u64 size, u8* out_ptr); // This can only be called once at least one byte of data has been read bool ApplyHashExceptions(VolumeWii::HashBlock hash_blocks[VolumeWii::BLOCKS_PER_GROUP], u64 exception_list_index) const; template bool ReadAll(std::vector* vector) { return Read(0, vector->size() * sizeof(T), reinterpret_cast(vector->data())); } private: bool HandleExceptions(const u8* data, size_t bytes_allocated, size_t bytes_written, size_t* bytes_used, bool align); DecompressionBuffer m_in; DecompressionBuffer m_out; size_t m_in_bytes_read = 0; std::unique_ptr m_decompressor = nullptr; File::IOFile* m_file = nullptr; u64 m_offset_in_file = 0; size_t m_out_bytes_allocated_for_exceptions = 0; size_t m_out_bytes_used_for_exceptions = 0; size_t m_in_bytes_used_for_exceptions = 0; u32 m_exception_lists = 0; bool m_compressed_exception_lists = false; }; explicit WIAFileReader(File::IOFile file, const std::string& path); bool Initialize(const std::string& path); bool HasDataOverlap() const; bool ReadFromGroups(u64* offset, u64* size, u8** out_ptr, u64 chunk_size, u32 sector_size, u64 data_offset, u64 data_size, u32 group_index, u32 number_of_groups, u32 exception_lists); Chunk& ReadCompressedData(u64 offset_in_file, u64 compressed_size, u64 decompressed_size, u32 exception_lists); static std::string VersionToString(u32 version); static u32 LZMA2DictionarySize(u8 p); struct ReuseID { bool operator==(const ReuseID& other) const { return std::tie(partition_key, data_size, decrypted, value) == std::tie(other.partition_key, other.data_size, other.decrypted, other.value); } bool operator<(const ReuseID& other) const { return std::tie(partition_key, data_size, decrypted, value) < std::tie(other.partition_key, other.data_size, other.decrypted, other.value); } bool operator>(const ReuseID& other) const { return std::tie(partition_key, data_size, decrypted, value) > std::tie(other.partition_key, other.data_size, other.decrypted, other.value); } bool operator!=(const ReuseID& other) const { return !operator==(other); } bool operator>=(const ReuseID& other) const { return !operator<(other); } bool operator<=(const ReuseID& other) const { return !operator>(other); } const WiiKey* partition_key; u64 data_size; bool decrypted; u8 value; }; struct CompressThreadState { using WiiBlockData = std::array; std::unique_ptr compressor; std::vector decryption_buffer = std::vector(VolumeWii::BLOCKS_PER_GROUP); std::vector hash_buffer = std::vector(VolumeWii::BLOCKS_PER_GROUP); std::vector exceptions_buffer; }; struct CompressParameters { std::vector data; const DataEntry* data_entry; u64 bytes_read; size_t group_index; }; struct OutputParameters { std::vector exception_lists; std::vector main_data; std::optional reuse_id; u64 bytes_read; size_t group_index; }; static bool PadTo4(File::IOFile* file, u64* bytes_written); static void AddRawDataEntry(u64 offset, u64 size, int chunk_size, u32* total_groups, std::vector* raw_data_entries, std::vector* data_entries); static PartitionDataEntry CreatePartitionDataEntry(u64 offset, u64 size, u32 index, int chunk_size, u32* total_groups, const std::vector& partition_entries, std::vector* data_entries); static ConversionResultCode SetUpDataEntriesForWriting(const VolumeDisc* volume, int chunk_size, u64 iso_size, u32* total_groups, std::vector* partition_entries, std::vector* raw_data_entries, std::vector* data_entries); static std::optional> Compress(Compressor* compressor, const u8* data, size_t size); static bool WriteHeader(File::IOFile* file, const u8* data, size_t size, u64 upper_bound, u64* bytes_written, u64* offset_out); static void SetUpCompressor(std::unique_ptr* compressor, WIACompressionType compression_type, int compression_level, WIAHeader2* header_2); static ConversionResult ProcessAndCompress(CompressThreadState* state, CompressParameters parameters, const std::vector& partition_entries, const std::vector& data_entries, std::map* reusable_groups, std::mutex* reusable_groups_mutex, u64 exception_lists_per_chunk, bool compressed_exception_lists); static ConversionResultCode Output(const OutputParameters& parameters, File::IOFile* outfile, std::map* reusable_groups, std::mutex* reusable_groups_mutex, GroupEntry* group_entry, u64* bytes_written); static ConversionResultCode RunCallback(size_t groups_written, u64 bytes_read, u64 bytes_written, u32 total_groups, u64 iso_size, CompressCB callback, void* arg); template static void PushBack(std::vector* vector, const T& x) { const size_t offset_in_vector = vector->size(); vector->resize(offset_in_vector + sizeof(T)); const u8* x_ptr = reinterpret_cast(&x); std::copy(x_ptr, x_ptr + sizeof(T), vector->data() + offset_in_vector); } bool m_valid; bool m_rvz; WIACompressionType m_compression_type; File::IOFile m_file; Chunk m_cached_chunk; u64 m_cached_chunk_offset = std::numeric_limits::max(); WiiEncryptionCache m_encryption_cache; WIAHeader1 m_header_1; WIAHeader2 m_header_2; std::vector m_partition_entries; std::vector m_raw_data_entries; std::vector m_group_entries; std::map m_data_entries; // Perhaps we could set WIA_VERSION_WRITE_COMPATIBLE to 0.9, but WIA version 0.9 was never in // any official release of wit, and interim versions (either source or binaries) are hard to find. // Since we've been unable to check if we're write compatible with 0.9, we set it 1.0 to be safe. static constexpr u32 WIA_VERSION = 0x01000000; static constexpr u32 WIA_VERSION_WRITE_COMPATIBLE = 0x01000000; static constexpr u32 WIA_VERSION_READ_COMPATIBLE = 0x00080000; static constexpr u32 RVZ_VERSION = 0x00010000; static constexpr u32 RVZ_VERSION_WRITE_COMPATIBLE = 0x00010000; static constexpr u32 RVZ_VERSION_READ_COMPATIBLE = 0x00010000; }; } // namespace DiscIO