如何在Linux上通过编程获取USB设备的供应商和产品信息?

20

使用udev,我已经能够获取特定USB设备的以下信息:

idVendor: 13b1
idProduct: 0018
manufacturer:  
product: USB 2.0 Network Adapter ver.2 
serial: 00FFFF

现在我想获取与供应商和产品ID相关联的完整字符串。 我发现文件/usr/share/misc/usb.ids包含我正在寻找的信息:

13b1  Linksys
        000b  WUSB11 v4.0 802.11b Adapter
        000d  WUSB54G Wireless Adapter
        0011  WUSB54GP v4.0 802.11g Adapter
        0018  USB200M 10/100 Ethernet Adapter
        001a  HU200TS Wireless Adapter
        001e  WUSBF54G 802.11bg
        0020  WUSB54GC 802.11g Adapter [ralink rt73]
        0023  WUSB54GR
        0024  WUSBF54G v1.1 802.11bg

然而,我不清楚我应该如何在我的应用程序中检索这些数据。是否有可用的 API,还是我只需要解析文件?如果我选择解析它,那么/usr/share/misc/usb.ids是否始终是正确的位置?


2
在我的系统上,它位于/usr/share/usb.ids。找不到任何规范来管理此文件的位置。 - P Shved
2
以下是最新版本:http://www.linux-usb.org/usb.ids,以及提交它们的接口:http://www.linux-usb.org/usb-ids.html。 - endolith
在Android上,我的C程序使用libusbhost,它提供了有关设备插入/移除的回调功能。所有设备描述符都由该库提供。 - m-ric
5个回答

26

lsusb 命令可查询当前已连接的 USB 设备信息。您可以使用其 -d 选项查询特定的供应商/产品(但似乎仅适用于当前已连接的设备):

$ lsusb -d 0e21:0750
Bus 001 Device 005: ID 0e21:0750 Cowon Systems, Inc.

你可以展示所有设备的信息:

$ lsusb
Bus 001 Device 001: ID 1d6b:0002 Linux Foundation 2.0 root hub
Bus 001 Device 004: ID 0421:01c7 Nokia Mobile Phones
Bus 001 Device 003: ID 0bda:8187 Realtek Semiconductor Corp. RTL8187 Wireless Adapter
Bus 001 Device 005: ID 0e21:0750 Cowon Systems, Inc.
Bus 002 Device 001: ID 1d6b:0002 Linux Foundation 2.0 root hub
Bus 003 Device 001: ID 1d6b:0001 Linux Foundation 1.1 root hub
Bus 004 Device 001: ID 1d6b:0001 Linux Foundation 1.1 root hub
Bus 005 Device 001: ID 1d6b:0001 Linux Foundation 1.1 root hub
Bus 006 Device 001: ID 1d6b:0001 Linux Foundation 1.1 root hub
Bus 006 Device 002: ID 046d:c01b Logitech, Inc. MX310 Optical Mouse
Bus 007 Device 001: ID 1d6b:0001 Linux Foundation 1.1 root hub
Bus 008 Device 001: ID 1d6b:0001 Linux Foundation 1.1 root hub

你也可以设置成冗长模式(lsusb -v)并打印出大量信息。

请注意,当在Linux操作系统中访问有关系统的信息时,最好通过Shell命令(比如lsusb)来获取,而不是直接解析这些命令所访问的系统文件。


那我需要使用popen调用lsusb吗? - StackedCrooked
1
@Stacked,是的(对于C语言),但这取决于您开发的语言。但无论如何,您都将从流中读取某些内容,并通过popen调用lsusb -d ...似乎比解析带有描述的整个文件需要更少的按键。 - P Shved

6
我虽未尝试过,但 libudev的udev_device_get_property_value 应该就是它了;在 pulseaudio的udev-util.c 中使用了它,如下所示:udev_device_get_property_value(card, "ID_VENDOR_FROM_DATABASE"))
以下是一个小例子,我基于 udev-util.c 编写,注意我使用了带有 FTDI FT232 芯片的 Arduino Duemillanove,我使用 udevadm 找到了其 udev 路径(请参阅下面代码中的注释),然后将其硬编码到了下面的程序中:udevl.c
// sudo apt-get install libudev-dev
// build with: gcc -o udevl -ludev -Wall -g udevl.c

