mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-11 08:39:13 +01:00
c238e49119
also make it more thread safe (avoid rare deadlock) and fix it trying to add devices before the CI has init
223 lines
7.1 KiB
Plaintext
223 lines
7.1 KiB
Plaintext
// Copyright 2013 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#include <thread>
|
|
|
|
#include <Cocoa/Cocoa.h>
|
|
#include <Foundation/Foundation.h>
|
|
#include <IOKit/hid/IOHIDLib.h>
|
|
|
|
#include "Common/Logging/Log.h"
|
|
#include "Common/StringUtil.h"
|
|
#include "Common/Thread.h"
|
|
|
|
#include "InputCommon/ControllerInterface/ControllerInterface.h"
|
|
#include "InputCommon/ControllerInterface/OSX/OSX.h"
|
|
#include "InputCommon/ControllerInterface/OSX/OSXJoystick.h"
|
|
#include "InputCommon/ControllerInterface/OSX/RunLoopStopper.h"
|
|
|
|
namespace ciface::OSX
|
|
{
|
|
constexpr CFTimeInterval FOREVER = 1e20;
|
|
static std::thread s_hotplug_thread;
|
|
static RunLoopStopper s_stopper;
|
|
static IOHIDManagerRef HIDManager = nullptr;
|
|
static CFStringRef OurRunLoop = CFSTR("DolphinOSXInput");
|
|
|
|
void DeviceElementDebugPrint(const void* value, void* context)
|
|
{
|
|
IOHIDElementRef e = (IOHIDElementRef)value;
|
|
bool recurse = false;
|
|
if (context)
|
|
recurse = *(bool*)context;
|
|
|
|
std::string type = "";
|
|
switch (IOHIDElementGetType(e))
|
|
{
|
|
case kIOHIDElementTypeInput_Axis:
|
|
type = "axis";
|
|
break;
|
|
case kIOHIDElementTypeInput_Button:
|
|
type = "button";
|
|
break;
|
|
case kIOHIDElementTypeInput_Misc:
|
|
type = "misc";
|
|
break;
|
|
case kIOHIDElementTypeInput_ScanCodes:
|
|
type = "scancodes";
|
|
break;
|
|
case kIOHIDElementTypeOutput:
|
|
type = "output";
|
|
break;
|
|
case kIOHIDElementTypeFeature:
|
|
type = "feature";
|
|
break;
|
|
case kIOHIDElementTypeCollection:
|
|
type = "collection";
|
|
break;
|
|
}
|
|
|
|
std::string c_type = "";
|
|
if (type == "collection")
|
|
{
|
|
switch (IOHIDElementGetCollectionType(e))
|
|
{
|
|
case kIOHIDElementCollectionTypePhysical:
|
|
c_type = "physical";
|
|
break;
|
|
case kIOHIDElementCollectionTypeApplication:
|
|
c_type = "application";
|
|
break;
|
|
case kIOHIDElementCollectionTypeLogical:
|
|
c_type = "logical";
|
|
break;
|
|
case kIOHIDElementCollectionTypeReport:
|
|
c_type = "report";
|
|
break;
|
|
case kIOHIDElementCollectionTypeNamedArray:
|
|
c_type = "namedArray";
|
|
break;
|
|
case kIOHIDElementCollectionTypeUsageSwitch:
|
|
c_type = "usageSwitch";
|
|
break;
|
|
case kIOHIDElementCollectionTypeUsageModifier:
|
|
c_type = "usageModifier";
|
|
break;
|
|
}
|
|
}
|
|
|
|
c_type.append(" ");
|
|
NSLog(@"%s%s%spage: 0x%x usage: 0x%x name: %@ "
|
|
"lmin: %ld lmax: %ld pmin: %ld pmax: %ld",
|
|
type.c_str(), type == "collection" ? ":" : "", type == "collection" ? c_type.c_str() : " ",
|
|
IOHIDElementGetUsagePage(e), IOHIDElementGetUsage(e),
|
|
IOHIDElementGetName(e), // usually just nullptr
|
|
IOHIDElementGetLogicalMin(e), IOHIDElementGetLogicalMax(e), IOHIDElementGetPhysicalMin(e),
|
|
IOHIDElementGetPhysicalMax(e));
|
|
|
|
if ((type == "collection") && recurse)
|
|
{
|
|
CFArrayRef elements = IOHIDElementGetChildren(e);
|
|
CFRange range = {0, CFArrayGetCount(elements)};
|
|
// this leaks...but it's just debug code, right? :D
|
|
CFArrayApplyFunction(elements, range, DeviceElementDebugPrint, nullptr);
|
|
}
|
|
}
|
|
|
|
static void DeviceDebugPrint(IOHIDDeviceRef device)
|
|
{
|
|
#if 0
|
|
#define shortlog(x) NSLog(@"%s: %@", x, IOHIDDeviceGetProperty(device, CFSTR(x)));
|
|
NSLog(@"-------------------------");
|
|
NSLog(@"Got Device: %@",
|
|
IOHIDDeviceGetProperty(device, CFSTR(kIOHIDProductKey)));
|
|
shortlog(kIOHIDTransportKey)
|
|
shortlog(kIOHIDVendorIDKey)
|
|
shortlog(kIOHIDVendorIDSourceKey)
|
|
shortlog(kIOHIDProductIDKey)
|
|
shortlog(kIOHIDVersionNumberKey)
|
|
shortlog(kIOHIDManufacturerKey)
|
|
shortlog(kIOHIDProductKey)
|
|
shortlog(kIOHIDSerialNumberKey)
|
|
shortlog(kIOHIDCountryCodeKey)
|
|
shortlog(kIOHIDLocationIDKey)
|
|
shortlog(kIOHIDDeviceUsageKey)
|
|
shortlog(kIOHIDDeviceUsagePageKey)
|
|
shortlog(kIOHIDDeviceUsagePairsKey)
|
|
shortlog(kIOHIDPrimaryUsageKey)
|
|
shortlog(kIOHIDPrimaryUsagePageKey)
|
|
shortlog(kIOHIDMaxInputReportSizeKey)
|
|
shortlog(kIOHIDMaxOutputReportSizeKey)
|
|
shortlog(kIOHIDMaxFeatureReportSizeKey)
|
|
shortlog(kIOHIDReportIntervalKey)
|
|
shortlog(kIOHIDReportDescriptorKey)
|
|
#endif
|
|
}
|
|
|
|
static std::string GetDeviceRefName(IOHIDDeviceRef inIOHIDDeviceRef)
|
|
{
|
|
const NSString* name = reinterpret_cast<const NSString*>(
|
|
IOHIDDeviceGetProperty(inIOHIDDeviceRef, CFSTR(kIOHIDProductKey)));
|
|
return (name != nullptr) ? std::string(StripSpaces([name UTF8String])) : "Unknown device";
|
|
}
|
|
|
|
static void DeviceRemovalCallback(void* inContext, IOReturn inResult, void* inSender,
|
|
IOHIDDeviceRef inIOHIDDeviceRef)
|
|
{
|
|
g_controller_interface.RemoveDevice([&inIOHIDDeviceRef](const auto* device) {
|
|
const Joystick* joystick = dynamic_cast<const Joystick*>(device);
|
|
if (joystick && joystick->IsSameDevice(inIOHIDDeviceRef))
|
|
return true;
|
|
|
|
return false;
|
|
});
|
|
}
|
|
|
|
static void DeviceMatchingCallback(void* inContext, IOReturn inResult, void* inSender,
|
|
IOHIDDeviceRef inIOHIDDeviceRef)
|
|
{
|
|
DeviceDebugPrint(inIOHIDDeviceRef);
|
|
std::string name = GetDeviceRefName(inIOHIDDeviceRef);
|
|
|
|
// Add a device if it's of a type we want
|
|
if (IOHIDDeviceConformsTo(inIOHIDDeviceRef, kHIDPage_GenericDesktop, kHIDUsage_GD_Joystick) ||
|
|
IOHIDDeviceConformsTo(inIOHIDDeviceRef, kHIDPage_GenericDesktop, kHIDUsage_GD_GamePad) ||
|
|
IOHIDDeviceConformsTo(inIOHIDDeviceRef, kHIDPage_GenericDesktop,
|
|
kHIDUsage_GD_MultiAxisController))
|
|
{
|
|
g_controller_interface.AddDevice(std::make_shared<Joystick>(inIOHIDDeviceRef, name));
|
|
}
|
|
}
|
|
|
|
void Init()
|
|
{
|
|
HIDManager = IOHIDManagerCreate(kCFAllocatorDefault, kIOHIDOptionsTypeNone);
|
|
if (!HIDManager)
|
|
ERROR_LOG_FMT(CONTROLLERINTERFACE, "Failed to create HID Manager reference");
|
|
|
|
IOHIDManagerSetDeviceMatching(HIDManager, nullptr);
|
|
if (IOHIDManagerOpen(HIDManager, kIOHIDOptionsTypeNone) != kIOReturnSuccess)
|
|
ERROR_LOG_FMT(CONTROLLERINTERFACE, "Failed to open HID Manager");
|
|
|
|
// Callbacks for acquisition or loss of a matching device
|
|
IOHIDManagerRegisterDeviceMatchingCallback(HIDManager, DeviceMatchingCallback, nullptr);
|
|
IOHIDManagerRegisterDeviceRemovalCallback(HIDManager, DeviceRemovalCallback, nullptr);
|
|
|
|
// Match devices that are plugged in right now
|
|
IOHIDManagerScheduleWithRunLoop(HIDManager, CFRunLoopGetCurrent(), OurRunLoop);
|
|
while (CFRunLoopRunInMode(OurRunLoop, 0, TRUE) == kCFRunLoopRunHandledSource)
|
|
{
|
|
};
|
|
IOHIDManagerUnscheduleFromRunLoop(HIDManager, CFRunLoopGetCurrent(), OurRunLoop);
|
|
|
|
// Enable hotplugging
|
|
s_hotplug_thread = std::thread([] {
|
|
Common::SetCurrentThreadName("IOHIDManager Hotplug Thread");
|
|
NOTICE_LOG_FMT(CONTROLLERINTERFACE, "IOHIDManager hotplug thread started");
|
|
|
|
IOHIDManagerScheduleWithRunLoop(HIDManager, CFRunLoopGetCurrent(), OurRunLoop);
|
|
s_stopper.AddToRunLoop(CFRunLoopGetCurrent(), OurRunLoop);
|
|
CFRunLoopRunInMode(OurRunLoop, FOREVER, FALSE);
|
|
s_stopper.RemoveFromRunLoop(CFRunLoopGetCurrent(), OurRunLoop);
|
|
IOHIDManagerUnscheduleFromRunLoop(HIDManager, CFRunLoopGetCurrent(), OurRunLoop);
|
|
|
|
NOTICE_LOG_FMT(CONTROLLERINTERFACE, "IOHIDManager hotplug thread stopped");
|
|
});
|
|
}
|
|
|
|
void DeInit()
|
|
{
|
|
if (HIDManager)
|
|
{
|
|
s_stopper.Signal();
|
|
s_hotplug_thread.join();
|
|
|
|
// This closes all devices as well
|
|
IOHIDManagerClose(HIDManager, kIOHIDOptionsTypeNone);
|
|
CFRelease(HIDManager);
|
|
HIDManager = nullptr;
|
|
}
|
|
}
|
|
} // namespace ciface::OSX
|