【北京迅为】《i.MX8MM嵌入式Linux开发指南》-第三篇 嵌入式Linux驱动开发篇-第六十一章 Linux内核定时器

i.MX8MM处理器采用了先进的14LPCFinFET工艺,提供更快的速度和更高的电源效率;四核Cortex-A53,单核Cortex-M4,多达五个内核 ,主频高达1.8GHz,2G DDR4内存、8G EMMC存储。千兆工业级以太网、MIPI-DSI、USB HOST、WIFI/BT、4G模块、CAN、RS485等接口一应俱全。H264、VP8视频硬编码,H.264、H.265、VP8、VP9视频硬解码,并提供相关历程,支持8路PDM接口、5路SAI接口、2路Speaker。系统支持Android9.0(支持获取root限)Linux4.14.78+Qt5.10.1、Yocto、Ubuntu20、Debian9系统。适用于智能充电桩,物联网,工业控制,医疗,智能交通等,可用于任何通用工业和物联网应用、

【公众号】迅为电子

【粉丝群】258811263


六十一Linux内核定时器

本章导读

定时器是我们最常用到的功能,一般用来完成定时功能,本章我们就来学习一下 Linux 内核提供的定时器 API 函数,通过这些定时器 API 函数我们可以完成很多要求定时的应用。Linux内核也提供了短延时函数,比如微秒、纳秒、毫秒延时函数,本章我们就来学习一下这些和时间有关的功能。

61.1章节讲解了Linux内核时间管理的基础知识

61.2章节讲解了内核定时器简介和相关函数介绍

61.3章节以实验的形式进行Linux内核定时器实验,实现内核定时器每隔一秒打印“This is timer_function \n”。

本章内容对应视频讲解链接(在线观看):

内核定时器  https://www.bilibili.com/video/BV1Vy4y1B7ta?p=40

程序源码在网盘资料“iTOP-i.MX8MM开发板\02-i.MX8MM开发板网盘资料汇总(不含光盘内容)\嵌入式Linux开发指南(iTOP-i.MX8MM)手册配套资料\2.驱动程序例程\018-Linux内核定时器”路径下。

61.1 Linux 内核时间管理

时间管理在内核中占有非常重要的地位。相对于事件驱动,内核中有大量的函数都是基于时间驱动的。

内核必须管理系统的运行时间以及当前的日期和时间。

周期产生的事件都是由系统定时器驱动的。系统定时器是一种可编程硬件芯片,它已固定频率产生中

断。该中断就是所谓的定时器中断,它所对应的中断处理程序负责更新系统时间,还负责执行需要周期性

运行的任务。系统定时器和时钟中断处理程序是 Linux 系统内核管理机制中的中枢。

61.1.1 内核中的时间概念

硬件为内核提供了一个系统定时器用以计算流逝的时间,系统定时器以某种频率自行触发时钟中断,该频率可以通过编程预定,称节拍率。当时钟中断发生时,内核就通过一种特殊中断处理程序对其进行处

理。内核知道连续两次时钟中断的间隔时间。这个间隔时间称为节拍(tick),内核就是靠这种已知的时钟

中断来计算墙上时间和系统运行时间。墙上时间即实际时间,内核提供了一组系统调用以获取实际日期和

实际时间。系统运行时间——自系统启动开始所经过的时间——对用户和内核都很有用,因为许多程序都必

须清楚流逝过的时间。

61.1.2 节拍率

系统定时器频率是通过静态预处理定义的,也就是 HZ,在系统启动时按照 Hz 对硬件进行设置。体系结构不同,HZ 的值也不同。内核在文件 <include/asm-generic/param.h> 中定义了 HZ 的实际值,节拍率就是 HZ,周期为 1/HZ。一般 ARM 体系结构的节拍率多数都等于 100,3399默认为1000。在编译 Linux 内核的时候可以通过图形化界面设置系统节拍率,按照如下路径打开配置界面:

-> Kernel Features

-> Timer frequency (<choice> [=y])

 选中“Timer frequency”,打开以后如下图所示:、

 从上图可以看出可选的系统节拍率为 100Hz、200Hz、250Hz、300Hz、500Hz 和 1000Hz,默认情况下

