mirror of
https://github.com/xenia-project/xenia.git
synced 2025-12-06 07:12:03 +01:00
- [XAM] Raise the size of the dummy HDD to 20GB. - [XAM] Rename unknown field to device_type in X_CONTENT_DEVICE_DATA.
506 lines
18 KiB
C++
506 lines
18 KiB
C++
/**
|
|
******************************************************************************
|
|
* Xenia : Xbox 360 Emulator Research Project *
|
|
******************************************************************************
|
|
* Copyright 2020 Ben Vanik. All rights reserved. *
|
|
* Released under the BSD license - see LICENSE in the root for more details. *
|
|
******************************************************************************
|
|
*/
|
|
|
|
#include "xenia/base/logging.h"
|
|
#include "xenia/base/math.h"
|
|
#include "xenia/kernel/kernel_state.h"
|
|
#include "xenia/kernel/util/shim_utils.h"
|
|
#include "xenia/kernel/xam/xam_private.h"
|
|
#include "xenia/kernel/xenumerator.h"
|
|
#include "xenia/xbox.h"
|
|
|
|
DEFINE_int32(
|
|
license_mask, 0,
|
|
"Set license mask for activated content.\n"
|
|
" 0 = No licenses enabled.\n"
|
|
" 1 = First license enabled. Generally the full version license in\n"
|
|
" Xbox Live Arcade titles.\n"
|
|
" -1 or 0xFFFFFFFF = All possible licenses enabled. Generally a\n"
|
|
" bad idea, could lead to undefined behavior.",
|
|
"Content");
|
|
|
|
namespace xe {
|
|
namespace kernel {
|
|
namespace xam {
|
|
|
|
struct DeviceInfo {
|
|
uint32_t device_id;
|
|
uint32_t device_type;
|
|
uint64_t total_bytes;
|
|
uint64_t free_bytes;
|
|
char16_t name[28];
|
|
};
|
|
|
|
// TODO(gibbed): real information.
|
|
//
|
|
// Until we expose real information about a HDD device, we
|
|
// claim there is 3GB free on a 4GB dummy HDD.
|
|
//
|
|
// There is a possibility that certain games are bugged in that
|
|
// they incorrectly only look at the lower 32-bits of free_bytes,
|
|
// when it is a 64-bit value. Which means any size above ~4GB
|
|
// will not be recognized properly.
|
|
#define ONE_GB (1024ull * 1024ull * 1024ull)
|
|
static const DeviceInfo dummy_device_info_ = {
|
|
0x00000001, // id
|
|
1, // 1=HDD
|
|
20ull * ONE_GB, // 20GB
|
|
3ull * ONE_GB, // 3GB, so it looks a little used.
|
|
u"Dummy HDD",
|
|
};
|
|
#undef ONE_GB
|
|
|
|
dword_result_t XamContentGetLicenseMask(lpdword_t mask_ptr,
|
|
lpunknown_t overlapped_ptr) {
|
|
// Each bit in the mask represents a granted license. Available licenses
|
|
// seems to vary from game to game, but most appear to use bit 0 to indicate
|
|
// if the game is purchased or not.
|
|
*mask_ptr = static_cast<uint32_t>(cvars::license_mask);
|
|
|
|
if (overlapped_ptr) {
|
|
kernel_state()->CompleteOverlappedImmediate(overlapped_ptr,
|
|
X_ERROR_SUCCESS);
|
|
return X_ERROR_IO_PENDING;
|
|
} else {
|
|
return X_ERROR_SUCCESS;
|
|
}
|
|
}
|
|
DECLARE_XAM_EXPORT2(XamContentGetLicenseMask, kContent, kStub, kHighFrequency);
|
|
|
|
dword_result_t XamContentGetDeviceName(dword_t device_id,
|
|
lpu16string_t name_buffer,
|
|
dword_t name_capacity) {
|
|
if ((device_id & 0x0000000F) != dummy_device_info_.device_id) {
|
|
return X_ERROR_DEVICE_NOT_CONNECTED;
|
|
}
|
|
|
|
auto name = std::u16string(dummy_device_info_.name);
|
|
if (name_capacity < name.size() + 1) {
|
|
return X_ERROR_INSUFFICIENT_BUFFER;
|
|
}
|
|
|
|
xe::store_and_swap<std::u16string>(name_buffer, name);
|
|
((char16_t*)name_buffer)[name.size()] = 0;
|
|
return X_ERROR_SUCCESS;
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentGetDeviceName, kContent, kImplemented);
|
|
|
|
dword_result_t XamContentGetDeviceState(dword_t device_id,
|
|
lpunknown_t overlapped_ptr) {
|
|
if ((device_id & 0x0000000F) != dummy_device_info_.device_id) {
|
|
if (overlapped_ptr) {
|
|
kernel_state()->CompleteOverlappedImmediateEx(
|
|
overlapped_ptr, X_ERROR_FUNCTION_FAILED, X_ERROR_DEVICE_NOT_CONNECTED,
|
|
0);
|
|
return X_ERROR_IO_PENDING;
|
|
} else {
|
|
return X_ERROR_DEVICE_NOT_CONNECTED;
|
|
}
|
|
}
|
|
|
|
if (overlapped_ptr) {
|
|
kernel_state()->CompleteOverlappedImmediate(overlapped_ptr,
|
|
X_ERROR_SUCCESS);
|
|
return X_ERROR_IO_PENDING;
|
|
} else {
|
|
return X_ERROR_SUCCESS;
|
|
}
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentGetDeviceState, kContent, kStub);
|
|
|
|
typedef struct {
|
|
xe::be<uint32_t> device_id;
|
|
xe::be<uint32_t> device_type;
|
|
xe::be<uint64_t> total_bytes;
|
|
xe::be<uint64_t> free_bytes;
|
|
xe::be<uint16_t> name[28];
|
|
} X_CONTENT_DEVICE_DATA;
|
|
static_assert_size(X_CONTENT_DEVICE_DATA, 0x50);
|
|
|
|
dword_result_t XamContentGetDeviceData(
|
|
dword_t device_id, pointer_t<X_CONTENT_DEVICE_DATA> device_data) {
|
|
if ((device_id & 0x0000000F) != dummy_device_info_.device_id) {
|
|
// TODO(benvanik): memset 0 the data?
|
|
return X_ERROR_DEVICE_NOT_CONNECTED;
|
|
}
|
|
|
|
device_data.Zero();
|
|
const auto& device_info = dummy_device_info_;
|
|
device_data->device_id = device_info.device_id;
|
|
device_data->device_type = device_info.device_type;
|
|
device_data->total_bytes = device_info.total_bytes;
|
|
device_data->free_bytes = device_info.free_bytes;
|
|
xe::store_and_swap<std::u16string>(&device_data->name[0], device_info.name);
|
|
return X_ERROR_SUCCESS;
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentGetDeviceData, kContent, kImplemented);
|
|
|
|
dword_result_t XamContentResolve(dword_t user_index, lpvoid_t content_data_ptr,
|
|
lpunknown_t buffer_ptr, dword_t buffer_size,
|
|
dword_t unk1, dword_t unk2, dword_t unk3) {
|
|
auto content_data = XCONTENT_DATA((uint8_t*)content_data_ptr);
|
|
|
|
// Result of buffer_ptr is sent to RtlInitAnsiString.
|
|
// buffer_size is usually 260 (max path).
|
|
// Games expect zero if resolve was successful.
|
|
assert_always();
|
|
XELOGW("XamContentResolve unimplemented!");
|
|
return X_ERROR_NOT_FOUND;
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentResolve, kContent, kStub);
|
|
|
|
// https://github.com/MrColdbird/gameservice/blob/master/ContentManager.cpp
|
|
// https://github.com/LestaD/SourceEngine2007/blob/master/se2007/engine/xboxsystem.cpp#L499
|
|
dword_result_t XamContentCreateEnumerator(dword_t user_index, dword_t device_id,
|
|
dword_t content_type,
|
|
dword_t content_flags,
|
|
dword_t items_per_enumerate,
|
|
lpdword_t buffer_size_ptr,
|
|
lpdword_t handle_out) {
|
|
assert_not_null(handle_out);
|
|
if ((device_id && (device_id & 0x0000000F) != dummy_device_info_.device_id) ||
|
|
!handle_out) {
|
|
if (buffer_size_ptr) {
|
|
*buffer_size_ptr = 0;
|
|
}
|
|
|
|
// TODO(benvanik): memset 0 the data?
|
|
return X_E_INVALIDARG;
|
|
}
|
|
|
|
if (buffer_size_ptr) {
|
|
*buffer_size_ptr = (uint32_t)XCONTENT_DATA::kSize * items_per_enumerate;
|
|
}
|
|
|
|
auto e = new XStaticEnumerator(kernel_state(), items_per_enumerate,
|
|
XCONTENT_DATA::kSize);
|
|
e->Initialize();
|
|
|
|
// Get all content data.
|
|
auto content_datas = kernel_state()->content_manager()->ListContent(
|
|
device_id ? static_cast<uint32_t>(device_id)
|
|
: dummy_device_info_.device_id,
|
|
content_type);
|
|
for (auto& content_data : content_datas) {
|
|
auto ptr = e->AppendItem();
|
|
assert_not_null(ptr);
|
|
content_data.Write(ptr);
|
|
}
|
|
|
|
XELOGD("XamContentCreateEnumerator: added {} items to enumerator",
|
|
e->item_count());
|
|
|
|
*handle_out = e->handle();
|
|
return X_ERROR_SUCCESS;
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentCreateEnumerator, kContent, kImplemented);
|
|
|
|
dword_result_t XamContentCreateDeviceEnumerator(dword_t content_type,
|
|
dword_t content_flags,
|
|
dword_t max_count,
|
|
lpdword_t buffer_size_ptr,
|
|
lpdword_t handle_out) {
|
|
assert_not_null(handle_out);
|
|
|
|
if (buffer_size_ptr) {
|
|
*buffer_size_ptr = sizeof(DeviceInfo) * max_count;
|
|
}
|
|
|
|
auto e = new XStaticEnumerator(kernel_state(), max_count, sizeof(DeviceInfo));
|
|
e->Initialize();
|
|
|
|
// Copy our dummy device into the enumerator
|
|
DeviceInfo* dev = (DeviceInfo*)e->AppendItem();
|
|
if (dev) {
|
|
xe::store_and_swap(&dev->device_id, dummy_device_info_.device_id);
|
|
xe::store_and_swap(&dev->device_type, dummy_device_info_.device_type);
|
|
xe::store_and_swap(&dev->total_bytes, dummy_device_info_.total_bytes);
|
|
xe::store_and_swap(&dev->free_bytes, dummy_device_info_.free_bytes);
|
|
xe::copy_and_swap(dev->name, dummy_device_info_.name,
|
|
xe::countof(dev->name));
|
|
}
|
|
|
|
*handle_out = e->handle();
|
|
return X_ERROR_SUCCESS;
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentCreateDeviceEnumerator, kNone, kImplemented);
|
|
|
|
dword_result_t XamContentCreateEx(dword_t user_index, lpstring_t root_name,
|
|
lpvoid_t content_data_ptr, dword_t flags,
|
|
lpdword_t disposition_ptr,
|
|
lpdword_t license_mask_ptr,
|
|
dword_t cache_size, qword_t content_size,
|
|
lpvoid_t overlapped_ptr) {
|
|
X_RESULT result = X_ERROR_INVALID_PARAMETER;
|
|
auto content_data = XCONTENT_DATA((uint8_t*)content_data_ptr);
|
|
|
|
auto content_manager = kernel_state()->content_manager();
|
|
bool create = false;
|
|
bool open = false;
|
|
switch (flags & 0xF) {
|
|
case 1: // CREATE_NEW
|
|
// Fail if exists.
|
|
if (content_manager->ContentExists(content_data)) {
|
|
result = X_ERROR_ALREADY_EXISTS;
|
|
} else {
|
|
create = true;
|
|
}
|
|
break;
|
|
case 2: // CREATE_ALWAYS
|
|
// Overwrite existing, if any.
|
|
if (content_manager->ContentExists(content_data)) {
|
|
content_manager->DeleteContent(content_data);
|
|
create = true;
|
|
} else {
|
|
create = true;
|
|
}
|
|
break;
|
|
case 3: // OPEN_EXISTING
|
|
// Open only if exists.
|
|
if (!content_manager->ContentExists(content_data)) {
|
|
result = X_ERROR_PATH_NOT_FOUND;
|
|
} else {
|
|
open = true;
|
|
}
|
|
break;
|
|
case 4: // OPEN_ALWAYS
|
|
// Create if needed.
|
|
if (!content_manager->ContentExists(content_data)) {
|
|
create = true;
|
|
} else {
|
|
open = true;
|
|
}
|
|
break;
|
|
case 5: // TRUNCATE_EXISTING
|
|
// Fail if doesn't exist, if does exist delete and recreate.
|
|
if (!content_manager->ContentExists(content_data)) {
|
|
result = X_ERROR_PATH_NOT_FOUND;
|
|
} else {
|
|
content_manager->DeleteContent(content_data);
|
|
create = true;
|
|
}
|
|
break;
|
|
default:
|
|
assert_unhandled_case(flags & 0xF);
|
|
break;
|
|
}
|
|
|
|
// creation result
|
|
// 0 = ?
|
|
// 1 = created
|
|
// 2 = opened
|
|
uint32_t disposition = create ? 1 : 2;
|
|
if (disposition_ptr) {
|
|
if (overlapped_ptr) {
|
|
// If async always set to zero, but don't set to a real value.
|
|
*disposition_ptr = 0;
|
|
} else {
|
|
*disposition_ptr = disposition;
|
|
}
|
|
}
|
|
|
|
if (create) {
|
|
result = content_manager->CreateContent(root_name.value(), content_data);
|
|
} else if (open) {
|
|
result = content_manager->OpenContent(root_name.value(), content_data);
|
|
}
|
|
|
|
if (license_mask_ptr && XSUCCEEDED(result)) {
|
|
*license_mask_ptr = 0; // Stub!
|
|
}
|
|
|
|
if (overlapped_ptr) {
|
|
kernel_state()->CompleteOverlappedImmediateEx(overlapped_ptr, result, 0,
|
|
disposition);
|
|
return X_ERROR_IO_PENDING;
|
|
} else {
|
|
return result;
|
|
}
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentCreateEx, kContent, kImplemented);
|
|
|
|
dword_result_t XamContentCreate(dword_t user_index, lpstring_t root_name,
|
|
lpvoid_t content_data_ptr, dword_t flags,
|
|
lpdword_t disposition_ptr,
|
|
lpdword_t license_mask_ptr,
|
|
lpvoid_t overlapped_ptr) {
|
|
return XamContentCreateEx(user_index, root_name, content_data_ptr, flags,
|
|
disposition_ptr, license_mask_ptr, 0, 0,
|
|
overlapped_ptr);
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentCreate, kContent, kImplemented);
|
|
|
|
dword_result_t XamContentCreateInternal(
|
|
lpstring_t root_name, lpvoid_t content_data_ptr, dword_t flags,
|
|
lpdword_t disposition_ptr, lpdword_t license_mask_ptr, dword_t cache_size,
|
|
qword_t content_size, lpvoid_t overlapped_ptr) {
|
|
return XamContentCreateEx(0xFE, root_name, content_data_ptr, flags,
|
|
disposition_ptr, license_mask_ptr, cache_size,
|
|
content_size, overlapped_ptr);
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentCreateInternal, kContent, kImplemented);
|
|
|
|
dword_result_t XamContentOpenFile(dword_t user_index, lpstring_t root_name,
|
|
lpstring_t path, dword_t flags,
|
|
lpdword_t disposition_ptr,
|
|
lpdword_t license_mask_ptr,
|
|
lpvoid_t overlapped_ptr) {
|
|
// TODO(gibbed): arguments assumed based on XamContentCreate.
|
|
return X_ERROR_FILE_NOT_FOUND;
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentOpenFile, kContent, kStub);
|
|
|
|
dword_result_t XamContentFlush(lpstring_t root_name,
|
|
lpunknown_t overlapped_ptr) {
|
|
X_RESULT result = X_ERROR_SUCCESS;
|
|
if (overlapped_ptr) {
|
|
kernel_state()->CompleteOverlappedImmediate(overlapped_ptr, result);
|
|
return X_ERROR_IO_PENDING;
|
|
} else {
|
|
return result;
|
|
}
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentFlush, kContent, kStub);
|
|
|
|
dword_result_t XamContentClose(lpstring_t root_name,
|
|
lpunknown_t overlapped_ptr) {
|
|
// Closes a previously opened root from XamContentCreate*.
|
|
auto result =
|
|
kernel_state()->content_manager()->CloseContent(root_name.value());
|
|
|
|
if (overlapped_ptr) {
|
|
kernel_state()->CompleteOverlappedImmediate(overlapped_ptr, result);
|
|
return X_ERROR_IO_PENDING;
|
|
} else {
|
|
return result;
|
|
}
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentClose, kContent, kImplemented);
|
|
|
|
dword_result_t XamContentGetCreator(dword_t user_index,
|
|
lpvoid_t content_data_ptr,
|
|
lpdword_t is_creator_ptr,
|
|
lpqword_t creator_xuid_ptr,
|
|
lpunknown_t overlapped_ptr) {
|
|
auto result = X_ERROR_SUCCESS;
|
|
|
|
auto content_data = XCONTENT_DATA((uint8_t*)content_data_ptr);
|
|
|
|
if (content_data.content_type == 1) {
|
|
// User always creates saves.
|
|
*is_creator_ptr = 1;
|
|
if (creator_xuid_ptr) {
|
|
*creator_xuid_ptr = kernel_state()->user_profile()->xuid();
|
|
}
|
|
} else {
|
|
*is_creator_ptr = 0;
|
|
if (creator_xuid_ptr) {
|
|
*creator_xuid_ptr = 0;
|
|
}
|
|
}
|
|
|
|
if (overlapped_ptr) {
|
|
kernel_state()->CompleteOverlappedImmediate(overlapped_ptr, result);
|
|
return X_ERROR_IO_PENDING;
|
|
} else {
|
|
return result;
|
|
}
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentGetCreator, kContent, kImplemented);
|
|
|
|
dword_result_t XamContentGetThumbnail(dword_t user_index,
|
|
lpvoid_t content_data_ptr,
|
|
lpvoid_t buffer_ptr,
|
|
lpdword_t buffer_size_ptr,
|
|
lpunknown_t overlapped_ptr) {
|
|
assert_not_null(buffer_size_ptr);
|
|
uint32_t buffer_size = *buffer_size_ptr;
|
|
auto content_data = XCONTENT_DATA((uint8_t*)content_data_ptr);
|
|
|
|
// Get thumbnail (if it exists).
|
|
std::vector<uint8_t> buffer;
|
|
auto result = kernel_state()->content_manager()->GetContentThumbnail(
|
|
content_data, &buffer);
|
|
|
|
*buffer_size_ptr = uint32_t(buffer.size());
|
|
|
|
if (XSUCCEEDED(result)) {
|
|
// Write data, if we were given a pointer.
|
|
// This may have just been a size query.
|
|
if (buffer_ptr) {
|
|
if (buffer_size < buffer.size()) {
|
|
// Dest buffer too small.
|
|
result = X_ERROR_INSUFFICIENT_BUFFER;
|
|
} else {
|
|
// Copy data.
|
|
std::memcpy((uint8_t*)buffer_ptr, buffer.data(), buffer.size());
|
|
}
|
|
}
|
|
}
|
|
|
|
if (overlapped_ptr) {
|
|
kernel_state()->CompleteOverlappedImmediate(overlapped_ptr, result);
|
|
return X_ERROR_IO_PENDING;
|
|
} else {
|
|
return result;
|
|
}
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentGetThumbnail, kContent, kImplemented);
|
|
|
|
dword_result_t XamContentSetThumbnail(dword_t user_index,
|
|
lpvoid_t content_data_ptr,
|
|
lpvoid_t buffer_ptr, dword_t buffer_size,
|
|
lpunknown_t overlapped_ptr) {
|
|
auto content_data = XCONTENT_DATA((uint8_t*)content_data_ptr);
|
|
|
|
// Buffer is PNG data.
|
|
auto buffer = std::vector<uint8_t>((uint8_t*)buffer_ptr,
|
|
(uint8_t*)buffer_ptr + buffer_size);
|
|
auto result = kernel_state()->content_manager()->SetContentThumbnail(
|
|
content_data, std::move(buffer));
|
|
|
|
if (overlapped_ptr) {
|
|
kernel_state()->CompleteOverlappedImmediate(overlapped_ptr, result);
|
|
return X_ERROR_IO_PENDING;
|
|
} else {
|
|
return result;
|
|
}
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentSetThumbnail, kContent, kImplemented);
|
|
|
|
dword_result_t XamContentDelete(dword_t user_index, lpvoid_t content_data_ptr,
|
|
lpunknown_t overlapped_ptr) {
|
|
auto content_data = XCONTENT_DATA((uint8_t*)content_data_ptr);
|
|
|
|
auto result = kernel_state()->content_manager()->DeleteContent(content_data);
|
|
|
|
if (overlapped_ptr) {
|
|
kernel_state()->CompleteOverlappedImmediate(overlapped_ptr, result);
|
|
return X_ERROR_IO_PENDING;
|
|
} else {
|
|
return result;
|
|
}
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentDelete, kContent, kImplemented);
|
|
|
|
dword_result_t XamContentDeleteInternal(lpvoid_t content_data_ptr,
|
|
lpunknown_t overlapped_ptr) {
|
|
// INFO: Analysis of xam.xex shows that "internal" functions are wrappers with
|
|
// 0xFE as user_index
|
|
return XamContentDelete(0xFE, content_data_ptr, overlapped_ptr);
|
|
}
|
|
DECLARE_XAM_EXPORT1(XamContentDeleteInternal, kContent, kImplemented);
|
|
|
|
void RegisterContentExports(xe::cpu::ExportResolver* export_resolver,
|
|
KernelState* kernel_state) {}
|
|
|
|
} // namespace xam
|
|
} // namespace kernel
|
|
} // namespace xe
|