2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2008 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2014-02-10 13:54:46 -05:00
|
|
|
#pragma once
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2020-01-23 16:47:49 +01:00
|
|
|
#include <array>
|
2020-01-27 16:12:56 +01:00
|
|
|
#include <functional>
|
2015-04-09 17:44:53 +02:00
|
|
|
#include <map>
|
2014-08-31 09:34:58 -04:00
|
|
|
#include <memory>
|
2017-06-03 21:29:08 +02:00
|
|
|
#include <optional>
|
2014-02-21 01:47:53 +01:00
|
|
|
#include <string>
|
2015-12-19 17:52:10 -05:00
|
|
|
#include <vector>
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2020-01-23 16:47:49 +01:00
|
|
|
#include <mbedtls/aes.h>
|
|
|
|
|
2014-02-21 01:47:53 +01:00
|
|
|
#include "Common/CommonTypes.h"
|
2017-06-24 18:42:36 +02:00
|
|
|
#include "Common/Lazy.h"
|
2017-02-11 08:57:47 +01:00
|
|
|
#include "Core/IOS/ES/Formats.h"
|
2017-08-02 18:16:56 +02:00
|
|
|
#include "DiscIO/Filesystem.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "DiscIO/Volume.h"
|
2020-06-04 20:33:52 +02:00
|
|
|
#include "DiscIO/VolumeDisc.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
|
2008-12-08 04:46:09 +00:00
|
|
|
namespace DiscIO
|
|
|
|
{
|
2017-06-06 11:49:01 +02:00
|
|
|
class BlobReader;
|
2016-07-06 20:33:05 +02:00
|
|
|
enum class BlobType;
|
|
|
|
enum class Country;
|
2017-08-02 18:16:56 +02:00
|
|
|
class FileSystem;
|
2016-07-06 20:33:05 +02:00
|
|
|
enum class Language;
|
2016-12-23 18:41:21 +01:00
|
|
|
enum class Region;
|
2016-07-06 20:33:05 +02:00
|
|
|
enum class Platform;
|
|
|
|
|
2019-07-14 15:49:42 +02:00
|
|
|
class VolumeWii : public VolumeDisc
|
2008-12-08 04:46:09 +00:00
|
|
|
{
|
|
|
|
public:
|
2020-01-23 16:47:49 +01:00
|
|
|
static constexpr size_t AES_KEY_SIZE = 16;
|
|
|
|
static constexpr size_t SHA1_SIZE = 20;
|
|
|
|
|
|
|
|
static constexpr u32 H3_TABLE_SIZE = 0x18000;
|
|
|
|
static constexpr u32 BLOCKS_PER_GROUP = 0x40;
|
|
|
|
|
|
|
|
static constexpr u64 BLOCK_HEADER_SIZE = 0x0400;
|
|
|
|
static constexpr u64 BLOCK_DATA_SIZE = 0x7C00;
|
|
|
|
static constexpr u64 BLOCK_TOTAL_SIZE = BLOCK_HEADER_SIZE + BLOCK_DATA_SIZE;
|
|
|
|
|
|
|
|
static constexpr u64 GROUP_HEADER_SIZE = BLOCK_HEADER_SIZE * BLOCKS_PER_GROUP;
|
|
|
|
static constexpr u64 GROUP_DATA_SIZE = BLOCK_DATA_SIZE * BLOCKS_PER_GROUP;
|
|
|
|
static constexpr u64 GROUP_TOTAL_SIZE = GROUP_HEADER_SIZE + GROUP_DATA_SIZE;
|
|
|
|
|
2020-01-30 21:29:11 +01:00
|
|
|
struct HashBlock
|
|
|
|
{
|
|
|
|
u8 h0[31][SHA1_SIZE];
|
|
|
|
u8 padding_0[20];
|
|
|
|
u8 h1[8][SHA1_SIZE];
|
|
|
|
u8 padding_1[32];
|
|
|
|
u8 h2[8][SHA1_SIZE];
|
|
|
|
u8 padding_2[32];
|
|
|
|
};
|
|
|
|
static_assert(sizeof(HashBlock) == BLOCK_HEADER_SIZE);
|
|
|
|
|
2017-06-06 11:49:01 +02:00
|
|
|
VolumeWii(std::unique_ptr<BlobReader> reader);
|
|
|
|
~VolumeWii();
|
2018-05-30 02:37:13 -04:00
|
|
|
bool Read(u64 offset, u64 length, u8* buffer, const Partition& partition) const override;
|
2018-05-22 22:39:52 +02:00
|
|
|
bool IsEncryptedAndHashed() const override;
|
2015-06-13 12:51:24 +02:00
|
|
|
std::vector<Partition> GetPartitions() const override;
|
|
|
|
Partition GetGamePartition() const override;
|
2017-06-20 21:51:40 +02:00
|
|
|
std::optional<u32> GetPartitionType(const Partition& partition) const override;
|
2017-06-03 21:29:08 +02:00
|
|
|
std::optional<u64> GetTitleID(const Partition& partition) const override;
|
2017-05-20 18:33:36 +02:00
|
|
|
const IOS::ES::TicketReader& GetTicket(const Partition& partition) const override;
|
|
|
|
const IOS::ES::TMDReader& GetTMD(const Partition& partition) const override;
|
2019-03-21 23:04:44 +01:00
|
|
|
const std::vector<u8>& GetCertificateChain(const Partition& partition) const override;
|
2017-08-02 18:16:56 +02:00
|
|
|
const FileSystem* GetFileSystem(const Partition& partition) const override;
|
2018-05-22 22:39:52 +02:00
|
|
|
static u64 EncryptedPartitionOffsetToRawOffset(u64 offset, const Partition& partition,
|
|
|
|
u64 partition_data_offset);
|
|
|
|
u64 PartitionOffsetToRawOffset(u64 offset, const Partition& partition) const override;
|
2019-03-22 20:47:05 +01:00
|
|
|
std::string GetGameTDBID(const Partition& partition = PARTITION_NONE) const override;
|
2016-07-06 20:33:05 +02:00
|
|
|
std::map<Language, std::string> GetLongNames() const override;
|
2018-03-10 22:41:49 +01:00
|
|
|
std::vector<u32> GetBanner(u32* width, u32* height) const override;
|
2015-02-24 00:03:59 -05:00
|
|
|
|
2016-07-06 20:33:05 +02:00
|
|
|
Platform GetVolumeType() const override;
|
2020-07-04 14:51:13 +02:00
|
|
|
bool IsDatelDisc() const override;
|
2019-03-26 17:22:18 +01:00
|
|
|
bool SupportsIntegrityCheck() const override { return m_encrypted; }
|
|
|
|
bool CheckH3TableIntegrity(const Partition& partition) const override;
|
2021-03-07 13:48:51 +01:00
|
|
|
bool CheckBlockIntegrity(u64 block_index, const u8* encrypted_data,
|
2019-08-05 13:58:27 +02:00
|
|
|
const Partition& partition) const override;
|
2019-03-26 17:22:18 +01:00
|
|
|
bool CheckBlockIntegrity(u64 block_index, const Partition& partition) const override;
|
2015-02-24 00:03:59 -05:00
|
|
|
|
2016-12-23 18:41:21 +01:00
|
|
|
Region GetRegion() const override;
|
2016-06-24 10:43:46 +02:00
|
|
|
BlobType GetBlobType() const override;
|
|
|
|
u64 GetSize() const override;
|
2019-03-21 22:20:23 +01:00
|
|
|
bool IsSizeAccurate() const override;
|
2016-06-24 10:43:46 +02:00
|
|
|
u64 GetRawSize() const override;
|
2020-06-04 20:33:52 +02:00
|
|
|
const BlobReader& GetBlobReader() const override;
|
2020-06-07 22:58:03 +02:00
|
|
|
std::array<u8, 20> GetSyncHash() const override;
|
2014-03-08 01:54:44 +01:00
|
|
|
|
2020-04-15 21:15:08 +02:00
|
|
|
// The in parameter can either contain all the data to begin with,
|
|
|
|
// or read_function can write data into the in parameter when called.
|
|
|
|
// The latter lets reading run in parallel with hashing.
|
|
|
|
// This function returns false iff read_function returns false.
|
|
|
|
static bool HashGroup(const std::array<u8, BLOCK_DATA_SIZE> in[BLOCKS_PER_GROUP],
|
|
|
|
HashBlock out[BLOCKS_PER_GROUP],
|
|
|
|
const std::function<bool(size_t block)>& read_function = {});
|
|
|
|
|
2020-01-23 16:47:49 +01:00
|
|
|
static bool EncryptGroup(u64 offset, u64 partition_data_offset, u64 partition_data_decrypted_size,
|
|
|
|
const std::array<u8, AES_KEY_SIZE>& key, BlobReader* blob,
|
2020-01-27 16:12:56 +01:00
|
|
|
std::array<u8, GROUP_TOTAL_SIZE>* out,
|
|
|
|
const std::function<void(HashBlock hash_blocks[BLOCKS_PER_GROUP])>&
|
|
|
|
hash_exception_callback = {});
|
2015-02-08 10:31:17 +01:00
|
|
|
|
2020-04-15 21:15:08 +02:00
|
|
|
static void DecryptBlockHashes(const u8* in, HashBlock* out, mbedtls_aes_context* aes_context);
|
2020-04-14 11:40:32 +02:00
|
|
|
static void DecryptBlockData(const u8* in, u8* out, mbedtls_aes_context* aes_context);
|
|
|
|
|
2017-06-07 11:49:34 +02:00
|
|
|
protected:
|
|
|
|
u32 GetOffsetShift() const override { return 2; }
|
2018-04-12 14:18:04 +02:00
|
|
|
|
2017-02-10 18:57:58 +01:00
|
|
|
private:
|
2017-07-04 16:08:12 +02:00
|
|
|
struct PartitionDetails
|
|
|
|
{
|
2017-06-24 18:42:36 +02:00
|
|
|
Common::Lazy<std::unique_ptr<mbedtls_aes_context>> key;
|
|
|
|
Common::Lazy<IOS::ES::TicketReader> ticket;
|
|
|
|
Common::Lazy<IOS::ES::TMDReader> tmd;
|
2019-03-21 23:04:44 +01:00
|
|
|
Common::Lazy<std::vector<u8>> cert_chain;
|
2019-03-26 17:22:18 +01:00
|
|
|
Common::Lazy<std::vector<u8>> h3_table;
|
2017-08-02 18:16:56 +02:00
|
|
|
Common::Lazy<std::unique_ptr<FileSystem>> file_system;
|
2018-05-22 22:39:52 +02:00
|
|
|
Common::Lazy<u64> data_offset;
|
2017-07-04 16:08:12 +02:00
|
|
|
u32 type;
|
|
|
|
};
|
|
|
|
|
2018-05-30 02:37:13 -04:00
|
|
|
std::unique_ptr<BlobReader> m_reader;
|
2017-07-04 16:08:12 +02:00
|
|
|
std::map<Partition, PartitionDetails> m_partitions;
|
2015-06-13 12:51:24 +02:00
|
|
|
Partition m_game_partition;
|
2018-05-22 22:39:52 +02:00
|
|
|
bool m_encrypted;
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2015-04-21 11:17:39 +02:00
|
|
|
mutable u64 m_last_decrypted_block;
|
|
|
|
mutable u8 m_last_decrypted_block_data[BLOCK_DATA_SIZE];
|
2008-12-08 04:46:09 +00:00
|
|
|
};
|
|
|
|
|
2019-05-05 23:48:12 +00:00
|
|
|
} // namespace DiscIO
|