以前写的一些输入设备的驱动都是采用字符设备处理的。问题由此而来,Linux开源社区的大神们看到了这大量输入设备如此分散不堪,有木有可以实现一种机制,可以对分散的、不同类别的输入设备进行统一的驱动,所以才出现了输入子系统。

输入子系统引入的好处:

(1)统一了物理形态各异的相似的输入设备的处理功能。例如,各种鼠标,不论PS/2、USB、还是蓝牙,都被同样处理。

(2)提供了用于分发输入报告给用户应用程序的简单的事件(event)接口。你的驱动不必创建、管理/dev节点以及相关的访问方法。因此它能够很方便的调用输入API以发送鼠标移动、键盘按键,或触摸事件给用户空间。X windows这样的应用程序能够无缝地运行于输入子系统提供的event接口之上。

(3)抽取出了输入驱动的通用部分,简化了驱动,并提供了一致性。例如,输入子系统提供了一个底层驱动(成为serio)的集合,支持对串口和键盘控制器等硬件输入的访问。

子系统包括一前一后运行的两类驱动:输入事件(event)驱动和输入设备(device)驱动。

输入事件驱动负责和应用程序的接口;

而输入设备驱动负责和底层输入设备的通信。

输入事件驱动和输入设备驱动都可以利用输入子系统的高效、可重用的核心提供的服务。

而我们需要实现的就是输入设备驱动。

首先在linux内核源代码的drivers/input/input.c中就是输入子系统,它的file_operation中只有一个open函数。而在open函数中却对file_operation结构体进行了重新指定,从而实现read、write等其他功能。

驱动程序代码:

#include <linux/module.h>
#include <linux/version.h> #include <linux/init.h>
#include <linux/fs.h>
#include <linux/interrupt.h>
#include <linux/irq.h>
#include <linux/sched.h>
#include <linux/pm.h>
#include <linux/sysctl.h>
#include <linux/proc_fs.h>
#include <linux/delay.h>
#include <linux/platform_device.h>
#include <linux/input.h>
#include <linux/irq.h>
#include <asm/gpio.h> struct pin_desc *pin_timer;
static struct timer_list keys_timer; static struct input_dev *keys_dev;
struct pin_desc{
int irq;
char *name;
unsigned int pin;
unsigned int key_val;
}; struct pin_desc pins_desc[] = {
{ IRQ_EINT0, "key1", S3C2410_GPF0, KEY_L },
{ IRQ_EINT2, "key2", S3C2410_GPF2, KEY_S },
{ IRQ_EINT11, "key3", S3C2410_GPG3, KEY_ENTER },
{ IRQ_EINT19, "key4", S3C2410_GPG11,KEY_LEFTSHIFT }, }; static irqreturn_t keys_irq(int irq, void *dev_id)
{
pin_timer = (struct pin_desc *)dev_id;
mod_timer(&keys_timer, jiffies+HZ/ );
return IRQ_HANDLED; } static void keys_timer_fun(unsigned long t)
{
struct pin_desc *pindesc = (struct pin_desc *)pin_timer;
unsigned int pinval;
if( !pindesc )
return ;
pinval = s3c2410_gpio_getpin(pindesc->pin);
if(pinval)
{
input_event(keys_dev,EV_KEY,pindesc->key_val,); /* 有事件产生时上报事件 */
input_sync(keys_dev);
}
else
{
input_event(keys_dev,EV_KEY,pindesc->key_val,); /* 有事件产生时上报事件 */
input_sync(keys_dev);
}
} static int key_init(void)
{
int i;
/* 分配一个input_dev结构体*/
keys_dev = input_allocate_device();
if (!keys_dev)
return -ENOMEM; /*设置*/
set_bit(EV_KEY, keys_dev->evbit);
set_bit(EV_REP, keys_dev->evbit); set_bit(KEY_L,keys_dev->keybit);
set_bit(KEY_S,keys_dev->keybit);
set_bit(KEY_ENTER,keys_dev->keybit);
set_bit(KEY_LEFTSHIFT,keys_dev->keybit);
/*注册*/
input_register_device(keys_dev);
/*硬件相关的操作*/
for(i=;i<;i++)
{
request_irq( pins_desc[i].irq, keys_irq, IRQT_BOTHEDGE, pins_desc[i].name, &pins_desc[i] );
}
init_timer(&keys_timer);
keys_timer.function = keys_timer_fun;
keys_timer.expires = ;
add_timer(&keys_timer);
return ;
}
static void key_exit(void)
{
int i;
for( i=; i<; i++ )
{
free_irq( pins_desc[i].irq, &pins_desc[i] );
}
del_timer(&keys_timer);
input_unregister_device(keys_dev);
printk("key Module exit\n");
} module_init( key_init );
module_exit( key_exit );
MODULE_LICENSE("GPL");

会在/dev/目录下对一个event设备。

测试:

运行 cat /dev/tty1

sd

