mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-24 15:01:16 +01:00
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.
This commit is contained in:
parent
8a559f2e58
commit
4c5e283e75
@ -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<std::mutex> 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<WiimoteScannerBackend> backend)
|
||||
{
|
||||
m_scanner_backends.emplace_back(std::move(backend));
|
||||
}
|
||||
|
||||
static void CheckForDisconnectedWiimotes()
|
||||
{
|
||||
std::lock_guard<std::mutex> 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<std::mutex> lg(m_backends_mutex);
|
||||
|
||||
m_backends.emplace_back(std::make_unique<WiimoteScannerLinux>());
|
||||
m_backends.emplace_back(std::make_unique<WiimoteScannerAndroid>());
|
||||
m_backends.emplace_back(std::make_unique<WiimoteScannerWindows>());
|
||||
m_backends.emplace_back(std::make_unique<WiimoteScannerDarwin>());
|
||||
m_backends.emplace_back(std::make_unique<WiimoteScannerHidapi>());
|
||||
}
|
||||
|
||||
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<std::mutex> 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<WiimoteScannerLinux>());
|
||||
g_wiimote_scanner.AddScannerBackend(std::make_unique<WiimoteScannerAndroid>());
|
||||
g_wiimote_scanner.AddScannerBackend(std::make_unique<WiimoteScannerWindows>());
|
||||
g_wiimote_scanner.AddScannerBackend(std::make_unique<WiimoteScannerDarwin>());
|
||||
g_wiimote_scanner.AddScannerBackend(std::make_unique<WiimoteScannerHidapi>());
|
||||
g_wiimote_scanner.StartThread();
|
||||
}
|
||||
|
||||
|
@ -138,18 +138,18 @@ public:
|
||||
void StopThread();
|
||||
void SetScanMode(WiimoteScanMode scan_mode);
|
||||
|
||||
void AddScannerBackend(std::unique_ptr<WiimoteScannerBackend> backend);
|
||||
bool IsReady() const;
|
||||
|
||||
private:
|
||||
void ThreadFunc();
|
||||
|
||||
std::vector<std::unique_ptr<WiimoteScannerBackend>> 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<WiimoteScanMode> m_scan_mode{WiimoteScanMode::DO_NOT_SCAN};
|
||||
|
||||
std::vector<std::unique_ptr<WiimoteScannerBackend>> m_scanner_backends;
|
||||
};
|
||||
|
||||
extern std::mutex g_wiimotes_mutex;
|
||||
|
Loading…
x
Reference in New Issue
Block a user