如何在macOS下找到串口的父USB设备?

3

这是我尝试枚举在 Mac 上找到的所有串口,并遍历设备节点树以查找 USB-串行适配器的父 USB 设备的方法:

#import <Foundation/Foundation.h>
#import <IOKit/IOKitLib.h>
#import <IOKit/serial/IOSerialKeys.h>
#import <IOKit/usb/IOUSBLib.h>

int main(int argc, const char * argv[])
{
    @autoreleasepool
    {
        CFMutableDictionaryRef keywordDict = IOServiceMatching(kIOSerialBSDServiceValue);
        io_object_t port = 0;
        io_iterator_t iterator = 0;
        kern_return_t result = IOServiceGetMatchingServices(kIOMasterPortDefault, keywordDict, &iterator);
        if (result)
        {
            NSLog(@"Failed to find any serial ports.");
            exit(EXIT_FAILURE);
        }

        while ((port = IOIteratorNext(iterator)))
        {
            @autoreleasepool
            {
                NSMutableArray<NSDictionary *> *array = [NSMutableArray array];
                CFMutableDictionaryRef dict = NULL;
                result = IORegistryEntryCreateCFProperties(port, &dict, kCFAllocatorDefault, 0);
                if (!result)
                    [array addObject:CFBridgingRelease(dict)];

                io_object_t parent = 0;
                io_object_t parents = port;
                while (IORegistryEntryGetParentEntry(parents, kIOServicePlane, &parent))
                {
                    result = IORegistryEntryCreateCFProperties(parent, &dict, kCFAllocatorDefault, 0);
                    if (!result)
                        [array addObject:CFBridgingRelease(dict)];

                    if (parents != port)
                        IOObjectRelease(parents);
                    parents = parent;
                }

                NSLog(@"0x%08lx = %@", (uintptr_t)port, array);
            }
        }
    }
    return 0;
}

然而,虽然这段代码可以找到所有串口(包括我的Hackintosh上的USB-串口适配器和Super IO),但遍历IOService树并不能正常工作。请帮忙解决问题。目标操作系统是macOS 10.13.6。

编辑

以下是我一个USB-串口适配器的输出:

2018-08-18 19:05:39.860794+0800 lstty[98738:5197725] 0x00002a07 = (
        {
        CFBundleIdentifier = "com.apple.iokit.IOSerialFamily";
        IOCalloutDevice = "/dev/cu.usbmodem1751";
        IOClass = IOSerialBSDClient;
        IODialinDevice = "/dev/tty.usbmodem1751";
        IOGeneralInterest = "IOCommand is not serializable";
        IOMatchCategory = IODefaultMatchCategory;
        IOProbeScore = 1000;
        IOProviderClass = IOSerialStreamSync;
        IOResourceMatch = IOBSD;
        IOSerialBSDClientType = IOModemSerialStream;
        IOTTYBaseName = usbmodem;
        IOTTYDevice = usbmodem1751;
        IOTTYSuffix = 1751;
    }
)

以下是同一设备的 ioreg -itrc IOSerialBSDClient -w0 输出:

+-o Root  <class IORegistryEntry, id 0x100000100, retain 15>
  +-o iMac14,2  <class IORegistryEntry:IOService:IOPlatformExpertDevice, id 0x100000110, registered, matched, active, busy 0 (51894 ms), retain 45>
    +-o AppleACPIPlatformExpert  <class IORegistryEntry:IOService:IOPlatformExpert:IODTPlatformExpert:IOACPIPlatformExpert:AppleACPIPlatformExpert, id 0x100000111, registered, matched, active, busy 0 (41509 ms), retain 50>
      +-o PCI0@0  <class IORegistryEntry:IOService:IOPlatformDevice:IOACPIPlatformDevice, id 0x10000013e, registered, matched, active, busy 0 (41377 ms), retain 54>
        +-o AppleACPIPCI  <class IORegistryEntry:IOService:IOPCIBridge:AppleACPIPCI, id 0x100000216, registered, matched, active, busy 0 (41332 ms), retain 31>
          +-o XHC@14  <class IORegistryEntry:IOService:IOPCIDevice, id 0x1000001f0, registered, matched, active, busy 0 (35582 ms), retain 11>
            +-o XHC@14000000  <class IORegistryEntry:IOService:AppleUSBHostController:AppleUSBXHCI:AppleUSBXHCIPCI:AppleUSBXHCILPT:AppleUSBXHCILPTH:AppleUSBXHCILPTHB, id 0x10000023d, registered, matched, active, busy 0 (35072 ms), retain 1249>
              +-o HS10@14a00000  <class IORegistryEntry:IOService:AppleUSBHostPort:AppleUSBXHCIPort:AppleUSB20XHCIPort, id 0x100000277, registered, matched, active, busy 0 (22180 ms), retain 17>
                +-o IOUSBHostDevice@14a00000  <class IORegistryEntry:IOService:IOUSBNub:IOUSBDevice, id 0x100007a40, registered, matched, active, busy 0 (1604 ms), retain 31>
                  +-o AppleUSB20Hub@14a00000  <class IORegistryEntry:IOService:AppleUSBHub:AppleUSB20Hub, id 0x100007a4e, registered, matched, active, busy 0 (1600 ms), retain 23>
                    +-o AppleUSB20HubPort@14a10000  <class IORegistryEntry:IOService:AppleUSBHostPort:AppleUSBHubPort:AppleUSB20HubPort, id 0x100007a52, registered, matched, active, busy 0 (964 ms), retain 17>
                      +-o IOUSBHostDevice@14a10000  <class IORegistryEntry:IOService:IOUSBNub:IOUSBDevice, id 0x1000082bb, registered, matched, active, busy 0 (323 ms), retain 32>
                        +-o AppleUSB20Hub@14a10000  <class IORegistryEntry:IOService:AppleUSBHub:AppleUSB20Hub, id 0x1000082d4, registered, matched, active, busy 0 (316 ms), retain 23>
                          +-o AppleUSB20HubPort@14a12000  <class IORegistryEntry:IOService:AppleUSBHostPort:AppleUSBHubPort:AppleUSB20HubPort, id 0x1000082d8, registered, matched, active, busy 0 (269 ms), retain 17>
                            +-o ComboDebug USB Serial@14a12000  <class IORegistryEntry:IOService:IOUSBNub:IOUSBDevice, id 0x10000837e, registered, matched, active, busy 0 (268 ms), retain 28>
                              +-o IOUSBHostInterface@1  <class IORegistryEntry:IOService:IOUSBNub:IOUSBInterface, id 0x100008390, registered, matched, active, busy 0 (259 ms), retain 8>
                                +-o AppleUSBACMData  <class IORegistryEntry:IOService:IOSerialDriverSync:AppleUSBACMData, id 0x1000083a0, registered, matched, active, busy 0 (1 ms), retain 6>
                                  +-o IOModemSerialStreamSync  <class IORegistryEntry:IOService:IOSerialStreamSync:IOModemSerialStreamSync, id 0x100008417, registered, matched, active, busy 0 (1 ms), retain 6>
                                    +-o IOSerialBSDClient  <class IORegistryEntry:IOService:IOSerialBSDClient, id 0x100008418, registered, matched, active, busy 0 (0 ms), retain 5>
                                        {
                                          "IOClass" = "IOSerialBSDClient"
                                          "CFBundleIdentifier" = "com.apple.iokit.IOSerialFamily"
                                          "IOProviderClass" = "IOSerialStreamSync"
                                          "IOTTYBaseName" = "usbmodem"
                                          "IOSerialBSDClientType" = "IOModemSerialStream"
                                          "IOProbeScore" = 1000
                                          "IOCalloutDevice" = "/dev/cu.usbmodem1751"
                                          "IODialinDevice" = "/dev/tty.usbmodem1751"
                                          "IOMatchCategory" = "IODefaultMatchCategory"
                                          "IOTTYDevice" = "usbmodem1751"
                                          "IOResourceMatch" = "IOBSD"
                                          "IOGeneralInterest" = "IOCommand is not serializable"
                                          "IOTTYSuffix" = "1751"
                                        }

在你期望找到USB设备的端口之一,从遍历循环中输出一些调试信息(例如IORegistryEntryGetNameIOObjectGetClass)。然后发布来自ioreg -itrc IOSerialBSDClient的相应输出,以便我们可以看到它们的差异在哪里。 - pmdj
@pmdj 已添加一个设备的输出。该步行路线从未深入到足以显示 USB 设备的位置。 - Maxthon Chan
1个回答

1

你的早期终止问题在于循环的退出条件:

     while (IORegistryEntryGetParentEntry(parents, kIOServicePlane, &parent))

IORegistryEntryGetParentEntry() 返回一个 kern_return_t - KERN_SUCCESS 是 0,这意味着 while 条件的计算结果为假,循环不会运行。尝试

     while (KERN_SUCCESS == IORegistryEntryGetParentEntry(parents, kIOServicePlane, &parent))

这行代码的作用是:while (!IORegistryEntryGetParentEntry(parents, kIOServicePlane, &parent))。我成功地从循环中转储了相同级别的详细信息以及更多信息。现在只需要找到USB设备并提取信息即可。 - Maxthon Chan
2
是的,你的功能与我的建议在功能上是等效的。(我只是更喜欢我的清晰度,特别是考虑到你原来的符号表示使错误不明显。) - pmdj
我必须承认,我写那个不明显的代码是出于习惯,因为在小型设备编程中,每个时钟周期都很重要。一些编译器会为 if (!func()) {}if (func() == 0) {} 发出不同的代码,前者通常比后者少几条指令。 - Maxthon Chan
@MaxthonChan 如果启用了任何优化,现代版本的clang和GCC为这两个变体发出相同的代码:https://godbolt.org/z/vYWAnI - pmdj
我曾经需要处理MPLAB XC8 v1.35一段时间,那个东西不是任何衍生物。此外,AVR处理器通常使用的GCC 4.3.5版本。 - Maxthon Chan

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接