嵌入式Linux驱动学习之路(十六)输入子系统的更多相关文章

  1. 嵌入式Linux驱动学习之路(十八)LCD驱动

    驱动代码: /************************************************************************* > File Name: lcd ...

  2. 嵌入式Linux驱动学习之路(十五)按键驱动-定时器防抖

    在之前的定时器驱动程序中,我们发现在连续按下按键的时候,正常情况下应该是一次按下对应一次松开.而程序有时候会显示是两次按下,一次松开.这个问题是因为在按下的时候,因为是机械按键,所以电压信号会产生一定 ...

  3. 嵌入式Linux驱动学习之路(十四)按键驱动-同步、互斥、阻塞

    目的:同一个时刻,只能有一个应用程序打开我们的驱动程序. ①原子操作: v = ATOMIC_INIT( i )  定义原子变量v并初始化为i atomic_read(v)        返回原子变量 ...

  4. 嵌入式Linux驱动学习之路(十二)按键驱动-poll机制

    实现的功能是在读取按键信息的时候,如果没有产生按键,则程序休眠在read函数中,利用poll机制,可以在没有退出的情况下让程序自动退出. 下面的程序就是在读取按键信息的时候,如果5000ms内没有按键 ...

  5. 嵌入式Linux驱动学习之路(十)字符设备驱动-my_led

    首先贴上代码: 字符设备驱动代码: /** *file name: led.c */#include <linux/sched.h> #include <linux/signal.h ...

  6. 嵌入式Linux驱动学习之路(十九)触摸屏驱动、tslib测试

    触摸屏使用流程: 1. 按下产生中断. 2.在中断处理程序中启动AD转换XY坐标. 3.AD转换结束并产生AD中断. 4. 在AD的中断处理函数中上报信息,启动定时器. 5. 定时器时间到后进入中断, ...

  7. 嵌入式Linux驱动学习之路(二十一)字符设备驱动程序总结和块设备驱动程序的引入

    字符设备驱动程序 应用程序是调用C库中的open read write等函数.而为了操作硬件,所以引入了驱动模块. 构建一个简单的驱动,有一下步骤. 1. 创建file_operations 2. 申 ...

  8. 嵌入式Linux驱动学习之路(二十六)DM9000C网卡驱动程序

    基于DM9000C的原厂代码修改dm9000c的驱动程序. 首先确认内存的基地址 iobase. 确定中断号码. 打开模块的初始化函数定义. 配置内存控制器的相应时序(结合DM9000C.C的手册). ...

  9. 嵌入式Linux驱动学习之路(二十五)虚拟网卡驱动程序

    一.协议栈层次对比 设备无关层到驱动层的体系结构 1).网络协议接口层向网络层协议提供提供统一的数据包收发接口,不论上层协议为ARP还是IP,都通过dev_queue_xmit()函数发送数据,并通过 ...

随机推荐

  1. What does "size" in int(size) of MySQL mean?

    What does "size" in int(size) of MySQL mean? https://alexander.kirk.at/2007/08/24/what-doe ...

  2. C#语言和数据库基础

    第一章 第一个C#程序 Vs2012密钥 RBCXF-CVBGR-382MK-DFHJ4-C69G8 01..net和C#的区别 大概在2000年,微软推出了一种革命性的产品--.NET(战略) 目标 ...

  3. Netty(一)引题

    本文介绍Java BIO(同步阻塞IO),伪异步IO,NIO(非阻塞IO),AIO(异步IO)这四种IO的情况,并对不同IO模型作比较. 目录 1.BIO 2.伪异步IO 3.NIO 4.AIO 5. ...

  4. jQuery带tab切换搜索框样式代码

    效果体验:http://hovertree.com/texiao/jquery/23/ 代码如下,保存到HTML文件也可以查看效果: <!DOCTYPE html> <html la ...

  5. Javascript中call,apply,bind方法的详解与总结

    在 javascript之 this 关键字详解 文章中,谈及了如下内容,做一个简单的回顾: 1.this对象的涵义就是指向当前对象中的属性和方法. 2.this指向的可变性.当在全局作用域时,thi ...

  6. 阻止默认事件event.preventDefault();

    阻止浏览器默认事件.什么是默认事件,例如浏览器默认右键菜单.a标签默认连接跳转...,如何阻止呢? Firefox中,event必须作为参数传入.  IE中,event是window对象的属性. ev ...

  7. C#读取ini文件的方法

    最近项目用到ini文件,读取ini文件,方法如下: using System; using System.Collections.Generic; using System.Linq; using S ...

  8. [转]利用JitPack发布自己项目让别人可以在dependencies中compile的简单方法

    JitPack的地址为:https://jitpack.io/ 发布 登陆 jitpack 在编辑框中输入github的网址,完成后点击LookUp 使用 点击 Get it. 之后网页就会自动导航到 ...

  9. 丰富eclipse注解的内容

    如何丰富eclipse注解的内容 eclipse -> Window -> Preferences -> Code Templates -> Comments (Comment ...

  10. Linux 克隆虚拟机引起的“Device eth0 does not seem to be present, delaying initialization”

    虚拟机Vmware上克隆了一个Red Hat Enterprise Linx启动时发现找不到网卡,如下所示,如果你在命令窗口启动网络服务就会遇到"Device eth0 does not s ...