本节目标:

通过定时器来防止按键抖动,测试程序是使用上节的:阻塞操作的测试程序

一、引入

如下图所示,在没有定时器防抖情况下,按键没有稳定之前会多次进入中断,使得输出多个相同信息出来

按键波形图,如下所示:

问:如何消去按键抖动?

答:通过定时器延时10ms,然后每当按键进入中断时就更新定时器延时10ms,若延时10ms到了说明已经过了抖动范围,然后再打印按键电平信息

二、定时器结构体和函数介绍

我们先来看看两个全局变量:

jiffies: 是系统时钟,全局变量,默认每隔10ms加1

HZ:是每S的频率,通过系统时钟换算出来,比如每隔10ms加1,那么HZ就等于100。

2.1、定时器结构体timer_list

timer_list常用结构体成员如下所示:

1)data //传递到*function超时处理函数的参数,可以通过参数来获取信息

2)expires //定时器到期的时间,当expires小于等于jiffies时,这个定时器便到期并调用定时器超时处理函数,然后就不会再调用了,比如要使用10ms后到期,赋值(jiffies+HZ/100)即可

3)void (*function)(unsigned long) //定时器超时处理函数。

2.2、定时器常用函数

init_timer(struct timer_list*) //定时器初始化结构体函数,

add_timer(struct timer_list*) //往系统添加定时器,告诉内核有个定时器结构体

mod_timer(struct timer_list *, unsigned long jiffier_timerout) //修改定时器的超时时间为jiffies_timerout, 当expires小于等于jiffies时,便调用定时器超时处理函数。

timer_pending(struct timer_list *) //定时器状态查询,如果在系统的定时器列表中则返回1,否则返回0;

del_timer(struct timer_list*) //删除定时器,在本驱动程序出口函数sixth_drv_exit()里添加

三、修改驱动程序实现定时器消抖动并测试

3.1、首先定义一个定时器结构体:

static struct timer_list buttons_timer; //定义定时器结构体

3.2、在init入口函数中初始化定时器结构体:

init_timer(&buttons_timer); //初始化结构体

/*成员.data未使用

不需要定时器到期时间,所以成员.expires无需初始化,默认为0,由于小于等于jiffies,会进入一次定时器超时函数*/

buttons_timer. function= buttons_timer_ function;

add_timer(&buttons_timer); //告诉内核,有一个定时器

注:以上3步可以用函数 setup_timer(time,func,data)代替,该函数实现赋值并初始化定时器,比手动设置更方便

3.3、在exit出口函数中删除定时器:

del_timer(&buttons_timer); //删除定时器

3.4、定义全局变量*irq_dev_id并在中断服务函数中获取dev_id

struct pin_desc *irq_dev_id ; //定义全局变量获取dev_id

并修改中断服务函数:

static irqreturn_t buttons_irq (int irq, void *dev_id) //中断服务函数

{

irq_dev_id =(struct pin_desc *)dev_id; //获取引脚描述结构体

/*每产生一次中断,则更新定时器10ms超时 */

mod_timer(&buttons_timer, jiffies+HZ/100);

return IRQ_RETVAL(IRQ_HANDLED);

}

注意: jiffies+HZ/100 也可以直接换成 jiffies + msecs_to_jiffies(10),更加方便

3.5、超时函数

当10ms超时到了,进入定时器超时函数,处理*irq_dev_id来判断是哪个按键按下的

static void buttons_timer_function(unsigned long data) //定时器超时函数

{

unsigned int pin_val=0;

if(!irq_dev_id) //初始化时,由于定时器.expires成员=0,会进入一次,若irq_dev_id为0则退出

{

printk("expires: timer out\n");

return ;

}

pin_val=s3c2410_gpio_getpin(irq_dev_id->pin); //获取按键值

if(pin_val)

{

/*按下 (下降沿),清除0x80*/

key_val=irq_dev_id->pin_status&0xef;

}

else

{

/*没有按下(上升沿),加上0x80*/

key_val=irq_dev_id->pin_status|0x80;

}

even_press=1; //退出等待队列

wake_up_interruptible(&button_wait); //唤醒 中断

kill_fasync(&button_async, SIGIO, POLL_IN); //发送SIGIO信号给应用层

}

3.6、测试效果

如下图所示,我们运行测试程序,来快速按下按键试试:

四、代码总览

测试程序代码使用阻塞操作的测试程序。

驱动程序sixth.c代码:

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

static struct timer_list buttons_timer; //定义定时器结构体

struct pin_desc *irq_dev_id ; //定义全局变量获取dev_id

static struct class *sixthdrv_class;

static struct class_device *sixthdrv_class_devs;

/*定义互斥锁button_lock,被用来后面的down()和up()使用 */

static DECLARE_MUTEX(button_lock);

/* 声明等待队列类型中断 button_wait */

static DECLARE_WAIT_QUEUE_HEAD(button_wait);

/* 异步信号结构体变量 */

static struct fasync_struct * button_async;

/*

* 定义中断事件标志

* 0:进入等待队列 1:退出等待队列

*/

static int even_press=0;

