如何成为一个HID设备(模拟键盘)
1、基础知识
通过《USB HID 设备类协议入门》一文和上一节的实例我们知道决定HID设备“身份”的因素有
1)5个标准描述符中与HID设备有关的部分有:
- 设备描述符中bDeviceClass、bDeviceSubClass和bDeviceProtocol三个字段的值必须为零。
- 接口描述符中bInterfaceClass的值必须为0x03,bInterfaceSubClass的值为0或1,为1表示HID设备符是一个启动设备(Boot Device,一般对PC机而言才有意义,意思是BIOS启动时能识别并使用您的HID设备,且只有标准鼠标或键盘类设备才能成为Boot Device。 bInterfaceProtocol的取值含义如下表所示:
HID接口描述符中bInterfaceProtocol的含义 bInterfaceProtocol的取值(十进制) 含义 0 NONE 1 鼠标 2 键盘 3~255 保留
2)HID设备的描述符除了5个USB的标准描述符(设备描述符、配置描述符、接口描述符、端点描述符、字符串描述符,见百合电子工作室的另一篇文章:USB开发基础--USB命令(请求)和USB描述符)外,还包括3个HID设备类特定描述符:HID描述符、报告描述符、实体描述符。
2、在上一节实例的基础上作一些修改来将Easy USB 51 Programer改造成键盘
1)下载上一节实例
2)修改接口描述符中bInterfaceProtocol的值为0x02
在Descriptor.c中找到以下代码
- 1, //bInterfaceProtocol为1代表鼠标
将其修改为
- 2, //bInterfaceProtocol为2代表键盘
3)在上一节中也提到“虽然我们将接口描述符中的bInterfaceProtocol设为1(代表鼠标),但这只是针对启动设备(Boot Device)而言有才有效果(即PC机的BIOS加载后能识别和使用),但真正对HID设备的数据流格式进行描述的是报告描述符,所以 bInterfaceProtocol的取值实际意义不大”,所以现在我们来修改报告描述符
在Descriptor.c中找到以下代码:
- code char MouseReportDescriptor[52] = {
- 0x05, 0x01, // USAGE_PAGE (Generic Desktop)
- 0x09, 0x02, // USAGE (Mouse)
- 0xa1, 0x01, // COLLECTION (Application)
- 0x09, 0x01, // USAGE (Pointer)
- 0xa1, 0x00, // COLLECTION (Physical)
- 0x05, 0x09, // USAGE_PAGE (Button)
- 0x19, 0x01, // USAGE_MINIMUM (Button 1)
- 0x29, 0x03, // USAGE_MAXIMUM (Button 3)
- 0x15, 0x00, // LOGICAL_MINIMUM (0)
- 0x25, 0x01, // LOGICAL_MAXIMUM (1)
- 0x95, 0x03, // REPORT_COUNT (3)
- 0x75, 0x01, // REPORT_SIZE (1)
- 0x81, 0x02, // INPUT (Data,Var,Abs)
- 0x95, 0x01, // REPORT_COUNT (1)
- 0x75, 0x05, // REPORT_SIZE (5)
- 0x81, 0x03, // INPUT (Cnst,Var,Abs)
- 0x05, 0x01, // USAGE_PAGE (Generic Desktop)
- 0x09, 0x30, // USAGE (X)
- 0x09, 0x31, // USAGE (Y)
- 0x09, 0x38, // USAGE (Wheel)
- 0x15, 0x81, // LOGICAL_MINIMUM (-127)
- 0x25, 0x7f, // LOGICAL_MAXIMUM (127)
- 0x75, 0x08, // REPORT_SIZE (8)
- 0x95, 0x03, // REPORT_COUNT (3)
- 0x81, 0x06, // INPUT (Data,Var,Rel)
- 0xc0, // END_COLLECTION
- 0xc0 // END_COLLECTION
- };
将其修改为
- code char MouseReportDescriptor[63] = {
- //表示用途页为通用桌面设备
- 0x05, 0x01, // USAGE_PAGE (Generic Desktop)
- //表示用途为键盘
- 0x09, 0x06, // USAGE (Keyboard)
- //表示应用集合,必须要以END_COLLECTION来结束它,见最后的END_COLLECTION
- 0xa1, 0x01, // COLLECTION (Application)
- //表示用途页为按键
- 0x05, 0x07, // USAGE_PAGE (Keyboard)
- //用途最小值,这里为左ctrl键
- 0x19, 0xe0, // USAGE_MINIMUM (Keyboard LeftControl)
- //用途最大值,这里为右GUI键,即window键
- 0x29, 0xe7, // USAGE_MAXIMUM (Keyboard Right GUI)
- //逻辑最小值为0
- 0x15, 0x00, // LOGICAL_MINIMUM (0)
- //逻辑最大值为1
- 0x25, 0x01, // LOGICAL_MAXIMUM (1)
- //报告大小(即这个字段的宽度)为1bit,所以前面的逻辑最小值为0,逻辑最大值为1
- 0x75, 0x01, // REPORT_SIZE (1)
- //报告的个数为8,即总共有8个bits
- 0x95, 0x08, // REPORT_COUNT (8)
- //输入用,变量,值,绝对值。像键盘这类一般报告绝对值,
- //而鼠标移动这样的则报告相对值,表示鼠标移动多少
- 0x81, 0x02, // INPUT (Data,Var,Abs)
- //上面这这几项描述了一个输入用的字段,总共为8个bits,每个bit表示一个按键
- //分别从左ctrl键到右GUI键。这8个bits刚好构成一个字节,它位于报告的第一个字节。
- //它的最低位,即bit-0对应着左ctrl键,如果返回的数据该位为1,则表示左ctrl键被按下,
- //否则,左ctrl键没有按下。最高位,即bit-7表示右GUI键的按下情况。中间的几个位,
- //需要根据HID协议中规定的用途页表(HID Usage Tables)来确定。这里通常用来表示
- //特殊键,例如ctrl,shift,del键等
- //这样的数据段个数为1
- 0x95, 0x01, // REPORT_COUNT (1)
- //每个段长度为8bits
- 0x75, 0x08, // REPORT_SIZE (8)
- //输入用,常量,值,绝对值
- 0x81, 0x03, // INPUT (Cnst,Var,Abs)
- //上面这8个bit是常量,设备必须返回0
- //这样的数据段个数为5
- 0x95, 0x05, // REPORT_COUNT (5)
- //每个段大小为1bit
- 0x75, 0x01, // REPORT_SIZE (1)
- //用途是LED,即用来控制键盘上的LED用的,因此下面会说明它是输出用
- 0x05, 0x08, // USAGE_PAGE (LEDs)
- //用途最小值是Num Lock,即数字键锁定灯
- 0x19, 0x01, // USAGE_MINIMUM (Num Lock)
- //用途最大值是Kana,这个是什么灯我也不清楚^_^
- 0x29, 0x05, // USAGE_MAXIMUM (Kana)
- //如前面所说,这个字段是输出用的,用来控制LED。变量,值,绝对值。
- //1表示灯亮,0表示灯灭
- 0x91, 0x02, // OUTPUT (Data,Var,Abs)
- //这样的数据段个数为1
- 0x95, 0x01, // REPORT_COUNT (1)
- //每个段大小为3bits
- 0x75, 0x03, // REPORT_SIZE (3)
- //输出用,常量,值,绝对
- 0x91, 0x03, // OUTPUT (Cnst,Var,Abs)
- //由于要按字节对齐,而前面控制LED的只用了5个bit,
- //所以后面需要附加3个不用bit,设置为常量。
- //报告个数为6
- 0x95, 0x06, // REPORT_COUNT (6)
- //每个段大小为8bits
- 0x75, 0x08, // REPORT_SIZE (8)
- //逻辑最小值0
- 0x15, 0x00, // LOGICAL_MINIMUM (0)
- //逻辑最大值255
- 0x25, 0xFF, // LOGICAL_MAXIMUM (255)
- //用途页为按键
- 0x05, 0x07, // USAGE_PAGE (Keyboard)
- //使用最小值为0
- 0x19, 0x00, // USAGE_MINIMUM (Reserved (no event indicated))
- //使用最大值为0x65
- 0x29, 0x65, // USAGE_MAXIMUM (Keyboard Application)
- //输入用,变量,数组,绝对值
- 0x81, 0x00, // INPUT (Data,Ary,Abs)
- //以上定义了6个8bit宽的数组,每个8bit(即一个字节)用来表示一个按键,所以可以同时
- //有6个按键按下。没有按键按下时,全部返回0。如果按下的键太多,导致键盘扫描系统
- //无法区分按键时,则全部返回0x01,即6个0x01。如果有一个键按下,则这6个字节中的第一
- //个字节为相应的键值(具体的值参看HID Usage Tables),如果两个键按下,则第1、2两个
- //字节分别为相应的键值,以次类推。
- //关集合,跟上面的对应
- 0xc0 // END_COLLECTION
- };
这们还需要将Descriptor.h中的以下代码
- extern code char MouseReportDescriptor[52];
修改为
- extern code char MouseReportDescriptor[63];
上面的报告描述符中只有一个报告,所以没有报告ID,
因此返回的都是实际使用的数据。总共有8字节输入,1字节输出。其中输入的
第一字节用来表示特殊按键,第二字节保留,后面的六字节为普通按键。如果
只有左ctrl键按下,则返回01 00 00 00 00 00 00 00(十六进制),如果
只有数字键1 按下,则返回00 00 59 00 00 00 00 00,如果数字
键1 和2 同时按下,则返回00 00 59 5A 00 00 00 00,如果
再按下左shift 键,则返回02 00 59 5A 00 00 00 00,
然后再释放1 键,则返回02 00 5A 00 00 00 00 00,
然后全部按键释放,则返回00 00 00 00 00 00 00 00。
这些数据(即报告)都是通过中断端点返回的。当按下Num Lock键时,PC会发送
输出报告,从报告描述符中我们知道,Num Lock的LED对应着输出报告的最低位,
当数字小键盘打开时,输出xxxxxxx1(二进制,打x的由其它的LED状态决定);
当数字小键盘关闭时,输出xxxxxxx0(同前)。取出最低位就可以控制数字键锁定LED了。
(注:以上说明摘自computer00的《USB HID报告及报告描述符简介》一文)
5)将Descriptor中如下语句
- 0x66,0x02, //设备制造商定的产品ID
修改为
- 0x66,0x03, //设备制造商定的产品ID
4)模拟NumLock键盘和Windows键
我们定义扩展板EXT-BOARD-A上的K1键对应Windows键(即USB HID Usage Table中定义的GUI键,Left GUI或Right GUI,我们这里就选Left GUI吧),而K2键盘对应NumLock键,当NumLock使能时应点亮NumLock指示灯,我定义D0为NumLock指示灯。
根据报告描述符的定义,再参考USB HID Usage Table,要模拟Windows键盘,应将送给主机的8个字节的第一个字节置为0x04,要模拟NumLock键,应将第三个字节置为0x53。如果要模拟NumLock指示灯,不应该在主控芯片里判断当到K2按下后就打开或熄灭LED,其正确的方法是:当主机接收到NumLock按键信息后,会根据系统当前NumLock的状态决定打开还是关闭LED指示灯,然后将这一信息通过传给设备(发送一个字节的数据给设置,最低位表示NumLock的状态)。
更改Main.c文件中的main函数为:
- void main()
- {
- unsigned char i = 0;
- signed char cKeyIn[8];
- static bit bKeyPressed = 0; //键按下标志,防止重入
- if (Init_D12()!=0) //初始化D12
- return; //如果初始化不成功,返回
- IT0 = 0; //外部中断0为电平触发方式
- EX0 = 1; //开外部中断0
- PX0 = 0; //设置外部中断0中断优先级
- EA = 1; //开80C51总中断
- P0 = 0;
- while(1)
- {
- usbserve(); //处理USB事件
- if(bEPPflags.bits.configuration)
- {
- //在这里添加端点操作代码
- if(bEPPflags.bits.ep2_rxdone ) //主端点接收到数据(从主机发往设备的数据)
- {
- bEPPflags.bits.ep2_rxdone = 0;
- //判断NumLock状态
- if(EpBuf[0] & 0x01) //EpBuf为接收缓冲
- {
- P0 = 0x01;
- }
- else
- {
- P0 = 0x00;
- }
- }
- K1 = 1; //P3.5
- K2 = 1; //P3.6
- for(i=0;i<100;i++); //延时
- if(~K1 & K2) //K1按下(模拟左Windows键)
- {
- if(!bKeyPressed)
- {
- bKeyPressed = 1;
- cKeyIn[0]=0x08;
- cKeyIn[1]=0; //保留
- cKeyIn[2]=0;
- cKeyIn[3]=0;
- cKeyIn[4]=0;
- cKeyIn[5]=0;
- cKeyIn[6]=0;
- cKeyIn[7]=0;
- D12_WriteEndpoint(5,8,cKeyIn); //发8个字节到PC机
- }
- }
- else if(K1 & ~K2) //K2按下(模拟NumLock键)
- {
- if(!bKeyPressed)
- {
- bKeyPressed = 1;
- cKeyIn[0]=0;
- cKeyIn[1]=0; //保留
- cKeyIn[2]=0x53;
- cKeyIn[3]=0;
- cKeyIn[4]=0;
- cKeyIn[5]=0;
- cKeyIn[6]=0;
- cKeyIn[7]=0;
- D12_WriteEndpoint(5,8,cKeyIn); //发8个字节到PC机
- }
- }
- else if(~K1 & ~K2) //K1和K2同时按下(Window和NumLock同时按下)
- {
- if(!bKeyPressed)
- {
- bKeyPressed = 1;
- cKeyIn[0]=0x08;
- cKeyIn[1]=0; //保留
- cKeyIn[2]=0x53;
- cKeyIn[3]=0;
- cKeyIn[4]=0;
- cKeyIn[5]=0;
- cKeyIn[6]=0;
- cKeyIn[7]=0;
- D12_WriteEndpoint(5,8,cKeyIn); //发8个字节到PC机
- }
- }
- else if(K1 & K2)
- {
- if(bKeyPressed)
- {
- bKeyPressed = 0;
- cKeyIn[0]=0;
- cKeyIn[1]=0; //保留
- cKeyIn[2]=0;
- cKeyIn[3]=0;
- cKeyIn[4]=0;
- cKeyIn[5]=0;
- cKeyIn[6]=0;
- cKeyIn[7]=0;
- D12_WriteEndpoint(5,8,cKeyIn); //发8个字节到PC机
- }
- }
- }
- }
- }
在测试这个例子时,按下EXT-BOARD-A上的K1键,会弹出开始菜单,按K2键,D0的状态会改变,同时原有键盘上的NumLock指示灯也会同EXT-BOARD-A上的D0状态同步,相反,按原有键盘上的NumLock键,D0的状态也会跟着改变。