Windows与自定义USB HID设备通信说明.

.   所使用的典型 Windows API

CreateFile

ReadFile

WriteFile

以下函数是 DDK 的内容:

HidD_SetFeature

HidD_GetFeature

HidD_SetOutputReport

HidD_GetInputReport

其中, CreateFile 用于打开设备; ReadFile 、 HidD_GetFeature 、 HidD_GetInputReport 用于设备到主机方向的数据通信; WriteFile 、 HidD_SetFeature 、 HidD_SetOutputReport 用于主机到设备方向的数据通信。鉴于实际应用,后文主要讨论 CreateFile , WriteFile , ReadFile ,HidD_SetFeature 四个函数,明白了这四个函数,其它的可以类推之。

2 .   几个常见错误

当使用以上 API 时,如果操作失败,调用 GetLastError() 会得到以下常见错误:

6 :          句柄无效

23 :        数据错误(循环冗余码检查)

87 :        参数错误

1784 :     用户提供的 buffer 无效

后文将会详细说明这些错误情况。

3.         主机端设备枚举程序流程

4.         函数使用说明

CreateFile(devDetail->DevicePath,                                         // 设备路径

               GENERIC_READ | GENERIC_WRITE,                    // 访问方式

               FILE_SHARE_READ | FILE_SHARE_WRITE,         // 共享模式

               NULL,

               OPEN_EXISTING,                                           // 文件不存在时,返回失败

               FILE_FLAG_OVERLAPPED,                                 // 以重叠(异步)模式打开

               NULL);

在这里, CreateFile 用于打开 HID 设备,其中设备路径通过函数SetupDiGetInterfaceDeviceDetail 取得。 CreateFile 有以下几点需要注意:

-     访问方式: 如果是系统独占设备,例如鼠标、键盘等等,应将此参数设置为 0 ,否则后续函数操作将失败(譬如 HidD_GetAttributes );也就是说,不能对独占设备进行除了查询以外的任何操作,所以能够使用的函数也是很有限的,下文的一些函数并不一定适合这些设备。在此顺便列出 MSDN 上关于此参数的说明:

If this parameter is zero, the application can query file and device attributes without accessing the device. This is useful if an application wants to determine the size of a floppy disk drive and the formats it supports without requiring a floppy in the drive. It can also be used to test for the file‘s or directory‘s existence without opening it for read or write access 。

-          重叠(异步)模式:此参数并不会在此处表现出明显的意义,它主要是对后续的 WriteFile ,ReadFile 有影响。如果这里设置为重叠(异步)模式,那么在使用 WriteFile , ReadFile 时也应该使用重叠(异步)模式,反之亦然。这首先要求 WriteFile , ReadFile 的最后一个参数不能为空(NULL )。否则,便会返回 87 (参数错误)错误号。当然, 87 号错误并不代表就是此参数不正确,更多的信息将在具体讲述这两个函数时指出。此参数为 0 时,代表同步模式,即 WriteFile , ReadFile操作会在数据处理完成之后才返回,否则阻塞在函数内部。

ReadFile(hDev,                                 // 设备句柄,即 CreateFile 的返回值

              recvBuffer,                          // 用于接收数据的 buffer

              IN_REPORT_LEN,              // 要读取数据的长度

              &recvBytes,                         // 实际收到的数据的字节数

              &ol);                                  // 异步模式

在这里, ReadFile 用于读取 HID 设备通过中断 IN 传输发来的输入报告 。有以下几点要注意:

1 、 ReadFile 的调用不会引起设备的任何反应,即 HID 设备与主机之间的中断 IN 传输不与ReadFile 打交道。实际上主机会在最大间隔时间(由设备的端点描述符来指定)内轮询设备,发出中断 IN 传输的请求。“读取”即意味着从某个 buffer 里面取回数据,实际上这个 buffer 就是 HID 设备驱动中的buffer 。这个 buffer 的大小可以通过 HidD_SetNumInputBuffers 来改变。在 XP 上缺省值是 32(个报告)。

2 、读取的数据对象是输入报告,也即通过中断输入管道传入的数据。所以,如果设备不支持中断 IN 传输,那么是无法使用此函数来得到预期结果的。实际上这种情况不可能在 HID 中出现,因为协议指明了至少要有一个中断 IN 端点。

