2013-10-24 03:42:24 +00:00
|
|
|
/**
|
|
|
|
******************************************************************************
|
|
|
|
* Xenia : Xbox 360 Emulator Research Project *
|
|
|
|
******************************************************************************
|
|
|
|
* Copyright 2013 Ben Vanik. All rights reserved. *
|
|
|
|
* Released under the BSD license - see LICENSE in the root for more details. *
|
|
|
|
******************************************************************************
|
|
|
|
*/
|
|
|
|
|
2015-02-01 06:49:47 +00:00
|
|
|
#include "xenia/emulator.h"
|
2013-10-24 03:42:24 +00:00
|
|
|
|
2016-12-01 04:54:58 +00:00
|
|
|
#include <cinttypes>
|
2015-05-27 05:20:46 +00:00
|
|
|
|
2019-04-17 19:49:29 +00:00
|
|
|
#include "config.h"
|
2015-06-27 18:25:45 +00:00
|
|
|
#include "xenia/apu/audio_system.h"
|
2015-05-02 10:42:51 +00:00
|
|
|
#include "xenia/base/assert.h"
|
2015-12-01 23:25:47 +00:00
|
|
|
#include "xenia/base/byte_stream.h"
|
2015-05-27 05:20:46 +00:00
|
|
|
#include "xenia/base/clock.h"
|
2019-04-17 19:49:29 +00:00
|
|
|
#include "xenia/base/cvar.h"
|
2015-09-11 22:09:25 +00:00
|
|
|
#include "xenia/base/debugging.h"
|
|
|
|
#include "xenia/base/exception_handler.h"
|
2015-06-27 23:27:24 +00:00
|
|
|
#include "xenia/base/logging.h"
|
2015-12-01 23:25:47 +00:00
|
|
|
#include "xenia/base/mapped_memory.h"
|
2015-11-07 19:25:53 +00:00
|
|
|
#include "xenia/base/profiling.h"
|
2015-05-02 10:42:51 +00:00
|
|
|
#include "xenia/base/string.h"
|
2015-09-11 22:09:25 +00:00
|
|
|
#include "xenia/cpu/backend/code_cache.h"
|
2017-02-11 05:54:10 +00:00
|
|
|
#include "xenia/cpu/backend/x64/x64_backend.h"
|
|
|
|
#include "xenia/cpu/cpu_flags.h"
|
2016-12-01 04:54:58 +00:00
|
|
|
#include "xenia/cpu/thread_state.h"
|
2015-06-27 16:11:03 +00:00
|
|
|
#include "xenia/gpu/graphics_system.h"
|
2015-11-08 23:02:24 +00:00
|
|
|
#include "xenia/hid/input_driver.h"
|
2015-06-27 16:42:30 +00:00
|
|
|
#include "xenia/hid/input_system.h"
|
2015-02-01 06:49:47 +00:00
|
|
|
#include "xenia/kernel/kernel_state.h"
|
2015-12-01 23:25:47 +00:00
|
|
|
#include "xenia/kernel/user_module.h"
|
2016-07-23 17:27:13 +00:00
|
|
|
#include "xenia/kernel/util/gameinfo_utils.h"
|
2016-01-10 19:04:55 +00:00
|
|
|
#include "xenia/kernel/util/xdbf_utils.h"
|
2015-09-07 01:07:52 +00:00
|
|
|
#include "xenia/kernel/xam/xam_module.h"
|
2017-01-09 10:51:49 +00:00
|
|
|
#include "xenia/kernel/xbdm/xbdm_module.h"
|
2015-09-07 01:07:52 +00:00
|
|
|
#include "xenia/kernel/xboxkrnl/xboxkrnl_module.h"
|
2015-02-01 06:49:47 +00:00
|
|
|
#include "xenia/memory.h"
|
2015-12-27 04:25:24 +00:00
|
|
|
#include "xenia/ui/imgui_dialog.h"
|
2015-06-27 23:27:24 +00:00
|
|
|
#include "xenia/vfs/devices/disc_image_device.h"
|
|
|
|
#include "xenia/vfs/devices/host_path_device.h"
|
|
|
|
#include "xenia/vfs/devices/stfs_container_device.h"
|
2015-09-07 01:07:52 +00:00
|
|
|
#include "xenia/vfs/virtual_file_system.h"
|
2013-10-24 03:42:24 +00:00
|
|
|
|
2015-05-27 05:20:46 +00:00
|
|
|
DEFINE_double(time_scalar, 1.0,
|
2019-04-17 19:49:29 +00:00
|
|
|
"Scalar used to speed or slow time (1x, 2x, 1/2x, etc).",
|
|
|
|
"General");
|
2019-09-14 17:30:04 +00:00
|
|
|
DEFINE_string(
|
|
|
|
launch_module, "",
|
|
|
|
"Executable to launch from the .iso or the package instead of default.xex "
|
|
|
|
"or the module specified by the game. Leave blank to launch the default "
|
|
|
|
"module.",
|
|
|
|
"General");
|
2015-05-27 05:20:46 +00:00
|
|
|
|
2014-08-17 20:13:03 +00:00
|
|
|
namespace xe {
|
2013-10-24 03:42:24 +00:00
|
|
|
|
2018-11-22 00:04:43 +00:00
|
|
|
Emulator::Emulator(const std::wstring& command_line,
|
2020-03-21 16:21:00 +00:00
|
|
|
const std::wstring& storage_root,
|
2018-11-22 00:04:43 +00:00
|
|
|
const std::wstring& content_root)
|
2019-07-29 18:22:45 +00:00
|
|
|
: on_launch(),
|
2019-08-03 13:12:37 +00:00
|
|
|
on_terminate(),
|
2019-07-29 18:22:45 +00:00
|
|
|
on_exit(),
|
|
|
|
command_line_(command_line),
|
2020-03-21 16:21:00 +00:00
|
|
|
storage_root_(storage_root),
|
2019-07-29 18:22:45 +00:00
|
|
|
content_root_(content_root),
|
|
|
|
game_title_(),
|
|
|
|
display_window_(nullptr),
|
|
|
|
memory_(),
|
|
|
|
audio_system_(),
|
|
|
|
graphics_system_(),
|
|
|
|
input_system_(),
|
|
|
|
export_resolver_(),
|
|
|
|
file_system_(),
|
|
|
|
kernel_state_(),
|
2019-07-29 18:24:36 +00:00
|
|
|
main_thread_(),
|
2019-07-29 18:22:45 +00:00
|
|
|
title_id_(0),
|
|
|
|
paused_(false),
|
|
|
|
restoring_(false),
|
|
|
|
restore_fence_() {}
|
2013-10-24 03:42:24 +00:00
|
|
|
|
|
|
|
Emulator::~Emulator() {
|
|
|
|
// Note that we delete things in the reverse order they were initialized.
|
|
|
|
|
2015-05-20 05:20:49 +00:00
|
|
|
// Give the systems time to shutdown before we delete them.
|
2015-12-13 19:22:47 +00:00
|
|
|
if (graphics_system_) {
|
|
|
|
graphics_system_->Shutdown();
|
|
|
|
}
|
|
|
|
if (audio_system_) {
|
|
|
|
audio_system_->Shutdown();
|
|
|
|
}
|
2013-10-24 03:42:24 +00:00
|
|
|
|
2014-08-21 06:26:46 +00:00
|
|
|
input_system_.reset();
|
|
|
|
graphics_system_.reset();
|
|
|
|
audio_system_.reset();
|
2014-01-03 02:58:44 +00:00
|
|
|
|
2015-05-25 03:44:27 +00:00
|
|
|
kernel_state_.reset();
|
|
|
|
file_system_.reset();
|
|
|
|
|
2014-08-21 06:26:46 +00:00
|
|
|
processor_.reset();
|
2013-10-24 03:42:24 +00:00
|
|
|
|
2014-08-21 06:26:46 +00:00
|
|
|
export_resolver_.reset();
|
2015-09-21 04:31:05 +00:00
|
|
|
|
|
|
|
ExceptionHandler::Uninstall(Emulator::ExceptionCallbackThunk, this);
|
2013-10-24 03:42:24 +00:00
|
|
|
}
|
|
|
|
|
2015-11-08 23:02:24 +00:00
|
|
|
X_STATUS Emulator::Setup(
|
|
|
|
ui::Window* display_window,
|
|
|
|
std::function<std::unique_ptr<apu::AudioSystem>(cpu::Processor*)>
|
|
|
|
audio_system_factory,
|
|
|
|
std::function<std::unique_ptr<gpu::GraphicsSystem>()>
|
|
|
|
graphics_system_factory,
|
|
|
|
std::function<std::vector<std::unique_ptr<hid::InputDriver>>(ui::Window*)>
|
|
|
|
input_driver_factory) {
|
2013-10-24 03:42:24 +00:00
|
|
|
X_STATUS result = X_STATUS_UNSUCCESSFUL;
|
|
|
|
|
2015-07-13 05:03:47 +00:00
|
|
|
display_window_ = display_window;
|
|
|
|
|
2015-05-27 05:20:46 +00:00
|
|
|
// Initialize clock.
|
|
|
|
// 360 uses a 50MHz clock.
|
2015-05-28 02:04:47 +00:00
|
|
|
Clock::set_guest_tick_frequency(50000000);
|
2015-05-27 05:20:46 +00:00
|
|
|
// We could reset this with save state data/constant value to help replays.
|
|
|
|
Clock::set_guest_system_time_base(Clock::QueryHostSystemTime());
|
|
|
|
// This can be adjusted dynamically, as well.
|
2019-04-17 19:49:29 +00:00
|
|
|
Clock::set_guest_time_scalar(cvars::time_scalar);
|
2015-05-27 05:20:46 +00:00
|
|
|
|
|
|
|
// Before we can set thread affinity we must enable the process to use all
|
|
|
|
// logical processors.
|
2015-07-16 05:09:19 +00:00
|
|
|
xe::threading::EnableAffinityConfiguration();
|
2015-05-16 06:52:48 +00:00
|
|
|
|
2013-10-24 03:42:24 +00:00
|
|
|
// Create memory system first, as it is required for other systems.
|
2014-08-21 06:26:46 +00:00
|
|
|
memory_ = std::make_unique<Memory>();
|
2015-12-03 01:37:48 +00:00
|
|
|
if (!memory_->Initialize()) {
|
|
|
|
return false;
|
2014-08-21 06:26:46 +00:00
|
|
|
}
|
2013-10-24 03:42:24 +00:00
|
|
|
|
|
|
|
// Shared export resolver used to attach and query for HLE exports.
|
2015-05-02 09:11:11 +00:00
|
|
|
export_resolver_ = std::make_unique<xe::cpu::ExportResolver>();
|
2013-10-24 03:42:24 +00:00
|
|
|
|
2017-02-11 05:54:10 +00:00
|
|
|
std::unique_ptr<xe::cpu::backend::Backend> backend;
|
|
|
|
if (!backend) {
|
|
|
|
#if defined(XENIA_HAS_X64_BACKEND) && XENIA_HAS_X64_BACKEND
|
2019-04-17 19:49:29 +00:00
|
|
|
if (cvars::cpu == "x64") {
|
2017-02-11 05:54:10 +00:00
|
|
|
backend.reset(new xe::cpu::backend::x64::X64Backend());
|
|
|
|
}
|
|
|
|
#endif // XENIA_HAS_X64_BACKEND
|
2019-04-17 19:49:29 +00:00
|
|
|
if (cvars::cpu == "any") {
|
2017-02-11 05:54:10 +00:00
|
|
|
#if defined(XENIA_HAS_X64_BACKEND) && XENIA_HAS_X64_BACKEND
|
|
|
|
if (!backend) {
|
|
|
|
backend.reset(new xe::cpu::backend::x64::X64Backend());
|
|
|
|
}
|
|
|
|
#endif // XENIA_HAS_X64_BACKEND
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-24 03:42:24 +00:00
|
|
|
// Initialize the CPU.
|
2016-01-18 19:48:21 +00:00
|
|
|
processor_ = std::make_unique<xe::cpu::Processor>(memory_.get(),
|
|
|
|
export_resolver_.get());
|
2017-02-11 05:54:10 +00:00
|
|
|
if (!processor_->Setup(std::move(backend))) {
|
2015-11-04 12:47:38 +00:00
|
|
|
return X_STATUS_UNSUCCESSFUL;
|
|
|
|
}
|
2013-10-24 03:42:24 +00:00
|
|
|
|
|
|
|
// Initialize the APU.
|
2015-11-08 23:02:24 +00:00
|
|
|
if (audio_system_factory) {
|
|
|
|
audio_system_ = audio_system_factory(processor_.get());
|
|
|
|
if (!audio_system_) {
|
|
|
|
return X_STATUS_NOT_IMPLEMENTED;
|
|
|
|
}
|
2014-08-21 06:26:46 +00:00
|
|
|
}
|
2013-10-24 03:42:24 +00:00
|
|
|
|
|
|
|
// Initialize the GPU.
|
2015-11-08 23:02:24 +00:00
|
|
|
graphics_system_ = graphics_system_factory();
|
2014-08-21 06:26:46 +00:00
|
|
|
if (!graphics_system_) {
|
|
|
|
return X_STATUS_NOT_IMPLEMENTED;
|
|
|
|
}
|
2013-10-24 04:47:36 +00:00
|
|
|
|
|
|
|
// Initialize the HID.
|
2015-11-08 23:02:24 +00:00
|
|
|
input_system_ = std::make_unique<xe::hid::InputSystem>(display_window_);
|
2014-08-21 06:26:46 +00:00
|
|
|
if (!input_system_) {
|
|
|
|
return X_STATUS_NOT_IMPLEMENTED;
|
|
|
|
}
|
2015-11-08 23:02:24 +00:00
|
|
|
if (input_driver_factory) {
|
|
|
|
auto input_drivers = input_driver_factory(display_window_);
|
|
|
|
for (size_t i = 0; i < input_drivers.size(); ++i) {
|
|
|
|
input_system_->AddDriver(std::move(input_drivers[i]));
|
|
|
|
}
|
|
|
|
}
|
2014-01-05 01:12:46 +00:00
|
|
|
|
2013-10-24 04:47:36 +00:00
|
|
|
result = input_system_->Setup();
|
2014-08-21 06:26:46 +00:00
|
|
|
if (result) {
|
|
|
|
return result;
|
|
|
|
}
|
2013-10-24 03:42:24 +00:00
|
|
|
|
|
|
|
// Bring up the virtual filesystem used by the kernel.
|
2015-08-07 03:17:01 +00:00
|
|
|
file_system_ = std::make_unique<xe::vfs::VirtualFileSystem>();
|
2013-10-24 03:42:24 +00:00
|
|
|
|
2014-01-05 01:12:46 +00:00
|
|
|
// Shared kernel state.
|
2015-08-07 03:17:01 +00:00
|
|
|
kernel_state_ = std::make_unique<xe::kernel::KernelState>(this);
|
2015-05-20 05:20:49 +00:00
|
|
|
|
|
|
|
// Setup the core components.
|
2015-11-08 19:54:36 +00:00
|
|
|
result = graphics_system_->Setup(processor_.get(), kernel_state_.get(),
|
2015-07-13 05:03:47 +00:00
|
|
|
display_window_);
|
2015-05-20 05:20:49 +00:00
|
|
|
if (result) {
|
|
|
|
return result;
|
|
|
|
}
|
2014-01-05 01:12:46 +00:00
|
|
|
|
2015-11-08 23:02:24 +00:00
|
|
|
if (audio_system_) {
|
|
|
|
result = audio_system_->Setup(kernel_state_.get());
|
|
|
|
if (result) {
|
|
|
|
return result;
|
|
|
|
}
|
2015-06-21 07:25:24 +00:00
|
|
|
}
|
|
|
|
|
2019-07-29 18:26:09 +00:00
|
|
|
#define LOAD_KERNEL_MODULE(t) \
|
|
|
|
static_cast<void>(kernel_state_->LoadKernelModule<kernel::t>())
|
2013-10-24 03:42:24 +00:00
|
|
|
// HLE kernel modules.
|
2019-07-29 18:26:09 +00:00
|
|
|
LOAD_KERNEL_MODULE(xboxkrnl::XboxkrnlModule);
|
|
|
|
LOAD_KERNEL_MODULE(xam::XamModule);
|
|
|
|
LOAD_KERNEL_MODULE(xbdm::XbdmModule);
|
|
|
|
#undef LOAD_KERNEL_MODULE
|
2013-10-24 03:42:24 +00:00
|
|
|
|
2015-09-21 04:31:05 +00:00
|
|
|
// Initialize emulator fallback exception handling last.
|
|
|
|
ExceptionHandler::Install(Emulator::ExceptionCallbackThunk, this);
|
|
|
|
|
2017-12-19 22:02:09 +00:00
|
|
|
if (display_window_) {
|
|
|
|
// Finish initializing the display.
|
|
|
|
display_window_->loop()->PostSynchronous([this]() {
|
|
|
|
xe::ui::GraphicsContextLock context_lock(display_window_->context());
|
|
|
|
Profiler::set_window(display_window_);
|
|
|
|
});
|
|
|
|
}
|
2015-07-13 05:03:47 +00:00
|
|
|
|
2013-10-24 03:42:24 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-11-23 17:00:10 +00:00
|
|
|
X_STATUS Emulator::TerminateTitle() {
|
|
|
|
if (!is_title_open()) {
|
|
|
|
return X_STATUS_UNSUCCESSFUL;
|
|
|
|
}
|
|
|
|
|
|
|
|
kernel_state_->TerminateTitle();
|
|
|
|
title_id_ = 0;
|
|
|
|
game_title_ = L"";
|
2019-08-03 13:12:37 +00:00
|
|
|
on_terminate();
|
2016-11-23 17:00:10 +00:00
|
|
|
return X_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2016-06-19 01:42:28 +00:00
|
|
|
X_STATUS Emulator::LaunchPath(std::wstring path) {
|
2015-06-27 23:27:24 +00:00
|
|
|
// Launch based on file type.
|
|
|
|
// This is a silly guess based on file extension.
|
2015-08-06 05:06:20 +00:00
|
|
|
auto last_slash = path.find_last_of(xe::kPathSeparator);
|
2015-06-27 23:27:24 +00:00
|
|
|
auto last_dot = path.find_last_of('.');
|
|
|
|
if (last_dot < last_slash) {
|
|
|
|
last_dot = std::wstring::npos;
|
|
|
|
}
|
|
|
|
if (last_dot == std::wstring::npos) {
|
|
|
|
// Likely an STFS container.
|
2016-06-19 01:42:28 +00:00
|
|
|
return LaunchStfsContainer(path);
|
2018-06-09 06:22:45 +00:00
|
|
|
};
|
|
|
|
auto extension = path.substr(last_dot);
|
|
|
|
std::transform(extension.begin(), extension.end(), extension.begin(),
|
|
|
|
tolower);
|
2018-06-09 07:24:11 +00:00
|
|
|
if (extension == L".xex" || extension == L".elf" || extension == L".exe") {
|
2015-06-27 23:27:24 +00:00
|
|
|
// Treat as a naked xex file.
|
2016-06-19 01:42:28 +00:00
|
|
|
return LaunchXexFile(path);
|
2015-06-27 23:27:24 +00:00
|
|
|
} else {
|
|
|
|
// Assume a disc image.
|
2016-06-19 01:42:28 +00:00
|
|
|
return LaunchDiscImage(path);
|
2015-06-27 23:27:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-19 01:42:28 +00:00
|
|
|
X_STATUS Emulator::LaunchXexFile(std::wstring path) {
|
2013-10-24 03:42:24 +00:00
|
|
|
// We create a virtual filesystem pointing to its directory and symlink
|
|
|
|
// that to the game filesystem.
|
|
|
|
// e.g., /my/files/foo.xex will get a local fs at:
|
|
|
|
// \\Device\\Harddisk0\\Partition1
|
|
|
|
// and then get that symlinked to game:\, so
|
|
|
|
// -> game:\foo.xex
|
|
|
|
|
2019-10-26 08:14:27 +00:00
|
|
|
// Register local directory as some commonly used mount paths
|
|
|
|
std::string mount_paths[] = {
|
|
|
|
"\\Device\\Harddisk0\\Partition0",
|
|
|
|
"\\Device\\Harddisk0\\Partition1",
|
|
|
|
"\\Device\\Harddisk0\\Partition1\\DEVKIT",
|
|
|
|
"\\Device\\LauncherData",
|
|
|
|
"\\SystemRoot",
|
|
|
|
};
|
2015-06-27 23:27:24 +00:00
|
|
|
|
|
|
|
auto parent_path = xe::find_base_path(path);
|
2019-10-26 08:14:27 +00:00
|
|
|
for (auto path : mount_paths) {
|
|
|
|
auto device =
|
|
|
|
std::make_unique<vfs::HostPathDevice>(path, parent_path, true);
|
|
|
|
if (!device->Initialize()) {
|
|
|
|
XELOGE("Unable to scan host path");
|
|
|
|
return X_STATUS_NO_SUCH_FILE;
|
|
|
|
}
|
|
|
|
if (!file_system_->RegisterDevice(std::move(device))) {
|
|
|
|
XELOGE("Unable to register host path as %s", path.c_str());
|
|
|
|
return X_STATUS_NO_SUCH_FILE;
|
|
|
|
}
|
2014-08-15 03:28:44 +00:00
|
|
|
}
|
|
|
|
|
2015-06-27 23:27:24 +00:00
|
|
|
// Create symlinks to the device.
|
2019-10-26 08:14:27 +00:00
|
|
|
file_system_->RegisterSymbolicLink("game:", mount_paths[0]);
|
|
|
|
file_system_->RegisterSymbolicLink("d:", mount_paths[0]);
|
2015-06-27 23:27:24 +00:00
|
|
|
|
2013-10-24 03:42:24 +00:00
|
|
|
// Get just the filename (foo.xex).
|
2015-06-27 23:27:24 +00:00
|
|
|
auto file_name = xe::find_name_from_path(path);
|
2013-10-24 03:42:24 +00:00
|
|
|
|
2014-08-15 17:19:59 +00:00
|
|
|
// Launch the game.
|
2015-05-02 10:42:51 +00:00
|
|
|
std::string fs_path = "game:\\" + xe::to_string(file_name);
|
2016-06-19 01:42:28 +00:00
|
|
|
return CompleteLaunch(path, fs_path);
|
2014-08-15 17:19:59 +00:00
|
|
|
}
|
2013-10-24 03:42:24 +00:00
|
|
|
|
2016-06-19 01:42:28 +00:00
|
|
|
X_STATUS Emulator::LaunchDiscImage(std::wstring path) {
|
2015-06-27 23:27:24 +00:00
|
|
|
auto mount_path = "\\Device\\Cdrom0";
|
|
|
|
|
|
|
|
// Register the disc image in the virtual filesystem.
|
|
|
|
auto device = std::make_unique<vfs::DiscImageDevice>(mount_path, path);
|
|
|
|
if (!device->Initialize()) {
|
2015-08-30 01:06:30 +00:00
|
|
|
xe::FatalError("Unable to mount disc image; file not found or corrupt.");
|
2015-06-27 23:27:24 +00:00
|
|
|
return X_STATUS_NO_SUCH_FILE;
|
|
|
|
}
|
|
|
|
if (!file_system_->RegisterDevice(std::move(device))) {
|
2015-08-30 01:06:30 +00:00
|
|
|
xe::FatalError("Unable to register disc image.");
|
2015-06-27 23:27:24 +00:00
|
|
|
return X_STATUS_NO_SUCH_FILE;
|
2013-10-24 03:42:24 +00:00
|
|
|
}
|
|
|
|
|
2015-06-27 23:27:24 +00:00
|
|
|
// Create symlinks to the device.
|
|
|
|
file_system_->RegisterSymbolicLink("game:", mount_path);
|
|
|
|
file_system_->RegisterSymbolicLink("d:", mount_path);
|
|
|
|
|
2013-10-24 03:42:24 +00:00
|
|
|
// Launch the game.
|
2016-07-23 17:27:13 +00:00
|
|
|
auto module_path(FindLaunchModule());
|
|
|
|
return CompleteLaunch(path, module_path);
|
2013-10-24 03:42:24 +00:00
|
|
|
}
|
|
|
|
|
2016-06-19 01:42:28 +00:00
|
|
|
X_STATUS Emulator::LaunchStfsContainer(std::wstring path) {
|
2015-06-27 23:27:24 +00:00
|
|
|
auto mount_path = "\\Device\\Cdrom0";
|
|
|
|
|
|
|
|
// Register the container in the virtual filesystem.
|
2015-06-29 17:38:51 +00:00
|
|
|
auto device = std::make_unique<vfs::StfsContainerDevice>(mount_path, path);
|
2015-06-27 23:27:24 +00:00
|
|
|
if (!device->Initialize()) {
|
2015-08-30 01:06:30 +00:00
|
|
|
xe::FatalError(
|
|
|
|
"Unable to mount STFS container; file not found or corrupt.");
|
2015-06-27 23:27:24 +00:00
|
|
|
return X_STATUS_NO_SUCH_FILE;
|
2013-10-24 03:42:24 +00:00
|
|
|
}
|
2015-06-27 23:27:24 +00:00
|
|
|
if (!file_system_->RegisterDevice(std::move(device))) {
|
2015-08-30 01:06:30 +00:00
|
|
|
xe::FatalError("Unable to register STFS container.");
|
2015-06-27 23:27:24 +00:00
|
|
|
return X_STATUS_NO_SUCH_FILE;
|
|
|
|
}
|
|
|
|
|
|
|
|
file_system_->RegisterSymbolicLink("game:", mount_path);
|
|
|
|
file_system_->RegisterSymbolicLink("d:", mount_path);
|
2013-10-24 03:42:24 +00:00
|
|
|
|
|
|
|
// Launch the game.
|
2016-07-23 17:27:13 +00:00
|
|
|
auto module_path(FindLaunchModule());
|
|
|
|
return CompleteLaunch(path, module_path);
|
2013-10-24 03:42:24 +00:00
|
|
|
}
|
2014-01-19 06:23:26 +00:00
|
|
|
|
2015-12-01 23:25:47 +00:00
|
|
|
void Emulator::Pause() {
|
|
|
|
if (paused_) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
paused_ = true;
|
|
|
|
|
2015-12-07 16:55:30 +00:00
|
|
|
// Don't hold the lock on this (so any waits follow through)
|
|
|
|
graphics_system_->Pause();
|
2015-12-08 05:34:34 +00:00
|
|
|
audio_system_->Pause();
|
2015-12-07 16:55:30 +00:00
|
|
|
|
|
|
|
auto lock = global_critical_region::AcquireDirect();
|
2015-12-01 23:25:47 +00:00
|
|
|
auto threads =
|
|
|
|
kernel_state()->object_table()->GetObjectsByType<kernel::XThread>(
|
|
|
|
kernel::XObject::kTypeThread);
|
2016-12-01 04:54:58 +00:00
|
|
|
auto current_thread = kernel::XThread::IsInThread()
|
|
|
|
? kernel::XThread::GetCurrentThread()
|
|
|
|
: nullptr;
|
2015-12-01 23:25:47 +00:00
|
|
|
for (auto thread : threads) {
|
2016-12-01 04:54:58 +00:00
|
|
|
// Don't pause ourself or host threads.
|
|
|
|
if (thread == current_thread || !thread->can_debugger_suspend()) {
|
2015-12-01 23:25:47 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-06-19 01:42:28 +00:00
|
|
|
if (thread->is_running()) {
|
|
|
|
thread->thread()->Suspend(nullptr);
|
|
|
|
}
|
2015-12-01 23:25:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
XELOGD("! EMULATOR PAUSED !");
|
|
|
|
}
|
|
|
|
|
|
|
|
void Emulator::Resume() {
|
|
|
|
if (!paused_) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
paused_ = false;
|
|
|
|
XELOGD("! EMULATOR RESUMED !");
|
|
|
|
|
2015-12-07 16:55:30 +00:00
|
|
|
graphics_system_->Resume();
|
2015-12-08 05:34:34 +00:00
|
|
|
audio_system_->Resume();
|
2015-12-07 16:55:30 +00:00
|
|
|
|
2015-12-01 23:25:47 +00:00
|
|
|
auto threads =
|
|
|
|
kernel_state()->object_table()->GetObjectsByType<kernel::XThread>(
|
|
|
|
kernel::XObject::kTypeThread);
|
|
|
|
for (auto thread : threads) {
|
|
|
|
if (!thread->can_debugger_suspend()) {
|
|
|
|
// Don't pause host threads.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-06-19 01:42:28 +00:00
|
|
|
if (thread->is_running()) {
|
|
|
|
thread->thread()->Resume(nullptr);
|
|
|
|
}
|
2015-12-01 23:25:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Emulator::SaveToFile(const std::wstring& path) {
|
|
|
|
Pause();
|
|
|
|
|
|
|
|
filesystem::CreateFile(path);
|
|
|
|
auto map = MappedMemory::Open(path, MappedMemory::Mode::kReadWrite, 0,
|
2016-09-28 23:33:25 +00:00
|
|
|
1024ull * 1024ull * 1024ull * 2ull);
|
2015-12-01 23:25:47 +00:00
|
|
|
if (!map) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save the emulator state to a file
|
|
|
|
ByteStream stream(map->data(), map->size());
|
|
|
|
stream.Write('XSAV');
|
2016-09-28 23:33:25 +00:00
|
|
|
stream.Write(title_id_);
|
2015-12-01 23:25:47 +00:00
|
|
|
|
2015-12-06 23:45:40 +00:00
|
|
|
// It's important we don't hold the global lock here! XThreads need to step
|
|
|
|
// forward (possibly through guarded regions) without worry!
|
2016-01-18 19:48:21 +00:00
|
|
|
processor_->Save(&stream);
|
2015-12-07 16:55:30 +00:00
|
|
|
graphics_system_->Save(&stream);
|
2015-12-08 05:34:34 +00:00
|
|
|
audio_system_->Save(&stream);
|
2015-12-01 23:25:47 +00:00
|
|
|
kernel_state_->Save(&stream);
|
|
|
|
memory_->Save(&stream);
|
|
|
|
map->Close(stream.offset());
|
|
|
|
|
|
|
|
Resume();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Emulator::RestoreFromFile(const std::wstring& path) {
|
|
|
|
// Restore the emulator state from a file
|
|
|
|
auto map = MappedMemory::Open(path, MappedMemory::Mode::kReadWrite);
|
|
|
|
if (!map) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
restoring_ = true;
|
|
|
|
|
|
|
|
// Terminate any loaded titles.
|
|
|
|
Pause();
|
|
|
|
kernel_state_->TerminateTitle();
|
|
|
|
|
2015-12-07 03:02:12 +00:00
|
|
|
auto lock = global_critical_region::AcquireDirect();
|
2015-12-01 23:25:47 +00:00
|
|
|
ByteStream stream(map->data(), map->size());
|
|
|
|
if (stream.Read<uint32_t>() != 'XSAV') {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-09-28 23:33:25 +00:00
|
|
|
auto title_id = stream.Read<uint32_t>();
|
|
|
|
if (title_id != title_id_) {
|
|
|
|
// Swapping between titles is unsupported at the moment.
|
|
|
|
assert_always();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-01-18 19:48:21 +00:00
|
|
|
if (!processor_->Restore(&stream)) {
|
|
|
|
XELOGE("Could not restore processor!");
|
|
|
|
return false;
|
|
|
|
}
|
2015-12-07 16:55:30 +00:00
|
|
|
if (!graphics_system_->Restore(&stream)) {
|
2015-12-08 05:34:34 +00:00
|
|
|
XELOGE("Could not restore graphics system!");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!audio_system_->Restore(&stream)) {
|
|
|
|
XELOGE("Could not restore audio system!");
|
2015-12-07 16:55:30 +00:00
|
|
|
return false;
|
|
|
|
}
|
2015-12-01 23:25:47 +00:00
|
|
|
if (!kernel_state_->Restore(&stream)) {
|
2015-12-08 05:34:34 +00:00
|
|
|
XELOGE("Could not restore kernel state!");
|
2015-12-01 23:25:47 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!memory_->Restore(&stream)) {
|
2015-12-08 05:34:34 +00:00
|
|
|
XELOGE("Could not restore memory!");
|
2015-12-01 23:25:47 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the main thread.
|
2015-12-29 21:13:34 +00:00
|
|
|
auto threads =
|
|
|
|
kernel_state_->object_table()->GetObjectsByType<kernel::XThread>();
|
2015-12-01 23:25:47 +00:00
|
|
|
for (auto thread : threads) {
|
|
|
|
if (thread->main_thread()) {
|
2019-07-29 18:24:36 +00:00
|
|
|
main_thread_ = thread;
|
2015-12-01 23:25:47 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Resume();
|
|
|
|
|
|
|
|
restore_fence_.Signal();
|
|
|
|
restoring_ = false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-06-19 02:17:37 +00:00
|
|
|
bool Emulator::TitleRequested() {
|
|
|
|
auto xam = kernel_state()->GetKernelModule<kernel::xam::XamModule>("xam.xex");
|
|
|
|
return xam->loader_data().launch_data_present;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Emulator::LaunchNextTitle() {
|
|
|
|
auto xam = kernel_state()->GetKernelModule<kernel::xam::XamModule>("xam.xex");
|
|
|
|
auto next_title = xam->loader_data().launch_path;
|
|
|
|
|
|
|
|
CompleteLaunch(L"", next_title);
|
|
|
|
}
|
|
|
|
|
2015-09-21 04:31:05 +00:00
|
|
|
bool Emulator::ExceptionCallbackThunk(Exception* ex, void* data) {
|
|
|
|
return reinterpret_cast<Emulator*>(data)->ExceptionCallback(ex);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Emulator::ExceptionCallback(Exception* ex) {
|
2015-09-11 22:09:25 +00:00
|
|
|
// Check to see if the exception occurred in guest code.
|
|
|
|
auto code_cache = processor()->backend()->code_cache();
|
|
|
|
auto code_base = code_cache->base_address();
|
|
|
|
auto code_end = code_base + code_cache->total_size();
|
|
|
|
|
2016-01-18 19:48:21 +00:00
|
|
|
if (!processor()->is_debugger_attached() && debugging::IsDebuggerAttached()) {
|
2015-09-11 22:09:25 +00:00
|
|
|
// If Xenia's debugger isn't attached but another one is, pass it to that
|
|
|
|
// debugger.
|
|
|
|
return false;
|
2016-01-18 19:48:21 +00:00
|
|
|
} else if (processor()->is_debugger_attached()) {
|
2015-09-21 04:31:05 +00:00
|
|
|
// Let the debugger handle this exception. It may decide to continue past it
|
|
|
|
// (if it was a stepping breakpoint, etc).
|
2016-01-18 19:48:21 +00:00
|
|
|
return processor()->OnUnhandledException(ex);
|
2015-09-11 22:09:25 +00:00
|
|
|
}
|
|
|
|
|
2015-09-21 04:31:05 +00:00
|
|
|
if (!(ex->pc() >= code_base && ex->pc() < code_end)) {
|
|
|
|
// Didn't occur in guest code. Let it pass.
|
|
|
|
return false;
|
|
|
|
}
|
2015-09-11 22:09:25 +00:00
|
|
|
|
2015-09-21 04:31:05 +00:00
|
|
|
// Within range. Pause the emulator and eat the exception.
|
2016-12-01 04:54:58 +00:00
|
|
|
Pause();
|
|
|
|
|
|
|
|
// Dump information into the log.
|
2015-09-21 04:31:05 +00:00
|
|
|
auto current_thread = kernel::XThread::GetCurrentThread();
|
2016-12-01 04:54:58 +00:00
|
|
|
assert_not_null(current_thread);
|
|
|
|
|
|
|
|
auto guest_function = code_cache->LookupFunction(ex->pc());
|
|
|
|
assert_not_null(guest_function);
|
|
|
|
|
|
|
|
auto context = current_thread->thread_state()->context();
|
|
|
|
|
|
|
|
XELOGE("==== CRASH DUMP ====");
|
|
|
|
XELOGE("Thread ID (Host: 0x%.8X / Guest: 0x%.8X)",
|
|
|
|
current_thread->thread()->system_id(), current_thread->thread_id());
|
|
|
|
XELOGE("Thread Handle: 0x%.8X", current_thread->handle());
|
|
|
|
XELOGE("PC: 0x%.8X", guest_function->MapMachineCodeToGuestAddress(ex->pc()));
|
|
|
|
XELOGE("Registers:");
|
|
|
|
for (int i = 0; i < 32; i++) {
|
|
|
|
XELOGE(" r%-3d = 0x%.16" PRIX64, i, context->r[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < 32; i++) {
|
|
|
|
XELOGE(" f%-3d = 0x%.16" PRIX64 " = (double)%f = (float)%f", i,
|
|
|
|
context->f[i], context->f[i], *(float*)&context->f[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < 128; i++) {
|
|
|
|
XELOGE(" v%-3d = [0x%.8X, 0x%.8X, 0x%.8X, 0x%.8X]", i, context->v[i].i32[0],
|
|
|
|
context->v[i].i32[1], context->v[i].i32[2], context->v[i].i32[3]);
|
2015-09-21 04:31:05 +00:00
|
|
|
}
|
2015-09-11 22:09:25 +00:00
|
|
|
|
2015-09-21 04:31:05 +00:00
|
|
|
// Display a dialog telling the user the guest has crashed.
|
|
|
|
display_window()->loop()->PostSynchronous([&]() {
|
2016-12-01 04:54:58 +00:00
|
|
|
xe::ui::ImGuiDialog::ShowMessageBox(
|
|
|
|
display_window(), "Uh-oh!",
|
|
|
|
"The guest has crashed.\n\n"
|
|
|
|
""
|
|
|
|
"Xenia has now paused itself.\n"
|
|
|
|
"A crash dump has been written into the log.");
|
2015-09-21 04:31:05 +00:00
|
|
|
});
|
2015-09-11 22:09:25 +00:00
|
|
|
|
2015-09-21 04:31:05 +00:00
|
|
|
// Now suspend ourself (we should be a guest thread).
|
|
|
|
current_thread->Suspend(nullptr);
|
2015-09-11 22:09:25 +00:00
|
|
|
|
2015-09-21 04:31:05 +00:00
|
|
|
// We should not arrive here!
|
|
|
|
assert_always();
|
2015-09-11 22:09:25 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-12-01 23:25:47 +00:00
|
|
|
void Emulator::WaitUntilExit() {
|
|
|
|
while (true) {
|
2019-08-04 04:48:05 +00:00
|
|
|
if (main_thread_) {
|
|
|
|
xe::threading::Wait(main_thread_->thread(), false);
|
|
|
|
}
|
2015-12-01 23:25:47 +00:00
|
|
|
|
|
|
|
if (restoring_) {
|
|
|
|
restore_fence_.Wait();
|
|
|
|
} else {
|
|
|
|
// Not restoring and the thread exited. We're finished.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2016-06-19 01:42:28 +00:00
|
|
|
|
|
|
|
on_exit();
|
2015-12-01 23:25:47 +00:00
|
|
|
}
|
|
|
|
|
2016-07-23 17:27:13 +00:00
|
|
|
std::string Emulator::FindLaunchModule() {
|
|
|
|
std::string path("game:\\");
|
2019-09-14 17:30:04 +00:00
|
|
|
|
|
|
|
if (!cvars::launch_module.empty()) {
|
|
|
|
return path + cvars::launch_module;
|
|
|
|
}
|
|
|
|
|
2016-07-23 17:27:13 +00:00
|
|
|
std::string default_module("default.xex");
|
|
|
|
|
|
|
|
auto gameinfo_entry(file_system_->ResolvePath(path + "GameInfo.bin"));
|
|
|
|
if (gameinfo_entry) {
|
2016-07-30 14:00:51 +00:00
|
|
|
vfs::File* file = nullptr;
|
2016-07-23 17:27:13 +00:00
|
|
|
X_STATUS result =
|
|
|
|
gameinfo_entry->Open(vfs::FileAccess::kGenericRead, &file);
|
|
|
|
if (XSUCCEEDED(result)) {
|
|
|
|
std::vector<uint8_t> buffer(gameinfo_entry->size());
|
|
|
|
size_t bytes_read = 0;
|
|
|
|
result = file->ReadSync(buffer.data(), buffer.size(), 0, &bytes_read);
|
|
|
|
if (XSUCCEEDED(result)) {
|
|
|
|
kernel::util::GameInfo info(buffer);
|
|
|
|
if (info.is_valid()) {
|
|
|
|
XELOGI("Found virtual title %s", info.virtual_title_id().c_str());
|
|
|
|
|
|
|
|
const std::string xna_id("584E07D1");
|
|
|
|
auto xna_id_entry(file_system_->ResolvePath(path + xna_id));
|
|
|
|
if (xna_id_entry) {
|
|
|
|
default_module = xna_id + "\\" + info.module_name();
|
|
|
|
} else {
|
|
|
|
XELOGE("Could not find fixed XNA path %s", xna_id.c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return path + default_module;
|
|
|
|
}
|
|
|
|
|
2016-01-10 19:04:55 +00:00
|
|
|
X_STATUS Emulator::CompleteLaunch(const std::wstring& path,
|
2016-06-19 01:42:28 +00:00
|
|
|
const std::string& module_path) {
|
2019-08-03 13:13:45 +00:00
|
|
|
// Reset state.
|
|
|
|
title_id_ = 0;
|
|
|
|
game_title_ = L"";
|
|
|
|
display_window_->SetIcon(nullptr, 0);
|
|
|
|
|
2015-07-05 20:44:46 +00:00
|
|
|
// Allow xam to request module loads.
|
2015-09-07 01:07:52 +00:00
|
|
|
auto xam = kernel_state()->GetKernelModule<kernel::xam::XamModule>("xam.xex");
|
2015-07-05 20:44:46 +00:00
|
|
|
|
2016-06-19 01:42:28 +00:00
|
|
|
XELOGI("Launching module %s", module_path.c_str());
|
|
|
|
auto module = kernel_state_->LoadUserModule(module_path.c_str());
|
|
|
|
if (!module) {
|
|
|
|
XELOGE("Failed to load user module %S", path.c_str());
|
|
|
|
return X_STATUS_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Grab the current title ID.
|
|
|
|
xex2_opt_execution_info* info = nullptr;
|
|
|
|
module->GetOptHeader(XEX_HEADER_EXECUTION_INFO, &info);
|
|
|
|
if (info) {
|
|
|
|
title_id_ = info->title_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try and load the resource database (xex only).
|
|
|
|
if (module->title_id()) {
|
|
|
|
char title_id[9] = {0};
|
2016-10-24 16:00:22 +00:00
|
|
|
std::snprintf(title_id, xe::countof(title_id), "%08X", module->title_id());
|
2019-04-17 19:49:58 +00:00
|
|
|
config::LoadGameConfig(xe::to_wstring(title_id));
|
2016-06-19 01:42:28 +00:00
|
|
|
uint32_t resource_data = 0;
|
|
|
|
uint32_t resource_size = 0;
|
|
|
|
if (XSUCCEEDED(
|
|
|
|
module->GetSection(title_id, &resource_data, &resource_size))) {
|
|
|
|
kernel::util::XdbfGameData db(
|
|
|
|
module->memory()->TranslateVirtual(resource_data), resource_size);
|
|
|
|
if (db.is_valid()) {
|
|
|
|
game_title_ = xe::to_wstring(db.title());
|
|
|
|
auto icon_block = db.icon();
|
|
|
|
if (icon_block) {
|
|
|
|
display_window_->SetIcon(icon_block.buffer, icon_block.size);
|
2016-01-07 20:51:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-07-05 20:44:46 +00:00
|
|
|
}
|
|
|
|
|
2020-03-21 16:21:00 +00:00
|
|
|
// Initializing the shader storage in a blocking way so the user doesn't miss
|
|
|
|
// the initial seconds - for instance, sound from an intro video may start
|
|
|
|
// playing before the video can be seen if doing this in parallel with the
|
|
|
|
// main thread.
|
|
|
|
on_shader_storage_initialization(true);
|
|
|
|
graphics_system_->InitializeShaderStorage(storage_root_, title_id_, true);
|
|
|
|
on_shader_storage_initialization(false);
|
|
|
|
|
2019-07-29 18:24:36 +00:00
|
|
|
auto main_thread = kernel_state_->LaunchModule(module);
|
|
|
|
if (!main_thread) {
|
2015-05-20 05:20:49 +00:00
|
|
|
return X_STATUS_UNSUCCESSFUL;
|
|
|
|
}
|
2019-07-29 18:24:36 +00:00
|
|
|
main_thread_ = main_thread;
|
2019-08-03 13:12:37 +00:00
|
|
|
on_launch(title_id_, game_title_);
|
2016-06-19 01:42:28 +00:00
|
|
|
|
|
|
|
return X_STATUS_SUCCESS;
|
2014-01-19 06:23:26 +00:00
|
|
|
}
|
2014-08-17 20:13:03 +00:00
|
|
|
|
|
|
|
} // namespace xe
|