linux HZ 值_Linux内核设备驱动之内核的时间管理笔记整理(推荐)

概述

今天小编给大家分享一篇关于Linux内核设备驱动之内核的时间管理笔记整理。

117962c443aa05a5158b2ddd518ab9fe.png

内核中的时间概念

时间管理在linux内核中占有非常重要的作用。

相对于事件驱动而言,内核中有大量函数是基于时间驱动的。

有些函数是周期执行的,比如每10毫秒刷新一次屏幕;

有些函数是推后一定时间执行的,比如内核在500毫秒后执行某项任务。

要区分:

  • *绝对时间和相对时间
  • *周期性产生的事件和推迟执行的事件

周期性事件是由系统系统定时器驱动的


HZ值

内核必须在硬件定时器的帮助下才能计算和管理时间。

定时器产生中断的频率称为节拍率(tick rate)。

在内核中指定了一个变量HZ,内核初始化的时候会根据这个值确定定时器的节拍率。

HZ定义在,在i386平台上,目前采用的HZ值是1000。

也就是时钟中断每秒发生1000次,周期为1毫秒。即:

#define HZ 1000

注意!HZ不是个固定不变的值,它是可以更改的,可以在内核源代码配置的时候输入。

不同的体系结构其HZ值是不一样的,比如arm就采用100。

如果在驱动中要使用系统的中断频率,直接使用HZ,而不要用100或1000

a.理想的HZ值

i386的HZ值一直采用100,直到2.5版后才改为1000。

提高节拍率意味着时钟中断产生的更加频繁,中断处理程序也会更频繁地执行。

带来的好处有:

  • *内核定时器能够以更高的频率和更高的准确度运行
  • *依赖定时器执行的系统调用,比如poll()和select(),运行的精度更高
  • *提高进程抢占的准确度

(缩短了调度延时,如果进程还剩2ms时间片,在10ms的调度周期下,进程会多运行8ms。

由于耽误了抢占,对于一些对时间要求严格的任务会产生影响)

坏处有:

*节拍率要高,系统负担越重。

中断处理程序将占用更多的处理器时间。


jiffies

全局变量jiffies用于记录系统启动以来产生的节拍的总数。

启动时,jiffies初始化为0,此后每次时钟中断处理程序都会增加该变量的值。

这样,系统启动后的运行时间就是jiffies/HZ秒

jiffies定义于中:

extern unsigned long volatile jiffies;

jiffies变量总是为unsigned long型。

因此在32位体系结构上是32位,而在64位体系上是64位。对于32位的jiffies,如果HZ为1000,49.7天后会溢出。虽然溢出的情况不常见,但程序在检测超时时仍然可能因为回绕而导致错误。linux提供了4个宏来比较节拍计数,它们能正确地处理节拍计数回绕。

3b7ba94c43bdc3df44c63433a6001f81.png

unknown通常是指jiffies,known是需要对比的值(常常是一个jiffies加减后计算出的相对值)例:

069c21107fa8f335fdf01ce0d00e757e.png

time_before可以理解为如果在超时(timeout)之前(before)完成

*系统中还声明了一个64位的值jiffies_64,在64位系统中jiffies_64和jiffies是一个值。

可以通过get_jiffies_64()获得这个值。

*使用

0e4ec3c6d4082bc75f03990a0b0ca11b.png

获得当前时间

驱动程序中一般不需要知道墙钟时间(也就是年月日的时间)。但驱动可能需要处理绝对时间。

为此,内核提供了两个结构体,都定义在:

2cdd2b7987b4252c4c4fc36f8cf085b6.png

do_gettimeofday()该函数用通常的秒或微秒来填充一个指向struct timeval的指针变量,原型如下:

6e5a03a873fff6d67bebded2c11dd76b.png

current_kernel_time()该函数可用于获得timespec

75afa03f348ec4d882eed5839e119a38.png
3800f4159573ce7813069d158a11a5c2.png

设备驱动程序经常需要将某些特定代码延迟一段时间后执行,通常是为了让硬件能完成某些任务。

长于定时器周期(也称为时钟嘀嗒)的延迟可以通过使用系统时钟完成,而非常短的延时则通过软件循环的方式完成

(1)短延时

对于那些最多几十个毫秒的延迟,无法借助系统定时器。

系统通过软件循环提供了下面的延迟函数:

a2ea77f1dd679ccda4907bcc794b5ddb.png

这三个延迟函数均是忙等待函数,在延迟过程中无法运行其他任务。

实际上,当前所有平台都无法达到纳秒精度。

(2)长延时

a.在延迟到期前让出处理器

08cd9da3e9e931547ca8ffae303f611c.png

在等待期间可以让出处理器,但系统无法进入空闲模式(因为这个进程始终在进行调度),不利于省电。

b.超时函数

6faa2131836519fc4e73a3e7528bd871.png

使用方式:

e935ced692517fe3d2c05aff50b3457e.png

进程经过2秒后会被唤醒。如果不希望被用户空间打断,可以将进程状态设置为TASK_UNINTERRUPTIBLE。

e1ffb7e0b2f8558f6f45cdb19710844f.png

(3)等待队列

