input子系统
input子系统:
像按键、键盘、鼠标、触摸屏、游戏摇杆等设备只有输入没有输出,而且在编程实现其对应的驱动程序时会有很多重复性的代码,内核的设计者将该部分代码抽象出来,驱动工程师只需要复用该部分代码,并且实现硬件相关的代码(中断号,中断触发条件),就可以可以很容易实现对应硬件的驱动程序 如何用复用input子系统中提供的通用函数功能模块, 要遵循以下步骤:
核心数据结构
struct input_dev
{
evbit//记录该设备将来会报告哪些事件
}
1)分配一个输入设备//定义一个input_dev类型变量
input_allocate_device()
2)设置input_dev
3)注册input_dev
input_register_device
4)硬件操作
注册中断服务程序
延时去抖
//保存按键值
5)向input核心模块报告事件
input_event(...)
6)注销input_dev
input_unregister_device
7)释放输入设备
input_free_device(...)
代码实例:
#include <linux/init.h>
#include <linux/module.h>
#include <linux/input.h>
#include <linux/irq.h>
#include <linux/interrupt.h>
#include <asm/gpio.h>
#include <plat/gpio-cfg.h>
//定义按键硬件私有数据结构
struct btn_resource {
char *name; //名称
int irq; //中断号
int gpio; //GPIO编号
int code; //键值
}; //初始化开发板按键信息
static struct btn_resource btn_info[] = {
[] = {
.name = "KEY_L",
.irq = IRQ_EINT(),
.gpio = S5PV210_GPH0(),
.code = KEY_L
},
[] = {
.name = "KEY_S",
.irq = IRQ_EINT(),
.gpio = S5PV210_GPH0(),
.code = KEY_S
},
[] = {
.name = "KEY_ENTER",
.irq = IRQ_EINT(),
.gpio = S5PV210_GPH0(),
.code = KEY_ENTER
},
}; //定义input_dev指针
static struct input_dev *btn_dev; //分配定时器
static struct timer_list btn_timer;
static struct btn_resource *pdata; //定时器的处理函数
static void btn_timer_func(unsigned long data)
{
unsigned int pinstate; //2.获取按键的状态
pinstate = gpio_get_value(pdata->gpio); //3.上报按键信息给核心层然后唤醒休眠的进程
if (pinstate == ) { //松开
//EV_KEY:上报按键类事件
//pdata->code:具体键值
//0:松开
input_event(btn_dev, EV_KEY, pdata->code, );
input_sync(btn_dev); //上报同步类事件
} else { //按下
//EV_KEY:上报按键类事件
//pdata->code:具体键值
//1:按下
input_event(btn_dev, EV_KEY, pdata->code, );
input_sync(btn_dev);//上报同步类事件
}
} //中断处理函数
static irqreturn_t button_isr(int irq, void *dev_id)
{
//1.获取按键对应的数据项
pdata = (struct btn_resource *)dev_id; //2.启动定时器,设置定时器的超时时间为10ms
mod_timer(&btn_timer, jiffies + msecs_to_jiffies());
return IRQ_HANDLED;
} static int btn_init(void)
{
int i; //1.分配input_dev
btn_dev = input_allocate_device(); //2.初始化input_dev
btn_dev->name = "wf_button";
//2.1设置上报按键类事件
set_bit(EV_KEY, btn_dev->evbit);
//2.2设置上报重复类事件
set_bit(EV_REP, btn_dev->evbit);
//2.3设置上报按键类事件中的哪些键值
for(i = ; i < ARRAY_SIZE(btn_info); i++)
set_bit(btn_info[i].code, btn_dev->keybit); //3.注册input_dev
input_register_device(btn_dev); //4.申请GPIO资源
//5.注册中断处理函数
for (i = ; i < ARRAY_SIZE(btn_info); i++) {
gpio_request(btn_info[i].gpio, btn_info[i].name);
request_irq(btn_info[i].irq, button_isr/*中断处理函数*/,
IRQF_TRIGGER_FALLING|IRQF_TRIGGER_RISING,/*下降沿和上升沿触发*/
btn_info[i].name, &btn_info[i]/*给中断处理函数传递的参数*/);
} //6.初始化定时器 处理按键的毛刺
init_timer(&btn_timer); //6.1指定定时器的处理函数
btn_timer.function = btn_timer_func;
add_timer(&btn_timer);//将定时器添加到内核中
return ;
} static void btn_exit(void)
{
int i; //1.释放中断,释放GPIO资源
for (i = ; i < ARRAY_SIZE(btn_info); i++) {
gpio_free(btn_info[i].gpio);
free_irq(btn_info[i].irq, &btn_info[i]);
} //2.卸载input_dev
input_unregister_device(btn_dev); //3.释放input_dev内存
input_free_device(btn_dev); //4.删除定时器
del_timer(&btn_timer);
} module_init(btn_init);
module_exit(btn_exit);
MODULE_LICENSE("GPL");
测试代码:
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <linux/input.h> //input_event,标准键值等 int main(int argc, char *argv[])
{
int fd;
struct input_event button; fd = open(argv[], O_RDWR); //./btn_test /dev/input/event3
if (fd < ) {
printf("open button failed.\n");
return -;
} while() {
read(fd, &button, sizeof(button));
printf("type = %#x, code = %#x, value = %#x\n",
button.type, button.code, button.value);
} close(fd);
return ;
}
input子系统的更多相关文章
- linux kernel input 子系统分析
Linux 内核为了处理各种不同类型的的输入设备 , 比如说鼠标 , 键盘 , 操纵杆 , 触摸屏 , 设计并实现了一个对上层应用统一的试图的抽象层 , 即是Linux 输入子系统 . 输入子系统的层 ...
- input子系统详解
一.初识linux输入子系统 linux输入子系统(linux input subsystem)从上到下由三层实现,分别为:输入子系统事件处理层(EventHandler).输入子系统核心层(Inpu ...
- 鼠标驱动之-sys节点-input子系统
首先需要了解sys节点和linux驱动编程的知识,在linux内核<linux/>下有着对应的实现.本例实现创建sys节点,外围程序通过input子系统控制鼠标位置. 第一步编写驱动代码, ...
- linux 输入子系统(4)---- input子系统的初始化
Input子系统的初始化函数为input_init(),如下: static int __init input_init(void) { int err; input_init_abs_bypass( ...
- input子系统 KeyPad-Touch上报数据格式与机制
-----------------------------------------------------------------------本文系本站原创,欢迎转载!转载请注明出处:http://b ...
- input子系统分析
------------------------------------------ 本文系本站原创,欢迎转载! 转载请注明出处:http://ericxiao.cublog.cn/ -------- ...
- Android驱动之 Linux Input子系统之TP——A/B(Slot)协议
将A/B协议这部分单独拿出来说一方面是因为这部分内容是比较容易忽视的,周围大多数用到input子系统的开发人员也不甚理解:另一方面是由于这部分知识一旦扩展到TP(触摸屏Touch Panel)的多点触 ...
- 基于input子系统的sensor驱动调试(一)
要想弄明白世界的本质,就要追根溯源:代码也是一样的道理: 最近调试几个sensor驱动,alps sensor驱动.compass sensor驱动.G-sensor驱动都是一样的架构: 一.基于in ...
- 基于input子系统的sensor驱动调试(二)
继上一篇:http://www.cnblogs.com/linhaostudy/p/8303628.html#_label1_1 一.驱动流程解析: 1.模块加载: static struct of_ ...
随机推荐
- POJ 3276 Face The Right Way 反转
大致题意:有n头牛,有些牛朝正面,有些牛朝背面.现在你能一次性反转k头牛(区间[i,i+k-1]),求使所有的牛都朝前的最小的反转次数,以及此时最小的k值. 首先,区间反转的顺序对结果没有影响,并且, ...
- PHP内核探索之变量(7)- 不平凡的字符串
切,一个字符串有什么好研究的. 别这么说,看过<平凡的世界>么,平凡的字符串也可以有不平凡的故事.试看: (1) 在C语言中,strlen计算字符串的时间复杂度是?PHP中呢? ...
- String类型中ToString hashCode equals compareTo等方法的经典实现
private final char value[]; private int hash; // Default to 0 public String(String original) { this. ...
- jquery.datepair日期时分秒选择器
jquery.datepair是一个轻量级的jQuery插件,智能选择日期和时间范围,灵感来自于谷歌日历.Datepair将保持开始和结束日期/时间同步,并可以根据用户的操作设置默认值.该插件不提供任 ...
- WPF实现强大的动态公式计算
数据库可以定义表不同列之间的计算公式,进行自动公式计算,但如何实现行上的动态公式计算呢?行由于可以动态扩展,在某些应用场景下将能很好的解决实际问题. 1.VS2012新建一个WPF应用程序WpfApp ...
- Tourist.js – 简单灵活的操作指南和导航插件
Tourist.js 是一个基于 Backbone 和 jQuery 开发的轻量库,帮助你在应用程序创建简单易用的操作指南和导航功能.相比网站,它更适合用于复杂的,单页网站类型的应用程序.Touris ...
- CSS盒子模型
2016-10-22 <css入门经典>第6章 1.每个HTML元素对应于一个显示盒子,但不是所有的元素都显示在屏幕上. 2.HTML元素显示为CSS显示盒子的真正方法称为"可视 ...
- Kotlin:Android世界的Swift
转自:http://www.infoq.com/cn/news/2015/06/Android-JVM-JetBrains-Kotlin Kotlin是一门与Swift类似的静态类型JVM语言,由Je ...
- 操作系统开发系列—解释typedef void (*int_handler) ();
于是我换了一个思路来理解这个typedef 我们首先看常规的变量定义: int INT//定义了一个名为INT的int型变量. char *c//定义了一个名为c的char型指针变量 void(*Fu ...
- Android 数据库 LiteOrm 的使用
1.什么是 LiteOrm LiteOrm是一个小巧.强大.比系统自带数据库操作性能快1倍的 android ORM 框架类库,开发者一行代码实现数据库的增删改查操作,以及实体关系的持久化和自动映射. ...