libusb中断传输

时间:2011-06-25 00:15:22

标签: c++ c linux usb libusb

我需要对定制的HID USB设备(控制面板上的某些按钮和指示灯)进行反向工程。该驱动程序仅在Windows上可用,我们需要* nix实现。

该设备显然是HID设备,但不属于特定类。它提供两个接口,每个接口都有一个中断端点。

我的设置目前涉及在Ubuntu主机上运行Windows的VirtualBox,以通过Wireshark捕获USB流量。协议相当简单,我已经获得了相当好的理解。

我在一个简单的C ++控制台程序中使用libusb-1.0进行原型设计。我已经设法通过发出SET_REPORT控制传输来切换LED,但是很难通过传输中断接收按钮按下。

实际上以下呼叫永远阻止:

unsigned char bytes[8] = { 0 };
int len = 0;
int ret = libusb_interrupt_transfer(handle, 0x81, bytes, 8, &len, 0); 

在Wireshark中检查生成的URB时,它看起来与Windows会话中捕获的等效内容完全相同。我仍然没有收到设备的回复。

我失败了,我错过了一些设置。请注意,设备已正确打开,并且设备提供的两个接口均已成功声明。即使在我的linux应用程序中,通过控制传输的输入报告即将到来。

感谢任何指针! ·阿尔

附录一: 我想知道在使用libusb_interrupt_transfer()时我应该如何指定我想要接收哪个报告ID?

附录二: 将Windows驱动程序发出的请求与Wireshark中上述代码生成的请求进行比较时,我看不出任何差异(URB中的值相同)。但是,只有当Windows驱动程序发出时才会返回中断传输。

在Wireshark中检查Windows驱动程序通信时,除了各种GET_DESCRIPTOR(...)之外,我没有看到任何控件传输。最重要的是:没有SET_INTERFACESET_CONFIGURATION因此我怀疑问题与库或我如何使用它有关,并且与设备相关。

2 个答案:

答案 0 :(得分:1)

您发布的代码存在问题。您为定义bytes而编写的语法不会产生8字节数组,但是您要求libusb将8个字节写入该地址,这样您可能会收到错误或内存损坏。试试这个:

unsigned char buffer[8];
int len = 0;
int ret = libusb_interrupt_transfer(handle, 0x81, buffer, sizeof(buffer), &len, 0);

每个HID报告都有自己的端点,因此您可以通过指定正确的端点来指定要接收的报告。您指定了端点1 IN(0x81)。您确定在设备的描述符中定义了端点吗?也许你应该得到描述符(在Ubuntu中使用lsusb -v)并将它们发布在这里,以便我们检查它们。

答案 1 :(得分:0)

我遇到了同样的问题,libusb_interrupt_transfer()永远阻止了读取端点(bEndpoinntAddress:0x81 EP 1 IN(lsusb -v的输出)),但是解决了。

就我而言,我是这样写的。

#define ENDPOINT_IN 0x81
unsigned char buffer[8];
int len = 0;
int ret = libusb_interrupt_transfer(handle, ENDPOINT_IN, buffer, sizeof(buffer), &len, 0);

但是,我的设备在读取数据之前需要发送一些代码,
尽管lsusb -v的输出bLength为7,并且需要sizeof 64缓冲区。

#define ENDPOINT_OUT 0x01
#define ENDPOINT_IN 0x81
unsigned char buffer[64] = {0x20, 0x01, 0x03, 0x02, 0x07};
int len = 0;
int send_ret = libusb_interrupt_transfer(handle, ENDPOINT_OUT, buffer, sizeof(buffer), &len, 0);
int recv_ret = libusb_interrupt_transfer(handle, ENDPOINT_IN, buffer, sizeof(buffer), &len, 0);

buffer[64] = {0x20, 0x01, 0x03, 0x02, 0x07}取决于设备规格。

希望这会有所帮助。