【linux驱动】定时器的使用

【linux驱动】定时器的使用

文章目录

  • 【linux驱动】定时器的使用
    • 1.介绍
      • 1.1相关名词
      • 1.2配置HZ的方法
    • 2.API
    • 3.示例
    • 4.调试

1.介绍

1.1相关名词

HZ、jiffies、tick

Linux系统启动后,每隔固定周期就会发出timer interrupt(IRQ 0),HZ用来定义每一秒发生多少次timer interrupt;
一般HZ的值并不确定,可以被修改设定;可供修改的值有:100HZ 200HZ 250HZ 300HZ 500HZ 1000HZ;
具体修改方法下面有
Tick是HZ的倒数,意即timer;
也就是发生一次中断的时间;比如HZ是250时,tick为1/250s,也就是4ms;
Jiffies为Linux核心变数(unsigned long)用来记录系统开启以来,发生的timer interrupt的次数;这里需要注意的就是32位的类型会出现溢出导致数据不准确(这个时间好像是30天还是40天,记不住,大概是这个时间);64位的类型就不会出现这个问题,溢出都要几百万年了;其他的倒是不用怎么关注,定时器的使用,程序里面都是套路;下面是一些其他的内容,提供给想了解的同志。

在这里插入图片描述

1.2配置HZ的方法

menuconfig

打开menuconfig配置界面

cd xxx/Linux-xxx.xxx 
make menuconfig

在这里插入图片描述

我的系统默认选择的是100HZ

其他选项:

在这里插入图片描述

.config

cd 到对应linux源码顶层目录下
vi .config

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
在这里插入图片描述

修改后,可以使用下面的命令:
esc 后  :wq!
要是只是看看,就直接esc :q!

2.API

头文件:

#include <linux/timer.h>

对应结构体:

    struct timer_list { struct hlist_node entry; //构成内核链表相关成员unsigned long  expires; //定时器到期时间void   (*function)(struct timer_list *); //定时器处理函数u32			flags; //一般填写为0};

初始化定时器:

方式1void timer_func(struct timer_list *timer){}mytimer.expires = jiffies+HZ;  //# define HZ CONFIG_HZ 定时1stimer_setup(&mytimer, timer_func, 0);方式2/*我的好像是linux版本太低,导致无法使用比较新的API,因此使用下面的方式*/void timer_func(struct timer_list *timer){}
init_timer(&mytimer);
mytimer.expires = jiffies + HZ/50;
mytimer.function = timer_func;  // 设置定时器到期时调用的回调函数还有其他的方式,这里就不一一赘述,有兴趣的可以多多探索

启动定时器:

    void add_timer(struct timer_list *timer)//启动定时器,定时器启动之后只会执行一次,add_timer只能调用一次,//如果第二次调用内核会崩溃int mod_timer(struct timer_list *timer, unsigned long expires)//功能再次启动定时器

删除定时器:

    int del_timer(struct timer_list *timer)//删除定时器

3.示例

key.h

#ifndef __KEY_CTRL_H__
#define __KEY_CTRL_H__#include <linux/timer.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/of_gpio.h>
#include <linux/device.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/io.h>
#include <linux/module.h>
#include <linux/uaccess.h>
#include <linux/gpio/consumer.h>
#include <linux/interrupt.h>
#include <linux/of_irq.h>
#include <linux/interrupt.h>
#include <linux/timer.h>#define KEY_NAME1 "KEY_USER1"
#define KEY_NAME2 "KEY_USER2"
#define CONSUMER_LABEL1 "user1"
#define CONSUMER_LABEL2 "user2"#define PATH_DTS_KEY_USR1 "/psd_key_irqs/user1"
#define PATH_DTS_KEY_USR2 "/psd_key_irqs/user2"#define KMD_ERR(str) \printk("%s %s line: %d %s \n", __FILE__, __FUNCTION__, __LINE__, str);typedef struct my_key {char *dev_name;struct device_node *key_node;unsigned int key_irq_no;int key_num;int key_status;//led开关状态int Level_state;//电平状态struct timer_list mytimer; // 分配定时器
} key_ctrl_t;/*key1设备树控制的初始化*/
int key1_ctrl_init(key_ctrl_t * key);
/*key2设备树控制的初始化*/
int key2_ctrl_init(key_ctrl_t * key);
/*led设备树控制的卸载处理函数*/
void key_ctrl_exit(key_ctrl_t * key);#endif

