一、输入系统的框架
二、分析输入设备
驱动程序上报的数据含义三项重要内容:
type:哪类?比如EV_KEY,按键类
code:哪个,比如KEY_A
value:值,如0按下,1松开,2长按
在linux中使用一个结构体叫做input_event来保存上报的数据(通常还有个表示时间的结构体)
可以使用hexdump来查看/dev/input/下节点的输入
可以使用cat /proc/bus/input/devices来查看输入系统的具体的设备信息
ev=b,代表二进制1011代表支持0号事件,1号事件,3号事件,可以通过linux内核源码查询
3号事件支持哪些事件
ABS=6e18000 0的二进制如下,它的每一位代表一个事件
0000011011100001100000000000000000000000000000000000000000000000
高位在前,低位在后,可以通过查看内核源码,找到他对应什么事件。
0003代表abs事件,后面代表他所支持的事件。
三、获取输入设备的信息
输入子系统支持完整的 API 操作
支持这些机制:阻塞、非阻塞、POLL/SELECT、异步通知。
#include<linux/input.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include<stdio.h>
int main(int argc,char **argv)
{
struct input_id id;
int err=0;;
unsigned int ev_bit[2];
int len=0;
if(argc != 2)
{
printf("usage:%s <dev>\n",argv[0]);
return -1;
}
int fd = open(argv[1],O_RDWR);
if(fd<0)
{
printf("open %s err\n",argv[1]);
perror("open:");
return -1;
}
err=ioctl(fd,EVIOCGID,&id);
通过宏命令EVIOCGID来得到参数信息将他保存到结构体struct input_id id中
if(0==err)
{
printf("bustype = 0x%x\n",id.bustype);
printf("vendor = 0x%x\n", id.vendor );
printf("product = 0x%x\n",id.product);
printf("version = 0x%x\n",id.version);
}
len=ioctl(fd,EVIOCGBIT(0,sizeof(ev_bit)),ev_bit);//通过宏命令EVIOCGBIT来得到信息将他保存到ev_bit,这个evbit里面的每一位代表所支持的事件,每一位的1代表支持,0代表不支持。
int bit=0;
int i=0;
unsigned char byte;
char *ev_names[] = {//查看内核输入子系统的源码可以找到所有的事件
"EV_SYN ",
"EV_KEY ",
"EV_REL ",
"EV_ABS ",
"EV_MSC ",
"EV_SW ",
"NULL ",
"NULL ",
"NULL ",
"NULL ",
"NULL ",
"NULL ",
"NULL ",
"NULL ",
"NULL ",
"NULL ",
"NULL ",
"EV_LED ",
"EV_SND ",
"NULL ",
"EV_REP ",
"EV_FF ",
"EV_PWR ",
};
if(len>0&&len<=sizeof(ev_bit))
{
for(i=0;i<len;i++)
{
for(bit=0;bit<8;bit++)//取出每一位然后判断是否为1
{
byte=((unsigned char*)ev_names)[i];
if(byte & (1<<bit))//将1每一次左移一位然后和byte作&运算可以得到这8个位里面所支持的事件位
{
printf("%s ",ev_names[i*8+bit]);//打印支持的事件,乘以8是为了得到上面字符串的的字符,因为每一次循环是8次
}
}
printf("\n");
}
}
return 0;
}