转自:https://blog.csdn.net/xiaopangzi313/article/details/52383226
版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/xiaopangzi313/article/details/52383226
通过前一节的分析得到,linux Input子系统上传数据本质上是将input_dev的数据,上报给input_handler,
当用户读入event时,驱动层只需要利用copy_to_user将数据传递至用户空间。当然,以上只是内核中Linux input
的机制,作为驱动工程师我们该如何使用input子系统呢?考虑到部分读者没有嵌入式设备,作者从一个虚拟嵌入式
设备的编写,描述Input subsystem 的使用。
实际的嵌入式设备中,用的input system的设备很多,如gsensor,psensor,touch panel等,本文着重模拟一个
virsual touch 设备,让其向上层空间报值,然后用户空间可以通过通用IO来捕获键值或者坐标。
一、 虚拟设备的编写流程
1.定义虚拟设备结构体
struct touch_dev{
struct platform_device *p_dev; //定义平台设备,这个不为必须
struct input_dev *input; //定义input设备结构体
int x; //定义坐标X
int y; //定义坐标Y,当然如果需要也可以添加键值
struct task_struct *run_thread; //定义内核线程,为了让用户空间随时都抓到坐标,我们可以开启线程
不断上报
};
2. 注册平台设备和驱动
platform_device_register_simple("v_touch",-1,NULL,0);
platform_device_register_simple("v_touch",-1,NULL,0);
3. probe 函数实现
kthread_run(vtouch_thread,vtouch_dev,"vtouch_thread");
4. 内核线程实现
static int vtouch_thread(void *data)
{
int x,y;
struct touch_dev *vtouch_dev = (struct touch_dev*)data;
printk(KERN_INFO "vtouch thread running\n");
do{
...
printk("vtouch thread report\n");
msleep(2000);
} while(!kthread_should_stop());//线程退出条件
return 0;
}
二、 input 设备的添加流程
1.在 probe中添加
//为input device申请内存
vtouch_dev->input = input_allocate_device();
//设置 vtouch 设备名称
vtouch_dev->input->name = "vtouch";
//设置设备支持坐标事件,包括X坐标,Y坐标事件,Z坐标事件。
set_bit(EV_ABS,vtouch_dev->input->evbit);
//对于X轴范围是-1024到+1024,数据误差是-2到+2,中心平滑位置是0
input_set_abs_params(vtouch_dev->input, ABS_X, -1024, 1024, 2, 0);
//同上
input_set_abs_params(vtouch_dev->input, ABS_Y, -1024, 1024, 2, 0);
//注册输入设备
ret = input_register_device(vtouch_dev->input);
if(ret < 0){
printk("%s register input device error\n",__func__);
goto input_register;
}
2.在线程中添加
//上报绝对坐标
input_report_abs(vtouch_dev->input,ABS_X,x);
input_report_abs(vtouch_dev->input,ABS_Y,y);
//上报同步通知
input_sync(vtouch_dev->input); //对于坐标必须添加
三、应用程序编写
int main(void)
{
struct input_event ev;
int count,x,y;
int fd = open(EVENT_DEV, O_RDWR);
while(1){
count = read(fd, &ev,sizeof(struct input_event));
if(EV_ABS == ev.type){
if(ev.code == ABS_X){
x = ev.value;
}else if(ev.code == ABS_Y){
y = ev.value;
}
printf("position: x=%d, y=%d\n",x,y);
}else if(EV_SYN == ev.type){
puts("sync!");
}
}
return 0;
}
整体源码如下:
驱动: input_simulate.c
应用: input_simulate_test.c
四、调试过程
1. 通过
cat /proc/bus/input/devices
查看与dev、input目录下的event对应的设备
2. ./a.out //运行应用程序
实验效果:
五、总结
就单纯使用而言,应用input subsystem 我们只需要做以下工作,
1. 定义input_device 并分配调用input_allocate_device(); 分配空间
2. 设置input_dev 支持的事件类型如:set_bit,input_set_abs_params
3. 调用input_register_device 注册input_dev
4. 调用input_report_abs,input_sync上报事件
---------------------
作者:xiaopangzi313
来源:CSDN
原文:https://blog.csdn.net/xiaopangzi313/article/details/52383226
版权声明:本文为博主原创文章,转载请附上博文链接!