标签:
本文章摘自下面的网友:
http://blog.sina.com.cn/s/blog_6e5b342e0100m87d.html
一、内核中如何记录时间
任何程序都需要时间控制,其主要目的是:
为达到这个目的,应用程序使用日历时间(年月日时分秒)或者自1970年1月1日零时零分零秒到当前的秒数来度量时间的流逝,但内核中需要更加有精度的时间度量,因此内核使用时钟嘀嗒来记录时间。时钟中断发生后内核内部时间计数器增加1(即:增加1个时钟嘀嗒),系统引导时为0,当前值为自上次系统引导以来的时钟滴答数,程序可通过内核定义的全局变量jiffies_64或jiffies来访问。真实硬件上每秒的嘀嗒数从 50 到 1200 不等 ,x86上默认为1000,而s3c2440上默认为200,出于统一编程接口的考虑,内核定义了一个宏HZ,它表示1秒钟的嘀嗒数,可供程序使用。
Jiffies 和 jiffies_64是 unsigned long 类型只读变量,其用法如下:
注意:32-位 平台上当 HZ 是 1000 时, 计数器只是每 50 天溢出一次, 必要时你的代码应当准备处理这个事件
比较2个时间的大小,常用内核的如下宏定义:
它们分别在时间a在时间b之后、之前、之后或相等、之前或相等的时候为真,反之为假
二、内核定时器
1、概述
一个cpu,一个ksoftirqd
ksoftirqd属于atomic context
ksoftirqd运行时,不禁用irq
2、定时器 API
#include <linux/timer.h> struct timer_list { unsigned long expires; void (*function)(unsigned long); unsigned long data; 其它字段 };
#include <linux/delay.h> void ndelay(unsigned long nsecs); void udelay(unsigned long usecs); void mdelay(unsigned long msecs);
2、长延时:
如果需要延后较长时间,就可以采用长延时。长延时可分为忙等待和让出CPU两种方式。
1)、忙等待:
unsigned long j1 = jiffies + 2*HZ;
while (time_before(jiffies, j1))
cpu_relax();
cpu_relex 的调用使用了一个特定于体系的方式,你此时没有用处理器做事情,比较浪费处理器的资源
2)、让出处理器
unsigned long j1 = jiffies + 3600*HZ;
while (time_before(jiffies, j))
{
set_current_state(TASK_INTERRUPTIBLE);
schedule_timeout(30*HZ);
}
3)此外,如果你的驱动使用一个等待队列来等待某些其他事件,但是你也想确保它在一个确定时间段内运行能够运行,而不是永久等待,那么可以使用超时
#include <linux/wait.h> 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);
后面会编写一个linux模块化驱动编程的实例和延时。
标签:
原文地址:http://www.cnblogs.com/hjj801006/p/4551824.html