选择 1000Hz。设置好以后打开 Linux 内核源码根目录下的.config 文件,在此文件中有如下图所示定义:

上图中的 CONFIG_HZ 为 1000,Linux 内核会使用 CONFIG_HZ 来设置自己的系统时钟。打开文件

include/asm-generic/param.h,有如下内容:

# undef HZ

# define HZ CONFIG_HZ

# define USER_HZ 100

# define CLOCKS_PER_SEC (USER_HZ)

第 7 行定义了一个宏 HZ,宏 HZ 就是 CONFIG_HZ,因此 HZ=1000,我们后面编写 Linux 驱动的时候

会常常用到 HZ,因为 HZ 表示一秒的节拍数,也就是频率。

61.1.3 jiffies

全局变量 jiffies 用来记录自系统启动以来产生的节拍的总数。启动时,内核将该变量初始化为 0,此后,

每次时钟中断处理程序都会增加该变量的值。因为一秒内时钟中断的次数等于 Hz,所以 jiffes 一秒内增加的值也就为 Hz,系统运行时间以秒为单位计算,就等于 jiffes/Hz。jiffes=seconds*HZ。jiffies 定义在文件include/linux/jiffies.h 中,定义如下:

 extern u64 __jiffy_data jiffies_64;

extern unsigned long volatile __jiffy_data jiffies;

函数

作用

time_after(unkown, known)

unkown 通常为 jiffies,known 通常是需要对比的值。

time_before(unkown, known)

time_after_eq(unkown, known)

time_before_eq(unkown, known)

如果 unkown 超过 known 的话,time_after 函数返回真,否则返回假。如果 unkown 没有超过 known

的话 time_before 函数返回真,否则返回假。time_after_eq 函数和 time_after 函数类似,只是多了判断等

于这个条件。同理,time_before_eq 函数和 time_before 函数也类似。

为了方便开发,Linux 内核提供了几个 jiffies 和 ms、us、ns 之间的转换函数,如下所示:

函数

作用

int jiffies_to_msecs(const unsigned long j)

将 jiffies 类型的参数 j 分别转换为对应的毫秒、微秒、纳秒。

int jiffies_to_usecs(const unsigned long j)

u64 jiffies_to_nsecs(const unsigned long j)

long msecs_to_jiffies(const unsigned int m)

将毫秒、微秒、纳秒转换为 jiffies 类型。

long usecs_to_jiffies(const unsigned int u)

unsigned long nsecs_to_jiffies(u64 n)

举例来说

<1>定时10ms

计算:jiffies +msecs_to_jiffies(10)

<2>定时10us

计算:jiffies +usecs_to_jiffies(10)

61.2 内核定时器简介

定时器,有时也称为动态定时器或内核定时器——是管理内核时间的基础。定时器的使用很简单。只需要执行一些初始化工作,设置一个超时时间,指定超时发生后执行的函数,然后激活定时器就可以了。指定的函数将在定时器到期时自动执行。定时器并不周期运行,它在超时后就自行销毁,这也正是这种定时器被称为动态定时器的一个原因。

Linux 内核使用 timer_list 结构体表示内核定时器,timer_list 定义在文件 include/linux/timer.h 中,定

义如下:

struct timer_list

{

    struct list_head entry;

    unsigned long expires/* 定时器超时时间,单位是节拍数 */

    struct tvec_base *base;

    void (*function)(unsigned long); /* 定时处理函数 */

    unsigned long data;              /* 要传递给 function 函数的参数 */

    int slack;

};

要使用内核定时器首先要先定义一个 timer_list 变量,表示定时器,tiemr_list 结构体的 expires 成员变量表示超时时间,单位为节拍数。比如我们现在需要定义一个周期为 2 秒的定时器,那么这个定时器的超时时间就是 jiffies+(2*HZ),因此 expires=jiffies+(2*HZ)。function 就是定时器超时以后的定时处理函数,我们要做的工作就放到这个函数里面,需要我们编写这个定时处理函数。

