2

我正在开发支持游戏控制器的 OS X 应用程序。它必须支持源自 IOKit HID 和 GameController.framework 的控制器。我面临的问题是大多数 MFi GameController.framework 兼容的控制器也是隐藏设备。因此,MFi 控制器在控制器列表中出现两次,分别为 GCController 和 IOHIDDevice。有什么方法可以在它们之间建立连接,忽略 HID 设备?

GCController 对象具有私有属性deviceRef,指向底层隐藏设备,从而可以识别和忽略隐藏层中的设备。问题是这deviceRef是一个私有财产,所以我不能在 App Store 应用程序中使用它。

理想的解决方案是一种识别 IOHIDDeviceRef 是 MFi 设备的方法,因此我可以在我的 HID 层中完全跳过它。

4

2 回答 2

5

我正在试验 GCController 并最终找到了一个 hacky 解决方案。这可能是区分使用 GameController 框架的控制器和使用 IOKit 的控制器的唯一方法:

  1. 每当一个新的控制器连接到 Mac 时,都会为 IOKit 和 GameController 分别调用 IOHIDDeviceRef 和 GCController 实例的连接回调。

  2. 获取 IOHIDDeviceRef 的供应商 ID 和产品 ID:

CFNumberRef vendor = static_cast<CFNumberRef>(IOHIDDeviceGetProperty(device, CFSTR(kIOHIDVendorIDKey));
if (vendor) CFNumberGetValue(vendor, kCFNumberSInt32Type, &vendorId);
CFNumberRef product = static_cast<CFNumberRef>(IOHIDDeviceGetProperty(device, CFSTR(kIOHIDProductIDKey)));
if (product) CFNumberGetValue(product, kCFNumberSInt32Type, &productId);
  1. 获取 GCController 实例的供应商 ID 和产品 ID 有点棘手(和 hacky),但我在几个版本的 macOS 上使用许多设备对其进行了测试,并且它可以工作。首先,您必须声明在 IOKit/hidsystem/IOHIDServiceClient.h 中定义的 IOHIDServiceClientCopyProperty 函数。IOHIDServiceClient.h 仅包含在 MacOSX sdk 10.12 中,因此您必须定义该函数以用于早期版本的 SDK:
typedef struct CF_BRIDGED_TYPE(id) __IOHIDServiceClient * IOHIDServiceClientRef;
extern "C" CFTypeRef _Nullable IOHIDServiceClientCopyProperty(IOHIDServiceClientRef service, CFStringRef key);
  1. 要获取实际 ID,您首先必须调用 GCController 的受保护方法“hidServices”,这将返回一个指向 GCCControllerHIDServiceInfo 实例的指针数组。GCCControllerHIDServiceInfo 是一个内部类,它有两个方法:“inputData”和“service”(我们感兴趣)。该数组通常只有一个元素,因此我们调用它的服务方法来获取设备的 IOHIDServiceClientRef 实例。您可以通过调用 IOHIDServiceClientCopyProperty 来获取它的供应商 ID 和产品 ID,其他一切都类似于 IOKit:
if (class_respondsToSelector(object_getClass(controller), sel_getUid("hidServices")))
{
    NSArray* hidServices = reinterpret_cast<NSArray* (*)(id, SEL)>(objc_msgSend)(controller, sel_getUid("hidServices"));

    if (hidServices && [hidServices count] > 0)
    {
        IOHIDServiceClientRef service = reinterpret_cast<IOHIDServiceClientRef (*)(id, SEL)>(objc_msgSend)([hidServices firstObject], sel_getUid("service"));

        CFNumberRef vendor = static_cast<CFNumberRef>(IOHIDServiceClientCopyProperty(service, CFSTR(kIOHIDVendorIDKey)));
        if (vendor)
        {
            CFNumberGetValue(vendor, kCFNumberSInt32Type, &vendorId);
            CFRelease(vendor);
        }

        CFNumberRef product = static_cast<CFNumberRef>(IOHIDServiceClientCopyProperty(service, CFSTR(kIOHIDProductIDKey)));
        if (product)
        {
            CFNumberGetValue(product, kCFNumberSInt32Type, &productId);
            CFRelease(product);
        }
    }
}
  1. 您要做的最后一件事实际上是将供应商 ID 和产品 ID 与支持一个或另一个框架的设备列表进行比较。目前,我只知道两种支持 GameController 框架的设备(如果您知道任何其他 GameController 框架兼容的设备,请告诉我):
    • SteelSeries Nimbus:供应商 ID = 0x1038,产品 ID = 0x1420
    • HoriPad Ultimate:供应商 ID = 0x0F0D,产品 ID = 0x0090

您可以在Ouzel 引擎中看到上面描述的完整代码。

于 2017-08-03T12:17:29.893 回答
2

macOS 11 及更高版本上的 GCController 具有+[GCController supportsHIDDevice:]采用IOHIDDeviceRef. 为了支持早期的系统,我会建议 Elviss 的建议。您还可以参考处理这两种情况的 WebKit 的源代码。

于 2020-11-25T04:30:38.657 回答