From c827fdd2b576d34af43a9cb3b80ecd9616493cf6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?L=C3=A9o=20Lam?= Date: Thu, 7 Jul 2016 15:40:14 +0200 Subject: [PATCH 1/2] WiimoteReal: Don't block on refresh This changes Refresh() to use the existing scanning thread to scan for devices, instead of running the scan on the UI thread and blocking it. Also makes the UI thread not block when Continuous Scanning is disabled and removes duplicated code. Should fix issue 8992. Under the hood: * The scanning thread is now always active, even when continuous scanning is disabled. * The initialize code which waits for Wiimotes to be connected also uses the scanning thread instead of scanning on yet another thread. * The scanning thread now always checks for disconnected devices, to avoid Dolphin thinking a Wiimote is still connected when it isn't. So we now check if we need new Wiimotes or a Balance Board at scan time. --- Source/Core/Core/Core.cpp | 4 +- Source/Core/Core/HW/Wiimote.cpp | 4 +- Source/Core/Core/HW/Wiimote.h | 10 +- .../Core/Core/HW/WiimoteReal/WiimoteReal.cpp | 156 ++++++------------ Source/Core/Core/HW/WiimoteReal/WiimoteReal.h | 23 ++- Source/Core/DolphinWX/ControllerConfigDiag.h | 2 +- Source/Core/DolphinWX/Frame.cpp | 6 +- 7 files changed, 79 insertions(+), 126 deletions(-) diff --git a/Source/Core/Core/Core.cpp b/Source/Core/Core/Core.cpp index f01b17beae..d4aae1f7ea 100644 --- a/Source/Core/Core/Core.cpp +++ b/Source/Core/Core/Core.cpp @@ -533,7 +533,9 @@ void EmuThread() if (core_parameter.bWii) { if (init_controllers) - Wiimote::Initialize(s_window_handle, !s_state_filename.empty()); + Wiimote::Initialize(s_window_handle, !s_state_filename.empty() ? + Wiimote::InitializeMode::DO_WAIT_FOR_WIIMOTES : + Wiimote::InitializeMode::DO_NOT_WAIT_FOR_WIIMOTES); else Wiimote::LoadConfig(); diff --git a/Source/Core/Core/HW/Wiimote.cpp b/Source/Core/Core/HW/Wiimote.cpp index 7b57eb4cbd..d10620c4fb 100644 --- a/Source/Core/Core/HW/Wiimote.cpp +++ b/Source/Core/Core/HW/Wiimote.cpp @@ -28,7 +28,7 @@ void Shutdown() g_controller_interface.Shutdown(); } -void Initialize(void* const hwnd, bool wait) +void Initialize(void* const hwnd, InitializeMode init_mode) { if (s_config.ControllersNeedToBeCreated()) { @@ -40,7 +40,7 @@ void Initialize(void* const hwnd, bool wait) s_config.LoadConfig(false); - WiimoteReal::Initialize(wait); + WiimoteReal::Initialize(init_mode); // Reload Wiimotes with our settings if (Movie::IsMovieActive()) diff --git a/Source/Core/Core/HW/Wiimote.h b/Source/Core/Core/HW/Wiimote.h index 96503f9905..1c77042342 100644 --- a/Source/Core/Core/HW/Wiimote.h +++ b/Source/Core/Core/HW/Wiimote.h @@ -35,8 +35,14 @@ extern unsigned int g_wiimote_sources[MAX_BBMOTES]; namespace Wiimote { +enum class InitializeMode +{ + DO_WAIT_FOR_WIIMOTES, + DO_NOT_WAIT_FOR_WIIMOTES, +}; + void Shutdown(); -void Initialize(void* const hwnd, bool wait = false); +void Initialize(void* const hwnd, InitializeMode init_mode); void ResetAllWiimotes(); void LoadConfig(); void Resume(); @@ -54,7 +60,7 @@ void Update(int _number, bool _connected); namespace WiimoteReal { -void Initialize(bool wait = false); +void Initialize(::Wiimote::InitializeMode init_mode); void Stop(); void Shutdown(); void Resume(); diff --git a/Source/Core/Core/HW/WiimoteReal/WiimoteReal.cpp b/Source/Core/Core/HW/WiimoteReal/WiimoteReal.cpp index f8343e5b43..ee81729042 100644 --- a/Source/Core/Core/HW/WiimoteReal/WiimoteReal.cpp +++ b/Source/Core/Core/HW/WiimoteReal/WiimoteReal.cpp @@ -430,34 +430,29 @@ static unsigned int CalculateWantedBB() return wanted_bb; } -void WiimoteScanner::WantWiimotes(bool do_want) +void WiimoteScanner::StartThread() { - m_want_wiimotes.store(do_want); + if (m_scan_thread_running.IsSet()) + return; + m_scan_thread_running.Set(); + m_scan_thread = std::thread(&WiimoteScanner::ThreadFunc, this); } -void WiimoteScanner::WantBB(bool do_want) +void WiimoteScanner::StopThread() { - m_want_bb.store(do_want); -} - -void WiimoteScanner::StartScanning() -{ - if (!m_run_thread.load()) - { - m_run_thread.store(true); - m_scan_thread = std::thread(&WiimoteScanner::ThreadFunc, this); - } -} - -void WiimoteScanner::StopScanning() -{ - m_run_thread.store(false); - if (m_scan_thread.joinable()) + if (m_scan_thread_running.TestAndClear()) { + SetScanMode(WiimoteScanMode::DO_NOT_SCAN); m_scan_thread.join(); } } +void WiimoteScanner::SetScanMode(WiimoteScanMode scan_mode) +{ + m_scan_mode.store(scan_mode); + m_scan_mode_changed_event.Set(); +} + static void CheckForDisconnectedWiimotes() { std::lock_guard lk(g_refresh_lock); @@ -471,41 +466,33 @@ void WiimoteScanner::ThreadFunc() { Common::SetCurrentThreadName("Wiimote Scanning Thread"); - NOTICE_LOG(WIIMOTE, "Wiimote scanning has started."); + NOTICE_LOG(WIIMOTE, "Wiimote scanning thread has started."); - while (m_run_thread.load()) + while (m_scan_thread_running.IsSet()) { - std::vector found_wiimotes; - Wiimote* found_board = nullptr; + m_scan_mode_changed_event.WaitFor(std::chrono::milliseconds(500)); - // NOTICE_LOG(WIIMOTE, "In loop"); - - if (m_want_wiimotes.load() || m_want_bb.load()) - { - FindWiimotes(found_wiimotes, found_board); - } - else - { - // Does stuff needed to detect disconnects on Windows - Update(); - } - - // NOTICE_LOG(WIIMOTE, "After update"); - - // TODO: this is a fairly lame place for this + Update(); // Does stuff needed to detect disconnects on Windows CheckForDisconnectedWiimotes(); - if (m_want_wiimotes.load()) + if (m_scan_mode.load() == WiimoteScanMode::DO_NOT_SCAN) + continue; + + if (CalculateWantedWiimotes() != 0 || CalculateWantedBB() != 0) + { + std::vector found_wiimotes; + Wiimote* found_board = nullptr; + FindWiimotes(found_wiimotes, found_board); HandleFoundWiimotes(found_wiimotes); + if (found_board) + TryToConnectBalanceBoard(found_board); + } - if (m_want_bb.load() && found_board) - TryToConnectBalanceBoard(found_board); - - // std::this_thread::yield(); - Common::SleepCurrentThread(500); + if (m_scan_mode.load() == WiimoteScanMode::SCAN_ONCE) + m_scan_mode.store(WiimoteScanMode::DO_NOT_SCAN); } - NOTICE_LOG(WIIMOTE, "Wiimote scanning has stopped."); + NOTICE_LOG(WIIMOTE, "Wiimote scanning thread has stopped."); } bool Wiimote::Connect(int index) @@ -621,33 +608,27 @@ void LoadSettings() } // config dialog calls this when some settings change -void Initialize(bool wait) +void Initialize(::Wiimote::InitializeMode init_mode) { + if (!g_real_wiimotes_initialized) + g_wiimote_scanner.StartThread(); + if (SConfig::GetInstance().m_WiimoteContinuousScanning) - g_wiimote_scanner.StartScanning(); + g_wiimote_scanner.SetScanMode(WiimoteScanMode::CONTINUOUSLY_SCAN); else - g_wiimote_scanner.StopScanning(); + g_wiimote_scanner.SetScanMode(WiimoteScanMode::DO_NOT_SCAN); std::lock_guard lk(g_refresh_lock); - g_wiimote_scanner.WantWiimotes(0 != CalculateWantedWiimotes()); - g_wiimote_scanner.WantBB(0 != CalculateWantedBB()); - // wait for connection because it should exist before state load - if (wait) + if (init_mode == ::Wiimote::InitializeMode::DO_WAIT_FOR_WIIMOTES) { int timeout = 100; - std::vector found_wiimotes; - Wiimote* found_board = nullptr; - g_wiimote_scanner.FindWiimotes(found_wiimotes, found_board); - if (SConfig::GetInstance().m_WiimoteContinuousScanning) + g_wiimote_scanner.SetScanMode(WiimoteScanMode::SCAN_ONCE); + while (CalculateWantedWiimotes() > CalculateConnectedWiimotes() && timeout) { - while (CalculateWantedWiimotes() && CalculateConnectedWiimotes() < found_wiimotes.size() && - timeout) - { - Common::SleepCurrentThread(100); - timeout--; - } + Common::SleepCurrentThread(100); + timeout--; } } @@ -670,17 +651,12 @@ void Stop() // called when the Dolphin app exits void Shutdown() { - g_wiimote_scanner.StopScanning(); + g_wiimote_scanner.StopThread(); std::lock_guard lk(g_refresh_lock); - if (!g_real_wiimotes_initialized) - return; - NOTICE_LOG(WIIMOTE, "WiimoteReal::Shutdown"); - g_real_wiimotes_initialized = false; - for (unsigned int i = 0; i < MAX_BBMOTES; ++i) HandleWiimoteDisconnect(i); } @@ -704,8 +680,6 @@ void ChangeWiimoteSource(unsigned int index, int source) { std::lock_guard lk(g_refresh_lock); g_wiimote_sources[index] = source; - g_wiimote_scanner.WantWiimotes(0 != CalculateWantedWiimotes()); - g_wiimote_scanner.WantBB(0 != CalculateWantedBB()); // kill real connection (or swap to different slot) DoneWithWiimote(index); } @@ -744,8 +718,6 @@ void TryToConnectWiimote(Wiimote* wm) } } - g_wiimote_scanner.WantWiimotes(0 != CalculateWantedWiimotes()); - lk.unlock(); delete wm; @@ -760,8 +732,6 @@ void TryToConnectBalanceBoard(Wiimote* wm) wm = nullptr; } - g_wiimote_scanner.WantBB(0 != CalculateWantedBB()); - lk.unlock(); delete wm; @@ -790,10 +760,7 @@ void HandleWiimoteDisconnect(int index) { std::lock_guard lk(g_refresh_lock); - std::swap(wm, g_wiimotes[index]); - g_wiimote_scanner.WantWiimotes(0 != CalculateWantedWiimotes()); - g_wiimote_scanner.WantBB(0 != CalculateWantedBB()); } if (wm) @@ -811,39 +778,8 @@ void HandleFoundWiimotes(const std::vector& wiimotes) // This is called from the GUI thread void Refresh() { - g_wiimote_scanner.StopScanning(); - - { - std::unique_lock lk(g_refresh_lock); - std::vector found_wiimotes; - Wiimote* found_board = nullptr; - - if (0 != CalculateWantedWiimotes() || 0 != CalculateWantedBB()) - { - // Don't hang Dolphin when searching - lk.unlock(); - g_wiimote_scanner.FindWiimotes(found_wiimotes, found_board); - lk.lock(); - } - - CheckForDisconnectedWiimotes(); - - // Brief rumble for already connected Wiimotes. - // Don't do this for Balance Board as it doesn't have rumble anyway. - for (int i = 0; i < MAX_WIIMOTES; ++i) - { - if (g_wiimotes[i]) - { - g_wiimotes[i]->Prepare(); - } - } - - HandleFoundWiimotes(found_wiimotes); - if (found_board) - TryToConnectBalanceBoard(found_board); - } - - Initialize(); + if (!SConfig::GetInstance().m_WiimoteContinuousScanning) + g_wiimote_scanner.SetScanMode(WiimoteScanMode::SCAN_ONCE); } void InterruptChannel(int _WiimoteNumber, u16 _channelID, const void* _pData, u32 _Size) diff --git a/Source/Core/Core/HW/WiimoteReal/WiimoteReal.h b/Source/Core/Core/HW/WiimoteReal/WiimoteReal.h index 4ac81de4c7..392745a5f8 100644 --- a/Source/Core/Core/HW/WiimoteReal/WiimoteReal.h +++ b/Source/Core/Core/HW/WiimoteReal/WiimoteReal.h @@ -12,7 +12,9 @@ #include #include "Common/Common.h" +#include "Common/Event.h" #include "Common/FifoQueue.h" +#include "Common/Flag.h" #include "Common/NonCopyable.h" #include "Core/HW/Wiimote.h" #include "Core/HW/WiimoteReal/WiimoteRealBase.h" @@ -112,6 +114,13 @@ private: Common::FifoQueue m_write_reports; }; +enum class WiimoteScanMode +{ + DO_NOT_SCAN, + CONTINUOUSLY_SCAN, + SCAN_ONCE +}; + class WiimoteScanner { public: @@ -120,11 +129,9 @@ public: bool IsReady() const; - void WantWiimotes(bool do_want); - void WantBB(bool do_want); - - void StartScanning(); - void StopScanning(); + void StartThread(); + void StopThread(); + void SetScanMode(WiimoteScanMode scan_mode); void FindWiimotes(std::vector&, Wiimote*&); @@ -136,9 +143,9 @@ private: std::thread m_scan_thread; - std::atomic m_run_thread{false}; - std::atomic m_want_wiimotes{false}; - std::atomic m_want_bb{false}; + Common::Event m_scan_mode_changed_event; + Common::Flag m_scan_thread_running; + std::atomic m_scan_mode{WiimoteScanMode::DO_NOT_SCAN}; #if defined(_WIN32) void CheckDeviceType(std::basic_string& devicepath, WinWriteMethod& write_method, diff --git a/Source/Core/DolphinWX/ControllerConfigDiag.h b/Source/Core/DolphinWX/ControllerConfigDiag.h index 98c7a1af47..e91b405330 100644 --- a/Source/Core/DolphinWX/ControllerConfigDiag.h +++ b/Source/Core/DolphinWX/ControllerConfigDiag.h @@ -59,7 +59,7 @@ private: void OnContinuousScanning(wxCommandEvent& event) { SConfig::GetInstance().m_WiimoteContinuousScanning = event.IsChecked(); - WiimoteReal::Initialize(); + WiimoteReal::Initialize(Wiimote::InitializeMode::DO_NOT_WAIT_FOR_WIIMOTES); event.Skip(); } diff --git a/Source/Core/DolphinWX/Frame.cpp b/Source/Core/DolphinWX/Frame.cpp index c785fe0dd2..608a6fe507 100644 --- a/Source/Core/DolphinWX/Frame.cpp +++ b/Source/Core/DolphinWX/Frame.cpp @@ -348,12 +348,14 @@ bool CFrame::InitControllers() Window win = X11Utils::XWindowFromHandle(GetHandle()); Pad::Initialize(reinterpret_cast(win)); Keyboard::Initialize(reinterpret_cast(win)); - Wiimote::Initialize(reinterpret_cast(win)); + Wiimote::Initialize(reinterpret_cast(win), + Wiimote::InitializeMode::DO_NOT_WAIT_FOR_WIIMOTES); HotkeyManagerEmu::Initialize(reinterpret_cast(win)); #else Pad::Initialize(reinterpret_cast(GetHandle())); Keyboard::Initialize(reinterpret_cast(GetHandle())); - Wiimote::Initialize(reinterpret_cast(GetHandle())); + Wiimote::Initialize(reinterpret_cast(GetHandle()), + Wiimote::InitializeMode::DO_NOT_WAIT_FOR_WIIMOTES); HotkeyManagerEmu::Initialize(reinterpret_cast(GetHandle())); #endif return true; From 80fc5e2814257676ad5551c7da4453cb51761aa6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?L=C3=A9o=20Lam?= Date: Sun, 10 Jul 2016 20:41:00 +0200 Subject: [PATCH 2/2] WiimoteReal: Don't use a recursive mutex This replaces a recursive mutex with a normal mutex. --- .../Core/HW/WiimoteEmu/EmuSubroutines.cpp | 2 +- Source/Core/Core/HW/WiimoteEmu/WiimoteEmu.cpp | 2 +- Source/Core/Core/HW/WiimoteReal/IOdarwin.mm | 4 +- .../Core/Core/HW/WiimoteReal/WiimoteReal.cpp | 121 +++++++----------- Source/Core/Core/HW/WiimoteReal/WiimoteReal.h | 2 +- 5 files changed, 51 insertions(+), 80 deletions(-) diff --git a/Source/Core/Core/HW/WiimoteEmu/EmuSubroutines.cpp b/Source/Core/Core/HW/WiimoteEmu/EmuSubroutines.cpp index ea6327b32b..da22b8e514 100644 --- a/Source/Core/Core/HW/WiimoteEmu/EmuSubroutines.cpp +++ b/Source/Core/Core/HW/WiimoteEmu/EmuSubroutines.cpp @@ -231,7 +231,7 @@ void Wiimote::RequestStatus(const wm_request_status* const rs) { using namespace WiimoteReal; - std::lock_guard lk(g_refresh_lock); + std::lock_guard lk(g_wiimotes_mutex); if (g_wiimotes[m_index]) { diff --git a/Source/Core/Core/HW/WiimoteEmu/WiimoteEmu.cpp b/Source/Core/Core/HW/WiimoteEmu/WiimoteEmu.cpp index 9adc810833..6cde18e55a 100644 --- a/Source/Core/Core/HW/WiimoteEmu/WiimoteEmu.cpp +++ b/Source/Core/Core/HW/WiimoteEmu/WiimoteEmu.cpp @@ -662,7 +662,7 @@ void Wiimote::Update() { using namespace WiimoteReal; - std::lock_guard lk(g_refresh_lock); + std::lock_guard lk(g_wiimotes_mutex); if (g_wiimotes[m_index]) { const Report& rpt = g_wiimotes[m_index]->ProcessReadQueue(); diff --git a/Source/Core/Core/HW/WiimoteReal/IOdarwin.mm b/Source/Core/Core/HW/WiimoteReal/IOdarwin.mm index 3471896951..afc066e326 100644 --- a/Source/Core/Core/HW/WiimoteReal/IOdarwin.mm +++ b/Source/Core/Core/HW/WiimoteReal/IOdarwin.mm @@ -500,7 +500,7 @@ void WiimoteDarwinHid::RemoveCallback(void* context, IOReturn result, void*) IOBluetoothDevice* device = [l2capChannel device]; WiimoteReal::WiimoteDarwin* wm = nullptr; - std::lock_guard lk(WiimoteReal::g_refresh_lock); + std::lock_guard lk(WiimoteReal::g_wiimotes_mutex); for (int i = 0; i < MAX_WIIMOTES; i++) { @@ -541,7 +541,7 @@ void WiimoteDarwinHid::RemoveCallback(void* context, IOReturn result, void*) IOBluetoothDevice* device = [l2capChannel device]; WiimoteReal::WiimoteDarwin* wm = nullptr; - std::lock_guard lk(WiimoteReal::g_refresh_lock); + std::lock_guard lk(WiimoteReal::g_wiimotes_mutex); for (int i = 0; i < MAX_WIIMOTES; i++) { diff --git a/Source/Core/Core/HW/WiimoteReal/WiimoteReal.cpp b/Source/Core/Core/HW/WiimoteReal/WiimoteReal.cpp index ee81729042..8994695fd3 100644 --- a/Source/Core/Core/HW/WiimoteReal/WiimoteReal.cpp +++ b/Source/Core/Core/HW/WiimoteReal/WiimoteReal.cpp @@ -25,15 +25,13 @@ unsigned int g_wiimote_sources[MAX_BBMOTES]; namespace WiimoteReal { -void HandleFoundWiimotes(const std::vector&); void TryToConnectBalanceBoard(Wiimote*); void TryToConnectWiimote(Wiimote*); void HandleWiimoteDisconnect(int index); -void DoneWithWiimote(int index); static bool g_real_wiimotes_initialized = false; -std::recursive_mutex g_refresh_lock; +std::mutex g_wiimotes_mutex; Wiimote* g_wiimotes[MAX_BBMOTES]; WiimoteScanner g_wiimote_scanner; @@ -402,6 +400,7 @@ void Wiimote::EmuPause() static unsigned int CalculateConnectedWiimotes() { + std::lock_guard lk(g_wiimotes_mutex); unsigned int connected_wiimotes = 0; for (unsigned int i = 0; i < MAX_WIIMOTES; ++i) if (g_wiimotes[i]) @@ -412,6 +411,7 @@ static unsigned int CalculateConnectedWiimotes() static unsigned int CalculateWantedWiimotes() { + std::lock_guard lk(g_wiimotes_mutex); // Figure out how many real Wiimotes are required unsigned int wanted_wiimotes = 0; for (unsigned int i = 0; i < MAX_WIIMOTES; ++i) @@ -423,6 +423,7 @@ static unsigned int CalculateWantedWiimotes() static unsigned int CalculateWantedBB() { + std::lock_guard lk(g_wiimotes_mutex); unsigned int wanted_bb = 0; if (WIIMOTE_SRC_REAL & g_wiimote_sources[WIIMOTE_BALANCE_BOARD] && !g_wiimotes[WIIMOTE_BALANCE_BOARD]) @@ -455,8 +456,7 @@ void WiimoteScanner::SetScanMode(WiimoteScanMode scan_mode) static void CheckForDisconnectedWiimotes() { - std::lock_guard lk(g_refresh_lock); - + std::lock_guard lk(g_wiimotes_mutex); for (unsigned int i = 0; i < MAX_BBMOTES; ++i) if (g_wiimotes[i] && !g_wiimotes[i]->IsConnected()) HandleWiimoteDisconnect(i); @@ -483,9 +483,12 @@ void WiimoteScanner::ThreadFunc() std::vector found_wiimotes; Wiimote* found_board = nullptr; FindWiimotes(found_wiimotes, found_board); - HandleFoundWiimotes(found_wiimotes); - if (found_board) - TryToConnectBalanceBoard(found_board); + { + std::lock_guard lk(g_wiimotes_mutex); + std::for_each(found_wiimotes.begin(), found_wiimotes.end(), TryToConnectWiimote); + if (found_board) + TryToConnectBalanceBoard(found_board); + } } if (m_scan_mode.load() == WiimoteScanMode::SCAN_ONCE) @@ -618,8 +621,6 @@ void Initialize(::Wiimote::InitializeMode init_mode) else g_wiimote_scanner.SetScanMode(WiimoteScanMode::DO_NOT_SCAN); - std::lock_guard lk(g_refresh_lock); - // wait for connection because it should exist before state load if (init_mode == ::Wiimote::InitializeMode::DO_WAIT_FOR_WIIMOTES) { @@ -653,10 +654,9 @@ void Shutdown() { g_wiimote_scanner.StopThread(); - std::lock_guard lk(g_refresh_lock); - NOTICE_LOG(WIIMOTE, "WiimoteReal::Shutdown"); + std::lock_guard lk(g_wiimotes_mutex); for (unsigned int i = 0; i < MAX_BBMOTES; ++i) HandleWiimoteDisconnect(i); } @@ -677,11 +677,22 @@ void Pause() void ChangeWiimoteSource(unsigned int index, int source) { + g_wiimote_sources[index] = source; { - std::lock_guard lk(g_refresh_lock); - g_wiimote_sources[index] = source; // kill real connection (or swap to different slot) - DoneWithWiimote(index); + std::lock_guard lk(g_wiimotes_mutex); + + Wiimote* wm = g_wiimotes[index]; + + if (wm) + { + g_wiimotes[index] = nullptr; + // First see if we can use this real Wiimote in another slot. + TryToConnectWiimote(wm); + } + + // else, just disconnect the Wiimote + HandleWiimoteDisconnect(index); } // reconnect to the emulator @@ -690,7 +701,7 @@ void ChangeWiimoteSource(unsigned int index, int source) Host_ConnectWiimote(index, true); } -static bool TryToConnectWiimoteN(Wiimote* wm, unsigned int i) +static bool TryToConnectWiimoteToSlot(Wiimote* wm, unsigned int i) { if (WIIMOTE_SRC_REAL & g_wiimote_sources[i] && !g_wiimotes[i]) { @@ -707,62 +718,30 @@ static bool TryToConnectWiimoteN(Wiimote* wm, unsigned int i) void TryToConnectWiimote(Wiimote* wm) { - std::unique_lock lk(g_refresh_lock); - for (unsigned int i = 0; i < MAX_WIIMOTES; ++i) { - if (TryToConnectWiimoteN(wm, i)) + if (TryToConnectWiimoteToSlot(wm, i)) { wm = nullptr; break; } } - - lk.unlock(); - delete wm; } void TryToConnectBalanceBoard(Wiimote* wm) { - std::unique_lock lk(g_refresh_lock); - - if (TryToConnectWiimoteN(wm, WIIMOTE_BALANCE_BOARD)) + if (TryToConnectWiimoteToSlot(wm, WIIMOTE_BALANCE_BOARD)) { wm = nullptr; } - - lk.unlock(); - delete wm; } -void DoneWithWiimote(int index) -{ - std::lock_guard lk(g_refresh_lock); - - Wiimote* wm = g_wiimotes[index]; - - if (wm) - { - g_wiimotes[index] = nullptr; - // First see if we can use this real Wiimote in another slot. - TryToConnectWiimote(wm); - } - - // else, just disconnect the Wiimote - HandleWiimoteDisconnect(index); -} - void HandleWiimoteDisconnect(int index) { Wiimote* wm = nullptr; - - { - std::lock_guard lk(g_refresh_lock); - std::swap(wm, g_wiimotes[index]); - } - + std::swap(wm, g_wiimotes[index]); if (wm) { delete wm; @@ -770,11 +749,6 @@ void HandleWiimoteDisconnect(int index) } } -void HandleFoundWiimotes(const std::vector& wiimotes) -{ - std::for_each(wiimotes.begin(), wiimotes.end(), TryToConnectWiimote); -} - // This is called from the GUI thread void Refresh() { @@ -784,54 +758,51 @@ void Refresh() void InterruptChannel(int _WiimoteNumber, u16 _channelID, const void* _pData, u32 _Size) { - std::lock_guard lk(g_refresh_lock); + std::lock_guard lk(g_wiimotes_mutex); if (g_wiimotes[_WiimoteNumber]) g_wiimotes[_WiimoteNumber]->InterruptChannel(_channelID, _pData, _Size); } void ControlChannel(int _WiimoteNumber, u16 _channelID, const void* _pData, u32 _Size) { - std::lock_guard lk(g_refresh_lock); - + std::lock_guard lk(g_wiimotes_mutex); if (g_wiimotes[_WiimoteNumber]) g_wiimotes[_WiimoteNumber]->ControlChannel(_channelID, _pData, _Size); } // Read the Wiimote once -void Update(int _WiimoteNumber) +void Update(int wiimote_number) { // Try to get a lock and return without doing anything if we fail - // This avoids deadlocks when adding a Wiimote during continuous scan - if (!g_refresh_lock.try_lock()) + // This avoids blocking the CPU thread + if (!g_wiimotes_mutex.try_lock()) return; - if (g_wiimotes[_WiimoteNumber]) - g_wiimotes[_WiimoteNumber]->Update(); + if (g_wiimotes[wiimote_number]) + g_wiimotes[wiimote_number]->Update(); // Wiimote::Update() may remove the Wiimote if it was disconnected. - if (!g_wiimotes[_WiimoteNumber]) + if (!g_wiimotes[wiimote_number]) { - Host_ConnectWiimote(_WiimoteNumber, false); + Host_ConnectWiimote(wiimote_number, false); } - g_refresh_lock.unlock(); + + g_wiimotes_mutex.unlock(); } -void ConnectOnInput(int _WiimoteNumber) +void ConnectOnInput(int wiimote_number) { - // see Update() above - if (!g_refresh_lock.try_lock()) + if (!g_wiimotes_mutex.try_lock()) return; - if (g_wiimotes[_WiimoteNumber]) - g_wiimotes[_WiimoteNumber]->ConnectOnInput(); + if (g_wiimotes[wiimote_number]) + g_wiimotes[wiimote_number]->ConnectOnInput(); - g_refresh_lock.unlock(); + g_wiimotes_mutex.unlock(); } void StateChange(EMUSTATE_CHANGE newState) { - // std::lock_guard lk(g_refresh_lock); - // TODO: disable/enable auto reporting, maybe } diff --git a/Source/Core/Core/HW/WiimoteReal/WiimoteReal.h b/Source/Core/Core/HW/WiimoteReal/WiimoteReal.h index 392745a5f8..e4e823bece 100644 --- a/Source/Core/Core/HW/WiimoteReal/WiimoteReal.h +++ b/Source/Core/Core/HW/WiimoteReal/WiimoteReal.h @@ -156,7 +156,7 @@ private: #endif }; -extern std::recursive_mutex g_refresh_lock; +extern std::mutex g_wiimotes_mutex; extern WiimoteScanner g_wiimote_scanner; extern Wiimote* g_wiimotes[MAX_BBMOTES];