/** ****************************************************************************** * 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/app/discord/discord_presence.h" #include "xenia/app/emulator_window.h" #include "xenia/base/cvar.h" #include "xenia/base/debugging.h" #include "xenia/base/logging.h" #include "xenia/base/main.h" #include "xenia/base/profiling.h" #include "xenia/base/threading.h" #include "xenia/config.h" #include "xenia/debug/ui/debug_window.h" #include "xenia/emulator.h" #include "xenia/ui/file_picker.h" #include "xenia/vfs/devices/host_path_device.h" // Available audio systems: #include "xenia/apu/nop/nop_audio_system.h" #include "xenia/apu/sdl/sdl_audio_system.h" #if XE_PLATFORM_WIN32 #include "xenia/apu/xaudio2/xaudio2_audio_system.h" #endif // XE_PLATFORM_WIN32 // Available graphics systems: #include "xenia/gpu/null/null_graphics_system.h" #include "xenia/gpu/vulkan/vulkan_graphics_system.h" #if XE_PLATFORM_WIN32 #include "xenia/gpu/d3d12/d3d12_graphics_system.h" #endif // XE_PLATFORM_WIN32 // Available input drivers: #include "xenia/hid/nop/nop_hid.h" #include "xenia/hid/sdl/sdl_hid.h" #if XE_PLATFORM_WIN32 #include "xenia/hid/winkey/winkey_hid.h" #include "xenia/hid/xinput/xinput_hid.h" #endif // XE_PLATFORM_WIN32 #include "third_party/fmt/include/fmt/format.h" #include "third_party/xbyak/xbyak/xbyak_util.h" DEFINE_string(apu, "any", "Audio system. Use: [any, nop, sdl, xaudio2]", "APU"); DEFINE_string(gpu, "any", "Graphics system. Use: [any, d3d12, vulkan, null]", "GPU"); DEFINE_string(hid, "any", "Input system. Use: [any, nop, sdl, winkey, xinput]", "HID"); DEFINE_bool(fullscreen, false, "Toggles fullscreen", "GPU"); DEFINE_path( storage_root, "", "Root path for persistent internal data storage (config, etc.), or empty " "to use the path preferred for the OS, such as the documents folder, or " "the emulator executable directory if portable.txt is present in it.", "Storage"); DEFINE_path( content_root, "", "Root path for guest content storage (saves, etc.), or empty to use the " "content folder under the storage root.", "Storage"); DEFINE_bool(mount_scratch, false, "Enable scratch mount", "Storage"); DEFINE_bool(mount_cache, false, "Enable cache mount", "Storage"); DEFINE_transient_path(target, "", "Specifies the target .xex or .iso to execute.", "General"); DEFINE_transient_bool(portable, false, "Specifies if Xenia should run in portable mode.", "General"); DECLARE_bool(debug); DEFINE_bool(discord, true, "Enable Discord rich presence", "General"); namespace xe { namespace app { template class Factory { private: struct Creator { std::string name; std::function is_available; std::function(Args...)> instantiate; }; std::vector creators_; public: void Add(const std::string_view name, std::function is_available, std::function(Args...)> instantiate) { creators_.push_back({std::string(name), is_available, instantiate}); } void Add(const std::string_view name, std::function(Args...)> instantiate) { auto always_available = []() { return true; }; Add(name, always_available, instantiate); } template void Add(const std::string_view name) { Add(name, DT::IsAvailable, [](Args... args) { return std::make_unique
(std::forward(args)...); }); } std::unique_ptr Create(const std::string_view name, Args... args) { if (!name.empty() && name != "any") { auto it = std::find_if( creators_.cbegin(), creators_.cend(), [&name](const auto& f) { return name.compare(f.name) == 0; }); if (it != creators_.cend() && (*it).is_available()) { return (*it).instantiate(std::forward(args)...); } return nullptr; } else { for (const auto& creator : creators_) { if (!creator.is_available()) continue; auto instance = creator.instantiate(std::forward(args)...); if (!instance) continue; return instance; } return nullptr; } } std::vector> CreateAll(const std::string_view name, Args... args) { std::vector> instances; if (!name.empty() && name != "any") { auto it = std::find_if( creators_.cbegin(), creators_.cend(), [&name](const auto& f) { return name.compare(f.name) == 0; }); if (it != creators_.cend() && (*it).is_available()) { auto instance = (*it).instantiate(std::forward(args)...); if (instance) { instances.emplace_back(std::move(instance)); } } } else { for (const auto& creator : creators_) { if (!creator.is_available()) continue; auto instance = creator.instantiate(std::forward(args)...); if (instance) { instances.emplace_back(std::move(instance)); } } } return instances; } }; std::unique_ptr CreateAudioSystem(cpu::Processor* processor) { Factory factory; #if XE_PLATFORM_WIN32 factory.Add("xaudio2"); #endif // XE_PLATFORM_WIN32 factory.Add("sdl"); factory.Add("nop"); return factory.Create(cvars::apu, processor); } std::unique_ptr CreateGraphicsSystem() { Factory factory; #if XE_PLATFORM_WIN32 factory.Add("d3d12"); #endif // XE_PLATFORM_WIN32 factory.Add("vulkan"); factory.Add("null"); return factory.Create(cvars::gpu); } std::vector> CreateInputDrivers( ui::Window* window) { std::vector> drivers; if (cvars::hid.compare("nop") == 0) { drivers.emplace_back(xe::hid::nop::Create(window)); } else { Factory factory; #if XE_PLATFORM_WIN32 factory.Add("xinput", xe::hid::xinput::Create); // WinKey input driver should always be the last input driver added! factory.Add("winkey", xe::hid::winkey::Create); #endif // XE_PLATFORM_WIN32 factory.Add("sdl", xe::hid::sdl::Create); for (auto& driver : factory.CreateAll(cvars::hid, window)) { if (XSUCCEEDED(driver->Setup())) { drivers.emplace_back(std::move(driver)); } } if (drivers.empty()) { // Fallback to nop if none created. drivers.emplace_back(xe::hid::nop::Create(window)); } } return drivers; } int xenia_main(const std::vector& args) { Profiler::Initialize(); Profiler::ThreadEnter("main"); // Figure out where internal files and content should go. std::filesystem::path storage_root = cvars::storage_root; if (storage_root.empty()) { storage_root = xe::filesystem::GetExecutableFolder(); if (!cvars::portable && !std::filesystem::exists(storage_root / "portable.txt")) { storage_root = xe::filesystem::GetUserFolder(); #if defined(XE_PLATFORM_WIN32) || defined(XE_PLATFORM_GNU_LINUX) storage_root = storage_root / "Xenia"; #else #warning Unhandled platform for the data root. storage_root = storage_root / "Xenia"; #endif } } storage_root = std::filesystem::absolute(storage_root); XELOGI("Storage root: {}", xe::path_to_utf8(storage_root)); config::SetupConfig(storage_root); std::filesystem::path content_root = cvars::content_root; if (content_root.empty()) { content_root = storage_root / "content"; } else { // If content root isn't an absolute path, then it should be relative to the // storage root. if (!content_root.is_absolute()) { content_root = storage_root / content_root; } } content_root = std::filesystem::absolute(content_root); XELOGI("Content root: {}", xe::path_to_utf8(content_root)); if (cvars::discord) { discord::DiscordPresence::Initialize(); discord::DiscordPresence::NotPlaying(); } // Create the emulator but don't initialize so we can setup the window. auto emulator = std::make_unique("", storage_root, content_root); // Main emulator display window. auto emulator_window = EmulatorWindow::Create(emulator.get()); // Setup and initialize all subsystems. If we can't do something // (unsupported system, memory issues, etc) this will fail early. X_STATUS result = emulator->Setup(emulator_window->window(), CreateAudioSystem, CreateGraphicsSystem, CreateInputDrivers); if (XFAILED(result)) { XELOGE("Failed to setup emulator: {:08X}", result); return 1; } if (cvars::mount_scratch) { auto scratch_device = std::make_unique( "\\SCRATCH", "scratch", false); if (!scratch_device->Initialize()) { XELOGE("Unable to scan scratch path"); } else { if (!emulator->file_system()->RegisterDevice(std::move(scratch_device))) { XELOGE("Unable to register scratch path"); } else { emulator->file_system()->RegisterSymbolicLink("scratch:", "\\SCRATCH"); } } } if (cvars::mount_cache) { auto cache0_device = std::make_unique("\\CACHE0", "cache0", false); if (!cache0_device->Initialize()) { XELOGE("Unable to scan cache0 path"); } else { if (!emulator->file_system()->RegisterDevice(std::move(cache0_device))) { XELOGE("Unable to register cache0 path"); } else { emulator->file_system()->RegisterSymbolicLink("cache0:", "\\CACHE0"); } } auto cache1_device = std::make_unique("\\CACHE1", "cache1", false); if (!cache1_device->Initialize()) { XELOGE("Unable to scan cache1 path"); } else { if (!emulator->file_system()->RegisterDevice(std::move(cache1_device))) { XELOGE("Unable to register cache1 path"); } else { emulator->file_system()->RegisterSymbolicLink("cache1:", "\\CACHE1"); } } } // Set a debug handler. // This will respond to debugging requests so we can open the debug UI. std::unique_ptr debug_window; if (cvars::debug) { emulator->processor()->set_debug_listener_request_handler( [&](xe::cpu::Processor* processor) { if (debug_window) { return debug_window.get(); } emulator_window->loop()->PostSynchronous([&]() { debug_window = xe::debug::ui::DebugWindow::Create( emulator.get(), emulator_window->loop()); debug_window->window()->on_closed.AddListener( [&](xe::ui::UIEvent* e) { emulator->processor()->set_debug_listener(nullptr); emulator_window->loop()->Post( [&]() { debug_window.reset(); }); }); }); return debug_window.get(); }); } auto evt = xe::threading::Event::CreateAutoResetEvent(false); emulator->on_launch.AddListener([&](auto title_id, const auto& game_title) { if (cvars::discord) { discord::DiscordPresence::PlayingTitle( game_title.empty() ? "Unknown Title" : std::string(game_title)); } emulator_window->UpdateTitle(); evt->Set(); }); emulator->on_shader_storage_initialization.AddListener( [&](bool initializing) { emulator_window->SetInitializingShaderStorage(initializing); }); emulator->on_terminate.AddListener([&]() { if (cvars::discord) { discord::DiscordPresence::NotPlaying(); } }); emulator_window->window()->on_closing.AddListener([&](ui::UIEvent* e) { // This needs to shut down before the graphics context. Profiler::Shutdown(); }); bool exiting = false; emulator_window->loop()->on_quit.AddListener([&](ui::UIEvent* e) { exiting = true; evt->Set(); if (cvars::discord) { discord::DiscordPresence::Shutdown(); } // TODO(DrChat): Remove this code and do a proper exit. XELOGI("Cheap-skate exit!"); exit(0); }); // Enable the main menu now that the emulator is properly loaded emulator_window->window()->EnableMainMenu(); // Grab path from the flag or unnamed argument. std::filesystem::path path; if (!cvars::target.empty()) { path = cvars::target; } // Toggles fullscreen if (cvars::fullscreen) emulator_window->ToggleFullscreen(); if (!path.empty()) { // Normalize the path and make absolute. auto abs_path = std::filesystem::absolute(path); result = emulator->LaunchPath(abs_path); if (XFAILED(result)) { xe::FatalError(fmt::format("Failed to launch target: {:08X}", result)); emulator.reset(); emulator_window.reset(); return 1; } } // Now, we're going to use the main thread to drive events related to // emulation. while (!exiting) { xe::threading::Wait(evt.get(), false); while (true) { emulator->WaitUntilExit(); if (emulator->TitleRequested()) { emulator->LaunchNextTitle(); } else { break; } } } debug_window.reset(); emulator.reset(); if (cvars::discord) { discord::DiscordPresence::Shutdown(); } Profiler::Dump(); Profiler::Shutdown(); emulator_window.reset(); return 0; } } // namespace app } // namespace xe DEFINE_ENTRY_POINT("xenia", xe::app::xenia_main, "[Path to .iso/.xex]", "target");