3 、 IN_REPORT_LEN 代表要读取的数据的长度(实际的数据正文 + 一个 byte 的报告 ID ),这里是一个常数,主要是因为设备固件的信息我是完全知道的,当然知道要读取多少数据(也就是报告的长度);不过也可以通过另外的函数( HidD_GetPreparsedData )来事先取得报告的长度,这里不做详细讨论。因为很难想象在不了解固件信息的情况下来做自定义设备的 HID 通信,在实际应用中一般来说就是固件与 PC 程序匹配着来开发。此参数如果设置过大,不会有实质性的错误,在 recvBytes 参数中会输出实际读到的长度;如果设置过小,即小于报告的长度,会返回 1784 号错误(用户提供的 buffer 无效)。

4 、关于异步模式。前面已经提过,此参数的设置必须与 CreateFile 时的设置相对应,否则会返回 87号错误(参数错误)。如果不需要异步模式,此参数需置为 NULL 。在这种情况下, ReadFile 会一直等待直到数据读取成功,所以会阻塞住程序的当前过程。

WriteFile(hDev,                                 // 设备句柄,即 CreateFile 的返回值

                     reportBuf,                           // 存有待发送数据的 buffer

                     OUT_REPORT_LEN,           // 待发送数据的长度

                     &sendBytes,                        // 实际收到的数据的字节数

                     &ol);                                  // 异步模式

在这里, WriteFile 用于传输一个输出报告 给 HID 设备。有以下几点要注意:

1、  与 ReadFile 不同, WriteFile 函数被调用后,虽然也是经过驱动程序,但是最终会反映到设备中。也就是说,调用 WriteFile 后,设备会接收到输出报告的请求。如果设备使用了中断 OUT 传输,则WriteFile 会通过中断 OUT 管道来进行传输;否则会使用 SetReport 请求通过控制管道来传输。

2、  OUT_REPORT_LEN 代表要写入的数据长度(实际的数据正文 + 一个 byte 的报告 ID )。如果大于实际报告的长度,则使用实际报告长度;如果小于实际报告长度,会返回 1784 号错误(用户提供的 buffer 无效)。

3、  reportBuf [0] 必须存有待发送报告的 ID ,并且此报告 ID 指示的必须是输出报告,否则会返回 87号错误(参数错误)。这种情况可能容易被程序员忽略,结果不知错误号所反映的是什么,网上也经常有类似疑问的帖子。顺便指出,输入报告、输入报告、特征报告这些报告类型,是反映在 HID 设备的报告描述符中。后文将做举例讨论。

4、  关于异步模式。前面已经提过,此参数的设置必须与 CreateFile 时的设置相对应,否则会返回 87 号错误(参数错误)。如果不需要异步模式,此参数需置为 NULL 。在这种情况下, WriteFile 会一直等待直到数据读取成功,所以会阻塞住程序的当前过程。

HidD_SetFeature(hDev,                                    // 设备句柄,即CreateFile 的返回值

                     reportBuf,                                   // 存有待发送数据的 buffer

                     FEATURE_REPORT_LEN);        //buffer 的长度

HidD_SetOutputReport(hDev,                            // 设备句柄,即 CreateFile的返回值

                     reportBuf,                                   // 存有待发送数据的 buffer

                     OUT_REPORT_LEN);                //buffer 的长度

HidD_SetFeature 发送一个特征报告 给设备, HidD_ SetOutputReport 发送一个输出报告 给设备。注意以下几点:

1、  跟 WriteFile 类似,必须在 reportBuf [0] 中指明要发送的报告的 ID ,并且和各自适合的类型相对应。也就是说, HidD_SetFeature 只能发送特征报告,因此报告 ID 必须是特征报告的 ID ;HidD_SetOutputReport 只能发送输出报告,因此报告 ID 只能是输出报告的 ID 。

2、  这两个函数最常返回的错误代码是 23 (数据错误)。包括但不仅限于以下情况:

- 报告 ID 与固件描述的不符。

- 传入的 buffer 长度少于固件描述的报告的长度。

据有关资料反映(非官方文档),只要是驱动程序对请求无反应,都会产生此错误。

5.         常见错误汇总

- HID ReadFile

- Error Code 6 (handle is invalid)

传入的句柄无效

- Error Code 87 ( 参数错误 )

