一.测试中断或者Bulk传输:

  首先要使用Libusb打印出HID设备的Endpoint查看是否支持中断或者Bulk传输模式;如果支持的话才可以进一步测试;

  因为HID设备在插入的时候无需安装,并且一般会被OS直接占用,所以如果直接使用Interrupt传输(通常只有一个断点)会发生超时

  所以建议使用zadig给对应的HID设备安装WinUSB来防止直接被占用,然后就可以收到数据了:

  对于收到的数据该如何解析(以及如果想要修改Device来发送不同的字母)可以查看:HID键盘对照表   和 字母的ASCII码表

  这是我的一部分解析数据的代码(传入参数就是接收缓冲区的首地址指针)

 1 void parse_mouse_data(char* data)
2 {
3 char c1 = *data;
4 char c2 = *(data + 1);
5 char c3 = *(data + 2);
6 char c4 = *(data + 3);
7 if (c1 & LEFT_PRESS)
8 printf("Left key press\n");
9 if (c1 & RIGHT_PRESS)
10 printf("Right Key Press\n");
11 if (c1 & MIDLE_PRESS)
12 printf("Middle Key Press\n");
13 if (c2 & LEFT_SIDE)
14 printf("Go Left : %d\n", (c2 & 0x7f));
15 else
16 printf("Go Right : %d\n", (c2 & 0x7f));
17 if (c3 & DOWN_SIDE)
18 printf("Go Down : %d\n", (c3 & 0x7f));
19 else
20 printf("Go Up : %d\n", (c3 & 0x7f));
21 }
22 void parse_keyboard_data(char* data)
23 {
24 char temp1 = *data;
25 if (temp1 & (0x1 << 0))
26 printf("Left Control Presed");
27 if (temp1 & (0x1 << 1))
28 printf("Left Shift Pressed\n");
29 if (temp1 & (0x1 << 2))
30 printf("Left Alt Pressed\n");
31 if (temp1 & (0x1 << 4))
32 printf("Right Control Pressed\n");
33 if (temp1 & (0x1 << 5))
34 printf("Right Shift Pressed\n");
35 if (temp1 & (0x1 << 6))
36 printf("Right Alt Pressed\n");
37 char alpha = data + 3;
38 printf("input: %c\n", (char)((int)alpha + ASCII_LOWERCASE_SUB));
39 }

二.如果失败了之后之后测试就会一直失败的问题:

  查找资料:如果不是EP0端点传输时,Device收到了不支持或者无效的请求,对应的EP将会在data或者status阶段返回STALL故障 —— 需要Host的EP0发送clear Halt指令才能重新启用这个EP( 也就是需要使用Libusb提供的clear_halt函数)

三. HID report

  这是标志一个设备是HID的重要方式:report数组:

  注意要区分:报告和报告描述符:前者是用来传输数据(在get report和set report命令,控制EP中传输)后者是对数据用途的说明,是一个数组;

  可以根据 HID设备描述符的官方文档 来写自己的HID report array;当然更方便的是结合使用官方提供的 Dt工具 生成更保险;

  是我对device,interface,endpoint的修改:

 1 /*
2 * for maxPakcetSize0 for device descriptor
3 * it means the mas packet size for endpoint 0
4 * for low: 8; for full: 8 or 16, 32, 65; for high : only can be 64
5 */
6 device_descriptor{
7 .bcdUSB = 0x0210, //表示这个是HID设备
8 .bDeviceClass = 0x00,
9 .bDeviceSubClass = 0x00,
10 .bDeviceProtocol = 0x00,
11 }
12
13 /*
14 * bConfiguration Value must >= 0x01
15 * if == 0x00, after set configuration, enter not configured state
16 * bmAttributes in Config desc
17 * for usb2.0 bit7 reserved must 1
18 * bit6 = 1:means self-powered
19 * bit5 = 1:means Remote Wakeup
20 * rest must be 0
21 */
22 config_descriptor{
23 .bmAttributes = 0xA0,
24 .bMaxPower = 0x32,
25 }
26
27 /*
28 * interfaceClass: class code : delivered by USB-IF; == bDeviceClass
29 * 0x03 means HID
30 * iInterface == 0 means there is no string descriptor
31 */
32 interface_descriptor{
33 .bInterfaceClass = 0x03,
34 .bInterfaceSubClass = 0x00,
35 .bInterfaceProtocol = 0x00,
36 .iInterface = 0x00,
37 }
38
39 /*
40 * for hid descriptor type == 0x21
41 * bDesccriptorType: 0x21 means HID;
42 * bDescriptorType0 : Lower level desciptor : 0x22 means report
43 */
44 hid_descriptor{
45 .bDescriptorType = 0x21,
46 .bDescriptorType0 = 0x22,
47 .wDescriptorLength0 = REPORT_DESC_SIZE,
48 }
49
50 // 下面是使用的HID report结构:
51 0x05, 0x0c, // USAGE PAGE 使用 Consumer Devices 这样就不会被OS直接占用,可以进行bulk等传输
52 0x09, 0x01, // USAGE 使用 Consumer Control
53 0xa1, 0x01, // Collection 选择 Application;
54 0x09, 0x00, // USAGE 使用 Unassigned定义自己的report;
55 0xc0 // End of Collection

四. Get. Set report 指令的使用:

  这一部分才是使用的精髓:之前使用多个都是中断传输;但是HID设备更常用或者说更应该使用控制节点传输的report命令:

  因为前者中断传输算是异步:需要另外开辟一个线程来等待监听;但是控制节点的类似于同步,对于Host来说更方便:

  参考:

  

  

  所以两者分别是 0xa1 0x01 和 0x21 0x09

  然后通过控制节点传输的方法就可以传输了

  并且注意:如果是对于HID设备传输时候(在Windows中)一定要将buffer的第一个Byte写为在report 数组中龟腚的ReportID(默认是0x00)才能被传输