key.c

#include"key_ctrl.h"
#include <linux/timer.h>void my_timer_callback1(unsigned long para)
{printk("key1 down...\n");
}void my_timer_callback2(unsigned long para)
{printk("key2 down...\n");
}irqreturn_t key1_irq_handle(int irq, void* dev)
{key_ctrl_t *key1 = (key_ctrl_t *)dev;// 启动定时器mod_timer(&key1->mytimer, jiffies + HZ/50);return IRQ_HANDLED;
}irqreturn_t key2_irq_handle(int irq, void* dev)
{key_ctrl_t * key2 = (key_ctrl_t *)dev;// 启动定时器mod_timer(&key2->mytimer, jiffies + HZ/50);return IRQ_HANDLED;
}int key1_ctrl_init(key_ctrl_t * key)
{int ret;key->key_node = of_find_node_by_path(PATH_DTS_KEY_USR1);if(IS_ERR(key->key_node)){KMD_ERR("of_find_node_by_path ERR");ret = -ENODATA;goto exit;}// 2.解析得到软中断号key->key_irq_no = irq_of_parse_and_map(key->key_node, 0);if (key->key_irq_no == 0) {printk("irq_of_parse_and_map error\n");ret = -EAGAIN;goto exit_node;// 资源暂时不可用}init_timer(&key->mytimer);key->mytimer.expires = jiffies + HZ/50;// timer_setup_on_stack(&key->mytimer, my_timer_callback1, 0);key->mytimer.function = my_timer_callback1;  // 设置定时器到期时调用的回调函数add_timer(&key->mytimer);              //将定时器加入到系统定时器链表中ret = request_irq(key->key_irq_no, key1_irq_handle,IRQF_TRIGGER_LOW, KEY_NAME1, key);if (ret) {printk("request_irq key1 error\n");goto exit_irq;}return 0;
exit_irq:free_irq(key->key_irq_no, NULL);
exit_node:of_node_put(key->key_node);
exit:return ret;  // 出错返回
}int key2_ctrl_init(key_ctrl_t * key)
{int ret;key->key_node = of_find_node_by_path(PATH_DTS_KEY_USR2);if(IS_ERR(key->key_node)){KMD_ERR("of_find_node_by_path ERR");ret = -ENODATA;goto exit;}// 2.解析得到软中断号key->key_irq_no = irq_of_parse_and_map(key->key_node, 0);if (key->key_irq_no == 0) {printk("irq_of_parse_and_map error\n");of_node_put(key->key_node); // 清理已获取的节点ret = -EAGAIN;goto exit_node;// 资源暂时不可用}init_timer(&key->mytimer);        key->mytimer.expires = jiffies + HZ/50;// timer_setup(&key->mytimer, my_timer_callback2, 0);key->mytimer.function = my_timer_callback2;  // 设置定时器到期时调用的回调函数add_timer(&key->mytimer);                //将定时器加入到系统定时器链表中ret = request_irq(key->key_irq_no, key2_irq_handle,IRQF_TRIGGER_LOW, KEY_NAME2, key);if (ret) {printk("request_irq key2 error\n");goto exit_irq;}return 0;
exit_irq:free_irq(key->key_irq_no, NULL);
exit_node:of_node_put(key->key_node);
exit:return ret;  // 出错返回
}void key_ctrl_exit(key_ctrl_t * key)
{free_irq(key->key_irq_no,key);if (timer_pending(&key->mytimer)) {del_timer_sync(&key->mytimer);}}MODULE_DESCRIPTION("key_ctrl_driver");
MODULE_LICENSE("GPL");

4.调试