定义好定时器以后还需要通过一系列的 API 函数来初始化此定时器,这些函数如下:

函数

作用

void init_timer(struct timer_list *timer)

初始化定时器

#define DEFINE_TIMER(_name, _function, _expires, _data)

该宏会静态创建一个名叫 timer_name 内核定时器,并初始化其 function, expires, name 和 base 字段。

void add_timer(struct timer_list *timer)

向 Linux 内核注册定时器,使用 add_timer 函数向内核注册定时器以后,定时器就会开始运行

int del_timer(struct timer_list * timer)

删除一个定时器

int del_timer_sync(struct timer_list *timer)

等待其他处理器使用完定时器再删除

int mod_timer(struct timer_list *timer,unsigned long expires)

修改定时值,如果定时器还没有激活的话,mod_timer 函数会激活定时器

#define DEFINE_TIMER(_name, _function, _expires, _data)

_name

变量名

_function

超时处理函数

_data

传递给超时处理函数的参数

_expires

到点时间,一般在启动定时前需要重新初始化。

功能

静态定义结构体变量并且初始化初始化function, expires, data 成员。

内核定时器一般的使用流程如下所示:

struct timer_list timer; /* 定义定时器 */

/* 定时器回调函数 */

void function(unsigned long arg)

{

/*

* 定时器处理代码

*/

/* 如果需要定时器周期性运行的话就使用 mod_timer

* 函数重新设置超时值并且启动定时器。

*/

mod_timer(&dev->timertest, jiffies + msecs_to_jiffies(2000));

}

/* 初始化函数 */

void init(void)

{

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

timer.function = function; /* 设置定时处理函数 */

timer.expires=jffies + msecs_to_jiffies(2000);/* 超时时间 2 秒 */

timer.data = (unsigned long)&dev; /* 将设备结构体作为参数 */

add_timer(&timer); /* 启动定时器 */

}

/* 退出函数 */

void exit(void)

{

del_timer(&timer); /* 删除定时器 */

/* 或者使用 */

del_timer_sync(&timer);

}

有时候我们需要在内核中实现短延时,尤其是在 Linux 驱动中。Linux 内核提供了毫秒、微秒和纳秒

延时函数,这三个函数如下所示:

函数

作用

void ndelay(unsigned long nsecs)

纳秒、微秒和毫秒延时函数。

void udelay(unsigned long usecs)

void mdelay(unsigned long mseces)

61.3 实验测试

我们拷贝最简单的驱动文件helloworld.c和Makefile到Ubuntu的/home/topeet/imx8mm/18目录下,我们在此基础上进行修改,编写驱动代码如下所示:

#include <linux/init.h>
#include <linux/module.h>
#include <linux/timer.h>
// 声明时间处理函数
static void timer_function(unsigned long data);
// 该宏会静态创建一个名叫 timer_name 内核定时器,
// 并初始化其 function, expires, name 和 base 字段。
DEFINE_TIMER(test_timer, timer_function, 0, 0);

/**
 * @description:超时处理函数 
 * @param {*}
 * @return {*}
 */
static void timer_function(unsigned long data)
{
    printk(" This is timer_function \n");
    /**
  * @description: 修改定时值,如果定时器还没有激活的话,mod_timer 函数会激活定时器
  * @param {1} *
  * @return {*}
  */
    mod_timer(&test_timer, jiffies + 1 * HZ);
}
static int hello_init(void)
{
    printk("hello world! \n");
    //初始化test_timer.expires意为超时时间
    test_timer.expires = jiffies + 1 * HZ;
    //定时器注册到内核里面,启动定时器
    add_timer(&test_timer);
    return 0;
}

static void hello_exit(void)
{
    printk("gooodbye! \n");
    // 删除一个定时器
    del_timer(&test_timer);
}
module_init(hello_init);
module_exit(hello_exit);
MODULE_LICENSE("GPL");

编译完加载驱动模块,如下图所示:

如上图所示,内核定时器每隔一秒打印“This is timer_function \n”。 

  • 17
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值