很可能是 createfile 时声明了异步方式,但是读取时按同步读取。

- Error Code 1784 ( 用户提供的 buffer 无效 ):

传参时传入的“读取 buffer 长度”与实际的报告长度不符。

- HID WriteFile

- Error Code 6 (handle is invalid)

传入的句柄无效

- Error Code 87 (参数错误)

- CreateFile 时声明的同步 / 异步方式与实际调用 WriteFile 时传入的不同。

- 报告 ID 与固件中定义的不一致( buffer 的首字节是报告 ID )

- Error Code 1784 ( 用户提供的 buffer 无效 )

传参时传入的“写入 buffer 长度”与实际的报告长度不符。

- HidD_SetFeature

- HidD_SetOutputReport

- Error Code 1 (incorrect function)

不支持此函数,很可能是设备的报告描述符中未定义这样的报告类型(输入、输出、特征)

- Error Code 6 (handle is invalid)

传入的句柄无效

- Error Code 23 (数据错误(循环冗余码检查))

- 报告 ID 与固件中定义的不相符( buffer 的首字节是报告 ID )

- 传入的 buffer 长度少于固件定义的报告长度(报告正文 +1byte, 1byte 为报告 ID )

- 据相关资料反映(非官方文档),只要是驱动程序不接受此请求(对请求无反应),都会产生此错误

6.         报告描述符及数据通信程序示例

报告描述符(由于是汇编代码,所以不必留意其语法,仅需注意表中的每个数据都占 1 个字节):

_ReportDescriptor:                              // 报告描述符

.dw 0x06,  0x00, 0xff               // 用法页

.dw 0x09,  0x01                     // 用法 ( 供应商用法 1)

.dw 0xa1,  0x01                      // 集合开始

.dw 0x85,  0x01                         // 报告 ID(1)

.dw 0x09,  0x01                  // 用法 ( 供应商用法 1)

.dw 0x15,  0x00                 // 逻辑最小值 (0)

.dw 0x26,  0xff, 0x0                     // 逻辑最大值 (255)

.dw 0x75,  0x08               // 报告大小 (8)

.dw 0x95,  0x07                        // 报告计数 (7)

.dw 0x81,  0x06                // 输入 (数据,变量,相对值)

.dw 0x09,  0x01                     // 用法 ( 供应商用法 1)

.dw 0x85,  0x03                         // 报告 ID ( 3 

.dw 0xb1,   0x06                         // 特征 (数据,变量,相对值)

.dw 0x09,  0x01                    // 用法 ( 供应商用法 1)

.dw 0x85,  0x02                         // 报告 ID ( 2 

.dw 0xb1,  0x06                         // 特征 (数据,变量,相对值)

.dw 0x09,  0x01                     // 用法 ( 供应商用法 1)

.dw 0x85,  0x04                         // 报告 ID ( 4 

.dw 0x91,   0x06                         // 输出 (数据,变量,相对值)

.dw   0xc0                    // 结合结束

_ReportDescriptor_End:

这个报告描述符,定义了 4 个不同的报告:输入报告 1 ,特征报告 2 ,特征报告 3 ,输出报告 4 (数字代表其报告 ID )。为了简化,每个报告都是 7 个字节(加上报告 ID 就是 8 个字节)。下面用一个简单的示例来描述 PC 端与 USB HID 设备进行通信的一般方法。