#include <stdio.h>

#include <libudev.h>

int main( int argc, char **argv )
{
  const char *v;
  char t[256];
  struct udev *udev;
  struct udev_device *card = NULL;

  if (!(udev = udev_new())) {
      fprintf(stderr, "Failed to allocate udev context.\n");
      return -1;
  }

  // $ lsusb | grep FT232
  // Bus 002 Device 002: ID 0403:6001 Future Technology Devices International, Ltd FT232 USB-Serial (UART) IC    
  // $ udevadm info --name=/dev/ttyUSB0 --attribute-walk | grep "looking at device"
  // looking at device '/devices/pci0000:00/0000:00:1d.0/usb2/2-2/2-2:1.0/ttyUSB0/tty/ttyUSB0'
  // (that one is under /sys)
  // hardcode that path below:

  // udev_get_sys_path(udev) for me: '/sys'
  sprintf(t, "%s/devices/pci0000:00/0000:00:1d.0/usb2/2-2/2-2:1.0/ttyUSB0/tty/ttyUSB0", udev_get_sys_path(udev));
  fprintf(stdout, " path: %s\n", t);

  card = udev_device_new_from_syspath(udev, t);
  fprintf(stdout, " udev_device: 0x%08X\n", (unsigned int)card);

  if ((v = udev_device_get_property_value(card, "ID_MODEL_FROM_DATABASE")) )
    fprintf(stdout, "got ID_MODEL_FROM_DATABASE: %s\n", v);
  else
    fprintf(stdout, "failed getting ID_MODEL_FROM_DATABASE: %s\n", v);

  fprintf(stdout, "Done.\n");

  if (card)
    udev_device_unref(card);

  if (udev)
    udev_unref(udev);

  return 0;
}

这个程序(连接了Arduino)输出:
$ ./udevl 
 path: /sys/devices/pci0000:00/0000:00:1d.0/usb2/2-2/2-2:1.0/ttyUSB0/tty/ttyUSB0
 udev_device: 0x09FBF080
got ID_MODEL_FROM_DATABASE: FT232 USB-Serial (UART) IC
Done.

... "FT232 USB-Serial (UART) IC"是具有VID:PID 0403:6001的正确条目。请参阅usb.ids

希望这可以帮到你,
祝好!


如何初始化结构体udev * udev;?struct udev * udev = NULL?因为我的代码检查器报错了。谢谢。 - uss

4
在我的Ubuntu系统上,lsusb(1)手册中说/var/lib/usbutils/usb.ids是id文件的位置;实际上,有两个符号链接,其中一个是/usr/share/misc/usb.ids。在信任符号链接之前,我会相信实际位置:
$ ls -l /usr/share/misc/usb.ids /var/lib/misc/usb.ids /var/lib/usbutils/usb.ids
lrwxrwxrwx 1 root root     25 2010-04-29 18:08 /usr/share/misc/usb.ids -> /var/lib/usbutils/usb.ids
lrwxrwxrwx 1 root root     19 2010-04-29 18:08 /var/lib/misc/usb.ids -> ../usbutils/usb.ids
-rw-r--r-- 1 root root 368377 2009-11-06 09:26 /var/lib/usbutils/usb.ids

3
lsusb

您会得到类似这样的内容。
Bus 002 Device 001: ID 1d6b:0003 Linux Foundation 3.0 root hub
Bus 001 Device 009: ID 138a:0010 Validity Sensors, Inc. VFS Fingerprint sensor
Bus 001 Device 008: ID 13d3:3491 IMC Networks 

那么:ID 1d6b:0003 可以被理解为:厂商 = 1d6b产品 = 0003


1

您的USB设备不需要与实际正确名称匹配供应商和产品ID。

最好使用类似于libusblsusb的工具从设备本身获取此信息。


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