使用libusb中的interrupt_transfer如果判断device是hid设备将会转给hid_transfer函数,所以将ReportID设置为第一个字符非常重要。

五. hidapi踩坑:

  相比起Libusb:Hidapi并没有caim interface;以及判断是否被占用和clear_halt等命令;对于mouse之类的无能为力;

  并且提供的set report函数功能也很有限;不建议使用;但是比较简单。

  

使用Libusb和hidapi测试HID设备的更多相关文章

  1. android usb Host模式下与usb Hid 设备的通信

    做android 与USB HID设备的通信有段时间了,总结一下遇到的问题和解决方法: 1,第一次遇到的问题:android 版本低不支持usb hid, 被要求做相关项目的时候,就从mUsbMana ...

  2. 使用LIBUSB实现和自定义通讯设备通讯--MFC代码在末尾

    LIBUSB是一款简单好用的USB通讯开发库,一般HID设备用该库通讯能大大降低开发周期,使用如下,首先需要为设备安装驱动 在libusb的bin目录下有一个inf_wirzed.exe的文件,该文件 ...

  3. C# 访问USB(HID)设备

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

  4. HIDKomponente使用读写Hid设备一瞥

    HIDKomponente 是delphi中使用的第三方Hid控件库,可以检测.控制连接到电脑的Hid设备.一般情况下多为usb设备.HIDKomponente的使用实际上很简单,只是因为第一次使用, ...

  5. 第五章 HID设备

    5.1 HID介绍 为简化USB设备的开发过程,USB提出了设备类的概念.所有设备类都必须支持标准USB描述符和标准USB设备请求.如果有必要,设备类还可以自行定义其专用的描述符和设备请求,这分别被称 ...

  6. Android USB Host 与 Hid 设备通信bulkTransfer()返回-1问题的原因

    近期一直在做Android USB Host 与USB Hid设备(STM32FXXX)的通信,遇到了很多问题.项目源码以及所遇到的其他问题可以见本博客其他相关文章,这里重点讲一下bulkTransf ...

  7. [Android] hid设备按键流程简述

    hexdump /dev/hidraw0就能看到usbhid设备传输过来的裸流 如:按下Input键 003ae60 0000 0096 8000 006b 0000 0000 0000 0000 * ...

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

    1 .   所使用的典型 Windows API CreateFile ReadFile WriteFile 以下函数是 DDK 的内容: HidD_SetFeature HidD_GetFeatur ...

  9. USB自定义HID设备实现-LPC1768

    首先在之前鼠标的基础上修改设备描述符 #include "usbdesc.h" //usb标准设备描述符 const U8 USB_DeviceDescriptor[] = { U ...

  10. USB自定义HID设备实现-STM32

    该文档使用USB固件库,在其基础上进行了自己的定制,完成了一个USB-HID设备,首先是usb_desc.c文件,里面存放了usb各种描述符的存在 #include "usb_desc.h& ...

随机推荐

  1. 如何找到CSDN中关注的用户和粉丝?

    如何找到CSDN中关注的用户和粉丝? 刚刚在CSDN个人账号里找了半天都没找到自己关注的人 对CSDN的页面更新感到很迷, 个人账号管理很不人性化, 或者说是根本找不到自己关注的用户以及关注自己的粉丝 ...

  2. Grafana 系列文章(六):Grafana Explore 中的日志

    ️URL: https://grafana.com/docs/grafana/latest/explore/logs-integration/#labels-and-detected-fields D ...

  3. NetCoreWebApi3.0-------MiniProfiler使用教程

    参考博客:ASP.NET Core WebAPI中的分析工具MiniProfiler - LamondLu - 博客园 (cnblogs.com) 注意事项: 1.不要盲目copy别人的代码 var ...

  4. python学习day04

    1.基本数据类型之布尔值bool 1.用来判断事物的对错,是否可行,用于流程控制中 2.只有两种状态: True:对的.真的.可行的 False:错的.假的.不可行的 3.python中所有的数据都自 ...

  5. BIO和NIO的基本用法和API讲解

    1 BIO 可以理解为Blocking IO 是同步阻塞的IO,也就是说,当有多个请求过来的时候,请求会呈现为链状结构,遵循先进先出的原则 1.1 单线程版本 1.1.1 服务端 //服务端单线程处理 ...

  6. 使用NAT网络模式搭建内网,修改IP地址

    使用NAT网络模式搭建内网,修改IP地址 首先说明一下虚拟机的三种联网方式: 桥接模式: 简单来说就是使这台虚拟机成为一台在互联网中的有独立IP的一台新设备和Mac地址(不够都是虚拟的) NAT模式: ...

  7. GPS地图生成04之数据预处理

      1. 引言¶   下载的轨迹数据来源真实,并非特意模拟的轨迹数据,所以质量问题十分严重,进行预处理就显得尤为重要   2. 裁剪¶   我们将下载的岳麓山轨迹数据加载入QGIS,并使用OSM作为底 ...

  8. Postgresql模板数据库之template1 和 template0

    一.简介 template1和template0是PostgreSQL的模板数据库.所谓模板数据库就是创建新database时,PostgreSQL会基于模板数据库制作一份副本,其中会包含所有的数据库 ...

  9. PostgreSQL维护年龄的处理

    1.错误信息 WARNING: database "postgres" must be vacuumed within 3330803 transactions 最常见的方法是通过 ...

  10. java学习日记20230302-字符

    JAVA字符 char c1 = 97 System.out.println(c1)// a 会输出97代表的字符(字符编码) 字符类型细节: 字符常量用单引号 java中允许使用\转义字符代表一个字 ...