ControllerInterface: invoke callbacks in AddDevice/RemoveDevice

Some backends already cause this to happen, so make it consistent across
systems.
This commit is contained in:
Michael M 2017-11-09 12:14:21 -08:00
parent 126b7ea01c
commit 7355b5f70d
6 changed files with 47 additions and 30 deletions

View File

@ -2,9 +2,11 @@
// Licensed under GPLv2+ // Licensed under GPLv2+
// Refer to the license.txt file included. // Refer to the license.txt file included.
#include "InputCommon/ControllerInterface/ControllerInterface.h"
#include <mutex> #include <mutex>
#include "InputCommon/ControllerInterface/ControllerInterface.h" #include "Common/Logging/Log.h"
#ifdef CIFACE_USE_XINPUT #ifdef CIFACE_USE_XINPUT
#include "InputCommon/ControllerInterface/XInput/XInput.h" #include "InputCommon/ControllerInterface/XInput/XInput.h"
@ -165,30 +167,45 @@ void ControllerInterface::Shutdown()
void ControllerInterface::AddDevice(std::shared_ptr<ciface::Core::Device> device) void ControllerInterface::AddDevice(std::shared_ptr<ciface::Core::Device> device)
{ {
std::lock_guard<std::mutex> lk(m_devices_mutex);
// Try to find an ID for this device
int id = 0;
while (true)
{ {
const auto it = std::find_if(m_devices.begin(), m_devices.end(), [&device, &id](const auto& d) { std::lock_guard<std::mutex> lk(m_devices_mutex);
return d->GetSource() == device->GetSource() && d->GetName() == device->GetName() && // Try to find an ID for this device
d->GetId() == id; int id = 0;
}); while (true)
if (it == m_devices.end()) // no device with the same name with this ID, so we can use it {
break; const auto it =
else std::find_if(m_devices.begin(), m_devices.end(), [&device, &id](const auto& d) {
id++; return d->GetSource() == device->GetSource() && d->GetName() == device->GetName() &&
d->GetId() == id;
});
if (it == m_devices.end()) // no device with the same name with this ID, so we can use it
break;
else
id++;
}
device->SetId(id);
NOTICE_LOG(SERIALINTERFACE, "Added device: %s", device->GetQualifiedName().c_str());
m_devices.emplace_back(std::move(device));
} }
device->SetId(id); InvokeHotplugCallbacks();
m_devices.emplace_back(std::move(device));
} }
void ControllerInterface::RemoveDevice(std::function<bool(const ciface::Core::Device*)> callback) void ControllerInterface::RemoveDevice(std::function<bool(const ciface::Core::Device*)> callback)
{ {
std::lock_guard<std::mutex> lk(m_devices_mutex); {
m_devices.erase(std::remove_if(m_devices.begin(), m_devices.end(), std::lock_guard<std::mutex> lk(m_devices_mutex);
[&callback](const auto& dev) { return callback(dev.get()); }), auto it = std::remove_if(m_devices.begin(), m_devices.end(), [&callback](const auto& dev) {
m_devices.end()); if (callback(dev.get()))
{
NOTICE_LOG(SERIALINTERFACE, "Removed device: %s", dev->GetQualifiedName().c_str());
return true;
}
return false;
});
m_devices.erase(it, m_devices.end());
}
InvokeHotplugCallbacks();
} }
// //

View File

@ -54,9 +54,10 @@ public:
void UpdateInput(); void UpdateInput();
void RegisterHotplugCallback(std::function<void(void)> callback); void RegisterHotplugCallback(std::function<void(void)> callback);
void InvokeHotplugCallbacks() const;
private: private:
void InvokeHotplugCallbacks() const;
std::vector<std::function<void()>> m_hotplug_callbacks; std::vector<std::function<void()>> m_hotplug_callbacks;
bool m_is_init; bool m_is_init;
void* m_hwnd; void* m_hwnd;

View File