这里的按键防抖需要注意以下几点:首先不加定时器防抖看下,按键硬件是否已经存在防抖机制
其次,使用定时器的时候要确定linux的HZ是多少,确定一次timer irq是多久,再添加防抖代码
再一个就是一般来说,按键抖动会在20ms左右,
最后就是如果HZ是100hz、jiffies+1就是在计数的基础上延时10ms触发自定义的定时器中断函数

over,祝各位定时器使用愉快!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/754234.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Day50| 123 买卖股票的最佳时机III 188 买卖股票的最佳时机IV

目录 123 买卖股票的最佳时机III 188 买卖股票的最佳时机IV 123 买卖股票的最佳时机III class Solution { public:int maxProfit(vector<int>& prices) {vector<vector<int>> dp(prices.size() 1, vector<int>(5, 0));dp[0][0] 0;dp[0][…

模块化开发在不同编程语言中的实现方式有何异同?并以LabVIEW为例进行说明

模块化开发是一种软件设计方法&#xff0c;它将一个大型程序分解成独立的、可以单独开发和测试的模块或组件。这种方法提高了代码的可重用性、可维护性和可测试性。不同编程语言实现模块化开发的方式各有特色&#xff0c;但都遵循基本的设计原则&#xff0c;如封装、接口抽象和…

【机器学习】经典目标检测算法:RCNN、Fast RCNN、 Faster RCNN 基本思想和网络结构介绍

文章目录 三者的比较&#xff1a;RCNN、Fast RCNN、 Faster RCNN一、框架的对比1.三者都是二阶算法&#xff0c;网络框架比较&#xff1a;2.三者的优缺点比较&#xff1a; RCNN一、RCNN系列简介二、RCNN算法流程的4个步骤三、RCNN存在的问题四、论文解析补充1.R-CNN提出了两个问…

Odoo17免费开源ERP开发技巧:如何在表单视图中调用JS类

文/Odoo亚太金牌服务开源智造 老杨 在Odoo最新V17新版中&#xff0c;其突出功能之一是能够构建个性化视图&#xff0c;允许用户以独特的方式与数据互动。本文深入探讨了如何使用 JavaScript 类来呈现表单视图来创建自定义视图。通过学习本教程&#xff0c;你将获得关于开发Odo…

【ceph】配置 ceph dashboard 详细配置过程

本站以分享各种运维经验和运维所需要的技能为主 《python零基础入门》&#xff1a;python零基础入门学习 《python运维脚本》&#xff1a; python运维脚本实践 《shell》&#xff1a;shell学习 《terraform》持续更新中&#xff1a;terraform_Aws学习零基础入门到最佳实战 《k8…

C++ QT串口通信(1)-串口模块QtSerialPort详解

本文讲解C++ QT串口模块QtSerialPort。 目录 一、串口通信基础与QtSerialPort模块简介 1.1 串口通信基础 1.2 QtSerialPort模块简介

环境变量和Bash内置命令

Command Line Editing Ctrla#Move to the start of the line.(光标移到最前面) Ctrle#Move to the end of the line.(光标移到最后面) Ctrll#Clear the screen, reprinting the current line at the top.(不等同clear命令.会在顶部重新打印当前行,当前行还有内容时,还会显示) …

放慢音频速度的三个方法 享受慢音乐

如何让音频慢速播放&#xff1f;我们都知道&#xff0c;在观看视频时&#xff0c;我们可以选择快进播放&#xff0c;但是很少有软件支持慢速播放。然而&#xff0c;将音频慢速播放在某些情况下是非常必要的。例如&#xff0c;当我们学习一门新语言时&#xff0c;我们可以将音频…

Pytorch详细应用基础(全)

&#x1f525;博客主页&#xff1a; A_SHOWY&#x1f3a5;系列专栏&#xff1a;力扣刷题总结录 数据结构 云计算 数字图像处理 力扣每日一题_ 1.安装pytorch以及anaconda配置 尽量保持默认的通道&#xff0c;每次写指令把镜像地址写上就行。 defaults优先级是最低的&#…

动态代理IP在反爬虫策略中的实战运用与挑战

动态代理IP在反爬虫策略中的实战运用与挑战是现代网络数据抓取领域中一个核心议题。动态代理IP服务允许爬虫程序通过不断切换不同的IP地址来访问目标网站&#xff0c;以应对各种反爬虫技术措施&#xff0c;主要包括以下几点&#xff1a; 实战运用&#xff1a; 1. 绕过IP限制&a…

React——关于事件处理

如何注册事件 驼峰命名法&#xff0c;语法on事件名&#xff5b;事件处理程序&#xff5d; 比如onClick{this.buttonClick} class App extends React.Component {render() {return (<div><button onClick{this.buttonClick}>点击按钮触发事件</button></di…

深度学习神经网络相关记录《二》

如何判断模型是一个好模型&#xff1f; 模型预测效果&#xff0c;也就是模型预测的准确率运算速度&#xff1b;能够处理大量数据、短时间内急速学习、可以实时进行预测&#xff0c;是机器学习的重要优势&#xff1b;可解释性&#xff1b;深度学习已经不太关系这一点了&#xf…

macOS安装erlang以及rabbitMq详情版本

1.利用HomeBrew安装&#xff0c;如果你电脑没有HomeBrew可以跳转到HomeBrew安装教程 https://blog.csdn.net/weixin_50268501/article/details/136820299 2.要想运行rabbitMq要有Erlang 安装erlang brew install erlang3.安装RabbitMq brew install rabbitmq4.执行完上述命令…

基础:TCP三次握手做了什么,为什么要握手?

1. TCP 三次握手在做些什么 1. 第一次握手 &#xff1a; 1&#xff09;握手作用&#xff1a;客户端发出建立连接请求。 2&#xff09;数据处理&#xff1a;客户端发送连接请求报文段&#xff0c;将SYN位置为1&#xff0c;Sequence Number为x;然后&#xff0c;客户端进入SYN_S…

基于Matlab的视频人面检测识别,Matalb实现

博主简介&#xff1a; 专注、专一于Matlab图像处理学习、交流&#xff0c;matlab图像代码代做/项目合作可以联系&#xff08;QQ:3249726188&#xff09; 个人主页&#xff1a;Matlab_ImagePro-CSDN博客 原则&#xff1a;代码均由本人编写完成&#xff0c;非中介&#xff0c;提供…

分布式(计算机算法)

目录 分布式计算 分布式​编辑 分布式和集群 分布式和集群的应用场景 分布式应用场景 集群应用场景 哪种技术更优、更快、更好呢 性能 稳定性 以下概念来源于百度百科 分布式计算 分布式计算是近年提出的一种新的计算方式。所谓分布式计算就是在两个或多个软件互相共享信息…

【uniapp】uniapp设置改变全局字体大小功能:

文章目录 一、效果&#xff1a;二、文档&#xff1a;三、案例&#xff1a;引用插件&#xff0c;将项目中的字体单位rpx换成rem&#xff0c;结合官网提供的page-meta属性&#xff0c;通过控制根字体大小&#xff0c;达到调节字体大小要求。 一、效果&#xff1a; 二、文档&#…

Bash Shell中双引号中的感叹号问题详解

Bash Shell中双引号中的感叹号问题详解 在Bash Shell中&#xff0c;感叹号(!)是一个特殊字符&#xff0c;主要用于历史扩展。历史扩展允许你使用!来引用历史命令。然而&#xff0c;当你在双引号中使用感叹号时&#xff0c;如果你在双引号中直接使用感叹号&#xff0c;它可能会…

redis cpu百分百问题

系统使用spring cloud alibaba微服务框架&#xff0c;应用使用K8S发布&#xff0c;使用redis作为缓存数据库&#xff0c;运行一段时间之后开发反应早高峰时整个系统响应缓慢&#xff0c;排查发现服务网关gateway的pod健康状态频繁unhealthy&#xff0c;导致重启&#xff0c;且此…

冯米塞斯应力(von Mises stress)云图的MATLAB计算方法

关注 M r . m a t e r i a l , \color{Violet} \rm Mr.material\ , Mr.material