[VFS/STFS] Improved StfsHashEntry, minor struct improvements
Changed XEPACKEDSTRUCT to #pragma pack, added setters for some fields that didn't have them, named some anonymous structs which might help with other compilers.
This commit is contained in:
parent
aa155612fb
commit
2659b70c90
|
@ -176,16 +176,15 @@ StfsContainerDevice::Error StfsContainerDevice::ReadHeaderAndVerify(
|
||||||
const uint8_t* map_ptr, size_t map_size) {
|
const uint8_t* map_ptr, size_t map_size) {
|
||||||
// Copy header & check signature
|
// Copy header & check signature
|
||||||
memcpy(&header_, map_ptr, sizeof(StfsHeader));
|
memcpy(&header_, map_ptr, sizeof(StfsHeader));
|
||||||
if (header_.header.magic != XContentPackageType::kPackageTypeCon &&
|
if (!header_.header.is_magic_valid()) {
|
||||||
header_.header.magic != XContentPackageType::kPackageTypeLive &&
|
|
||||||
header_.header.magic != XContentPackageType::kPackageTypePirs) {
|
|
||||||
// Unexpected format.
|
// Unexpected format.
|
||||||
return Error::kErrorFileMismatch;
|
return Error::kErrorFileMismatch;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Pre-calculate some values used in block number calculations
|
// Pre-calculate some values used in block number calculations
|
||||||
blocks_per_hash_table_ =
|
blocks_per_hash_table_ =
|
||||||
header_.metadata.stfs_volume_descriptor.flags.read_only_format ? 1 : 2;
|
header_.metadata.volume_descriptor.stfs.flags.bits.read_only_format ? 1
|
||||||
|
: 2;
|
||||||
|
|
||||||
block_step[0] = kBlocksPerHashLevel[0] + blocks_per_hash_table_;
|
block_step[0] = kBlocksPerHashLevel[0] + blocks_per_hash_table_;
|
||||||
block_step[1] = kBlocksPerHashLevel[1] +
|
block_step[1] = kBlocksPerHashLevel[1] +
|
||||||
|
@ -202,7 +201,8 @@ StfsContainerDevice::Error StfsContainerDevice::ReadSVOD() {
|
||||||
const char* MEDIA_MAGIC = "MICROSOFT*XBOX*MEDIA";
|
const char* MEDIA_MAGIC = "MICROSOFT*XBOX*MEDIA";
|
||||||
|
|
||||||
// Check for EDGF layout
|
// Check for EDGF layout
|
||||||
if (header_.metadata.svod_volume_descriptor.features.enhanced_gdf_layout) {
|
if (header_.metadata.volume_descriptor.svod.features.bits
|
||||||
|
.enhanced_gdf_layout) {
|
||||||
// The STFS header has specified that this SVOD system uses the EGDF layout.
|
// The STFS header has specified that this SVOD system uses the EGDF layout.
|
||||||
// We can expect the magic block to be located immediately after the hash
|
// We can expect the magic block to be located immediately after the hash
|
||||||
// blocks. We also offset block address calculation by 0x1000 by shifting
|
// blocks. We also offset block address calculation by 0x1000 by shifting
|
||||||
|
@ -414,7 +414,7 @@ void StfsContainerDevice::BlockToOffsetSVOD(size_t block, size_t* out_address,
|
||||||
const size_t BLOCKS_PER_FILE = 0x14388;
|
const size_t BLOCKS_PER_FILE = 0x14388;
|
||||||
const size_t MAX_FILE_SIZE = 0xA290000;
|
const size_t MAX_FILE_SIZE = 0xA290000;
|
||||||
const size_t BLOCK_OFFSET =
|
const size_t BLOCK_OFFSET =
|
||||||
header_.metadata.svod_volume_descriptor.start_data_block();
|
header_.metadata.volume_descriptor.svod.start_data_block();
|
||||||
|
|
||||||
// Resolve the true block address and file index
|
// Resolve the true block address and file index
|
||||||
size_t true_block = block - (BLOCK_OFFSET * 2);
|
size_t true_block = block - (BLOCK_OFFSET * 2);
|
||||||
|
@ -463,7 +463,7 @@ StfsContainerDevice::Error StfsContainerDevice::ReadSTFS() {
|
||||||
std::vector<StfsContainerEntry*> all_entries;
|
std::vector<StfsContainerEntry*> all_entries;
|
||||||
|
|
||||||
// Load all listings.
|
// Load all listings.
|
||||||
auto& volume_descriptor = header_.metadata.stfs_volume_descriptor;
|
auto& volume_descriptor = header_.metadata.volume_descriptor.stfs;
|
||||||
uint32_t table_block_index = volume_descriptor.file_table_block_number();
|
uint32_t table_block_index = volume_descriptor.file_table_block_number();
|
||||||
for (size_t n = 0; n < volume_descriptor.file_table_block_count; n++) {
|
for (size_t n = 0; n < volume_descriptor.file_table_block_count; n++) {
|
||||||
const uint8_t* p = data + BlockToOffsetSTFS(table_block_index);
|
const uint8_t* p = data + BlockToOffsetSTFS(table_block_index);
|
||||||
|
@ -615,30 +615,29 @@ size_t StfsContainerDevice::BlockToHashBlockOffsetSTFS(
|
||||||
|
|
||||||
const StfsHashEntry* StfsContainerDevice::GetBlockHash(const uint8_t* map_ptr,
|
const StfsHashEntry* StfsContainerDevice::GetBlockHash(const uint8_t* map_ptr,
|
||||||
uint32_t block_index) {
|
uint32_t block_index) {
|
||||||
auto& volume_descriptor = header_.metadata.stfs_volume_descriptor;
|
auto& volume_descriptor = header_.metadata.volume_descriptor.stfs;
|
||||||
|
|
||||||
// Offset for selecting the secondary hash block, in packages that have them
|
// Offset for selecting the secondary hash block, in packages that have them
|
||||||
uint32_t secondary_table_offset =
|
uint32_t secondary_table_offset =
|
||||||
volume_descriptor.flags.root_active_index ? kSectorSize : 0;
|
volume_descriptor.flags.bits.root_active_index ? kSectorSize : 0;
|
||||||
|
|
||||||
auto hash_offset_lv0 = BlockToHashBlockOffsetSTFS(block_index, 0);
|
auto hash_offset_lv0 = BlockToHashBlockOffsetSTFS(block_index, 0);
|
||||||
if (!cached_hash_tables_.count(hash_offset_lv0)) {
|
if (!cached_hash_tables_.count(hash_offset_lv0)) {
|
||||||
// If this is read_only_format then it doesn't contain secondary blocks, no
|
// If this is read_only_format then it doesn't contain secondary blocks, no
|
||||||
// need to check upper hash levels
|
// need to check upper hash levels
|
||||||
if (volume_descriptor.flags.read_only_format) {
|
if (volume_descriptor.flags.bits.read_only_format) {
|
||||||
secondary_table_offset = 0;
|
secondary_table_offset = 0;
|
||||||
} else {
|
} else {
|
||||||
// Not a read-only package, need to check each levels active index flag to
|
// Not a read-only package, need to check each levels active index flag to
|
||||||
// see if we need to use secondary block or not
|
// see if we need to use secondary block or not
|
||||||
|
|
||||||
// Check level1 table if package has it
|
// Check level1 table if package has it
|
||||||
if (volume_descriptor.allocated_block_count > kBlocksPerHashLevel[0]) {
|
if (volume_descriptor.total_block_count > kBlocksPerHashLevel[0]) {
|
||||||
auto hash_offset_lv1 = BlockToHashBlockOffsetSTFS(block_index, 1);
|
auto hash_offset_lv1 = BlockToHashBlockOffsetSTFS(block_index, 1);
|
||||||
|
|
||||||
if (!cached_hash_tables_.count(hash_offset_lv1)) {
|
if (!cached_hash_tables_.count(hash_offset_lv1)) {
|
||||||
// Check level2 table if package has it
|
// Check level2 table if package has it
|
||||||
if (volume_descriptor.allocated_block_count >
|
if (volume_descriptor.total_block_count > kBlocksPerHashLevel[1]) {
|
||||||
kBlocksPerHashLevel[1]) {
|
|
||||||
auto hash_offset_lv2 = BlockToHashBlockOffsetSTFS(block_index, 2);
|
auto hash_offset_lv2 = BlockToHashBlockOffsetSTFS(block_index, 2);
|
||||||
|
|
||||||
if (!cached_hash_tables_.count(hash_offset_lv2)) {
|
if (!cached_hash_tables_.count(hash_offset_lv2)) {
|
||||||
|
@ -652,7 +651,7 @@ const StfsHashEntry* StfsContainerDevice::GetBlockHash(const uint8_t* map_ptr,
|
||||||
auto record_data =
|
auto record_data =
|
||||||
&cached_hash_tables_[hash_offset_lv2].entries[record];
|
&cached_hash_tables_[hash_offset_lv2].entries[record];
|
||||||
secondary_table_offset =
|
secondary_table_offset =
|
||||||
record_data->levelN_activeindex() ? kSectorSize : 0;
|
record_data->levelN_active_index() ? kSectorSize : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
cached_hash_tables_[hash_offset_lv1] =
|
cached_hash_tables_[hash_offset_lv1] =
|
||||||
|
@ -665,7 +664,7 @@ const StfsHashEntry* StfsContainerDevice::GetBlockHash(const uint8_t* map_ptr,
|
||||||
auto record_data =
|
auto record_data =
|
||||||
&cached_hash_tables_[hash_offset_lv1].entries[record];
|
&cached_hash_tables_[hash_offset_lv1].entries[record];
|
||||||
secondary_table_offset =
|
secondary_table_offset =
|
||||||
record_data->levelN_activeindex() ? kSectorSize : 0;
|
record_data->levelN_active_index() ? kSectorSize : 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -680,9 +679,10 @@ const StfsHashEntry* StfsContainerDevice::GetBlockHash(const uint8_t* map_ptr,
|
||||||
return record_data;
|
return record_data;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t StfsContainerDevice::ReadMagic(const std::filesystem::path& path) {
|
XContentPackageType StfsContainerDevice::ReadMagic(
|
||||||
|
const std::filesystem::path& path) {
|
||||||
auto map = MappedMemory::Open(path, MappedMemory::Mode::kRead, 0, 4);
|
auto map = MappedMemory::Open(path, MappedMemory::Mode::kRead, 0, 4);
|
||||||
return xe::load_and_swap<uint32_t>(map->data());
|
return XContentPackageType(xe::load_and_swap<uint32_t>(map->data()));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool StfsContainerDevice::ResolveFromFolder(const std::filesystem::path& path) {
|
bool StfsContainerDevice::ResolveFromFolder(const std::filesystem::path& path) {
|
||||||
|
@ -708,9 +708,9 @@ bool StfsContainerDevice::ResolveFromFolder(const std::filesystem::path& path) {
|
||||||
auto path = current_file.path / current_file.name;
|
auto path = current_file.path / current_file.name;
|
||||||
auto magic = ReadMagic(path);
|
auto magic = ReadMagic(path);
|
||||||
|
|
||||||
if (magic == XContentPackageType::kPackageTypeCon ||
|
if (magic == XContentPackageType::kCon ||
|
||||||
magic == XContentPackageType::kPackageTypeLive ||
|
magic == XContentPackageType::kLive ||
|
||||||
magic == XContentPackageType::kPackageTypePirs) {
|
magic == XContentPackageType::kPirs) {
|
||||||
host_path_ = current_file.path / current_file.name;
|
host_path_ = current_file.path / current_file.name;
|
||||||
XELOGI("STFS Package found: {}", xe::path_to_utf8(host_path_));
|
XELOGI("STFS Package found: {}", xe::path_to_utf8(host_path_));
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -80,7 +80,7 @@ class StfsContainerDevice : public Device {
|
||||||
kSingleFile = 0x4,
|
kSingleFile = 0x4,
|
||||||
};
|
};
|
||||||
|
|
||||||
uint32_t ReadMagic(const std::filesystem::path& path);
|
XContentPackageType ReadMagic(const std::filesystem::path& path);
|
||||||
bool ResolveFromFolder(const std::filesystem::path& path);
|
bool ResolveFromFolder(const std::filesystem::path& path);
|
||||||
|
|
||||||
Error MapFiles();
|
Error MapFiles();
|
||||||
|
|
|
@ -19,18 +19,21 @@ namespace vfs {
|
||||||
// Structs used for interchange between Xenia and actual Xbox360 kernel/XAM
|
// Structs used for interchange between Xenia and actual Xbox360 kernel/XAM
|
||||||
|
|
||||||
inline uint32_t load_uint24_be(const uint8_t* p) {
|
inline uint32_t load_uint24_be(const uint8_t* p) {
|
||||||
return (static_cast<uint32_t>(p[0]) << 16) |
|
return (uint32_t(p[0]) << 16) | (uint32_t(p[1]) << 8) | uint32_t(p[2]);
|
||||||
(static_cast<uint32_t>(p[1]) << 8) | static_cast<uint32_t>(p[2]);
|
|
||||||
}
|
}
|
||||||
inline uint32_t load_uint24_le(const uint8_t* p) {
|
inline uint32_t load_uint24_le(const uint8_t* p) {
|
||||||
return (static_cast<uint32_t>(p[2]) << 16) |
|
return (uint32_t(p[2]) << 16) | (uint32_t(p[1]) << 8) | uint32_t(p[0]);
|
||||||
(static_cast<uint32_t>(p[1]) << 8) | static_cast<uint32_t>(p[0]);
|
}
|
||||||
|
inline void store_uint24_le(uint8_t* p, uint32_t value) {
|
||||||
|
p[2] = uint8_t((value >> 16) & 0xFF);
|
||||||
|
p[1] = uint8_t((value >> 8) & 0xFF);
|
||||||
|
p[0] = uint8_t(value & 0xFF);
|
||||||
}
|
}
|
||||||
|
|
||||||
enum XContentPackageType : uint32_t {
|
enum class XContentPackageType : uint32_t {
|
||||||
kPackageTypeCon = 0x434F4E20,
|
kCon = 0x434F4E20,
|
||||||
kPackageTypePirs = 0x50495253,
|
kPirs = 0x50495253,
|
||||||
kPackageTypeLive = 0x4C495645,
|
kLive = 0x4C495645,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum XContentType : uint32_t {
|
enum XContentType : uint32_t {
|
||||||
|
@ -76,7 +79,8 @@ enum class XContentVolumeType : uint32_t {
|
||||||
};
|
};
|
||||||
|
|
||||||
/* STFS structures */
|
/* STFS structures */
|
||||||
XEPACKEDSTRUCT(StfsVolumeDescriptor, {
|
#pragma pack(push, 1)
|
||||||
|
struct StfsVolumeDescriptor {
|
||||||
uint8_t descriptor_length;
|
uint8_t descriptor_length;
|
||||||
uint8_t version;
|
uint8_t version;
|
||||||
union {
|
union {
|
||||||
|
@ -86,52 +90,77 @@ XEPACKEDSTRUCT(StfsVolumeDescriptor, {
|
||||||
// otherwise uses two
|
// otherwise uses two
|
||||||
uint8_t root_active_index : 1; // if set, uses secondary backing-block
|
uint8_t root_active_index : 1; // if set, uses secondary backing-block
|
||||||
// for the highest-level hash table
|
// for the highest-level hash table
|
||||||
|
|
||||||
uint8_t directory_overallocated : 1;
|
uint8_t directory_overallocated : 1;
|
||||||
uint8_t directory_index_bounds_valid : 1;
|
uint8_t directory_index_bounds_valid : 1;
|
||||||
};
|
} bits;
|
||||||
uint8_t as_byte;
|
uint8_t as_byte;
|
||||||
} flags;
|
} flags;
|
||||||
uint16_t file_table_block_count;
|
uint16_t file_table_block_count;
|
||||||
uint8_t file_table_block_number_0;
|
uint8_t file_table_block_number_raw[3];
|
||||||
uint8_t file_table_block_number_1;
|
|
||||||
uint8_t file_table_block_number_2;
|
|
||||||
uint8_t top_hash_table_hash[0x14];
|
uint8_t top_hash_table_hash[0x14];
|
||||||
be<uint32_t> allocated_block_count;
|
be<uint32_t> total_block_count;
|
||||||
be<uint32_t> free_block_count;
|
be<uint32_t> free_block_count;
|
||||||
|
|
||||||
uint32_t file_table_block_number() {
|
uint32_t file_table_block_number() const {
|
||||||
return uint32_t(file_table_block_number_0) |
|
return load_uint24_le(file_table_block_number_raw);
|
||||||
(uint32_t(file_table_block_number_1) << 8) |
|
|
||||||
(uint32_t(file_table_block_number_2) << 16);
|
|
||||||
}
|
}
|
||||||
});
|
|
||||||
|
void set_file_table_block_number(uint32_t value) {
|
||||||
|
store_uint24_le(file_table_block_number_raw, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool is_valid() const {
|
||||||
|
return descriptor_length == sizeof(StfsVolumeDescriptor);
|
||||||
|
}
|
||||||
|
};
|
||||||
static_assert_size(StfsVolumeDescriptor, 0x24);
|
static_assert_size(StfsVolumeDescriptor, 0x24);
|
||||||
|
#pragma pack(pop)
|
||||||
|
|
||||||
|
enum class StfsHashState : uint8_t {
|
||||||
|
kFree = 0, // unallocated but doesn't exist in package (needs to expand)?
|
||||||
|
kFree2 = 1, // unallocated but exists in package?
|
||||||
|
kInUse = 2,
|
||||||
|
};
|
||||||
|
|
||||||
struct StfsHashEntry {
|
struct StfsHashEntry {
|
||||||
uint8_t sha1[0x14];
|
uint8_t sha1[0x14];
|
||||||
|
|
||||||
uint8_t info0; // usually contains flags
|
xe::be<uint32_t> info_raw;
|
||||||
|
|
||||||
uint8_t info1;
|
uint32_t level0_next_block() const { return info_raw & 0xFFFFFF; }
|
||||||
uint8_t info2;
|
void set_level0_next_block(uint32_t value) {
|
||||||
uint8_t info3;
|
info_raw = (info_raw & ~0xFFFFFF) | (value & 0xFFFFFF);
|
||||||
|
|
||||||
// If this is a level0 entry, this points to the next block in the chain
|
|
||||||
uint32_t level0_next_block() const {
|
|
||||||
return uint32_t(info3) | (uint32_t(info2) << 8) | (uint32_t(info1) << 16);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void level0_next_block(uint32_t value) {
|
StfsHashState level0_allocation_state() const {
|
||||||
info3 = uint8_t(value & 0xFF);
|
return StfsHashState(uint8_t(((info_raw & 0xC0000000) >> 30) & 0xFF));
|
||||||
info2 = uint8_t((value >> 8) & 0xFF);
|
}
|
||||||
info1 = uint8_t((value >> 16) & 0xFF);
|
void set_level0_allocation_state(StfsHashState value) {
|
||||||
|
info_raw = (info_raw & ~0xC0000000) | (uint32_t(value) << 30);
|
||||||
}
|
}
|
||||||
|
|
||||||
// If this is level 1 or 2, this says whether the hash table this entry refers
|
uint32_t levelN_num_blocks_free() const { return info_raw & 0x7FFF; }
|
||||||
// to is using the secondary block or not
|
void set_levelN_num_blocks_free(uint32_t value) {
|
||||||
bool levelN_activeindex() const { return info0 & 0x40; }
|
info_raw = (info_raw & ~0x7FFF) | (value & 0x7FFF);
|
||||||
|
}
|
||||||
|
|
||||||
bool levelN_writeable() const { return info0 & 0x80; }
|
uint32_t levelN_num_blocks_unk() const {
|
||||||
|
return ((info_raw & 0x3FFF8000) >> 15) & 0x7FFF;
|
||||||
|
}
|
||||||
|
void set_levelN_num_blocks_unk(uint32_t value) {
|
||||||
|
info_raw = (info_raw & ~0x3FFF8000) | ((value & 0x7FFF) << 15);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool levelN_active_index() const { return (info_raw & 0x40000000) != 0; }
|
||||||
|
void set_levelN_active_index(bool value) {
|
||||||
|
info_raw = (info_raw & ~0x40000000) | (value ? 0x40000000 : 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool levelN_writeable() const { return (info_raw & 0x80000000) != 0; }
|
||||||
|
void set_levelN_writeable(bool value) {
|
||||||
|
info_raw = (info_raw & ~0x80000000) | (value ? 0x80000000 : 0);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
static_assert_size(StfsHashEntry, 0x18);
|
static_assert_size(StfsHashEntry, 0x18);
|
||||||
|
|
||||||
|
@ -154,30 +183,27 @@ struct SvodDeviceDescriptor {
|
||||||
uint8_t must_be_zero_for_future_usage : 6;
|
uint8_t must_be_zero_for_future_usage : 6;
|
||||||
uint8_t enhanced_gdf_layout : 1;
|
uint8_t enhanced_gdf_layout : 1;
|
||||||
uint8_t zero_for_downlevel_clients : 1;
|
uint8_t zero_for_downlevel_clients : 1;
|
||||||
};
|
} bits;
|
||||||
uint8_t as_byte;
|
uint8_t as_byte;
|
||||||
} features;
|
} features;
|
||||||
uint8_t num_data_blocks2;
|
uint8_t num_data_blocks_raw[3];
|
||||||
uint8_t num_data_blocks1;
|
uint8_t start_data_block_raw[3];
|
||||||
uint8_t num_data_blocks0;
|
|
||||||
uint8_t start_data_block0;
|
|
||||||
uint8_t start_data_block1;
|
|
||||||
uint8_t start_data_block2;
|
|
||||||
uint8_t reserved[5];
|
uint8_t reserved[5];
|
||||||
|
|
||||||
uint32_t num_data_blocks() {
|
uint32_t num_data_blocks() { return load_uint24_le(num_data_blocks_raw); }
|
||||||
return uint32_t(num_data_blocks0) | (uint32_t(num_data_blocks1) << 8) |
|
|
||||||
(uint32_t(num_data_blocks2) << 16);
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32_t start_data_block() {
|
uint32_t start_data_block() { return load_uint24_le(start_data_block_raw); }
|
||||||
return uint32_t(start_data_block0) | (uint32_t(start_data_block1) << 8) |
|
|
||||||
(uint32_t(start_data_block2) << 16);
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
static_assert_size(SvodDeviceDescriptor, 0x24);
|
static_assert_size(SvodDeviceDescriptor, 0x24);
|
||||||
|
|
||||||
/* XContent structures */
|
/* XContent structures */
|
||||||
|
struct XContentLicense {
|
||||||
|
be<uint64_t> licensee_id;
|
||||||
|
be<uint32_t> license_bits;
|
||||||
|
be<uint32_t> license_flags;
|
||||||
|
};
|
||||||
|
static_assert_size(XContentLicense, 0x10);
|
||||||
|
|
||||||
struct XContentMediaData {
|
struct XContentMediaData {
|
||||||
uint8_t series_id[0x10];
|
uint8_t series_id[0x10];
|
||||||
uint8_t season_id[0x10];
|
uint8_t season_id[0x10];
|
||||||
|
@ -206,7 +232,8 @@ struct XContentAttributes {
|
||||||
};
|
};
|
||||||
static_assert_size(XContentAttributes, 1);
|
static_assert_size(XContentAttributes, 1);
|
||||||
|
|
||||||
XEPACKEDSTRUCT(XContentMetadata, {
|
#pragma pack(push, 1)
|
||||||
|
struct XContentMetadata {
|
||||||
static const uint32_t kThumbLengthV1 = 0x4000;
|
static const uint32_t kThumbLengthV1 = 0x4000;
|
||||||
static const uint32_t kThumbLengthV2 = 0x3D00;
|
static const uint32_t kThumbLengthV2 = 0x3D00;
|
||||||
|
|
||||||
|
@ -221,9 +248,9 @@ XEPACKEDSTRUCT(XContentMetadata, {
|
||||||
uint8_t console_id[5];
|
uint8_t console_id[5];
|
||||||
be<uint64_t> profile_id;
|
be<uint64_t> profile_id;
|
||||||
union {
|
union {
|
||||||
StfsVolumeDescriptor stfs_volume_descriptor;
|
StfsVolumeDescriptor stfs;
|
||||||
SvodDeviceDescriptor svod_volume_descriptor;
|
SvodDeviceDescriptor svod;
|
||||||
};
|
} volume_descriptor;
|
||||||
be<uint32_t> data_file_count;
|
be<uint32_t> data_file_count;
|
||||||
be<uint64_t> data_file_size;
|
be<uint64_t> data_file_size;
|
||||||
be<XContentVolumeType> volume_type;
|
be<XContentVolumeType> volume_type;
|
||||||
|
@ -233,24 +260,24 @@ XEPACKEDSTRUCT(XContentMetadata, {
|
||||||
union {
|
union {
|
||||||
XContentMediaData media_data;
|
XContentMediaData media_data;
|
||||||
XContentAvatarAssetData avatar_asset_data;
|
XContentAvatarAssetData avatar_asset_data;
|
||||||
};
|
} metadata_v2;
|
||||||
uint8_t device_id[0x14];
|
uint8_t device_id[0x14];
|
||||||
union {
|
union {
|
||||||
be<uint16_t> display_name_raw[kNumLanguagesV1][128];
|
be<uint16_t> uint[kNumLanguagesV1][128];
|
||||||
char16_t display_name_chars[kNumLanguagesV1][128];
|
char16_t chars[kNumLanguagesV1][128];
|
||||||
};
|
} display_name_raw;
|
||||||
union {
|
union {
|
||||||
be<uint16_t> description_raw[kNumLanguagesV1][128];
|
be<uint16_t> uint[kNumLanguagesV1][128];
|
||||||
char16_t description_chars[kNumLanguagesV1][128];
|
char16_t chars[kNumLanguagesV1][128];
|
||||||
};
|
} description_raw;
|
||||||
union {
|
union {
|
||||||
be<uint16_t> publisher_raw[64];
|
be<uint16_t> uint[64];
|
||||||
char16_t publisher_chars[64];
|
char16_t chars[64];
|
||||||
};
|
} publisher_raw;
|
||||||
union {
|
union {
|
||||||
be<uint16_t> title_name_raw[64];
|
be<uint16_t> uint[64];
|
||||||
char16_t title_name_chars[64];
|
char16_t chars[64];
|
||||||
};
|
} title_name_raw;
|
||||||
union {
|
union {
|
||||||
XContentAttributes bits;
|
XContentAttributes bits;
|
||||||
uint8_t as_byte;
|
uint8_t as_byte;
|
||||||
|
@ -259,14 +286,14 @@ XEPACKEDSTRUCT(XContentMetadata, {
|
||||||
be<uint32_t> title_thumbnail_size;
|
be<uint32_t> title_thumbnail_size;
|
||||||
uint8_t thumbnail[kThumbLengthV2];
|
uint8_t thumbnail[kThumbLengthV2];
|
||||||
union {
|
union {
|
||||||
be<uint16_t> display_name_ex_raw[kNumLanguagesV2 - kNumLanguagesV1][128];
|
be<uint16_t> uint[kNumLanguagesV2 - kNumLanguagesV1][128];
|
||||||
char16_t display_name_ex_chars[kNumLanguagesV2 - kNumLanguagesV1][128];
|
char16_t chars[kNumLanguagesV2 - kNumLanguagesV1][128];
|
||||||
};
|
} display_name_ex_raw;
|
||||||
uint8_t title_thumbnail[kThumbLengthV2];
|
uint8_t title_thumbnail[kThumbLengthV2];
|
||||||
union {
|
union {
|
||||||
be<uint16_t> description_ex_raw[kNumLanguagesV2 - kNumLanguagesV1][128];
|
be<uint16_t> uint[kNumLanguagesV2 - kNumLanguagesV1][128];
|
||||||
char16_t description_ex_chars[kNumLanguagesV2 - kNumLanguagesV1][128];
|
char16_t chars[kNumLanguagesV2 - kNumLanguagesV1][128];
|
||||||
};
|
} description_ex_raw;
|
||||||
|
|
||||||
std::u16string display_name(XLanguage language) const {
|
std::u16string display_name(XLanguage language) const {
|
||||||
uint32_t lang_id = uint32_t(language) - 1;
|
uint32_t lang_id = uint32_t(language) - 1;
|
||||||
|
@ -279,10 +306,10 @@ XEPACKEDSTRUCT(XContentMetadata, {
|
||||||
|
|
||||||
const be<uint16_t>* str = 0;
|
const be<uint16_t>* str = 0;
|
||||||
if (lang_id >= 0 && lang_id < kNumLanguagesV1) {
|
if (lang_id >= 0 && lang_id < kNumLanguagesV1) {
|
||||||
str = display_name_raw[lang_id];
|
str = display_name_raw.uint[lang_id];
|
||||||
} else if (lang_id >= kNumLanguagesV1 && lang_id < kNumLanguagesV2 &&
|
} else if (lang_id >= kNumLanguagesV1 && lang_id < kNumLanguagesV2 &&
|
||||||
metadata_version >= 2) {
|
metadata_version >= 2) {
|
||||||
str = display_name_ex_raw[lang_id - kNumLanguagesV1];
|
str = display_name_ex_raw.uint[lang_id - kNumLanguagesV1];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!str) {
|
if (!str) {
|
||||||
|
@ -305,10 +332,10 @@ XEPACKEDSTRUCT(XContentMetadata, {
|
||||||
|
|
||||||
const be<uint16_t>* str = 0;
|
const be<uint16_t>* str = 0;
|
||||||
if (lang_id >= 0 && lang_id < kNumLanguagesV1) {
|
if (lang_id >= 0 && lang_id < kNumLanguagesV1) {
|
||||||
str = description_raw[lang_id];
|
str = description_raw.uint[lang_id];
|
||||||
} else if (lang_id >= kNumLanguagesV1 && lang_id < kNumLanguagesV2 &&
|
} else if (lang_id >= kNumLanguagesV1 && lang_id < kNumLanguagesV2 &&
|
||||||
metadata_version >= 2) {
|
metadata_version >= 2) {
|
||||||
str = description_ex_raw[lang_id - kNumLanguagesV1];
|
str = description_ex_raw.uint[lang_id - kNumLanguagesV1];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!str) {
|
if (!str) {
|
||||||
|
@ -321,11 +348,11 @@ XEPACKEDSTRUCT(XContentMetadata, {
|
||||||
}
|
}
|
||||||
|
|
||||||
std::u16string publisher() const {
|
std::u16string publisher() const {
|
||||||
return load_and_swap<std::u16string>(publisher_raw);
|
return load_and_swap<std::u16string>(publisher_raw.uint);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::u16string title_name() const {
|
std::u16string title_name() const {
|
||||||
return load_and_swap<std::u16string>(title_name_raw);
|
return load_and_swap<std::u16string>(title_name_raw.uint);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool set_display_name(XLanguage language, const std::u16string_view value) {
|
bool set_display_name(XLanguage language, const std::u16string_view value) {
|
||||||
|
@ -339,10 +366,10 @@ XEPACKEDSTRUCT(XContentMetadata, {
|
||||||
|
|
||||||
char16_t* str = 0;
|
char16_t* str = 0;
|
||||||
if (lang_id >= 0 && lang_id < kNumLanguagesV1) {
|
if (lang_id >= 0 && lang_id < kNumLanguagesV1) {
|
||||||
str = display_name_chars[lang_id];
|
str = display_name_raw.chars[lang_id];
|
||||||
} else if (lang_id >= kNumLanguagesV1 && lang_id < kNumLanguagesV2 &&
|
} else if (lang_id >= kNumLanguagesV1 && lang_id < kNumLanguagesV2 &&
|
||||||
metadata_version >= 2) {
|
metadata_version >= 2) {
|
||||||
str = display_name_ex_chars[lang_id - kNumLanguagesV1];
|
str = display_name_ex_raw.chars[lang_id - kNumLanguagesV1];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!str) {
|
if (!str) {
|
||||||
|
@ -352,7 +379,7 @@ XEPACKEDSTRUCT(XContentMetadata, {
|
||||||
}
|
}
|
||||||
|
|
||||||
string_util::copy_and_swap_truncating(str, value,
|
string_util::copy_and_swap_truncating(str, value,
|
||||||
countof(display_name_chars[0]));
|
countof(display_name_raw.chars[0]));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -367,10 +394,10 @@ XEPACKEDSTRUCT(XContentMetadata, {
|
||||||
|
|
||||||
char16_t* str = 0;
|
char16_t* str = 0;
|
||||||
if (lang_id >= 0 && lang_id < kNumLanguagesV1) {
|
if (lang_id >= 0 && lang_id < kNumLanguagesV1) {
|
||||||
str = description_chars[lang_id];
|
str = description_raw.chars[lang_id];
|
||||||
} else if (lang_id >= kNumLanguagesV1 && lang_id < kNumLanguagesV2 &&
|
} else if (lang_id >= kNumLanguagesV1 && lang_id < kNumLanguagesV2 &&
|
||||||
metadata_version >= 2) {
|
metadata_version >= 2) {
|
||||||
str = description_ex_chars[lang_id - kNumLanguagesV1];
|
str = description_ex_raw.chars[lang_id - kNumLanguagesV1];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!str) {
|
if (!str) {
|
||||||
|
@ -380,39 +407,37 @@ XEPACKEDSTRUCT(XContentMetadata, {
|
||||||
}
|
}
|
||||||
|
|
||||||
string_util::copy_and_swap_truncating(str, value,
|
string_util::copy_and_swap_truncating(str, value,
|
||||||
countof(description_chars[0]));
|
countof(description_raw.chars[0]));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool set_publisher(const std::u16string_view value) {
|
void set_publisher(const std::u16string_view value) {
|
||||||
string_util::copy_and_swap_truncating(publisher_chars, value,
|
string_util::copy_and_swap_truncating(publisher_raw.chars, value,
|
||||||
countof(publisher_chars));
|
countof(publisher_raw.chars));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool set_title_name(const std::u16string_view value) {
|
void set_title_name(const std::u16string_view value) {
|
||||||
string_util::copy_and_swap_truncating(title_name_chars, value,
|
string_util::copy_and_swap_truncating(title_name_raw.chars, value,
|
||||||
countof(title_name_chars));
|
countof(title_name_raw.chars));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
});
|
};
|
||||||
static_assert_size(XContentMetadata, 0x93D6);
|
static_assert_size(XContentMetadata, 0x93D6);
|
||||||
|
|
||||||
struct XContentLicense {
|
struct XContentHeader {
|
||||||
be<uint64_t> licensee_id;
|
|
||||||
be<uint32_t> license_bits;
|
|
||||||
be<uint32_t> license_flags;
|
|
||||||
};
|
|
||||||
static_assert_size(XContentLicense, 0x10);
|
|
||||||
|
|
||||||
XEPACKEDSTRUCT(XContentHeader, {
|
|
||||||
be<XContentPackageType> magic;
|
be<XContentPackageType> magic;
|
||||||
uint8_t signature[0x228];
|
uint8_t signature[0x228];
|
||||||
XContentLicense licenses[0x10];
|
XContentLicense licenses[0x10];
|
||||||
uint8_t content_id[0x14];
|
uint8_t content_id[0x14];
|
||||||
be<uint32_t> header_size;
|
be<uint32_t> header_size;
|
||||||
});
|
|
||||||
|
bool is_magic_valid() const {
|
||||||
|
return magic == XContentPackageType::kCon ||
|
||||||
|
magic == XContentPackageType::kLive ||
|
||||||
|
magic == XContentPackageType::kPirs;
|
||||||
|
}
|
||||||
|
};
|
||||||
static_assert_size(XContentHeader, 0x344);
|
static_assert_size(XContentHeader, 0x344);
|
||||||
|
#pragma pack(pop)
|
||||||
|
|
||||||
struct StfsHeader {
|
struct StfsHeader {
|
||||||
XContentHeader header;
|
XContentHeader header;
|
||||||
|
|
Loading…
Reference in New Issue