[Kernel] Merge Content*Data structs into XCONTENT*_DATA

Also moved XContentType to xbox.h so code can make use of it without needing to include STFS stuff
This commit is contained in:
emoose 2021-06-15 22:13:30 +01:00 committed by Rick Gibbed
parent e9f1a534df
commit 13f30d3492
6 changed files with 157 additions and 206 deletions

View File

@ -31,7 +31,7 @@ static int content_device_id_ = 0;
ContentPackage::ContentPackage(KernelState* kernel_state, ContentPackage::ContentPackage(KernelState* kernel_state,
const std::string_view root_name, const std::string_view root_name,
const ContentData& data, const XCONTENT_DATA& data,
const std::filesystem::path& package_path) const std::filesystem::path& package_path)
: kernel_state_(kernel_state), root_name_(root_name) { : kernel_state_(kernel_state), root_name_(root_name) {
device_path_ = fmt::format("\\Device\\Content\\{0}\\", ++content_device_id_); device_path_ = fmt::format("\\Device\\Content\\{0}\\", ++content_device_id_);
@ -58,48 +58,26 @@ ContentManager::ContentManager(KernelState* kernel_state,
ContentManager::~ContentManager() = default; ContentManager::~ContentManager() = default;
std::filesystem::path ContentManager::ResolvePackageRoot( std::filesystem::path ContentManager::ResolvePackageRoot(
uint32_t content_type) { XContentType content_type) {
auto title_id = fmt::format("{:8X}", kernel_state_->title_id()); auto title_id_str = fmt::format("{:8X}", kernel_state_->title_id());
auto content_type_str = fmt::format("{:08X}", uint32_t(content_type));
std::string type_name;
switch (content_type) {
case 1:
// Save games.
type_name = "00000001";
break;
case 2:
// DLC from the marketplace.
type_name = "00000002";
break;
case 3:
// Publisher content?
type_name = "00000003";
break;
case 0x000D0000:
// ???
type_name = "000D0000";
break;
default:
assert_unhandled_case(data.content_type);
return std::filesystem::path();
}
// Package root path: // Package root path:
// content_root/title_id/type_name/ // content_root/title_id/content_type/
return root_path_ / title_id / type_name; return root_path_ / title_id_str / content_type_str;
} }
std::filesystem::path ContentManager::ResolvePackagePath( std::filesystem::path ContentManager::ResolvePackagePath(
const ContentData& data) { const XCONTENT_DATA& data) {
// Content path: // Content path:
// content_root/title_id/type_name/data_file_name/ // content_root/title_id/content_type/data_file_name/
auto package_root = ResolvePackageRoot(data.content_type); auto package_root = ResolvePackageRoot(data.content_type);
return package_root / xe::to_path(data.file_name); return package_root / xe::to_path(data.file_name());
} }
std::vector<ContentData> ContentManager::ListContent(uint32_t device_id, std::vector<XCONTENT_DATA> ContentManager::ListContent(
uint32_t content_type) { uint32_t device_id, XContentType content_type) {
std::vector<ContentData> result; std::vector<XCONTENT_DATA> result;
// Search path: // Search path:
// content_root/title_id/type_name/* // content_root/title_id/type_name/*
@ -110,11 +88,11 @@ std::vector<ContentData> ContentManager::ListContent(uint32_t device_id,
// Directories only. // Directories only.
continue; continue;
} }
ContentData content_data; XCONTENT_DATA content_data;
content_data.device_id = device_id; content_data.device_id = device_id;
content_data.content_type = content_type; content_data.content_type = content_type;
content_data.display_name = xe::path_to_utf16(file_info.name); content_data.set_display_name(xe::path_to_utf16(file_info.name));
content_data.file_name = xe::path_to_utf8(file_info.name); content_data.set_file_name(xe::path_to_utf8(file_info.name));
result.emplace_back(std::move(content_data)); result.emplace_back(std::move(content_data));
} }
@ -122,7 +100,7 @@ std::vector<ContentData> ContentManager::ListContent(uint32_t device_id,
} }
std::unique_ptr<ContentPackage> ContentManager::ResolvePackage( std::unique_ptr<ContentPackage> ContentManager::ResolvePackage(
const std::string_view root_name, const ContentData& data) { const std::string_view root_name, const XCONTENT_DATA& data) {
auto package_path = ResolvePackagePath(data); auto package_path = ResolvePackagePath(data);
if (!std::filesystem::exists(package_path)) { if (!std::filesystem::exists(package_path)) {
return nullptr; return nullptr;
@ -135,13 +113,13 @@ std::unique_ptr<ContentPackage> ContentManager::ResolvePackage(
return package; return package;
} }
bool ContentManager::ContentExists(const ContentData& data) { bool ContentManager::ContentExists(const XCONTENT_DATA& data) {
auto path = ResolvePackagePath(data); auto path = ResolvePackagePath(data);
return std::filesystem::exists(path); return std::filesystem::exists(path);
} }
X_RESULT ContentManager::CreateContent(const std::string_view root_name, X_RESULT ContentManager::CreateContent(const std::string_view root_name,
const ContentData& data) { const XCONTENT_DATA& data) {
auto global_lock = global_critical_region_.Acquire(); auto global_lock = global_critical_region_.Acquire();
if (open_packages_.count(string_key(root_name))) { if (open_packages_.count(string_key(root_name))) {
@ -168,7 +146,7 @@ X_RESULT ContentManager::CreateContent(const std::string_view root_name,
} }
X_RESULT ContentManager::OpenContent(const std::string_view root_name, X_RESULT ContentManager::OpenContent(const std::string_view root_name,
const ContentData& data) { const XCONTENT_DATA& data) {
auto global_lock = global_critical_region_.Acquire(); auto global_lock = global_critical_region_.Acquire();
if (open_packages_.count(string_key(root_name))) { if (open_packages_.count(string_key(root_name))) {
@ -207,7 +185,7 @@ X_RESULT ContentManager::CloseContent(const std::string_view root_name) {
return X_ERROR_SUCCESS; return X_ERROR_SUCCESS;
} }
X_RESULT ContentManager::GetContentThumbnail(const ContentData& data, X_RESULT ContentManager::GetContentThumbnail(const XCONTENT_DATA& data,
std::vector<uint8_t>* buffer) { std::vector<uint8_t>* buffer) {
auto global_lock = global_critical_region_.Acquire(); auto global_lock = global_critical_region_.Acquire();
auto package_path = ResolvePackagePath(data); auto package_path = ResolvePackagePath(data);
@ -226,7 +204,7 @@ X_RESULT ContentManager::GetContentThumbnail(const ContentData& data,
} }
} }
X_RESULT ContentManager::SetContentThumbnail(const ContentData& data, X_RESULT ContentManager::SetContentThumbnail(const XCONTENT_DATA& data,
std::vector<uint8_t> buffer) { std::vector<uint8_t> buffer) {
auto global_lock = global_critical_region_.Acquire(); auto global_lock = global_critical_region_.Acquire();
auto package_path = ResolvePackagePath(data); auto package_path = ResolvePackagePath(data);
@ -242,7 +220,7 @@ X_RESULT ContentManager::SetContentThumbnail(const ContentData& data,
} }
} }
X_RESULT ContentManager::DeleteContent(const ContentData& data) { X_RESULT ContentManager::DeleteContent(const XCONTENT_DATA& data) {
auto global_lock = global_critical_region_.Acquire(); auto global_lock = global_critical_region_.Acquire();
if (IsContentOpen(data)) { if (IsContentOpen(data)) {
@ -267,7 +245,7 @@ std::filesystem::path ContentManager::ResolveGameUserContentPath() {
return root_path_ / title_id / kGameUserContentDirName / user_name; return root_path_ / title_id / kGameUserContentDirName / user_name;
} }
bool ContentManager::IsContentOpen(const ContentData& data) const { bool ContentManager::IsContentOpen(const XCONTENT_DATA& data) const {
return std::any_of(open_packages_.cbegin(), open_packages_.cend(), return std::any_of(open_packages_.cbegin(), open_packages_.cend(),
[data](std::pair<string_key, ContentPackage*> content) { [data](std::pair<string_key, ContentPackage*> content) {
return data == content.second->GetPackageContentData(); return data == content.second->GetPackageContentData();

View File

@ -33,105 +33,87 @@ namespace xam {
struct XCONTENT_DATA { struct XCONTENT_DATA {
be<uint32_t> device_id; be<uint32_t> device_id;
be<uint32_t> content_type; be<XContentType> content_type;
union { union {
be<uint16_t> display_name[128]; // this should be be<uint16_t>, but that stops copy constructor being
char16_t display_name_chars[128]; // generated...
}; uint16_t uint[128];
char file_name[42]; char16_t chars[128];
uint8_t padding[2]; } display_name_raw;
};
static_assert_size(XCONTENT_DATA, 308);
struct XCONTENT_AGGREGATE_DATA { char file_name_raw[42];
be<uint32_t> device_id;
be<uint32_t> content_type; // Some games use this padding field as a null-terminator, as eg.
union { // DLC packages usually fill the entire file_name_raw array
be<uint16_t> display_name[128]; // Not every game sets it to 0 though, so make sure any file_name_raw reads
char16_t display_name_chars[128]; // only go up to 42 chars!
};
char file_name[42];
uint8_t padding[2]; uint8_t padding[2];
bool operator==(const XCONTENT_DATA& other) const {
// Package is located via device_id/content_type/file_name, so only need to
// compare those
return device_id == other.device_id && content_type == other.content_type &&
file_name() == other.file_name();
}
std::u16string display_name() const {
return load_and_swap<std::u16string>(display_name_raw.uint);
}
std::string file_name() const {
std::string value;
value.assign(file_name_raw,
std::min(strlen(file_name_raw), countof(file_name_raw)));
return value;
}
void set_display_name(const std::u16string_view value) {
// Some games (eg Goldeneye XBLA) require multiple null-terminators for it
// to read the string properly, blanking the array should take care of that
std::fill_n(display_name_raw.chars, countof(display_name_raw.chars), 0);
string_util::copy_and_swap_truncating(display_name_raw.chars, value,
countof(display_name_raw.chars));
}
void set_file_name(const std::string_view value) {
std::fill_n(file_name_raw, countof(file_name_raw), 0);
string_util::copy_maybe_truncating<string_util::Safety::IKnowWhatIAmDoing>(
file_name_raw, value, xe::countof(file_name_raw));
// Some games rely on padding field acting as a null-terminator...
padding[0] = padding[1] = 0;
}
};
static_assert_size(XCONTENT_DATA, 0x134);
struct XCONTENT_AGGREGATE_DATA : XCONTENT_DATA {
be<uint32_t> title_id; be<uint32_t> title_id;
};
static_assert_size(XCONTENT_AGGREGATE_DATA, 312);
struct ContentData { bool operator==(const XCONTENT_AGGREGATE_DATA& other) const {
uint32_t device_id; // Package is located via device_id/title_id/content_type/file_name, so only
uint32_t content_type; // need to compare those
std::u16string display_name; return device_id == other.device_id && title_id == other.title_id &&
std::string file_name; content_type == other.content_type &&
file_name() == other.file_name();
ContentData() = default;
bool operator==(const ContentData& rhs) const {
return device_id == rhs.device_id && content_type == rhs.content_type &&
file_name == rhs.file_name;
}
explicit ContentData(const XCONTENT_DATA& data) {
device_id = data.device_id;
content_type = data.content_type;
display_name = xe::load_and_swap<std::u16string>(data.display_name);
file_name = xe::load_and_swap<std::string>(data.file_name);
}
void Write(XCONTENT_DATA* data) const {
data->device_id = device_id;
data->content_type = content_type;
xe::string_util::copy_and_swap_truncating(
data->display_name_chars, display_name,
xe::countof(data->display_name_chars));
xe::string_util::copy_maybe_truncating<
string_util::Safety::IKnowWhatIAmDoing>(data->file_name, file_name,
xe::countof(data->file_name));
}
};
struct ContentAggregateData {
uint32_t device_id;
uint32_t content_type;
std::u16string display_name;
std::string file_name;
uint32_t title_id;
ContentAggregateData() = default;
explicit ContentAggregateData(const XCONTENT_AGGREGATE_DATA& data) {
device_id = data.device_id;
content_type = data.content_type;
display_name = xe::load_and_swap<std::u16string>(data.display_name);
file_name = xe::load_and_swap<std::string>(data.file_name);
title_id = data.title_id;
}
void Write(XCONTENT_AGGREGATE_DATA* data) const {
data->device_id = device_id;
data->content_type = content_type;
xe::string_util::copy_and_swap_truncating(
data->display_name_chars, display_name,
xe::countof(data->display_name_chars));
xe::string_util::copy_maybe_truncating<
string_util::Safety::IKnowWhatIAmDoing>(data->file_name, file_name,
xe::countof(data->file_name));
data->title_id = title_id;
} }
}; };
static_assert_size(XCONTENT_AGGREGATE_DATA, 0x138);
class ContentPackage { class ContentPackage {
public: public:
ContentPackage(KernelState* kernel_state, const std::string_view root_name, ContentPackage(KernelState* kernel_state, const std::string_view root_name,
const ContentData& data, const XCONTENT_DATA& data,
const std::filesystem::path& package_path); const std::filesystem::path& package_path);
~ContentPackage(); ~ContentPackage();
const ContentData& GetPackageContentData() const { return content_data_; } const XCONTENT_DATA& GetPackageContentData() const { return content_data_; }
private: private:
KernelState* kernel_state_; KernelState* kernel_state_;
std::string root_name_; std::string root_name_;
std::string device_path_; std::string device_path_;
ContentData content_data_; XCONTENT_DATA content_data_;
}; };
class ContentManager { class ContentManager {
@ -140,30 +122,30 @@ class ContentManager {
const std::filesystem::path& root_path); const std::filesystem::path& root_path);
~ContentManager(); ~ContentManager();
std::vector<ContentData> ListContent(uint32_t device_id, std::vector<XCONTENT_DATA> ListContent(uint32_t device_id,
uint32_t content_type); XContentType content_type);
std::unique_ptr<ContentPackage> ResolvePackage( std::unique_ptr<ContentPackage> ResolvePackage(
const std::string_view root_name, const ContentData& data); const std::string_view root_name, const XCONTENT_DATA& data);
bool ContentExists(const ContentData& data); bool ContentExists(const XCONTENT_DATA& data);
X_RESULT CreateContent(const std::string_view root_name, X_RESULT CreateContent(const std::string_view root_name,
const ContentData& data); const XCONTENT_DATA& data);
X_RESULT OpenContent(const std::string_view root_name, X_RESULT OpenContent(const std::string_view root_name,
const ContentData& data); const XCONTENT_DATA& data);
X_RESULT CloseContent(const std::string_view root_name); X_RESULT CloseContent(const std::string_view root_name);
X_RESULT GetContentThumbnail(const ContentData& data, X_RESULT GetContentThumbnail(const XCONTENT_DATA& data,
std::vector<uint8_t>* buffer); std::vector<uint8_t>* buffer);
X_RESULT SetContentThumbnail(const ContentData& data, X_RESULT SetContentThumbnail(const XCONTENT_DATA& data,
std::vector<uint8_t> buffer); std::vector<uint8_t> buffer);
X_RESULT DeleteContent(const ContentData& data); X_RESULT DeleteContent(const XCONTENT_DATA& data);
std::filesystem::path ResolveGameUserContentPath(); std::filesystem::path ResolveGameUserContentPath();
bool IsContentOpen(const ContentData& data) const; bool IsContentOpen(const XCONTENT_DATA& data) const;
void CloseOpenedFilesFromContent(const std::string_view root_name); void CloseOpenedFilesFromContent(const std::string_view root_name);
private: private:
std::filesystem::path ResolvePackageRoot(uint32_t content_type); std::filesystem::path ResolvePackageRoot(XContentType content_type);
std::filesystem::path ResolvePackagePath(const ContentData& data); std::filesystem::path ResolvePackagePath(const XCONTENT_DATA& data);
KernelState* kernel_state_; KernelState* kernel_state_;
std::filesystem::path root_path_; std::filesystem::path root_path_;

View File

@ -96,11 +96,12 @@ dword_result_t XamContentCreateEnumerator(dword_t user_index, dword_t device_id,
if (!device_info || device_info->device_id == DummyDeviceId::HDD) { if (!device_info || device_info->device_id == DummyDeviceId::HDD) {
// Get all content data. // Get all content data.
auto content_datas = kernel_state()->content_manager()->ListContent( auto content_datas = kernel_state()->content_manager()->ListContent(
static_cast<uint32_t>(DummyDeviceId::HDD), content_type); static_cast<uint32_t>(DummyDeviceId::HDD),
XContentType(uint32_t(content_type)));
for (const auto& content_data : content_datas) { for (const auto& content_data : content_datas) {
auto item = reinterpret_cast<XCONTENT_DATA*>(e->AppendItem()); auto item = reinterpret_cast<XCONTENT_DATA*>(e->AppendItem());
assert_not_null(item); assert_not_null(item);
content_data.Write(item); *item = content_data;
} }
} }
@ -123,8 +124,7 @@ dword_result_t XamContentCreateEx(dword_t user_index, lpstring_t root_name,
dword_t cache_size, qword_t content_size, dword_t cache_size, qword_t content_size,
lpvoid_t overlapped_ptr) { lpvoid_t overlapped_ptr) {
X_RESULT result = X_ERROR_INVALID_PARAMETER; X_RESULT result = X_ERROR_INVALID_PARAMETER;
auto content_data = auto content_data = *content_data_ptr.as<XCONTENT_DATA*>();
static_cast<ContentData>(*content_data_ptr.as<XCONTENT_DATA*>());
auto content_manager = kernel_state()->content_manager(); auto content_manager = kernel_state()->content_manager();
bool create = false; bool create = false;
@ -277,14 +277,13 @@ dword_result_t XamContentGetCreator(dword_t user_index,
lpunknown_t overlapped_ptr) { lpunknown_t overlapped_ptr) {
auto result = X_ERROR_SUCCESS; auto result = X_ERROR_SUCCESS;
auto content_data = auto content_data = *content_data_ptr.as<XCONTENT_DATA*>();
static_cast<ContentData>(*content_data_ptr.as<XCONTENT_DATA*>());
bool content_exists = bool content_exists =
kernel_state()->content_manager()->ContentExists(content_data); kernel_state()->content_manager()->ContentExists(content_data);
if (content_exists) { if (content_exists) {
if (content_data.content_type == 1) { if (content_data.content_type == XContentType::kSavedGame) {
// User always creates saves. // User always creates saves.
*is_creator_ptr = 1; *is_creator_ptr = 1;
if (creator_xuid_ptr) { if (creator_xuid_ptr) {
@ -316,8 +315,7 @@ dword_result_t XamContentGetThumbnail(dword_t user_index,
lpunknown_t overlapped_ptr) { lpunknown_t overlapped_ptr) {
assert_not_null(buffer_size_ptr); assert_not_null(buffer_size_ptr);
uint32_t buffer_size = *buffer_size_ptr; uint32_t buffer_size = *buffer_size_ptr;
auto content_data = auto content_data = *content_data_ptr.as<XCONTENT_DATA*>();
static_cast<ContentData>(*content_data_ptr.as<XCONTENT_DATA*>());
// Get thumbnail (if it exists). // Get thumbnail (if it exists).
std::vector<uint8_t> buffer; std::vector<uint8_t> buffer;
@ -353,8 +351,7 @@ dword_result_t XamContentSetThumbnail(dword_t user_index,
lpvoid_t content_data_ptr, lpvoid_t content_data_ptr,
lpvoid_t buffer_ptr, dword_t buffer_size, lpvoid_t buffer_ptr, dword_t buffer_size,
lpunknown_t overlapped_ptr) { lpunknown_t overlapped_ptr) {
auto content_data = auto content_data = *content_data_ptr.as<XCONTENT_DATA*>();
static_cast<ContentData>(*content_data_ptr.as<XCONTENT_DATA*>());
// Buffer is PNG data. // Buffer is PNG data.
auto buffer = std::vector<uint8_t>((uint8_t*)buffer_ptr, auto buffer = std::vector<uint8_t>((uint8_t*)buffer_ptr,
@ -373,8 +370,7 @@ DECLARE_XAM_EXPORT1(XamContentSetThumbnail, kContent, kImplemented);
dword_result_t XamContentDelete(dword_t user_index, lpvoid_t content_data_ptr, dword_result_t XamContentDelete(dword_t user_index, lpvoid_t content_data_ptr,
lpunknown_t overlapped_ptr) { lpunknown_t overlapped_ptr) {
auto content_data = auto content_data = *content_data_ptr.as<XCONTENT_DATA*>();
static_cast<ContentData>(*content_data_ptr.as<XCONTENT_DATA*>());
auto result = kernel_state()->content_manager()->DeleteContent(content_data); auto result = kernel_state()->content_manager()->DeleteContent(content_data);

View File

@ -22,14 +22,15 @@ namespace xe {
namespace kernel { namespace kernel {
namespace xam { namespace xam {
void AddODDContentTest(object_ref<XStaticEnumerator> e, uint32_t content_type) { void AddODDContentTest(object_ref<XStaticEnumerator> e,
XContentType content_type) {
auto root_entry = kernel_state()->file_system()->ResolvePath( auto root_entry = kernel_state()->file_system()->ResolvePath(
"game:\\Content\\0000000000000000"); "game:\\Content\\0000000000000000");
if (!root_entry) { if (!root_entry) {
return; return;
} }
auto content_type_path = fmt::format("{:08X}", content_type); auto content_type_path = fmt::format("{:08X}", uint32_t(content_type));
xe::filesystem::WildcardEngine title_find_engine; xe::filesystem::WildcardEngine title_find_engine;
title_find_engine.SetRule("????????"); title_find_engine.SetRule("????????");
@ -62,14 +63,11 @@ void AddODDContentTest(object_ref<XStaticEnumerator> e, uint32_t content_type) {
auto item = reinterpret_cast<XCONTENT_AGGREGATE_DATA*>(e->AppendItem()); auto item = reinterpret_cast<XCONTENT_AGGREGATE_DATA*>(e->AppendItem());
assert_not_null(item); assert_not_null(item);
ContentAggregateData content_aggregate_data = {}; item->device_id = static_cast<uint32_t>(DummyDeviceId::ODD);
content_aggregate_data.device_id = item->content_type = content_type;
static_cast<uint32_t>(DummyDeviceId::ODD); item->set_display_name(to_utf16(content_entry->name()));
content_aggregate_data.content_type = content_type; item->set_file_name(content_entry->name());
content_aggregate_data.display_name = to_utf16(content_entry->name()); item->title_id = title_id;
content_aggregate_data.file_name = content_entry->name();
content_aggregate_data.title_id = title_id;
content_aggregate_data.Write(item);
} }
} }
} }
@ -98,25 +96,22 @@ dword_result_t XamContentAggregateCreateEnumerator(qword_t xuid,
extra->magic = kXObjSignature; extra->magic = kXObjSignature;
extra->handle = e->handle(); extra->handle = e->handle();
auto content_type_enum = XContentType(uint32_t(content_type));
if (!device_info || device_info->device_type == DeviceType::HDD) { if (!device_info || device_info->device_type == DeviceType::HDD) {
// Get all content data. // Get all content data.
auto content_datas = kernel_state()->content_manager()->ListContent( auto content_datas = kernel_state()->content_manager()->ListContent(
static_cast<uint32_t>(DummyDeviceId::HDD), content_type); static_cast<uint32_t>(DummyDeviceId::HDD), content_type_enum);
for (const auto& content_data : content_datas) { for (const auto& content_data : content_datas) {
auto item = reinterpret_cast<XCONTENT_AGGREGATE_DATA*>(e->AppendItem()); auto item = reinterpret_cast<XCONTENT_AGGREGATE_DATA*>(e->AppendItem());
assert_not_null(item); assert_not_null(item);
ContentAggregateData content_aggregate_data = {}; *item = {content_data};
content_aggregate_data.device_id = content_data.device_id; item->title_id = kernel_state()->title_id();
content_aggregate_data.content_type = content_data.content_type;
content_aggregate_data.display_name = content_data.display_name;
content_aggregate_data.file_name = content_data.file_name;
content_aggregate_data.title_id = kernel_state()->title_id();
content_aggregate_data.Write(item);
} }
} }
if (!device_info || device_info->device_type == DeviceType::ODD) { if (!device_info || device_info->device_type == DeviceType::ODD) {
AddODDContentTest(e, content_type); AddODDContentTest(e, content_type_enum);
} }
XELOGD("XamContentAggregateCreateEnumerator: added {} items to enumerator", XELOGD("XamContentAggregateCreateEnumerator: added {} items to enumerator",

View File

@ -36,43 +36,6 @@ enum class XContentPackageType : uint32_t {
kLive = 0x4C495645, kLive = 0x4C495645,
}; };
enum XContentType : uint32_t {
kSavedGame = 0x00000001,
kMarketplaceContent = 0x00000002,
kPublisher = 0x00000003,
kXbox360Title = 0x00001000,
kIptvPauseBuffer = 0x00002000,
kXNACommunity = 0x00003000,
kInstalledGame = 0x00004000,
kXboxTitle = 0x00005000,
kSocialTitle = 0x00006000,
kGamesOnDemand = 0x00007000,
kSUStoragePack = 0x00008000,
kAvatarItem = 0x00009000,
kProfile = 0x00010000,
kGamerPicture = 0x00020000,
kTheme = 0x00030000,
kCacheFile = 0x00040000,
kStorageDownload = 0x00050000,
kXboxSavedGame = 0x00060000,
kXboxDownload = 0x00070000,
kGameDemo = 0x00080000,
kVideo = 0x00090000,
kGameTitle = 0x000A0000,
kInstaller = 0x000B0000,
kGameTrailer = 0x000C0000,
kArcadeTitle = 0x000D0000,
kXNA = 0x000E0000,
kLicenseStore = 0x000F0000,
kMovie = 0x00100000,
kTV = 0x00200000,
kMusicVideo = 0x00300000,
kGameVideo = 0x00400000,
kPodcastVideo = 0x00500000,
kViralVideo = 0x00600000,
kCommunityGame = 0x02000000,
};
enum class XContentVolumeType : uint32_t { enum class XContentVolumeType : uint32_t {
kStfs = 0, kStfs = 0,
kSvod = 1, kSvod = 1,

View File

@ -344,6 +344,43 @@ enum class XLanguage : uint32_t {
kMaxLanguages = 13 kMaxLanguages = 13
}; };
enum class XContentType : uint32_t {
kSavedGame = 0x00000001,
kMarketplaceContent = 0x00000002,
kPublisher = 0x00000003,
kXbox360Title = 0x00001000,
kIptvPauseBuffer = 0x00002000,
kXNACommunity = 0x00003000,
kInstalledGame = 0x00004000,
kXboxTitle = 0x00005000,
kSocialTitle = 0x00006000,
kGamesOnDemand = 0x00007000,
kSUStoragePack = 0x00008000,
kAvatarItem = 0x00009000,
kProfile = 0x00010000,
kGamerPicture = 0x00020000,
kTheme = 0x00030000,
kCacheFile = 0x00040000,
kStorageDownload = 0x00050000,
kXboxSavedGame = 0x00060000,
kXboxDownload = 0x00070000,
kGameDemo = 0x00080000,
kVideo = 0x00090000,
kGameTitle = 0x000A0000,
kInstaller = 0x000B0000,
kGameTrailer = 0x000C0000,
kArcadeTitle = 0x000D0000,
kXNA = 0x000E0000,
kLicenseStore = 0x000F0000,
kMovie = 0x00100000,
kTV = 0x00200000,
kMusicVideo = 0x00300000,
kGameVideo = 0x00400000,
kPodcastVideo = 0x00500000,
kViralVideo = 0x00600000,
kCommunityGame = 0x02000000,
};
} // namespace xe } // namespace xe
// clang-format on // clang-format on