diff --git a/Source/Core/Core/Core.cpp b/Source/Core/Core/Core.cpp index 791656f89e..8916804cab 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/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 f8343e5b43..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]) @@ -430,38 +431,32 @@ 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); - + 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); @@ -471,41 +466,36 @@ 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()) - HandleFoundWiimotes(found_wiimotes); + if (m_scan_mode.load() == WiimoteScanMode::DO_NOT_SCAN) + continue; - if (m_want_bb.load() && found_board) - TryToConnectBalanceBoard(found_board); + if (CalculateWantedWiimotes() != 0 || CalculateWantedBB() != 0) + { + std::vector found_wiimotes; + Wiimote* found_board = nullptr; + FindWiimotes(found_wiimotes, 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); + } + } - // 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 +611,25 @@ 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(); - - std::lock_guard lk(g_refresh_lock); - - g_wiimote_scanner.WantWiimotes(0 != CalculateWantedWiimotes()); - g_wiimote_scanner.WantBB(0 != CalculateWantedBB()); + g_wiimote_scanner.SetScanMode(WiimoteScanMode::DO_NOT_SCAN); // 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 +652,11 @@ void Stop() // called when the Dolphin app exits void Shutdown() { - g_wiimote_scanner.StopScanning(); - - std::lock_guard lk(g_refresh_lock); - - if (!g_real_wiimotes_initialized) - return; + g_wiimote_scanner.StopThread(); NOTICE_LOG(WIIMOTE, "WiimoteReal::Shutdown"); - g_real_wiimotes_initialized = false; - + std::lock_guard lk(g_wiimotes_mutex); for (unsigned int i = 0; i < MAX_BBMOTES; ++i) HandleWiimoteDisconnect(i); } @@ -701,13 +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; - g_wiimote_scanner.WantWiimotes(0 != CalculateWantedWiimotes()); - g_wiimote_scanner.WantBB(0 != CalculateWantedBB()); // 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 @@ -716,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]) { @@ -733,69 +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; } } - - g_wiimote_scanner.WantWiimotes(0 != CalculateWantedWiimotes()); - - 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; } - - g_wiimote_scanner.WantBB(0 != CalculateWantedBB()); - - 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]); - g_wiimote_scanner.WantWiimotes(0 != CalculateWantedWiimotes()); - g_wiimote_scanner.WantBB(0 != CalculateWantedBB()); - } - + std::swap(wm, g_wiimotes[index]); if (wm) { delete wm; @@ -803,99 +749,60 @@ 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() { - 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) { - 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 4ac81de4c7..e4e823bece 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, @@ -149,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]; 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;