在Windows中,可以通过编程方式为设备识别物理USB端口吗

本文关键字:识别 USB Windows 可以通过 编程 方式 | 更新日期: 2023-09-27 17:59:36

我有一个USB设备,当被命令这样做时,它会使用不同的接口、VID、PID和序列号进行枚举,并且我希望在发生此更改后跟踪物理设备。我的想法是通过它的枢纽和港口位置来跟踪它。

Win32_PnPSignedDriver类有一个看似完美的"位置"字段(例如Port_#0001.Hub_#0010),但它只包含首次加载驱动程序时设备的位置。将硬件插入其他端口不会更新该字段。

但是,由于通过设备管理器查看设备时,"详细信息"选项卡下有一个"位置信息"字段,因此该信息在某些地方可用。可以通过WMI查询或其他方法检索此信息吗?有没有更好的方法来解决这个问题?

编辑:我知道这听起来很奇怪。这些设备中的微控制器包含一个ROM,该ROM列举为CDC设备(即串行端口)并允许编程。在制造过程中,当设备在制造商的ROM(唯一的VID/PID/序列号)和我的自定义固件接口(不同的VID/PID/序列号。

在Windows中,可以通过编程方式为设备识别物理USB端口吗

我知道已经有一段时间没有任何关于这个答案的活动了,但我正在进行一个项目,该项目也需要类似的功能,我可以告诉你这确实是可能的。据我所知,它确实需要DDK和PInvoke,这些信息没有C#或WMI接口。它需要打开低级USB根集线器设备,并直接向它们发送驱动程序IOCTL命令。

好消息是,微软提供了一个示例C++应用程序,它完全枚举了所有USB设备,并准确显示了它们连接到的端口。该应用程序就是USBView示例应用程序。

我想你会发现,如果你编译并运行这个应用程序,你会看到它会准确地显示你的设备插入的位置,如果你将任何设备插入该端口,它会显示在同一个位置。如果您创建一个非托管的C++DLL,它提供了一些C#应用程序可以用来获取所需信息的调用,那么这可能会更容易。

关于代码中的EnumerateHubPorts()函数,它有这样的说法:

给定开放集线器的句柄和上的下游端口数集线器,向集线器发送IOCTL_USB_GET_NODE_CONNECTION_INFORMATION_EX请求集线器的每个下游端口获取有关连接到每个端口的设备(如果有的话)。

为了了解这需要的一切(必须从顶部开始枚举所有内容,即使您只对一个端口感兴趣),以下是代码中enum.c文件顶部列出的注释:

/*
This source file contains the routines which enumerate the USB bus
and populate the TreeView control.
The enumeration process goes like this:
(1) Enumerate Host Controllers and Root Hubs
EnumerateHostControllers()
EnumerateHostController()
Host controllers currently have symbolic link names of the form HCDx,
where x starts at 0.  Use CreateFile() to open each host controller
symbolic link.  Create a node in the TreeView to represent each host
controller.
GetRootHubName()
After a host controller has been opened, send the host controller an
IOCTL_USB_GET_ROOT_HUB_NAME request to get the symbolic link name of
the root hub that is part of the host controller.
(2) Enumerate Hubs (Root Hubs and External Hubs)
EnumerateHub()
Given the name of a hub, use CreateFile() to map the hub.  Send the
hub an IOCTL_USB_GET_NODE_INFORMATION request to get info about the
hub, such as the number of downstream ports.  Create a node in the
TreeView to represent each hub.
(3) Enumerate Downstream Ports
EnumerateHubPorts()
Given an handle to an open hub and the number of downstream ports on
the hub, send the hub an IOCTL_USB_GET_NODE_CONNECTION_INFORMATION_EX
request for each downstream port of the hub to get info about the
device (if any) attached to each port.  If there is a device attached
to a port, send the hub an IOCTL_USB_GET_NODE_CONNECTION_NAME request
to get the symbolic link name of the hub attached to the downstream
port.  If there is a hub attached to the downstream port, recurse to
step (2).  
GetAllStringDescriptors()
GetConfigDescriptor()
Create a node in the TreeView to represent each hub port
and attached device.
*/

您尝试过SetupDi吗?您可以使用API函数的SetupDi类从DeviceManager获取信息。

设备管理器下的"位置信息"与您通过WMI获得的字符串完全相同。

您是否考虑过,当设备插入不同的端口时,Windows不会使用新位置更新元数据,而是创建一个新的驱动程序实例和新的元数据。尝试过滤Win32_PnPDevice对象实例,只查找当前插入的对象实例,我想您会找到当前位置信息。

例如,如果我将USB鼠标移动到另一个端口,则设备管理器下仍会列出与旧端口关联的鼠标副本,默认情况下它只是隐藏的。看见http://oreilly.com/pub/h/3105以获取查看这些断开连接的设备的说明。或者从提升的管理员命令提示符运行以下命令:

C:'Windows'system32>set devmgr_show_nonpresent_devices=1
C:'Windows'system32>devmgmt

REF:"Win32_PnPSignedDriver类具有一个";"位置";字段(例如Port_#00001.Hub_#0010),但它只包含首次加载驱动程序时设备的位置。将硬件插入其他端口不会更新该字段"

对我来说确实如此。但是,只要确保在端口交换之间刷新(F5)regedit应用程序,否则你就看不到变化。

以下是一些例子:

端口2集线器2
端口2集线器4
端口3集线器4

更好的方法是使用USB设备的唯一序列号。