使用等待队列也可以实现长延迟。

在延迟期间,当前进程在等待队列中睡眠。

进程在睡眠时,需要根据所等待的事件链接到某一个等待队列。

a.声明等待队列

等待队列实际上就是一个进程链表,链表中包含了等待某个特定事件的所有进程。

2fa68b176d5d2e606bf79b8d00545217.png

typedef struct __wait_queue_head wait_queue_head_t;

要想把进程加入等待队列,驱动首先要在模块中声明一个等待队列头,并将它初始化。

静态初始化

fd99d5fcb0997a7a6476ca63c7d45b7c.png

动态初始化

fec15d6e88575d8d3a6a869be12cf47f.png

b.等待函数

进程通过调用下面函数可以在某个等待队列中休眠固定的时间:

#include long wait_event_timeout(wait_queue_head_t q,condition, long timeout);long wait_event_interruptible_timeout(wait_queue_head_t q, condition, long timeout);

调用这两个函数后,进程会在给定的等待队列q上休眠,但会在超时(timeout)到期时返回。

如果超时到期,则返回0,如果进程被其他事件唤醒,则返回剩余的时间数。

如果没有等待条件,则将condition设为0

使用方式:

92dd870540d16080ddb3eea854b950d4.png

(4)内核定时器

还有一种将任务延迟执行的方法是采用内核定时器。与前面几种延迟方法不同,内核定时器并不会阻塞当前进程,启动一个内核定时器只是声明了要在未来的某个时刻执行一项任务,当前进程仍然继续执行。不要用定时器完成硬实时任务

定时器由结构timer_list表示,定义在

1dbca526d680689af2fe75093a18fb3b.png

内核在中提供了一系列管理定时器的接口。

a.创建定时器

struct timer_list my_timer;

b.初始化定时器

f3a0504f52c8561c5f95277ff55bb5f7.png

c.定时器的执行函数

超时处理函数的原型如下:

c00428eb41dfd5c20551d79ee50f433e.png

可以利用data参数用一个处理函数处理多个定时器。可以将data设为0

d.激活定时器

add_timer(&my_timer);

定时器一旦激活就开始运行。

e.更改已激活的定时器的超时时间

e7a3c54ea28fafd9a088981f96608dd8.png

可以用于那些已经初始化但还没激活的定时器,如果调用时定时器未被激活则返回0,否则返回1。一旦mod_timer返回,定时器将被激活。

f.删除定时器

del_timer(&my_timer);

被激活或未被激活的定时器都可以使用,如果调用时定时器未被激活则返回0,否则返回1。不需要为已经超时的定时器调用,它们被自动删除

g.同步删除

del_time_sync(&my_timer);

在smp系统中,确保返回时,所有的定时器处理函数都退出。不能在中断上下文使用。

eaccf6018f8cbe6fc80bd9393e061468.png

什么是不确定时间的延迟

前面介绍的是确定时间的延迟执行,但在写驱动的过程中经常遇到这种情况:用户空间程序调用read函数从设备读数据,但设备中当前没有产生数据。此时,驱动的read函数默认的操作是进入休眠,一直等待到设备中有了数据为止。

这种等待就是不定时的延迟,通常采用休眠机制来实现。


休眠

休眠是基于等待队列实现的,前面我们已经介绍过wait_event系列函数,但现在我们将不会有确定的休眠时间。

当进程被置入休眠时,会被标记为特殊状态并从调度器的运行队列中移走。

直到某些事件发生后,如设备接收到数据,则将进程重新设为运行态并进入运行队列进行调度。

休眠函数的头文件是,具体的实现函数在kernel/wait.c中。

a.休眠的规则

  • *永远不要在原子上下文中休眠
  • *当被唤醒时,我们无法知道睡眠了多少时间,也不知道醒来后是否获得了我们需要的资源
  • *除非知道有其他进程会在其他地方唤醒我们,否则进程不能休眠

b.等待队列的初始化

见前文

c.休眠函数

linux最简单的睡眠方式为wait_event宏。该宏在实现休眠的同时,检查进程等待的条件。

adaeb3a0b96e11d7e23c366c0c18c533.png
  • q: 是等待队列头,注意是采用值传递。
  • condition: 任意一个布尔表达式,在条件为真之前,进程会保持休眠。
  • 注意!进程需要通过唤醒函数才可能被唤醒,此时需要检测条件。
  • 如果条件满足,则被唤醒的进程真正醒来;
  • 如果条件不满足,则进程继续睡眠。

d.唤醒函数

当我们的进程睡眠后,需要由其他的某个执行线程(可能是另一个进程或中断处理例程)唤醒。唤醒函数:

65efab41e8fb9d05ebfa64f5e9190a96.png

wake_up会唤醒等待在给定queue上的所有进程。而wake_up_interruptible唤醒那些执行可中断休眠的进程。实践中,约定做法是在使用wait_event时使用wake_up,而使用wait_event_interruptible时使用wake_up_interruptible。


以上就是本文的全部内容,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。后面小编会分享更多运维方面的干货,感兴趣的朋友走一波关注哩~

664f152bdca796d51adc298212898517.gif
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值