/*

* 定义全局变量key_val,保存key状态

*/

static int key_val=0;

/*

*引脚描述结构体

*/

struct pin_desc{

unsigned int pin;

unsigned int pin_status;

};

/*

*key初始状态(没有按下): 0x81,0x82,0x83,0x84

*key状态(按下): 0x01,0x02,0x03,0x04

*/

struct pin_desc pins_desc[4]={

{S3C2410_GPF0,0x01 },

{S3C2410_GPF2, 0x02 },

{S3C2410_GPG3, 0x03 },

{S3C2410_GPG11,0x04},

} ;

int sixth_drv_class(struct inode *inode, struct file *file) //卸载中断

{

free_irq(IRQ_EINT0,&pins_desc[0]);

free_irq(IRQ_EINT2,&pins_desc[1]);

free_irq(IRQ_EINT11,&pins_desc[2]);

free_irq(IRQ_EINT19,&pins_desc[3]);

/*释放信号量*/

up(&button_lock);

return 0;

}

/*

* 确定是上升沿还是下降沿

*/

static irqreturn_t buttons_irq (int irq, void *dev_id) //中断服务函数

{

irq_dev_id =(struct pin_desc *)dev_id; //获取引脚描述结构体

/*每产生一次中断,则更新定时器10ms超时 */

mod_timer(&buttons_timer, jiffies+HZ/100);

return IRQ_RETVAL(IRQ_HANDLED);

}

static int sixth_drv_open(struct inode *inode, struct file *file)

{

if( file->f_flags & O_NONBLOCK ) //非阻塞操作,获取不到则退出

{

if(down_trylock(&button_lock) )

return -1;

}

else //阻塞操作,获取不到则进入休眠

{

down(&button_lock);

}

request_irq(IRQ_EINT0,buttons_irq,IRQT_BOTHEDGE,"S1",&pins_desc[0]);

request_irq(IRQ_EINT2, buttons_irq,IRQT_BOTHEDGE, "S2", &pins_desc[1]);

request_irq(IRQ_EINT11, buttons_irq,IRQT_BOTHEDGE, "S3", &pins_desc[2]);

request_irq(IRQ_EINT19, buttons_irq,IRQT_BOTHEDGE, "S4", &pins_desc[3]);

return 0;

}

static int sixth_drv_read(struct file *file, const char __user *buf, size_t count, loff_t * ppos)

{

if( file->f_flags & O_NONBLOCK ) //非阻塞操作,获取不到则退出

{

if(!even_press ) //没有按键按下

return -1;

}

/*阻塞操作,则直接进入休眠状态,直到有按键按下为止*/

/*进程 进入等待队列(休眠状态)*/

wait_event_interruptible(button_wait, even_press);

/*有按键按下,退出等待队列,上传key_val 给用户层*/

if(copy_to_user(buf,&key_val,sizeof(key_val)))

return EFAULT;

even_press=0;

return 0;

}

static unsigned sixth_poll(struct file *file, poll_table *wait)

{

unsigned int mask = 0;

poll_wait(file, &button_wait, wait); // 不会立即休眠

if (even_press)

mask |= POLLIN | POLLRDNORM;

return mask;

}

static int sixth_fasync (int fd, struct file *file, int on)

{

return fasync_helper(fd, file, on, & button_async); //初始化button_async结构体,就能使用kill_fasync()了

}

static struct file_operations sixth_drv_fops={

.owner = THIS_MODULE,

.open = sixth_drv_open,

.read = sixth_drv_read,

.release=sixth_drv_class, //里面添加free_irq函数,来释放中断服务函数

.poll = sixth_poll,

.fasync= sixth_fasync, //初始化异步信号函数

};

static void buttons_timer_function(unsigned long data) //定时器超时函数

{

unsigned int pin_val=0;

if(!irq_dev_id) //定时器.expires成员=0,会进入一次,若irq_dev_id为0则退出

{

printk("expires: timer out\n");

return ;

}

pin_val=s3c2410_gpio_getpin(irq_dev_id->pin);

if(pin_val)

{

/* 按下 (下降沿),清除0x80*/

key_val=irq_dev_id->pin_status&0xef;

}

else

{

/*没有按下(上升沿),加上0x80*/

key_val=irq_dev_id->pin_status|0x80;

}

even_press=1; //退出等待队列

wake_up_interruptible(&button_wait); //唤醒 中断

kill_fasync(&button_async, SIGIO, POLL_IN); //发送SIGIO信号给应用层

}

volatile int sixth_major;

static int sixth_drv_init(void)

{

init_timer(&buttons_timer); //初始化定时器

buttons_timer. function= buttons_timer_function; //定时器超时函数

add_timer(&buttons_timer); //添加到内核中

sixth_major=register_chrdev(0,"sixth_drv",&sixth_drv_fops); //创建驱动

sixthdrv_class=class_create(THIS_MODULE,"sixth_dev"); //创建类名

sixthdrv_class_devs=class_device_create(sixthdrv_class, NULL, MKDEV(sixth_major,0), NULL,"buttons");

return 0;

}

static int sixth_drv_exit(void)

