From 4c5e283e7519690724d81f851a9dd23f39a95721 Mon Sep 17 00:00:00 2001 From: Michael Maltese Date: Sat, 15 Apr 2017 19:20:51 -0700 Subject: [PATCH] WiimoteReal: init and destroy ScannerBackends in same thread This fixes an error condition on macOS when HIDAPI calls IOHIDManagerCreate and IOHIDManagerClose on different threads. The error behavior is non-deterministic, but can cause EXC_BAD_ACCES and kill the program. --- .../Core/Core/HW/WiimoteReal/WiimoteReal.cpp | 33 ++++++++++++------- Source/Core/Core/HW/WiimoteReal/WiimoteReal.h | 8 ++--- 2 files changed, 25 insertions(+), 16 deletions(-) diff --git a/Source/Core/Core/HW/WiimoteReal/WiimoteReal.cpp b/Source/Core/Core/HW/WiimoteReal/WiimoteReal.cpp index e896b6f40d..4b0a8bc6de 100644 --- a/Source/Core/Core/HW/WiimoteReal/WiimoteReal.cpp +++ b/Source/Core/Core/HW/WiimoteReal/WiimoteReal.cpp @@ -535,15 +535,11 @@ void WiimoteScanner::SetScanMode(WiimoteScanMode scan_mode) bool WiimoteScanner::IsReady() const { - return std::any_of(m_scanner_backends.begin(), m_scanner_backends.end(), + std::lock_guard lg(m_backends_mutex); + return std::any_of(m_backends.begin(), m_backends.end(), [](const auto& backend) { return backend->IsReady(); }); } -void WiimoteScanner::AddScannerBackend(std::unique_ptr backend) -{ - m_scanner_backends.emplace_back(std::move(backend)); -} - static void CheckForDisconnectedWiimotes() { std::lock_guard lk(g_wiimotes_mutex); @@ -558,6 +554,20 @@ void WiimoteScanner::ThreadFunc() NOTICE_LOG(WIIMOTE, "Wiimote scanning thread has started."); + // Create and destroy scanner backends here to ensure all operations stay on the same thread. The + // HIDAPI backend on macOS has an error condition when IOHIDManagerCreate and IOHIDManagerClose + // are called on different threads (and so reference different CFRunLoops) which can cause an + // EXC_BAD_ACCES crash. + { + std::lock_guard lg(m_backends_mutex); + + m_backends.emplace_back(std::make_unique()); + m_backends.emplace_back(std::make_unique()); + m_backends.emplace_back(std::make_unique()); + m_backends.emplace_back(std::make_unique()); + m_backends.emplace_back(std::make_unique()); + } + while (m_scan_thread_running.IsSet()) { m_scan_mode_changed_event.WaitFor(std::chrono::milliseconds(500)); @@ -567,7 +577,7 @@ void WiimoteScanner::ThreadFunc() if (m_scan_mode.load() == WiimoteScanMode::DO_NOT_SCAN) continue; - for (const auto& backend : m_scanner_backends) + for (const auto& backend : m_backends) { if (CalculateWantedWiimotes() != 0 || CalculateWantedBB() != 0) { @@ -593,6 +603,10 @@ void WiimoteScanner::ThreadFunc() m_scan_mode.store(WiimoteScanMode::DO_NOT_SCAN); } + { + std::lock_guard lg(m_backends_mutex); + m_backends.clear(); + } NOTICE_LOG(WIIMOTE, "Wiimote scanning thread has stopped."); } @@ -695,11 +709,6 @@ void Initialize(::Wiimote::InitializeMode init_mode) if (!g_real_wiimotes_initialized) { s_known_ids.clear(); - g_wiimote_scanner.AddScannerBackend(std::make_unique()); - g_wiimote_scanner.AddScannerBackend(std::make_unique()); - g_wiimote_scanner.AddScannerBackend(std::make_unique()); - g_wiimote_scanner.AddScannerBackend(std::make_unique()); - g_wiimote_scanner.AddScannerBackend(std::make_unique()); g_wiimote_scanner.StartThread(); } diff --git a/Source/Core/Core/HW/WiimoteReal/WiimoteReal.h b/Source/Core/Core/HW/WiimoteReal/WiimoteReal.h index bf5d978a11..431a9c1d1e 100644 --- a/Source/Core/Core/HW/WiimoteReal/WiimoteReal.h +++ b/Source/Core/Core/HW/WiimoteReal/WiimoteReal.h @@ -138,18 +138,18 @@ public: void StopThread(); void SetScanMode(WiimoteScanMode scan_mode); - void AddScannerBackend(std::unique_ptr backend); bool IsReady() const; private: void ThreadFunc(); + + std::vector> m_backends; + mutable std::mutex m_backends_mutex; + std::thread m_scan_thread; Common::Flag m_scan_thread_running; - Common::Event m_scan_mode_changed_event; std::atomic m_scan_mode{WiimoteScanMode::DO_NOT_SCAN}; - - std::vector> m_scanner_backends; }; extern std::mutex g_wiimotes_mutex;