@ -2,12 +2,14 @@
// Licensed under GPLv2+ // Licensed under GPLv2+
// Refer to the license.txt file included. // Refer to the license.txt file included.
#include "InputCommon/ControllerInterface/Device.h"
#include <memory> #include <memory>
#include <sstream> #include <sstream>
#include <string> #include <string>
#include <tuple> #include <tuple>
#include "InputCommon/ControllerInterface/Device.h" #include "Common/StringUtil.h"
namespace ciface namespace ciface
{ {
@ -39,6 +41,11 @@ void Device::AddOutput(Device::Output* const o)
m_outputs.push_back(o); m_outputs.push_back(o);
} }
std::string Device::GetQualifiedName() const
{
return StringFromFormat("%s/%i/%s", this->GetSource().c_str(), GetId(), this->GetName().c_str());
}
Device::Input* Device::FindInput(const std::string& name) const Device::Input* Device::FindInput(const std::string& name) const
{ {
for (Input* input : m_inputs) for (Input* input : m_inputs)

View File

@ -79,6 +79,7 @@ public:
void SetId(int id) { m_id = id; } void SetId(int id) { m_id = id; }
virtual std::string GetName() const = 0; virtual std::string GetName() const = 0;
virtual std::string GetSource() const = 0; virtual std::string GetSource() const = 0;
std::string GetQualifiedName() const;
virtual void UpdateInput() {} virtual void UpdateInput() {}
virtual bool IsValid() const { return true; } virtual bool IsValid() const { return true; }
const std::vector<Input*>& Inputs() const { return m_inputs; } const std::vector<Input*>& Inputs() const { return m_inputs; }

View File

@ -156,8 +156,6 @@ static void DeviceRemovalCallback(void* inContext, IOReturn inResult, void* inSe
return false; return false;
}); });
g_controller_interface.InvokeHotplugCallbacks();
NOTICE_LOG(SERIALINTERFACE, "Removed device: %s", GetDeviceRefName(inIOHIDDeviceRef).c_str());
} }
static void DeviceMatchingCallback(void* inContext, IOReturn inResult, void* inSender, static void DeviceMatchingCallback(void* inContext, IOReturn inResult, void* inSender,
@ -174,9 +172,6 @@ static void DeviceMatchingCallback(void* inContext, IOReturn inResult, void* inS
{ {
g_controller_interface.AddDevice(std::make_shared<Joystick>(inIOHIDDeviceRef, name)); g_controller_interface.AddDevice(std::make_shared<Joystick>(inIOHIDDeviceRef, name));
} }
NOTICE_LOG(SERIALINTERFACE, "Added device: %s", name.c_str());
g_controller_interface.InvokeHotplugCallbacks();
} }
void Init(void* window) void Init(void* window)

View File

@ -92,9 +92,7 @@ static void HotplugThreadFunc()
g_controller_interface.RemoveDevice([&name](const auto& device) { g_controller_interface.RemoveDevice([&name](const auto& device) {
return device->GetSource() == "evdev" && device->GetName() == name && !device->IsValid(); return device->GetSource() == "evdev" && device->GetName() == name && !device->IsValid();
}); });
NOTICE_LOG(SERIALINTERFACE, "Removed device: %s", name.c_str());
s_devnode_name_map.erase(devnode); s_devnode_name_map.erase(devnode);
g_controller_interface.InvokeHotplugCallbacks();
} }
// Only react to "device added" events for evdev devices that we can access. // Only react to "device added" events for evdev devices that we can access.
else if (strcmp(action, "add") == 0 && access(devnode, W_OK) == 0) else if (strcmp(action, "add") == 0 && access(devnode, W_OK) == 0)
@ -107,8 +105,6 @@ static void HotplugThreadFunc()
{ {
g_controller_interface.AddDevice(std::move(device)); g_controller_interface.AddDevice(std::move(device));
s_devnode_name_map.insert(std::pair<std::string, std::string>(devnode, name)); s_devnode_name_map.insert(std::pair<std::string, std::string>(devnode, name));
NOTICE_LOG(SERIALINTERFACE, "Added new device: %s", name.c_str());
g_controller_interface.InvokeHotplugCallbacks();
} }
} }
udev_device_unref(dev); udev_device_unref(dev);