蜕变成蝶~Linux设备驱动之按键设备驱动

在上述的驱动系列博客中,我们已经了解了关于阻塞和非阻塞、异步通知、轮询、内存和I/O口访问、并发控制等知识,按键设备驱动相对来说是比较简单的,本章内容可以加深我们对字符设备驱动架构、阻塞与非阻塞、中断定时器等相关知识的理解。在嵌入式的系统中,按键的硬件原理简单,就是通过一个上拉电阻将处理器的外部中断引脚拉高,电阻的另一端接按钮并接地就可以实现。
1.按键的确认流程如下

2 按键驱动中的有关数据结构
2.1 按键设备结构体以及定时器
#define MAX KEY BUF 16 // 键缓冲区大小
typedef unsigned char KEY RET; //设备结构体:
typedef struct
{
unsigned int keyStatus[KEY NUM]; //4个 键的 键状态
KEY RET buf[MAX KEY BUF]; // 键缓冲区
unsigned int head, tail; // 键缓冲区头和尾
wait queue head t wq; //等待队列
struct cdev cdev; //cdev 结构体
} KEY DEV; static struct timer list key timer[KEY NUM];//4个 键去抖定时器
2.2 按键硬件资源、键值信息结构体
static struct key info
{ int irq no; //中断号 unsigned int gpio port; //GPIO端口 int key no; //键值 } key info tab [4] =
{
/* 键所使用的CPU 资源*/
{ IRQ EINT10, GPIO G2, 1
}
,
{ IRQ EINT13, GPIO G5, 2
}
,
{ IRQ EINT14, GPIO G6, 3
}
,
{ IRQ EINT15, GPIO G7, 4
}
,
};
2.3 按键设备驱动文件操作结构体
static struct file operations s3c2410 key fops =
{ owner: THIS MODULE, open: s3c2410 key open, //启动设备 release: s3c2410 key release, //关闭设备 read: s3c2410 key read, //读取 键的键值
};
3 按键设备的模块加载和卸载函数
3.1 加载函数
static int init s3c2410 key init (void)
{
...//申请设备号,添加cdev request irqs(); //注册中断函数
keydev .head = keydev .tail = 0; //初始化结构体 for (i = 0; i < KEY NUM; i++) keydev.keyStatus[i] = KEYSTATUS UP; init waitqueue head (&(keydev .wq)); //等待队列 //初始化定时器,实现软件的去抖动 for (i = 0; i < KEY NUM; i++) setup timer (&key timer[i], key timer handler, i);
//把 键的序号作为传入定时器处理函数的参数
}
3.2 卸载函数
static void exit s3c2410 key exit (void)
{ free irqs(); //注销中断
...//释放设备号,删除cdev
}
3.3 中断申请函数
/*申请系统中断,中断方式为下降沿触发*/
static int request irqs(void)
{
struct key info *k;
int i;
for (i= 0; i < sizeof(key info tab) / sizeof(key info tab [1]); i++)
{
k = key info tab + i;
set external irq (k->irq no, EXT LOWLEVEL, GPIO PULLUP DIS);
//设置低电平触发
if (request irq (k->irq no, &buttons irq, SA INTERRUPT,
DEVICE NAME,
i)) //申请中断,将 键序号作为参数传入中断服务程序
{
return - 1;
}
}
return 0;
}
3.4 中断释放函数
/*释放中断*/
static void free irqs(void)
{
struct key info *k;
int i;
for (i= 0; i < sizeof(key info tab) / sizeof(key info tab [1]); i++)
{
k = key info tab + i;
free irq (k->irq no, buttons irq); //释放中断
}
}
4 按键设备驱动中断和定时器处理程序
在按键按下之后,将发生中断,在中断处理程序中,应该先关闭中断进去查询模式,延时以消抖如下中断处理过程只有顶半部,没有底半部。
4.1 中断处理程序
static void s3c2410 eint key (int irq, void *dev id, struct pt regs
*reg)
{ int key = dev id; disable irq (key info tab [key].irq no); //关中断,转入查询 式 keydev.keyStatus[key] = KEYSTATUS DOWNX;//状态为按下
_
key timer [key].expires == jiffies + KEY TIMER DELAY1;//延迟 add timer (&key timer[key]); //启动定时器
}
4.2 定时器处理流程
按键按下时,该按键将记录字啊缓冲区,同时定时器启动延时,每次记录新的键值时,等待队列被唤醒,其代码如下。
//按键设备驱动的定时器处理函数
static void key timer handler (unsigned long data)
{
int key = data; if (ISKEY DOWN (key))
{ if (keydev.keyStatus[key] == KEYSTATUS DOWNX)
//从中断进入
{ keydev .keyStatus[key] = KEYSTATUS DOWN; key timer[key].expires == jiffies + KEY TIMER DELAY; //延迟
keyEvent (); //记录键值,唤醒等待队列 add timer(&key timer [key]);
}
else
{ key timer[key].expires == jiffies + KEY TIMER DELAY; //延迟 add timer(&key timer [key]);
}
}
else //键已抬起
{ keydev.keyStatus[key] = KEYSTATUS UP; enable irq (key info tab [key].irq no);
}
5 打开和释放函数
这里主要是设置keydev.head和keydev.tail还有按键事件函数指针keyEvent的值,按键设备驱动的打开、释放函数如下:
static int s3c2410 key open (struct inode *inode, struct file *filp)
{
keydev .head = keydev .tail = 0; //清空 键动作缓冲区 keyEvent = keyEvent raw; //函数指针指向 键处理函数keyEvent raw
return 0;
} static int s3c2410 key release (struct inode *inode, struct file *filp)
{ keyEvent = keyEvent dummy; //函数指针指向空函数
return 0;
}
6 读函数
读函数主要是提供对按键设备结构体缓冲区的读并复制到用户空间,当keydev.head != keydev.tail时,说明缓冲区有数据,使用copy_to_user()函数拷贝到用户空间,反之根据用户空间是阻塞还是非阻塞读分为以下两种情况:
- 非阻塞读:没有按键缓存,直接返回- EAGAIN;
- 阻塞读:在keydev.wq等待队列上睡眠,直到有按键记录 到缓冲区后被唤醒。
//按键设备驱动的读函数
static ssize t s3c2410 key read (struct file *filp,char *buf,ssize t
count,
loff t*ppos)
{
retry: if (keydev.head != keydev .tail)
//当前循环队列中有数据
{
key ret = keyRead (); //读取按键
copy to user(..); //把数据从内核空间传送到用户空间
}
else
{
if (filp->f flags &O NONBLOCK)
//若用户采用非阻塞方式读取
{
return - EAGAIN;
}
interruptible sleep on (&(keydev .wq));
//用户采用阻塞方式读取,调用该函数使进程睡眠
goto retry;
}
return 0;
}
版权所有,转载请注明转载地址:http://www.cnblogs.com/lihuidashen/p/4498025.html
蜕变成蝶~Linux设备驱动之按键设备驱动的更多相关文章
- 蜕变成蝶~Linux设备驱动之字符设备驱动
一.linux系统将设备分为3类:字符设备.块设备.网络设备.使用驱动程序: 字符设备:是指只能一个字节一个字节读写的设备,不能随机读取设备内存中的某一数据,读取数据需要按照先后数据.字符设备是面向流 ...
- 蜕变成蝶~Linux设备驱动之watchdog设备驱动
看门狗(watchdog )分硬件看门狗和软件看门狗.硬件看门狗是利用一个定时器 电路,其定时输出连接到电路的复位端,程序在一定时间范围内对定时器清零 (俗称 “喂狗”),如果程序出现故障,不在定时周 ...
- 蜕变成蝶~Linux设备驱动之DMA
如果不曾相逢 也许 心绪永远不会沉重 如果真的失之交臂 恐怕一生也不得轻松 一个眼神 便足以让心海 掠过飓风 在贫瘠的土地上 更深地懂得风景 一次远行 便足以憔悴了一颗 羸弱的心 每望一眼秋水微澜 便 ...
- 蜕变成蝶~Linux设备驱动之CPU与内存和I/O
那是世上最远的距离 思念让我无法去呼吸 你的一动和一举 占据我心里 陪我每个孤独无尽的夜里 用我心中盛放的画笔 描绘你微笑时的绚丽 爱让人痛彻心底 我却不怀疑 你的存在是我生命的奇迹 感受你的每一次的 ...
- 蜕变成蝶~Linux设备驱动之中断与定时器
“我叮咛你的 你说 不会遗忘 你告诉我的 我也全部珍藏 对于我们来说 记忆是飘不落的日子 永远不会发黄 相聚的时候 总是很短 期待的时候 总是很长 岁月的溪水边 捡拾起多少闪亮的诗行 如果你要想念我 ...
- 蜕变成蝶~Linux设备驱动之异步通知和异步I/O
在设备驱动中使用异步通知可以使得对设备的访问可进行时,由驱动主动通知应用程序进行访问.因此,使用无阻塞I/O的应用程序无需轮询设备是否可访问,而阻塞访问也可以被类似“中断”的异步通知所取代.异步通知类 ...
- 蜕变成蝶~Linux设备驱动中的阻塞和非阻塞I/O
今天意外收到一个消息,真是惊呆我了,博客轩给我发了信息,说是俺的博客文章有特色可以出本书,,这简直让我受宠若惊,俺只是个大三的技术宅,写的博客也是自己所学的一些见解和在网上看到我一些博文以及帖子里综合 ...
- 蜕变成蝶~Linux设备驱动中的并发控制
并发和竞争发生在两类体系中: 对称多处理器(SMP)的多个CPU 内核可抢占的单CPU系统 访问共享资源的代码区域称为临界区(critical sections),临界区需要以某种互斥机制加以保护.在 ...
- 乾坤合一~Linux设备驱动之块设备驱动
1. 题外话 在蜕变成蝶的一系列学习当中,我们已经掌握了大部分Linux驱动的知识,在乾坤合一的分享当中,以综合实例为主要讲解,在一个月的蜕茧成蝶的学习探索当中,觉得数据结构,指针,链表等等占据了代码 ...
随机推荐
- (转)ConurrentHashMap和Hashtable的区别
集合类是Java API的核心,但是我觉得要用好它们是一种艺术.我总结了一些个人的经验,譬如使用ArrayList能够提高性能,而不再需要过时的Vector了,等等.JDK 1.5引入了一些好用的并发 ...
- PAT基础6-10
6-10 阶乘计算升级版 (20 分) 本题要求实现一个打印非负整数阶乘的函数. 函数接口定义: void Print_Factorial ( const int N ); 其中N是用户传入的参数,其 ...
- 穷举法、for循环、函数、作用域、斐波那契数
1.穷举法 枚举所有可能性,直到得到正确的答案或者尝试完所有值. 穷举法经常是解决问题的最实用的方法,它实现起来热别容易,并且易于理解. 2.for循环 for语句一般形式如下: for variab ...
- 转:2016年崛起的js项目
近几年 JS 社区创新和演化的速度是有目共睹的,几个月前比较时髦的技术很可能现在已经过时了. 2016 已经过去,你有没有担心错过了什么重要的内容?在这篇调查报告中我们会为你解读社区的主流趋势. 我们 ...
- jdk各种包安装方式
大家都知道,现在JAVA的发展可谓是如日中天,它覆盖面非常广泛,小到个人PC,大到商业应用都能见到它的身影.以前它是由SUN公司来维护的,现在已经归属到甲骨文旗下了. 今天我们来学习一下Java JD ...
- poj3253 Fence Repair(贪心+哈夫曼 经典)
https://vjudge.net/problem/POJ-3253 很经典的题,运用哈夫曼思想,想想很有道理!! 具体实现还是有点绕人,最后被long long卡了一下,看数据大小的时候单纯相乘了 ...
- hihocoder1148 February 29(区间闰年计数)
hihocoder1148https://hihocoder.com/problemset/problem/1148 因为题目没有给范围,我本来是这么写的. ; i <= ; i++){ ==& ...
- delphi button 实现下拉列表
unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms ...
- Unity中调用DLL库
DLL -- Dynamic Link Library(动态链接库文件),这里以Window平台为例. Unity支持的两种语言生成的DLL库(C++.C#),这里以C#为例,C++网上可以搜索很详细 ...
- AxWindowsMediaPlayer控件的使用
首先要知道如何将控件添加到工具箱中,步骤如下: “工具箱”中单击右键,选择“选择项”菜单,打开“选择工具箱项”窗口,选择“COM组件”标签,在列表中找到并勾选“Windows Media Player ...