{

unregister_chrdev(sixth_major,"sixth_drv"); //卸载驱动

class_device_unregister(sixthdrv_class_devs); //卸载类设

class_destroy(sixthdrv_class); //卸载类

del_timer(&buttons_timer); //删除定时器

return 0;

}

module_init(sixth_drv_init);

module_exit(sixth_drv_exit);

MODULE_LICENSE("GPL v2");

}

linux添加定时器防抖,linux驱动2.3按键中断-定时器防抖相关推荐

  1. linux驱动开发5 按键中断实验(定时器和中断)

    led:IO的输出 :key:IO的输入 法一:直接读写IO 使用while(1)无限读取,但CPU占用达到了99.6%,所以不行 #include <linux/types.h>#inc ...

  2. linux添加windows网络打印机,Linux Mint如何添加windows分享的网络打印机?

    1.安装samba sudo apt-get install samba 2.找到系统打印机选项 通过 Menu-->>控制中心-->>系统管理找到 Printers选项,双击 ...

  3. 嵌入式linux添加环境变量,Linux环境变量

    前言: 1. 环境变量是操作系统环境设置的变量,适用于整个系统的用户进程: 2. 环境变量可以在命令中设置,但是用户注销的时候将会丢失这些设置值: 3. 若要重复适用,则最好在.profile中定义: ...

  4. linux添加qcc,Alpine Linux添加Let’s Encrypt CA证书或者自签CA证书

    Alpine Linux添加Let's Encrypt CA证书或者自签CA证书 1. 用docker进行查看 进入容器, docker run -it alpine:latest /bin/sh 安 ...

  5. linux 添加宋体字符集,Linux下安装中文宋体

    1,#cd /usr/share/fonts/default 2,mkdir -p ./truetype/simsun 3,取得simsun.ttc文件:如果网上下载不到则在windows (c:/w ...

  6. 龙芯2k按键中断驱动_字符设备驱动-高级篇按键中断程序驱动

    驱动源码: #include "linux/module.h" #include"linux/kernel.h" #include"linux/fs. ...

  7. arch linux添加用户,Arch Linux配置教程

    以arch的基本系统为基础,我们可以对其进行各种配置操作,让其更符合个人喜好.下面介绍了一些常用的配置. 1.用户管理 新安装的arch只有一个root用户,使用root用户来进行日常系统管理是很危险 ...

  8. linux添加变量6,Linux下查看和添加环境变量

    $PATH:决定了shell将到哪些目录中寻找命令或程序,PATH的值是一系列目录,当您运行一个程序时,Linux在这些目录下进行搜寻编译链接. 编辑你的 PATH 声明,其格式为: PATH=$PA ...

  9. 满足lisheng需求 linux添加环境变量 linux批处理 2016.04.25小结

    看了下博客上的日期,有11天没有更新博客了,我觉得这个还是个比较适合我的习惯,还是要坚持下去!对学习知识的记忆非常好!那就从今天开始,重新开始博客! 1.昨天重新改了credit_results的字段 ...

最新文章

  1. Activity 启动模式以及常见的启动Flag
  2. 三十九、SPSS神器界面功能介绍,计算变量和个案计数和加权
  3. CVPR 2020 三篇有趣的论文解读
  4. 如何通过 Excel import 的方式导入测试数据到 SAP Commerce Cloud 服务器
  5. c malloc 头文件_干货笔记 | C/C++笔试面试详细总结(二)
  6. android切换到上个页面,Android 返回上一个界面刷新数据
  7. python——函数 11、命名空间
  8. 行云管家堡垒机 导入腾讯云主机
  9. 前后落差大用什么词语_形容落差很大的成语_四字词语 - 成梦词典
  10. iOS 人民币符号与日圆符号的混淆
  11. pytorch autograd.grad
  12. Java开发笔记(一百三十五)Swing的文件对话框
  13. 微信公众号吸粉软件助你快速增加粉丝数和文章阅读量!
  14. 一个国企老兵给后辈们的忠告:三十岁之前远离国企
  15. 长尾理论 推荐系统长尾理论
  16. uvalive4987
  17. HTML基本标签归纳总结
  18. oracle8616,ORACLE11G-数据库备份恢复之RMAN全库备份恢复
  19. 荣耀MagicBook Pro性能测试,“秀”出硬实力
  20. nginx正向代理转发https出现502

热门文章

  1. 属于程序员的黄金五年,把握这 5 年,或将迎来美好的职场生活!
  2. 零基础学习什么编程语言比较合适?别的不说,听说大佬都学了这个!
  3. 我编程很渣,但我就是喜欢编程,我该怎么做?放弃还是继续坚持?
  4. 等午饭吃过后的dwzjzx
  5. android 如何叠加view,如何将另一个叠加项添加到android mapview
  6. 我的世界python写游戏_快来试试Python写的游戏《我的世界》
  7. 重磅!这个生信神器助你文章秒出图——miRNA与基因互作数据库
  8. 小学生都学Python了,你还不知道怎么开始
  9. monterey系统怎么降级?macOS Monterey系统降回Big Sur的详细教程
  10. php算次方,php怎么计算几次方