WIA: Implement re-encryption of Wii partition data

This commit is contained in:
JosJuice 2020-01-23 16:47:49 +01:00
parent e3d291a529
commit 04089f24f9
2 changed files with 137 additions and 48 deletions

View File

@ -8,6 +8,7 @@
#include <array> #include <array>
#include <cstring> #include <cstring>
#include <limits> #include <limits>
#include <map>
#include <memory> #include <memory>
#include <utility> #include <utility>
@ -23,10 +24,12 @@
#include "Common/Swap.h" #include "Common/Swap.h"
#include "DiscIO/VolumeWii.h" #include "DiscIO/VolumeWii.h"
#include "DiscIO/WiiEncryptionCache.h"
namespace DiscIO namespace DiscIO
{ {
WIAFileReader::WIAFileReader(File::IOFile file, const std::string& path) : m_file(std::move(file)) WIAFileReader::WIAFileReader(File::IOFile file, const std::string& path)
: m_file(std::move(file)), m_encryption_cache(this)
{ {
m_valid = Initialize(path); m_valid = Initialize(path);
} }
@ -120,24 +123,33 @@ bool WIAFileReader::Initialize(const std::string& path)
std::memset(reinterpret_cast<u8*>(&m_partition_entries[i]) + copy_length, 0, memset_length); std::memset(reinterpret_cast<u8*>(&m_partition_entries[i]) + copy_length, 0, memset_length);
} }
for (const PartitionEntry& partition : m_partition_entries) for (size_t i = 0; i < m_partition_entries.size(); ++i)
{ {
if (Common::swap32(partition.data_entries[1].number_of_sectors) != 0) const std::array<PartitionDataEntry, 2>& entries = m_partition_entries[i].data_entries;
size_t non_empty_entries = 0;
for (size_t j = 0; j < entries.size(); ++j)
{ {
const u32 first_end = Common::swap32(partition.data_entries[0].first_sector) + const u32 number_of_sectors = Common::swap32(entries[j].number_of_sectors);
Common::swap32(partition.data_entries[0].number_of_sectors); if (number_of_sectors != 0)
const u32 second_start = Common::swap32(partition.data_entries[1].first_sector); {
++non_empty_entries;
const u32 last_sector = Common::swap32(entries[j].first_sector) + number_of_sectors;
m_data_entries.emplace(last_sector * VolumeWii::BLOCK_TOTAL_SIZE, DataEntry(i, j));
}
}
if (non_empty_entries > 1)
{
const u32 first_end =
Common::swap32(entries[0].first_sector) + Common::swap32(entries[0].number_of_sectors);
const u32 second_start = Common::swap32(entries[1].first_sector);
if (first_end > second_start) if (first_end > second_start)
return false; return false;
} }
} }
std::sort(m_partition_entries.begin(), m_partition_entries.end(),
[](const PartitionEntry& a, const PartitionEntry& b) {
return Common::swap32(a.data_entries[0].first_sector) <
Common::swap32(b.data_entries[0].first_sector);
});
const u32 number_of_raw_data_entries = Common::swap32(m_header_2.number_of_raw_data_entries); const u32 number_of_raw_data_entries = Common::swap32(m_header_2.number_of_raw_data_entries);
m_raw_data_entries.resize(number_of_raw_data_entries); m_raw_data_entries.resize(number_of_raw_data_entries);
Chunk& raw_data_entries = Chunk& raw_data_entries =
@ -147,10 +159,13 @@ bool WIAFileReader::Initialize(const std::string& path)
if (!raw_data_entries.ReadAll(&m_raw_data_entries)) if (!raw_data_entries.ReadAll(&m_raw_data_entries))
return false; return false;
std::sort(m_raw_data_entries.begin(), m_raw_data_entries.end(), for (size_t i = 0; i < m_raw_data_entries.size(); ++i)
[](const RawDataEntry& a, const RawDataEntry& b) { {
return Common::swap64(a.data_offset) < Common::swap64(b.data_offset); const RawDataEntry& entry = m_raw_data_entries[i];
}); const u64 data_size = Common::swap64(entry.data_size);
if (data_size != 0)
m_data_entries.emplace(Common::swap64(entry.data_offset) + data_size, DataEntry(i));
}
const u32 number_of_group_entries = Common::swap32(m_header_2.number_of_group_entries); const u32 number_of_group_entries = Common::swap32(m_header_2.number_of_group_entries);
m_group_entries.resize(number_of_group_entries); m_group_entries.resize(number_of_group_entries);
@ -184,21 +199,75 @@ bool WIAFileReader::Read(u64 offset, u64 size, u8* out_ptr)
} }
const u32 chunk_size = Common::swap32(m_header_2.chunk_size); const u32 chunk_size = Common::swap32(m_header_2.chunk_size);
for (RawDataEntry raw_data : m_raw_data_entries) while (size > 0)
{ {
if (size == 0) const auto it = m_data_entries.upper_bound(offset);
return true; if (it == m_data_entries.end())
if (!ReadFromGroups(&offset, &size, &out_ptr, chunk_size, VolumeWii::BLOCK_TOTAL_SIZE,
Common::swap64(raw_data.data_offset), Common::swap64(raw_data.data_size),
Common::swap32(raw_data.group_index),
Common::swap32(raw_data.number_of_groups), 0))
{
return false; return false;
const DataEntry& data = it->second;
if (data.is_partition)
{
const PartitionEntry& partition = m_partition_entries[it->second.index];
const u32 partition_first_sector = Common::swap32(partition.data_entries[0].first_sector);
const u64 partition_data_offset = partition_first_sector * VolumeWii::BLOCK_TOTAL_SIZE;
const u32 second_number_of_sectors =
Common::swap32(partition.data_entries[1].number_of_sectors);
const u32 partition_total_sectors =
second_number_of_sectors ? Common::swap32(partition.data_entries[1].first_sector) -
partition_first_sector + second_number_of_sectors :
Common::swap32(partition.data_entries[0].number_of_sectors);
for (const PartitionDataEntry& partition_data : partition.data_entries)
{
if (size == 0)
return true;
const u32 first_sector = Common::swap32(partition_data.first_sector);
const u32 number_of_sectors = Common::swap32(partition_data.number_of_sectors);
const u64 data_offset = first_sector * VolumeWii::BLOCK_TOTAL_SIZE;
const u64 data_size = number_of_sectors * VolumeWii::BLOCK_TOTAL_SIZE;
if (data_size == 0)
continue;
if (data_offset + data_size <= offset)
continue;
if (offset < data_offset)
return false;
const u64 bytes_to_read = std::min(data_size - (offset - data_offset), size);
if (!m_encryption_cache.EncryptGroups(
offset - partition_data_offset, bytes_to_read, out_ptr, partition_data_offset,
partition_total_sectors * VolumeWii::BLOCK_DATA_SIZE, partition.partition_key))
{
return false;
}
offset += bytes_to_read;
size -= bytes_to_read;
out_ptr += bytes_to_read;
}
}
else
{
const RawDataEntry& raw_data = m_raw_data_entries[data.index];
if (!ReadFromGroups(&offset, &size, &out_ptr, chunk_size, VolumeWii::BLOCK_TOTAL_SIZE,
Common::swap64(raw_data.data_offset), Common::swap64(raw_data.data_size),
Common::swap32(raw_data.group_index),
Common::swap32(raw_data.number_of_groups), 0))
{
return false;
}
} }
} }
return size == 0; return true;
} }
bool WIAFileReader::SupportsReadWiiDecrypted() const bool WIAFileReader::SupportsReadWiiDecrypted() const
@ -210,34 +279,35 @@ bool WIAFileReader::ReadWiiDecrypted(u64 offset, u64 size, u8* out_ptr, u64 part
{ {
const u64 chunk_size = Common::swap32(m_header_2.chunk_size) * VolumeWii::BLOCK_DATA_SIZE / const u64 chunk_size = Common::swap32(m_header_2.chunk_size) * VolumeWii::BLOCK_DATA_SIZE /
VolumeWii::BLOCK_TOTAL_SIZE; VolumeWii::BLOCK_TOTAL_SIZE;
for (const PartitionEntry& partition : m_partition_entries)
const auto it = m_data_entries.upper_bound(partition_data_offset);
if (it == m_data_entries.end() || !it->second.is_partition)
return false;
const PartitionEntry& partition = m_partition_entries[it->second.index];
const u32 partition_first_sector = Common::swap32(partition.data_entries[0].first_sector);
if (partition_data_offset != partition_first_sector * VolumeWii::BLOCK_TOTAL_SIZE)
return false;
for (const PartitionDataEntry& data : partition.data_entries)
{ {
const u32 partition_first_sector = Common::swap32(partition.data_entries[0].first_sector); if (size == 0)
if (partition_data_offset != partition_first_sector * VolumeWii::BLOCK_TOTAL_SIZE) return true;
continue;
for (const PartitionDataEntry& data : partition.data_entries) const u64 data_offset =
(Common::swap32(data.first_sector) - partition_first_sector) * VolumeWii::BLOCK_DATA_SIZE;
const u64 data_size = Common::swap32(data.number_of_sectors) * VolumeWii::BLOCK_DATA_SIZE;
if (!ReadFromGroups(&offset, &size, &out_ptr, chunk_size, VolumeWii::BLOCK_DATA_SIZE,
data_offset, data_size, Common::swap32(data.group_index),
Common::swap32(data.number_of_groups),
chunk_size / VolumeWii::GROUP_DATA_SIZE))
{ {
if (size == 0) return false;
return true;
const u64 data_offset =
(Common::swap32(data.first_sector) - partition_first_sector) * VolumeWii::BLOCK_DATA_SIZE;
const u64 data_size = Common::swap32(data.number_of_sectors) * VolumeWii::BLOCK_DATA_SIZE;
if (!ReadFromGroups(&offset, &size, &out_ptr, chunk_size, VolumeWii::BLOCK_DATA_SIZE,
data_offset, data_size, Common::swap32(data.group_index),
Common::swap32(data.number_of_groups),
chunk_size / VolumeWii::GROUP_DATA_SIZE))
{
return false;
}
} }
return size == 0;
} }
return false; return size == 0;
} }
bool WIAFileReader::ReadFromGroups(u64* offset, u64* size, u8** out_ptr, u64 chunk_size, bool WIAFileReader::ReadFromGroups(u64* offset, u64* size, u8** out_ptr, u64 chunk_size,

View File

@ -6,6 +6,7 @@
#include <array> #include <array>
#include <limits> #include <limits>
#include <map>
#include <memory> #include <memory>
#include <utility> #include <utility>
@ -17,6 +18,7 @@
#include "Common/File.h" #include "Common/File.h"
#include "Common/Swap.h" #include "Common/Swap.h"
#include "DiscIO/Blob.h" #include "DiscIO/Blob.h"
#include "DiscIO/WiiEncryptionCache.h"
namespace DiscIO namespace DiscIO
{ {
@ -274,6 +276,7 @@ private:
File::IOFile m_file; File::IOFile m_file;
Chunk m_cached_chunk; Chunk m_cached_chunk;
u64 m_cached_chunk_offset = std::numeric_limits<u64>::max(); u64 m_cached_chunk_offset = std::numeric_limits<u64>::max();
WiiEncryptionCache m_encryption_cache;
WIAHeader1 m_header_1; WIAHeader1 m_header_1;
WIAHeader2 m_header_2; WIAHeader2 m_header_2;
@ -281,6 +284,22 @@ private:
std::vector<RawDataEntry> m_raw_data_entries; std::vector<RawDataEntry> m_raw_data_entries;
std::vector<GroupEntry> m_group_entries; std::vector<GroupEntry> m_group_entries;
struct DataEntry
{
u32 index;
bool is_partition;
u8 partition_data_index;
DataEntry(size_t index_) : index(static_cast<u32>(index_)), is_partition(false) {}
DataEntry(size_t index_, size_t partition_data_index_)
: index(static_cast<u32>(index_)), is_partition(true),
partition_data_index(static_cast<u8>(partition_data_index_))
{
}
};
std::map<u64, DataEntry> m_data_entries;
static constexpr u32 WIA_VERSION = 0x01000000; static constexpr u32 WIA_VERSION = 0x01000000;
static constexpr u32 WIA_VERSION_WRITE_COMPATIBLE = 0x01000000; static constexpr u32 WIA_VERSION_WRITE_COMPATIBLE = 0x01000000;
static constexpr u32 WIA_VERSION_READ_COMPATIBLE = 0x00080000; static constexpr u32 WIA_VERSION_READ_COMPATIBLE = 0x00080000;