From 5834a42ef3fab307cc7765b30ee7c1cc5c0cd60a Mon Sep 17 00:00:00 2001 From: Ben Vanik Date: Sun, 8 Nov 2015 15:02:24 -0800 Subject: [PATCH] Dependency injection for apu/gpu/hid. --- src/xenia/app/xenia_main.cc | 96 ++++++++++++++++++++- src/xenia/apu/apu_flags.cc | 2 - src/xenia/apu/apu_flags.h | 2 - src/xenia/apu/audio_system.cc | 28 ------ src/xenia/apu/audio_system.h | 2 - src/xenia/emulator.cc | 36 ++++++-- src/xenia/emulator.h | 11 ++- src/xenia/gpu/gl4/gl4_graphics_system.cc | 4 - src/xenia/gpu/gl4/gl4_trace_viewer_main.cc | 4 + src/xenia/gpu/gpu_flags.cc | 2 - src/xenia/gpu/gpu_flags.h | 2 - src/xenia/gpu/graphics_system.cc | 21 ----- src/xenia/gpu/graphics_system.h | 1 - src/xenia/gpu/trace_viewer.cc | 4 +- src/xenia/gpu/trace_viewer.h | 2 + src/xenia/hid/hid_demo.cc | 45 +++++++++- src/xenia/hid/hid_flags.cc | 2 - src/xenia/hid/hid_flags.h | 2 - src/xenia/hid/input_driver.cc | 3 +- src/xenia/hid/input_driver.h | 10 ++- src/xenia/hid/input_system.cc | 45 ---------- src/xenia/hid/input_system.h | 8 +- src/xenia/hid/nop/nop_hid.cc | 4 +- src/xenia/hid/nop/nop_hid.h | 2 +- src/xenia/hid/nop/nop_input_driver.cc | 5 +- src/xenia/hid/nop/nop_input_driver.h | 2 +- src/xenia/hid/winkey/winkey_hid.cc | 4 +- src/xenia/hid/winkey/winkey_hid.h | 2 +- src/xenia/hid/winkey/winkey_input_driver.cc | 10 +-- src/xenia/hid/winkey/winkey_input_driver.h | 2 +- src/xenia/hid/xinput/xinput_hid.cc | 4 +- src/xenia/hid/xinput/xinput_hid.h | 2 +- src/xenia/hid/xinput/xinput_input_driver.cc | 4 +- src/xenia/hid/xinput/xinput_input_driver.h | 2 +- 34 files changed, 217 insertions(+), 158 deletions(-) diff --git a/src/xenia/app/xenia_main.cc b/src/xenia/app/xenia_main.cc index 85d73da50..865d28cf3 100644 --- a/src/xenia/app/xenia_main.cc +++ b/src/xenia/app/xenia_main.cc @@ -17,11 +17,103 @@ #include "xenia/emulator.h" #include "xenia/ui/file_picker.h" +// Available audio systems: +#include "xenia/apu/nop/nop_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/gl4/gl4_graphics_system.h" + +// Available input drivers: +#include "xenia/hid/nop/nop_hid.h" +#if XE_PLATFORM_WIN32 +#include "xenia/hid/winkey/winkey_hid.h" +#include "xenia/hid/xinput/xinput_hid.h" +#endif // XE_PLATFORM_WIN32 + +DEFINE_string(apu, "any", "Audio system. Use: [any, nop, xaudio2]"); +DEFINE_string(gpu, "any", "Graphics system. Use: [any, gl4]"); +DEFINE_string(hid, "any", "Input system. Use: [any, nop, winkey, xinput]"); + DEFINE_string(target, "", "Specifies the target .xex or .iso to execute."); namespace xe { namespace app { +std::unique_ptr CreateAudioSystem(cpu::Processor* processor) { + if (FLAGS_apu.compare("nop") == 0) { + return apu::nop::NopAudioSystem::Create(processor); +#if XE_PLATFORM_WIN32 + } else if (FLAGS_apu.compare("xaudio2") == 0) { + return apu::xaudio2::XAudio2AudioSystem::Create(processor); +#endif // XE_PLATFORM_WIN32 + } else { + // Create best available. + std::unique_ptr best; + +#if XE_PLATFORM_WIN32 + best = apu::xaudio2::XAudio2AudioSystem::Create(processor); + if (best) { + return best; + } +#endif // XE_PLATFORM_WIN32 + + // Fallback to nop. + return apu::nop::NopAudioSystem::Create(processor); + } +} + +std::unique_ptr CreateGraphicsSystem() { + if (FLAGS_gpu.compare("gl4") == 0) { + return std::unique_ptr( + new xe::gpu::gl4::GL4GraphicsSystem()); + } else { + // Create best available. + std::unique_ptr best; + + best = std::unique_ptr( + new xe::gpu::gl4::GL4GraphicsSystem()); + if (best) { + return best; + } + + // Nothing! + return nullptr; + } +} + +std::vector> CreateInputDrivers( + ui::Window* window) { + std::vector> drivers; + if (FLAGS_hid.compare("nop") == 0) { + drivers.emplace_back(xe::hid::nop::Create(window)); +#if XE_PLATFORM_WIN32 + } else if (FLAGS_hid.compare("winkey") == 0) { + drivers.emplace_back(xe::hid::winkey::Create(window)); + } else if (FLAGS_hid.compare("xinput") == 0) { + drivers.emplace_back(xe::hid::xinput::Create(window)); +#endif // XE_PLATFORM_WIN32 + } else { +#if XE_PLATFORM_WIN32 + auto xinput_driver = xe::hid::xinput::Create(window); + if (xinput_driver) { + drivers.emplace_back(std::move(xinput_driver)); + } + auto winkey_driver = xe::hid::winkey::Create(window); + if (winkey_driver) { + drivers.emplace_back(std::move(winkey_driver)); + } +#endif // XE_PLATFORM_WIN32 + 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"); @@ -34,7 +126,9 @@ int xenia_main(const std::vector& args) { // 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()); + X_STATUS result = + emulator->Setup(emulator_window->window(), CreateAudioSystem, + CreateGraphicsSystem, CreateInputDrivers); if (XFAILED(result)) { XELOGE("Failed to setup emulator: %.8X", result); return 1; diff --git a/src/xenia/apu/apu_flags.cc b/src/xenia/apu/apu_flags.cc index aac14e851..927526bb0 100644 --- a/src/xenia/apu/apu_flags.cc +++ b/src/xenia/apu/apu_flags.cc @@ -9,6 +9,4 @@ #include "xenia/apu/apu_flags.h" -DEFINE_string(apu, "any", "Audio system. Use: [any, nop, xaudio2]"); - DEFINE_bool(mute, false, "Mutes all audio output."); diff --git a/src/xenia/apu/apu_flags.h b/src/xenia/apu/apu_flags.h index 400a2064c..d85d67b7e 100644 --- a/src/xenia/apu/apu_flags.h +++ b/src/xenia/apu/apu_flags.h @@ -12,8 +12,6 @@ #include -DECLARE_string(apu); - DECLARE_bool(mute); #endif // XENIA_APU_APU_FLAGS_H_ diff --git a/src/xenia/apu/audio_system.cc b/src/xenia/apu/audio_system.cc index e0d4f0cb5..6fbbd87d0 100644 --- a/src/xenia/apu/audio_system.cc +++ b/src/xenia/apu/audio_system.cc @@ -20,11 +20,6 @@ #include "xenia/base/threading.h" #include "xenia/cpu/thread_state.h" -#include "xenia/apu/nop/nop_audio_system.h" -#if XE_PLATFORM_WIN32 -#include "xenia/apu/xaudio2/xaudio2_audio_system.h" -#endif // XE_PLATFORM_WIN32 - // As with normal Microsoft, there are like twelve different ways to access // the audio APIs. Early games use XMA*() methods almost exclusively to touch // decoders. Later games use XAudio*() and direct memory writes to the XMA @@ -40,29 +35,6 @@ namespace xe { namespace apu { -std::unique_ptr AudioSystem::Create(cpu::Processor* processor) { - if (FLAGS_apu.compare("nop") == 0) { - return nop::NopAudioSystem::Create(processor); -#if XE_PLATFORM_WIN32 - } else if (FLAGS_apu.compare("xaudio2") == 0) { - return xaudio2::XAudio2AudioSystem::Create(processor); -#endif // WIN32 - } else { - // Create best available. - std::unique_ptr best; - -#if XE_PLATFORM_WIN32 - best = xaudio2::XAudio2AudioSystem::Create(processor); - if (best) { - return best; - } -#endif // XE_PLATFORM_WIN32 - - // Fallback to nop. - return nop::NopAudioSystem::Create(processor); - } -} - AudioSystem::AudioSystem(cpu::Processor* processor) : memory_(processor->memory()), processor_(processor), diff --git a/src/xenia/apu/audio_system.h b/src/xenia/apu/audio_system.h index f0bd9d4a6..2a6ce9d62 100644 --- a/src/xenia/apu/audio_system.h +++ b/src/xenia/apu/audio_system.h @@ -30,8 +30,6 @@ class AudioSystem { public: virtual ~AudioSystem(); - static std::unique_ptr Create(cpu::Processor* processor); - Memory* memory() const { return memory_; } cpu::Processor* processor() const { return processor_; } XmaDecoder* xma_decoder() const { return xma_decoder_.get(); } diff --git a/src/xenia/emulator.cc b/src/xenia/emulator.cc index 4319249f1..10029702a 100644 --- a/src/xenia/emulator.cc +++ b/src/xenia/emulator.cc @@ -22,6 +22,7 @@ #include "xenia/base/string.h" #include "xenia/cpu/backend/code_cache.h" #include "xenia/gpu/graphics_system.h" +#include "xenia/hid/input_driver.h" #include "xenia/hid/input_system.h" #include "xenia/kernel/kernel_state.h" #include "xenia/kernel/xam/xam_module.h" @@ -68,7 +69,14 @@ Emulator::~Emulator() { ExceptionHandler::Uninstall(Emulator::ExceptionCallbackThunk, this); } -X_STATUS Emulator::Setup(ui::Window* display_window) { +X_STATUS Emulator::Setup( + ui::Window* display_window, + std::function(cpu::Processor*)> + audio_system_factory, + std::function()> + graphics_system_factory, + std::function>(ui::Window*)> + input_driver_factory) { X_STATUS result = X_STATUS_UNSUCCESSFUL; display_window_ = display_window; @@ -111,13 +119,15 @@ X_STATUS Emulator::Setup(ui::Window* display_window) { } // Initialize the APU. - audio_system_ = xe::apu::AudioSystem::Create(processor_.get()); - if (!audio_system_) { - return X_STATUS_NOT_IMPLEMENTED; + if (audio_system_factory) { + audio_system_ = audio_system_factory(processor_.get()); + if (!audio_system_) { + return X_STATUS_NOT_IMPLEMENTED; + } } // Initialize the GPU. - graphics_system_ = xe::gpu::GraphicsSystem::Create(); + graphics_system_ = graphics_system_factory(); if (!graphics_system_) { return X_STATUS_NOT_IMPLEMENTED; } @@ -127,10 +137,16 @@ X_STATUS Emulator::Setup(ui::Window* display_window) { }); // Initialize the HID. - input_system_ = xe::hid::InputSystem::Create(display_window_); + input_system_ = std::make_unique(display_window_); if (!input_system_) { return X_STATUS_NOT_IMPLEMENTED; } + 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])); + } + } result = input_system_->Setup(); if (result) { @@ -150,9 +166,11 @@ X_STATUS Emulator::Setup(ui::Window* display_window) { return result; } - result = audio_system_->Setup(kernel_state_.get()); - if (result) { - return result; + if (audio_system_) { + result = audio_system_->Setup(kernel_state_.get()); + if (result) { + return result; + } } // HLE kernel modules. diff --git a/src/xenia/emulator.h b/src/xenia/emulator.h index 628a3456f..f50729cb7 100644 --- a/src/xenia/emulator.h +++ b/src/xenia/emulator.h @@ -10,6 +10,7 @@ #ifndef XENIA_EMULATOR_H_ #define XENIA_EMULATOR_H_ +#include #include #include "xenia/base/exception_handler.h" @@ -32,6 +33,7 @@ namespace gpu { class GraphicsSystem; } // namespace gpu namespace hid { +class InputDriver; class InputSystem; } // namespace hid namespace ui { @@ -68,7 +70,14 @@ class Emulator { kernel::KernelState* kernel_state() const { return kernel_state_.get(); } - X_STATUS Setup(ui::Window* display_window); + X_STATUS Setup( + ui::Window* display_window, + std::function(cpu::Processor*)> + audio_system_factory, + std::function()> + graphics_system_factory, + std::function>(ui::Window*)> + input_driver_factory); X_STATUS LaunchPath(std::wstring path); X_STATUS LaunchXexFile(std::wstring path); diff --git a/src/xenia/gpu/gl4/gl4_graphics_system.cc b/src/xenia/gpu/gl4/gl4_graphics_system.cc index 210c5bdad..0279541fb 100644 --- a/src/xenia/gpu/gl4/gl4_graphics_system.cc +++ b/src/xenia/gpu/gl4/gl4_graphics_system.cc @@ -24,10 +24,6 @@ namespace xe { namespace gpu { namespace gl4 { -std::unique_ptr Create() { - return std::make_unique(); -} - std::unique_ptr GL4GraphicsSystem::CreateContext( ui::Window* target_window) { // Setup the GL control that actually does the drawing. diff --git a/src/xenia/gpu/gl4/gl4_trace_viewer_main.cc b/src/xenia/gpu/gl4/gl4_trace_viewer_main.cc index 48739ba1f..48a108be6 100644 --- a/src/xenia/gpu/gl4/gl4_trace_viewer_main.cc +++ b/src/xenia/gpu/gl4/gl4_trace_viewer_main.cc @@ -30,6 +30,10 @@ using namespace xe::gpu::xenos; class GL4TraceViewer : public TraceViewer { public: + std::unique_ptr CreateGraphicsSystem() override { + return std::unique_ptr(new GL4GraphicsSystem()); + } + uintptr_t GetColorRenderTarget(uint32_t pitch, MsaaSamples samples, uint32_t base, ColorRenderTargetFormat format) override { diff --git a/src/xenia/gpu/gpu_flags.cc b/src/xenia/gpu/gpu_flags.cc index 3252d0d79..1cfe1ad65 100644 --- a/src/xenia/gpu/gpu_flags.cc +++ b/src/xenia/gpu/gpu_flags.cc @@ -9,8 +9,6 @@ #include "xenia/gpu/gpu_flags.h" -DEFINE_string(gpu, "any", "Graphics system. Use: [any, gl4]"); - DEFINE_string(trace_gpu_prefix, "scratch/gpu/gpu_trace_", "Prefix path for GPU trace files."); DEFINE_bool(trace_gpu_stream, false, "Trace all GPU packets."); diff --git a/src/xenia/gpu/gpu_flags.h b/src/xenia/gpu/gpu_flags.h index 572006896..98c7e8f0c 100644 --- a/src/xenia/gpu/gpu_flags.h +++ b/src/xenia/gpu/gpu_flags.h @@ -12,8 +12,6 @@ #include -DECLARE_string(gpu); - DECLARE_string(trace_gpu_prefix); DECLARE_bool(trace_gpu_stream); diff --git a/src/xenia/gpu/graphics_system.cc b/src/xenia/gpu/graphics_system.cc index 0d7ddb989..a78560ecd 100644 --- a/src/xenia/gpu/graphics_system.cc +++ b/src/xenia/gpu/graphics_system.cc @@ -21,27 +21,6 @@ namespace xe { namespace gpu { -namespace gl4 { -std::unique_ptr Create(); -} // namespace gl4 - -std::unique_ptr GraphicsSystem::Create() { - if (FLAGS_gpu.compare("gl4") == 0) { - return xe::gpu::gl4::Create(); - } else { - // Create best available. - std::unique_ptr best; - - best = xe::gpu::gl4::Create(); - if (best) { - return best; - } - - // Nothing! - return nullptr; - } -} - GraphicsSystem::GraphicsSystem() : vsync_worker_running_(false) {} GraphicsSystem::~GraphicsSystem() = default; diff --git a/src/xenia/gpu/graphics_system.h b/src/xenia/gpu/graphics_system.h index 9ff96aa10..659eed964 100644 --- a/src/xenia/gpu/graphics_system.h +++ b/src/xenia/gpu/graphics_system.h @@ -34,7 +34,6 @@ class GraphicsSystem { public: virtual ~GraphicsSystem(); - static std::unique_ptr Create(); virtual std::unique_ptr CreateContext( ui::Window* target_window) = 0; diff --git a/src/xenia/gpu/trace_viewer.cc b/src/xenia/gpu/trace_viewer.cc index f73fc5714..53c750428 100644 --- a/src/xenia/gpu/trace_viewer.cc +++ b/src/xenia/gpu/trace_viewer.cc @@ -66,7 +66,9 @@ bool TraceViewer::Setup() { // Create the emulator but don't initialize so we can setup the window. emulator_ = std::make_unique(L""); - X_STATUS result = emulator_->Setup(window_.get()); + X_STATUS result = + emulator_->Setup(window_.get(), nullptr, + [this]() { return CreateGraphicsSystem(); }, nullptr); if (XFAILED(result)) { XELOGE("Failed to setup emulator: %.8X", result); return false; diff --git a/src/xenia/gpu/trace_viewer.h b/src/xenia/gpu/trace_viewer.h index eaa298df3..542a553cf 100644 --- a/src/xenia/gpu/trace_viewer.h +++ b/src/xenia/gpu/trace_viewer.h @@ -44,6 +44,8 @@ class TraceViewer { protected: TraceViewer(); + virtual std::unique_ptr CreateGraphicsSystem() = 0; + void DrawMultilineString(const std::string& str); virtual uintptr_t GetColorRenderTarget( diff --git a/src/xenia/hid/hid_demo.cc b/src/xenia/hid/hid_demo.cc index 90694e97d..5cb7e4f9f 100644 --- a/src/xenia/hid/hid_demo.cc +++ b/src/xenia/hid/hid_demo.cc @@ -22,12 +22,51 @@ #include "xenia/ui/imgui_drawer.h" #include "xenia/ui/window.h" +// Available input drivers: +#include "xenia/hid/nop/nop_hid.h" +#if XE_PLATFORM_WIN32 +#include "xenia/hid/winkey/winkey_hid.h" +#include "xenia/hid/xinput/xinput_hid.h" +#endif // XE_PLATFORM_WIN32 + +DEFINE_string(hid, "any", "Input system. Use: [any, nop, winkey, xinput]"); + namespace xe { namespace hid { std::unique_ptr imgui_drawer_; std::unique_ptr input_system_; +std::vector> CreateInputDrivers( + ui::Window* window) { + std::vector> drivers; + if (FLAGS_hid.compare("nop") == 0) { + drivers.emplace_back(xe::hid::nop::Create(window)); +#if XE_PLATFORM_WIN32 + } else if (FLAGS_hid.compare("winkey") == 0) { + drivers.emplace_back(xe::hid::winkey::Create(window)); + } else if (FLAGS_hid.compare("xinput") == 0) { + drivers.emplace_back(xe::hid::xinput::Create(window)); +#endif // XE_PLATFORM_WIN32 + } else { +#if XE_PLATFORM_WIN32 + auto xinput_driver = xe::hid::xinput::Create(window); + if (xinput_driver) { + drivers.emplace_back(std::move(xinput_driver)); + } + auto winkey_driver = xe::hid::winkey::Create(window); + if (winkey_driver) { + drivers.emplace_back(std::move(winkey_driver)); + } +#endif // XE_PLATFORM_WIN32 + if (drivers.empty()) { + // Fallback to nop if none created. + drivers.emplace_back(xe::hid::nop::Create(window)); + } + } + return drivers; +} + std::unique_ptr CreateDemoContext( xe::ui::Window* window) { return xe::ui::gl::GLContext::Create(window); @@ -70,7 +109,11 @@ int hid_demo_main(const std::vector& args) { imgui_drawer_->SetupDefaultInput(); // Initialize input system and all drivers. - input_system_ = xe::hid::InputSystem::Create(window.get()); + input_system_ = std::make_unique(window.get()); + auto drivers = CreateInputDrivers(window.get()); + for (size_t i = 0; i < drivers.size(); ++i) { + input_system_->AddDriver(std::move(drivers[i])); + } }); window->on_painting.AddListener([&](xe::ui::UIEvent* e) { diff --git a/src/xenia/hid/hid_flags.cc b/src/xenia/hid/hid_flags.cc index ccf00dd18..784b32f42 100644 --- a/src/xenia/hid/hid_flags.cc +++ b/src/xenia/hid/hid_flags.cc @@ -8,5 +8,3 @@ */ #include "xenia/hid/hid_flags.h" - -DEFINE_string(hid, "any", "Input system. Use: [any, nop, winkey, xinput]"); diff --git a/src/xenia/hid/hid_flags.h b/src/xenia/hid/hid_flags.h index def7711da..ec43d1d7d 100644 --- a/src/xenia/hid/hid_flags.h +++ b/src/xenia/hid/hid_flags.h @@ -12,6 +12,4 @@ #include -DECLARE_string(hid); - #endif // XENIA_HID_HID_FLAGS_H_ diff --git a/src/xenia/hid/input_driver.cc b/src/xenia/hid/input_driver.cc index b74c0035c..1627d5dc4 100644 --- a/src/xenia/hid/input_driver.cc +++ b/src/xenia/hid/input_driver.cc @@ -12,8 +12,7 @@ namespace xe { namespace hid { -InputDriver::InputDriver(InputSystem* input_system) - : input_system_(input_system) {} +InputDriver::InputDriver(xe::ui::Window* window) : window_(window) {} InputDriver::~InputDriver() = default; diff --git a/src/xenia/hid/input_driver.h b/src/xenia/hid/input_driver.h index 510c14c0b..6de322ab1 100644 --- a/src/xenia/hid/input_driver.h +++ b/src/xenia/hid/input_driver.h @@ -13,6 +13,12 @@ #include "xenia/hid/input.h" #include "xenia/xbox.h" +namespace xe { +namespace ui { +class Window; +} // namespace ui +} // namespace xe + namespace xe { namespace hid { @@ -33,9 +39,9 @@ class InputDriver { X_INPUT_KEYSTROKE* out_keystroke) = 0; protected: - explicit InputDriver(InputSystem* input_system); + explicit InputDriver(xe::ui::Window* window); - InputSystem* input_system_ = nullptr; + xe::ui::Window* window_ = nullptr; }; } // namespace hid diff --git a/src/xenia/hid/input_system.cc b/src/xenia/hid/input_system.cc index 9ca55bea8..a77d5b548 100644 --- a/src/xenia/hid/input_system.cc +++ b/src/xenia/hid/input_system.cc @@ -13,54 +13,9 @@ #include "xenia/hid/hid_flags.h" #include "xenia/hid/input_driver.h" -#include "xenia/hid/nop/nop_hid.h" -#if XE_PLATFORM_WIN32 -#include "xenia/hid/winkey/winkey_hid.h" -#include "xenia/hid/xinput/xinput_hid.h" -#endif // XE_PLATFORM_WIN32 - namespace xe { namespace hid { -std::unique_ptr InputSystem::Create(xe::ui::Window* window) { - std::unique_ptr input_system(new InputSystem(window)); - - if (FLAGS_hid.compare("nop") == 0) { - input_system->AddDriver(xe::hid::nop::Create(input_system.get())); -#if XE_PLATFORM_WIN32 - } else if (FLAGS_hid.compare("winkey") == 0) { - input_system->AddDriver(xe::hid::winkey::Create(input_system.get())); - } else if (FLAGS_hid.compare("xinput") == 0) { - input_system->AddDriver(xe::hid::xinput::Create(input_system.get())); -#endif // WIN32 - } else { - // Create all available. - bool any_created = false; - -// NOTE: in any mode we create as many as we can, falling back to nop. - -#if XE_PLATFORM_WIN32 - auto xinput_driver = xe::hid::xinput::Create(input_system.get()); - if (xinput_driver) { - input_system->AddDriver(std::move(xinput_driver)); - any_created = true; - } - auto winkey_driver = xe::hid::winkey::Create(input_system.get()); - if (winkey_driver) { - input_system->AddDriver(std::move(winkey_driver)); - any_created = true; - } -#endif // WIN32 - - // Fallback to nop if none created. - if (!any_created) { - input_system->AddDriver(xe::hid::nop::Create(input_system.get())); - } - } - - return input_system; -} - InputSystem::InputSystem(xe::ui::Window* window) : window_(window) {} InputSystem::~InputSystem() = default; diff --git a/src/xenia/hid/input_system.h b/src/xenia/hid/input_system.h index adcefcf85..7f26e0663 100644 --- a/src/xenia/hid/input_system.h +++ b/src/xenia/hid/input_system.h @@ -14,6 +14,7 @@ #include #include "xenia/hid/input.h" +#include "xenia/hid/input_driver.h" #include "xenia/xbox.h" namespace xe { @@ -25,14 +26,11 @@ class Window; namespace xe { namespace hid { -class InputDriver; - class InputSystem { public: + explicit InputSystem(xe::ui::Window* window); ~InputSystem(); - static std::unique_ptr Create(xe::ui::Window* window); - xe::ui::Window* window() const { return window_; } X_STATUS Setup(); @@ -47,8 +45,6 @@ class InputSystem { X_INPUT_KEYSTROKE* out_keystroke); private: - explicit InputSystem(xe::ui::Window* window); - xe::ui::Window* window_ = nullptr; std::vector> drivers_; diff --git a/src/xenia/hid/nop/nop_hid.cc b/src/xenia/hid/nop/nop_hid.cc index fa69c7390..54765df5f 100644 --- a/src/xenia/hid/nop/nop_hid.cc +++ b/src/xenia/hid/nop/nop_hid.cc @@ -15,8 +15,8 @@ namespace xe { namespace hid { namespace nop { -std::unique_ptr Create(InputSystem* input_system) { - return std::make_unique(input_system); +std::unique_ptr Create(xe::ui::Window* window) { + return std::make_unique(window); } } // namespace nop diff --git a/src/xenia/hid/nop/nop_hid.h b/src/xenia/hid/nop/nop_hid.h index 9c65c1e36..c2ea0a037 100644 --- a/src/xenia/hid/nop/nop_hid.h +++ b/src/xenia/hid/nop/nop_hid.h @@ -18,7 +18,7 @@ namespace xe { namespace hid { namespace nop { -std::unique_ptr Create(InputSystem* input_system); +std::unique_ptr Create(xe::ui::Window* window); } // namespace nop } // namespace hid diff --git a/src/xenia/hid/nop/nop_input_driver.cc b/src/xenia/hid/nop/nop_input_driver.cc index 4794b83f4..bd22d65e2 100644 --- a/src/xenia/hid/nop/nop_input_driver.cc +++ b/src/xenia/hid/nop/nop_input_driver.cc @@ -15,10 +15,9 @@ namespace xe { namespace hid { namespace nop { -NopInputDriver::NopInputDriver(InputSystem* input_system) - : InputDriver(input_system) {} +NopInputDriver::NopInputDriver(xe::ui::Window* window) : InputDriver(window) {} -NopInputDriver::~NopInputDriver() {} +NopInputDriver::~NopInputDriver() = default; X_STATUS NopInputDriver::Setup() { return X_STATUS_SUCCESS; } diff --git a/src/xenia/hid/nop/nop_input_driver.h b/src/xenia/hid/nop/nop_input_driver.h index 3609a1317..be949f016 100644 --- a/src/xenia/hid/nop/nop_input_driver.h +++ b/src/xenia/hid/nop/nop_input_driver.h @@ -18,7 +18,7 @@ namespace nop { class NopInputDriver : public InputDriver { public: - explicit NopInputDriver(InputSystem* input_system); + explicit NopInputDriver(xe::ui::Window* window); ~NopInputDriver() override; X_STATUS Setup() override; diff --git a/src/xenia/hid/winkey/winkey_hid.cc b/src/xenia/hid/winkey/winkey_hid.cc index 1b463dfef..7244f4c53 100644 --- a/src/xenia/hid/winkey/winkey_hid.cc +++ b/src/xenia/hid/winkey/winkey_hid.cc @@ -15,8 +15,8 @@ namespace xe { namespace hid { namespace winkey { -std::unique_ptr Create(InputSystem* input_system) { - return std::make_unique(input_system); +std::unique_ptr Create(xe::ui::Window* window) { + return std::make_unique(window); } } // namespace winkey diff --git a/src/xenia/hid/winkey/winkey_hid.h b/src/xenia/hid/winkey/winkey_hid.h index 540885bfa..a17e5ba98 100644 --- a/src/xenia/hid/winkey/winkey_hid.h +++ b/src/xenia/hid/winkey/winkey_hid.h @@ -18,7 +18,7 @@ namespace xe { namespace hid { namespace winkey { -std::unique_ptr Create(InputSystem* input_system); +std::unique_ptr Create(xe::ui::Window* window); } // namespace winkey } // namespace hid diff --git a/src/xenia/hid/winkey/winkey_input_driver.cc b/src/xenia/hid/winkey/winkey_input_driver.cc index aff500486..3e2f44a99 100644 --- a/src/xenia/hid/winkey/winkey_input_driver.cc +++ b/src/xenia/hid/winkey/winkey_input_driver.cc @@ -19,10 +19,10 @@ namespace xe { namespace hid { namespace winkey { -WinKeyInputDriver::WinKeyInputDriver(InputSystem* input_system) - : InputDriver(input_system), packet_number_(1) { +WinKeyInputDriver::WinKeyInputDriver(xe::ui::Window* window) + : InputDriver(window), packet_number_(1) { // Register a key listener. - input_system_->window()->on_key_down.AddListener([this](ui::KeyEvent* evt) { + window_->on_key_down.AddListener([this](ui::KeyEvent* evt) { auto global_lock = global_critical_region_.Acquire(); KeyEvent key; @@ -32,7 +32,7 @@ WinKeyInputDriver::WinKeyInputDriver(InputSystem* input_system) key.repeat_count = evt->repeat_count(); key_events_.push(key); }); - input_system_->window()->on_key_up.AddListener([this](ui::KeyEvent* evt) { + window_->on_key_up.AddListener([this](ui::KeyEvent* evt) { auto global_lock = global_critical_region_.Acquire(); KeyEvent key; @@ -89,7 +89,7 @@ X_RESULT WinKeyInputDriver::GetState(uint32_t user_index, int16_t thumb_rx = 0; int16_t thumb_ry = 0; - if (input_system_->window()->has_focus()) { + if (window_->has_focus()) { if (IS_KEY_TOGGLED(VK_CAPITAL)) { // dpad toggled if (IS_KEY_DOWN(0x41)) { diff --git a/src/xenia/hid/winkey/winkey_input_driver.h b/src/xenia/hid/winkey/winkey_input_driver.h index 65d2cd476..2bc923f39 100644 --- a/src/xenia/hid/winkey/winkey_input_driver.h +++ b/src/xenia/hid/winkey/winkey_input_driver.h @@ -21,7 +21,7 @@ namespace winkey { class WinKeyInputDriver : public InputDriver { public: - explicit WinKeyInputDriver(InputSystem* input_system); + explicit WinKeyInputDriver(xe::ui::Window* window); ~WinKeyInputDriver() override; X_STATUS Setup() override; diff --git a/src/xenia/hid/xinput/xinput_hid.cc b/src/xenia/hid/xinput/xinput_hid.cc index f08308b3e..093d266f0 100644 --- a/src/xenia/hid/xinput/xinput_hid.cc +++ b/src/xenia/hid/xinput/xinput_hid.cc @@ -15,8 +15,8 @@ namespace xe { namespace hid { namespace xinput { -std::unique_ptr Create(InputSystem* input_system) { - return std::make_unique(input_system); +std::unique_ptr Create(xe::ui::Window* window) { + return std::make_unique(window); } } // namespace xinput diff --git a/src/xenia/hid/xinput/xinput_hid.h b/src/xenia/hid/xinput/xinput_hid.h index a63b66350..c295cb434 100644 --- a/src/xenia/hid/xinput/xinput_hid.h +++ b/src/xenia/hid/xinput/xinput_hid.h @@ -18,7 +18,7 @@ namespace xe { namespace hid { namespace xinput { -std::unique_ptr Create(InputSystem* input_system); +std::unique_ptr Create(xe::ui::Window* window); } // namespace xinput } // namespace hid diff --git a/src/xenia/hid/xinput/xinput_input_driver.cc b/src/xenia/hid/xinput/xinput_input_driver.cc index e18ddef77..ca4951dfd 100644 --- a/src/xenia/hid/xinput/xinput_input_driver.cc +++ b/src/xenia/hid/xinput/xinput_input_driver.cc @@ -19,8 +19,8 @@ namespace xe { namespace hid { namespace xinput { -XInputInputDriver::XInputInputDriver(InputSystem* input_system) - : InputDriver(input_system) { +XInputInputDriver::XInputInputDriver(xe::ui::Window* window) + : InputDriver(window) { XInputEnable(TRUE); } diff --git a/src/xenia/hid/xinput/xinput_input_driver.h b/src/xenia/hid/xinput/xinput_input_driver.h index e206dd3c0..fbf75c422 100644 --- a/src/xenia/hid/xinput/xinput_input_driver.h +++ b/src/xenia/hid/xinput/xinput_input_driver.h @@ -18,7 +18,7 @@ namespace xinput { class XInputInputDriver : public InputDriver { public: - explicit XInputInputDriver(InputSystem* input_system); + explicit XInputInputDriver(xe::ui::Window* window); ~XInputInputDriver() override; X_STATUS Setup() override;