DirectoryBlob: Replace std::set<DiscContent> with a new class

This commit is contained in:
JosJuice 2017-08-01 17:45:46 +02:00
parent a8606f5d13
commit d6260e02ce
2 changed files with 95 additions and 76 deletions

View File

@ -7,7 +7,6 @@
#include <algorithm> #include <algorithm>
#include <array> #include <array>
#include <cinttypes> #include <cinttypes>
#include <cstddef>
#include <cstring> #include <cstring>
#include <locale> #include <locale>
#include <map> #include <map>
@ -33,10 +32,6 @@
namespace DiscIO namespace DiscIO
{ {
static const DiscContent& AddFileToContents(std::set<DiscContent>* contents,
const std::string& path, u64 offset,
u64 max_size = UINT64_MAX);
// Reads as many bytes as the vector fits (or less, if the file is smaller). // Reads as many bytes as the vector fits (or less, if the file is smaller).
// Returns the number of bytes read. // Returns the number of bytes read.
static size_t ReadFileToVector(const std::string& path, std::vector<u8>* vector); static size_t ReadFileToVector(const std::string& path, std::vector<u8>* vector);
@ -120,6 +115,58 @@ bool DiscContent::Read(u64* offset, u64* length, u8** buffer) const
return true; return true;
} }
const DiscContent& DiscContentContainer::Add(u64 offset, u64 size, const std::string& path)
{
return *(m_contents.emplace(offset, size, path).first);
}
const DiscContent& DiscContentContainer::Add(u64 offset, u64 size, const u8* data)
{
return *(m_contents.emplace(offset, size, data).first);
}
const DiscContent& DiscContentContainer::CheckSizeAndAdd(u64 offset, const std::string& path)
{
return Add(offset, File::GetSize(path), path);
}
const DiscContent& DiscContentContainer::CheckSizeAndAdd(u64 offset, u64 max_size,
const std::string& path)
{
return Add(offset, std::min(File::GetSize(path), max_size), path);
}
bool DiscContentContainer::Read(u64 offset, u64 length, u8* buffer) const
{
if (m_contents.empty())
return true;
// Determine which DiscContent the offset refers to
std::set<DiscContent>::const_iterator it = m_contents.lower_bound(DiscContent(offset));
if (it->GetOffset() > offset && it != m_contents.begin())
--it;
// zero fill to start of file data
PadToAddress(it->GetOffset(), &offset, &length, &buffer);
while (it != m_contents.end() && length > 0)
{
_dbg_assert_(DISCIO, it->GetOffset() <= offset);
if (!it->Read(&offset, &length, &buffer))
return false;
++it;
if (it != m_contents.end())
{
_dbg_assert_(DISCIO, it->GetOffset() >= offset);
PadToAddress(it->GetOffset(), &offset, &length, &buffer);
}
}
return true;
}
static std::optional<PartitionType> ParsePartitionDirectoryName(const std::string& name) static std::optional<PartitionType> ParsePartitionDirectoryName(const std::string& name)
{ {
if (name.size() < 2) if (name.size() < 2)
@ -322,45 +369,11 @@ DirectoryBlobReader::DirectoryBlobReader(const std::string& game_partition_root,
} }
} }
bool DirectoryBlobReader::ReadInternal(u64 offset, u64 length, u8* buffer,
const std::set<DiscContent>& contents)
{
if (contents.empty())
return true;
// Determine which DiscContent the offset refers to
std::set<DiscContent>::const_iterator it = contents.lower_bound(DiscContent(offset));
if (it->GetOffset() > offset && it != contents.begin())
--it;
// zero fill to start of file data
PadToAddress(it->GetOffset(), &offset, &length, &buffer);
while (it != contents.end() && length > 0)
{
_dbg_assert_(DISCIO, it->GetOffset() <= offset);
if (!it->Read(&offset, &length, &buffer))
return false;
++it;
if (it != contents.end())
{
_dbg_assert_(DISCIO, it->GetOffset() >= offset);
PadToAddress(it->GetOffset(), &offset, &length, &buffer);
}
}
return true;
}
bool DirectoryBlobReader::Read(u64 offset, u64 length, u8* buffer) bool DirectoryBlobReader::Read(u64 offset, u64 length, u8* buffer)
{ {
// TODO: We don't handle raw access to the encrypted area of Wii discs correctly. // TODO: We don't handle raw access to the encrypted area of Wii discs correctly.
return (m_is_wii ? m_nonpartition_contents : m_gamecube_pseudopartition.GetContents())
const std::set<DiscContent>& contents = .Read(offset, length, buffer);
m_is_wii ? m_nonpartition_contents : m_gamecube_pseudopartition.GetContents();
return ReadInternal(offset, length, buffer, contents);
} }
bool DirectoryBlobReader::SupportsReadWiiDecrypted() const bool DirectoryBlobReader::SupportsReadWiiDecrypted() const
@ -377,7 +390,7 @@ bool DirectoryBlobReader::ReadWiiDecrypted(u64 offset, u64 size, u8* buffer, u64
if (it == m_partitions.end()) if (it == m_partitions.end())
return false; return false;
return ReadInternal(offset, size, buffer, it->second.GetContents()); return it->second.GetContents().Read(offset, size, buffer);
} }
BlobType DirectoryBlobReader::GetBlobType() const BlobType DirectoryBlobReader::GetBlobType() const
@ -417,8 +430,7 @@ void DirectoryBlobReader::SetNonpartitionDiscHeader(const std::vector<u8>& parti
if (header_bin_bytes_read < 0x61) if (header_bin_bytes_read < 0x61)
m_disc_header_nonpartition[0x61] = 0; m_disc_header_nonpartition[0x61] = 0;
m_nonpartition_contents.emplace(NONPARTITION_DISCHEADER_ADDRESS, NONPARTITION_DISCHEADER_SIZE, m_nonpartition_contents.Add(NONPARTITION_DISCHEADER_ADDRESS, m_disc_header_nonpartition);
m_disc_header_nonpartition.data());
} }
void DirectoryBlobReader::SetWiiRegionData(const std::string& game_partition_root) void DirectoryBlobReader::SetWiiRegionData(const std::string& game_partition_root)
@ -436,8 +448,7 @@ void DirectoryBlobReader::SetWiiRegionData(const std::string& game_partition_roo
constexpr u64 WII_REGION_DATA_ADDRESS = 0x4E000; constexpr u64 WII_REGION_DATA_ADDRESS = 0x4E000;
constexpr u64 WII_REGION_DATA_SIZE = 0x20; constexpr u64 WII_REGION_DATA_SIZE = 0x20;
m_nonpartition_contents.emplace(WII_REGION_DATA_ADDRESS, WII_REGION_DATA_SIZE, m_nonpartition_contents.Add(WII_REGION_DATA_ADDRESS, m_wii_region_data);
m_wii_region_data.data());
} }
void DirectoryBlobReader::SetPartitions(std::vector<PartitionWithType>&& partitions) void DirectoryBlobReader::SetPartitions(std::vector<PartitionWithType>&& partitions)
@ -501,8 +512,7 @@ void DirectoryBlobReader::SetPartitions(std::vector<PartitionWithType>&& partiti
} }
m_data_size = partition_address; m_data_size = partition_address;
m_nonpartition_contents.emplace(PARTITION_TABLE_ADDRESS, m_partition_table.size(), m_nonpartition_contents.Add(PARTITION_TABLE_ADDRESS, m_partition_table);
m_partition_table.data());
} }
// This function sets the header that's shortly before the start of the encrypted // This function sets the header that's shortly before the start of the encrypted
@ -519,19 +529,19 @@ void DirectoryBlobReader::SetPartitionHeader(const DirectoryBlobPartition& parti
const std::string& partition_root = partition.GetRootDirectory(); const std::string& partition_root = partition.GetRootDirectory();
AddFileToContents(&m_nonpartition_contents, partition_root + "ticket.bin", m_nonpartition_contents.CheckSizeAndAdd(partition_address + TICKET_OFFSET, TICKET_SIZE,
partition_address + TICKET_OFFSET, TICKET_SIZE); partition_root + "ticket.bin");
const DiscContent& tmd = AddFileToContents(&m_nonpartition_contents, partition_root + "tmd.bin", const DiscContent& tmd = m_nonpartition_contents.CheckSizeAndAdd(
partition_address + TMD_OFFSET, MAX_TMD_SIZE); partition_address + TMD_OFFSET, MAX_TMD_SIZE, partition_root + "tmd.bin");
const u64 cert_offset = Common::AlignUp(TMD_OFFSET + tmd.GetSize(), 0x20ull); const u64 cert_offset = Common::AlignUp(TMD_OFFSET + tmd.GetSize(), 0x20ull);
const u64 max_cert_size = H3_OFFSET - cert_offset; const u64 max_cert_size = H3_OFFSET - cert_offset;
const DiscContent& cert = AddFileToContents(&m_nonpartition_contents, partition_root + "cert.bin", const DiscContent& cert = m_nonpartition_contents.CheckSizeAndAdd(
partition_address + cert_offset, max_cert_size); partition_address + cert_offset, max_cert_size, partition_root + "cert.bin");
AddFileToContents(&m_nonpartition_contents, partition_root + "h3.bin", m_nonpartition_contents.CheckSizeAndAdd(partition_address + H3_OFFSET, H3_SIZE,
partition_address + H3_OFFSET, H3_SIZE); partition_root + "h3.bin");
constexpr u32 PARTITION_HEADER_SIZE = 0x1c; constexpr u32 PARTITION_HEADER_SIZE = 0x1c;
constexpr u32 DATA_OFFSET = 0x20000; constexpr u32 DATA_OFFSET = 0x20000;
@ -546,8 +556,7 @@ void DirectoryBlobReader::SetPartitionHeader(const DirectoryBlobPartition& parti
Write32(DATA_OFFSET >> 2, 0x14, &partition_header); Write32(DATA_OFFSET >> 2, 0x14, &partition_header);
Write32(static_cast<u32>(data_size >> 2), 0x18, &partition_header); Write32(static_cast<u32>(data_size >> 2), 0x18, &partition_header);
m_nonpartition_contents.emplace(partition_address + TICKET_SIZE, PARTITION_HEADER_SIZE, m_nonpartition_contents.Add(partition_address + TICKET_SIZE, partition_header);
partition_header.data());
} }
DirectoryBlobPartition::DirectoryBlobPartition(const std::string& root_directory, DirectoryBlobPartition::DirectoryBlobPartition(const std::string& root_directory,
@ -569,7 +578,7 @@ void DirectoryBlobPartition::SetDiscHeaderAndDiscType(std::optional<bool> is_wii
if (ReadFileToVector(boot_bin_path, &m_disc_header) < 0x20) if (ReadFileToVector(boot_bin_path, &m_disc_header) < 0x20)
ERROR_LOG(DISCIO, "%s doesn't exist or is too small", boot_bin_path.c_str()); ERROR_LOG(DISCIO, "%s doesn't exist or is too small", boot_bin_path.c_str());
m_contents.emplace(DISCHEADER_ADDRESS, DISCHEADER_SIZE, m_disc_header.data()); m_contents.Add(DISCHEADER_ADDRESS, m_disc_header);
if (is_wii.has_value()) if (is_wii.has_value())
{ {
@ -600,7 +609,7 @@ void DirectoryBlobPartition::SetBI2()
if (!m_is_wii && bytes_read < 0x1C) if (!m_is_wii && bytes_read < 0x1C)
ERROR_LOG(DISCIO, "Couldn't read region from %s", bi2_path.c_str()); ERROR_LOG(DISCIO, "Couldn't read region from %s", bi2_path.c_str());
m_contents.emplace(BI2_ADDRESS, BI2_SIZE, m_bi2.data()); m_contents.Add(BI2_ADDRESS, m_bi2);
} }
u64 DirectoryBlobPartition::SetApploader() u64 DirectoryBlobPartition::SetApploader()
@ -633,7 +642,7 @@ u64 DirectoryBlobPartition::SetApploader()
constexpr u64 APPLOADER_ADDRESS = 0x2440; constexpr u64 APPLOADER_ADDRESS = 0x2440;
m_contents.emplace(APPLOADER_ADDRESS, m_apploader.size(), m_apploader.data()); m_contents.Add(APPLOADER_ADDRESS, m_apploader);
// Return DOL address, 32 byte aligned (plus 32 byte padding) // Return DOL address, 32 byte aligned (plus 32 byte padding)
return Common::AlignUp(APPLOADER_ADDRESS + m_apploader.size() + 0x20, 0x20ull); return Common::AlignUp(APPLOADER_ADDRESS + m_apploader.size() + 0x20, 0x20ull);
@ -642,7 +651,7 @@ u64 DirectoryBlobPartition::SetApploader()
u64 DirectoryBlobPartition::SetDOL(u64 dol_address) u64 DirectoryBlobPartition::SetDOL(u64 dol_address)
{ {
const DiscContent& dol = const DiscContent& dol =
AddFileToContents(&m_contents, m_root_directory + "sys/main.dol", dol_address); m_contents.CheckSizeAndAdd(dol_address, m_root_directory + "sys/main.dol");
Write32(static_cast<u32>(dol_address >> m_address_shift), 0x0420, &m_disc_header); Write32(static_cast<u32>(dol_address >> m_address_shift), 0x0420, &m_disc_header);
@ -685,7 +694,7 @@ void DirectoryBlobPartition::BuildFST(u64 fst_address)
Write32((u32)(m_fst_data.size() >> m_address_shift), 0x0428, &m_disc_header); Write32((u32)(m_fst_data.size() >> m_address_shift), 0x0428, &m_disc_header);
Write32((u32)(m_fst_data.size() >> m_address_shift), 0x042c, &m_disc_header); Write32((u32)(m_fst_data.size() >> m_address_shift), 0x042c, &m_disc_header);
m_contents.emplace(fst_address, m_fst_data.size(), m_fst_data.data()); m_contents.Add(fst_address, m_fst_data);
m_data_size = current_data_address; m_data_size = current_data_address;
} }
@ -747,8 +756,7 @@ void DirectoryBlobPartition::WriteDirectory(const File::FSTEntry& parent_entry,
WriteEntryName(name_offset, entry.virtualName, name_table_offset); WriteEntryName(name_offset, entry.virtualName, name_table_offset);
// write entry to virtual disc // write entry to virtual disc
auto result = m_contents.emplace(*data_offset, entry.size, entry.physicalName); auto result = m_contents.Add(*data_offset, entry.size, entry.physicalName);
_dbg_assert_(DISCIO, result.second); // Check that this offset wasn't already occupied
// 32 KiB aligned - many games are fine with less alignment, but not all // 32 KiB aligned - many games are fine with less alignment, but not all
*data_offset = Common::AlignUp(*data_offset + std::max<u64>(entry.size, 1ull), 0x8000ull); *data_offset = Common::AlignUp(*data_offset + std::max<u64>(entry.size, 1ull), 0x8000ull);
@ -756,12 +764,6 @@ void DirectoryBlobPartition::WriteDirectory(const File::FSTEntry& parent_entry,
} }
} }
static const DiscContent& AddFileToContents(std::set<DiscContent>* contents,
const std::string& path, u64 offset, u64 max_size)
{
return *(contents->emplace(offset, std::min(File::GetSize(path), max_size), path).first);
}
static size_t ReadFileToVector(const std::string& path, std::vector<u8>* vector) static size_t ReadFileToVector(const std::string& path, std::vector<u8>* vector)
{ {
File::IOFile file(path, "rb"); File::IOFile file(path, "rb");

View File

@ -58,6 +58,25 @@ private:
ContentSource m_content_source; ContentSource m_content_source;
}; };
class DiscContentContainer
{
public:
template <typename T>
const DiscContent& Add(u64 offset, const std::vector<T>& vector)
{
return Add(offset, vector.size() * sizeof(T), reinterpret_cast<const u8*>(vector.data()));
}
const DiscContent& Add(u64 offset, u64 size, const std::string& path);
const DiscContent& Add(u64 offset, u64 size, const u8* data);
const DiscContent& CheckSizeAndAdd(u64 offset, const std::string& path);
const DiscContent& CheckSizeAndAdd(u64 offset, u64 max_size, const std::string& path);
bool Read(u64 offset, u64 length, u8* buffer) const;
private:
std::set<DiscContent> m_contents;
};
class DirectoryBlobPartition class DirectoryBlobPartition
{ {
public: public:
@ -74,7 +93,7 @@ public:
u64 GetDataSize() const { return m_data_size; } u64 GetDataSize() const { return m_data_size; }
const std::string& GetRootDirectory() const { return m_root_directory; } const std::string& GetRootDirectory() const { return m_root_directory; }
const std::vector<u8>& GetHeader() const { return m_disc_header; } const std::vector<u8>& GetHeader() const { return m_disc_header; }
const std::set<DiscContent>& GetContents() const { return m_contents; } const DiscContentContainer& GetContents() const { return m_contents; }
private: private:
void SetDiscHeaderAndDiscType(std::optional<bool> is_wii); void SetDiscHeaderAndDiscType(std::optional<bool> is_wii);
void SetBI2(); void SetBI2();
@ -93,7 +112,7 @@ private:
void WriteDirectory(const File::FSTEntry& parent_entry, u32* fst_offset, u32* name_offset, void WriteDirectory(const File::FSTEntry& parent_entry, u32* fst_offset, u32* name_offset,
u64* data_offset, u32 parent_entry_index, u64 name_table_offset); u64* data_offset, u32 parent_entry_index, u64 name_table_offset);
std::set<DiscContent> m_contents; DiscContentContainer m_contents;
std::vector<u8> m_disc_header; std::vector<u8> m_disc_header;
std::vector<u8> m_bi2; std::vector<u8> m_bi2;
std::vector<u8> m_apploader; std::vector<u8> m_apploader;
@ -141,8 +160,6 @@ private:
explicit DirectoryBlobReader(const std::string& game_partition_root, explicit DirectoryBlobReader(const std::string& game_partition_root,
const std::string& true_root); const std::string& true_root);
bool ReadInternal(u64 offset, u64 length, u8* buffer, const std::set<DiscContent>& contents);
void SetNonpartitionDiscHeader(const std::vector<u8>& partition_header, void SetNonpartitionDiscHeader(const std::vector<u8>& partition_header,
const std::string& game_partition_root); const std::string& game_partition_root);
void SetWiiRegionData(const std::string& game_partition_root); void SetWiiRegionData(const std::string& game_partition_root);
@ -153,7 +170,7 @@ private:
DirectoryBlobPartition m_gamecube_pseudopartition; DirectoryBlobPartition m_gamecube_pseudopartition;
// For Wii: // For Wii:
std::set<DiscContent> m_nonpartition_contents; DiscContentContainer m_nonpartition_contents;
std::map<u64, DirectoryBlobPartition> m_partitions; std::map<u64, DirectoryBlobPartition> m_partitions;
bool m_is_wii; bool m_is_wii;