view plaincopy to clipboardprint?

  1. #define     USB_VID       0xFC0
  2. #define     USB_PID       0x420
  3. HANDLE OpenMyHIDDevice(int overlapped);
  4. void HIDSampleFunc()
  5. {
  6. HANDLE       hDev;
  7. BYTE         recvDataBuf[8];
  8. BYTE         reportBuf[8];
  9. DWORD        bytes;
  10. hDev = OpenMyHIDDevice(0);                                // 打开设备,不使用重叠(异步)方式 ;
  11. if (hDev == INVALID_HANDLE_VALUE)
  12. return;
  13. reportBuf[0] = 4;                                         // 输出报告的报告 ID 是 4
  14. memset(reportBuf, 0, 8);
  15. reportBuf[1] = 1;
  16. if (!WriteFile(hDev, reportBuf, 8, &bytes, NULL))         // 写入数据到设备
  17. return;
  18. ReadFile(hDev, recvDatatBuf, 8, &bytes, NULL);            // 读取设备发给主机的数据
  19. }
  20. HANDLE OpenMyHIDDevice(int overlapped)
  21. {
  22. HANDLE hidHandle;
  23. GUID hidGuid;
  24. HidD_GetHidGuid(&hidGuid);
  25. HDEVINFO hDevInfo = SetupDiGetClassDevs(&hidGuid,
  26. NULL,
  27. NULL,
  28. (DIGCF_PRESENT | DIGCF_DEVICEINTERFACE));
  29. if (hDevInfo == INVALID_HANDLE_VALUE)
  30. {
  31. return INVALID_HANDLE_VALUE;
  32. }
  33. SP_DEVICE_INTERFACE_DATA devInfoData;
  34. devInfoData.cbSize = sizeof (SP_DEVICE_INTERFACE_DATA);
  35. int deviceNo = 0;
  36. SetLastError(NO_ERROR);
  37. while (GetLastError() != ERROR_NO_MORE_ITEMS)
  38. {
  39. if (SetupDiEnumInterfaceDevice (hDevInfo,
  40. 0,
  41. &hidGuid,
  42. deviceNo,
  43. &devInfoData))
  44. {
  45. ULONG  requiredLength = 0;
  46. SetupDiGetInterfaceDeviceDetail(hDevInfo,
  47. &devInfoData,
  48. NULL,
  49. 0,
  50. &requiredLength,
  51. NULL);
  52. PSP_INTERFACE_DEVICE_DETAIL_DATA devDetail = (SP_INTERFACE_DEVICE_DETAIL_DATA*) malloc (requiredLength);
  53. devDetail->cbSize = sizeof(SP_INTERFACE_DEVICE_DETAIL_DATA);
  54. if(!SetupDiGetInterfaceDeviceDetail(hDevInfo,
  55. &devInfoData,
  56. devDetail,
  57. requiredLength,
  58. NULL,
  59. NULL))
  60. {
  61. free(devDetail);
  62. SetupDiDestroyDeviceInfoList(hDevInfo);
  63. return INVALID_HANDLE_VALUE;
  64. }
  65. if (overlapped)
  66. {
  67. hidHandle = CreateFile(devDetail->DevicePath,
  68. GENERIC_READ | GENERIC_WRITE,
  69. FILE_SHARE_READ | FILE_SHARE_WRITE,
  70. NULL,
  71. OPEN_EXISTING,
  72. FILE_FLAG_OVERLAPPED,
  73. NULL);
  74. }
  75. else
  76. {
  77. hidHandle = CreateFile(devDetail->DevicePath,
  78. GENERIC_READ | GENERIC_WRITE,
  79. FILE_SHARE_READ | FILE_SHARE_WRITE,
  80. NULL,
  81. OPEN_EXISTING,
  82. 0,
  83. NULL);
  84. }
  85. free(devDetail);
  86. if (hidHandle==INVALID_HANDLE_VALUE)
  87. {
  88. SetupDiDestroyDeviceInfoList(hDevInfo);
  89. free(devDetail);
  90. return INVALID_HANDLE_VALUE;
  91. }
  92. _HIDD_ATTRIBUTES hidAttributes;
  93. if(!HidD_GetAttributes(hidHandle, &hidAttributes))
  94. {
  95. CloseHandle(hidHandle);
  96. SetupDiDestroyDeviceInfoList(hDevInfo);
  97. return INVALID_HANDLE_VALUE;
时间: 2024-10-14 21:49:19

Windows与自定义USB HID设备通信说明.的相关文章

C# 访问USB(HID)设备

原文:C# 访问USB(HID)设备 二话不说,直接给代码,如果您真想做这方面的东西,还是稍微研究下,没有现成的好类用,就需要自己了解其原理 //引用空间 using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Text; using System.Windows.Forms; using S

stm32 USB hid设备与PC进行双向数据传输时PC不识别USB设备

stm32 USB hid设备与PC进行双向数据传输时PC不识别USB设备,或者开始时识别,拔出后再插入就没有反应了,就连鼠标U盘也没有反应. 我的问题是,我安装了VMware虚拟机,并进行USB设备的分配,使得虚拟机系统也识别USB设备. 所以,解决问题的办法如下: 在<属性>中选择禁用. 然后把下面的VMware Workstation Server 也禁用了.重启电脑就OK了.

[BLE--HID]USB HID设备类定义

简述 HID设备,即人机交互设备,常见的有鼠标,键盘,游戏手柄,等等.一般有线方式都是通过USB连线连接到机器设备,作为用户输入设备.在蓝牙技术中,HID设备的接入就是无线的了.不过HID的相关定义当然还是一样的,不然设备驱动,兼容等问题都会有很多麻烦了.蓝牙中有HID, HOGP profile,但是只是在蓝牙数据通信上做的规范,HID具体含义相关,还是需要看USB相关的HID文档. USB设备类? USB设备可以在启动时,或者是在之后的连接状态下,都能够被主机进行配置.USB设备种类是非常多

C# 访问USB(HID)设备方法 (转)

依次使用一下函数: 1.static public extern void HidD_GetHidGuid (ref System.Guid HidGuid); 获取HID的全局GUID 2.static public extern IntPtr SetupDiGetClassDevs(ref System.Guid ClassGuid, string Enumerator, int hwndParent, int Flags); 获取所有hid句柄 3.static public extern

转 关于USB HID报告描述符

原文地址 USB HID报告及报告描述符简介 在USB中,USBHOST是通过各种描述符来识别设备的,有设备描述符,配置描述符,接口描述符,端点描述符,字符串描述符,报告描述符等等.USB报 告描述符(ReportDescriptor)是HID设备中的一个描述符,它是比较复杂的一个描述符.USBHID设备是通过报告来给传送数据的,报告 有输入报告和输出报告.输入报告是USB设备发送给主机的,例如USB鼠标将鼠标移动和鼠标点击等信息返回给电脑,键盘将按键数据数据返回给电脑等:输出 报告是主机发送在

USB HID报告及报告描述符简介

在USB中,USB HOST是通过各种描述符来识别设备的,有设备描述符,配置描述符,接口描述符,端点描述符,字符串描述符,报告描述符等等.USB报告描述符(Report Descriptor)是HID设备中的一个描述符,它是比较复杂的一个描述符.        USB HID设备是通过报告来给传送数据的,报告有输入报告和输出报告.输入报告是USB设备发送给主机的,例如USB鼠标将鼠标移动和鼠标点击等信息返回给电脑,键盘将按键数据数据返回给电脑等:输出报告是主机发送给USB设备的,例如键盘上的数字

C#与USB&nbsp;HID间的通信

原文:C#与USB HID间的通信 C#与USBHID接口的通讯相对于与串口间的通讯较为复杂,其中需要多次调用到Windows的一些API.其原理编者尚未全部理清,以下提供简单的USBHID通讯流程.(参考网友资料) 一.获取所有连接HID的设备信息. 1.通过一个空的GUID来获取HID的全局GUID. Guid HIDGuid = Guid.Empty; /// /// The HidD_GetHidGuid routine returns the device interface GUID

重新想象 Windows 8.1 Store Apps (90) - 通信的新特性: 通过 HttpBaseProtocolFilter 实现 http 请求的缓存控制,以及 cookie 读写; 自定义 HttpFilter; 其他

[源码下载] 作者:webabcd 介绍重新想象 Windows 8.1 Store Apps 之通信的新特性 通过 HttpBaseProtocolFilter 控制缓存逻辑,以及如何通过 HttpBaseProtocolFilter 管理 cookie 自定义 HttpFilter 其他 示例HTTP 服务端WebServer/HttpDemo.aspx.cs /* * 用于响应 http 请求 */ using System; using System.IO; using System.T

usb由于其配置信息(注册表中的)不完整或已损坏,Windows 无法启动这个硬件设备(代码 19)

今天解决了一晚上,USB驱动的问题: 在设备管理器的usb设备的属性中,显示提示“由于其配置信息(注册表中的)不完整或已损坏,Windows 无法启动这个硬件设备”.注册表坏了.经过查询,解决方法如下: 方法:打开注册表编辑器(开始-->运行-->regedit),依次展开HKEY_LOCAL_MACHINE/SYSTEM/CurrentControlSet/Control/Class/在这下面有很多用“{}”括起来的项,一个一个的点开,看右面空口有没有“通用串行总线控制器”这些文字,在右面窗