Linux之定时器与时间管理 【转】

转自:http://blog.chinaunix.net/uid-23228758-id-154820.html

定时器与时间管理:

1、节拍率——HZ:在alpha体系结构上1024,而在其它平台上,都为10数量级倍。在嵌入式ARM上为100(2.6内核)。这个值的意义是什么呢,也就是在ARM平台上时钟中断100次,为一秒。一般的情况下编程者不要改变这个值,因为内核编很多代码都是有时间要求的,而且内核编写都在很多地方都做了相应的优化与折衷处理,改变HZ的值会对系统的性能有很大的影响。

2、jiffies:这个值是用来记录系统自系统启动以来产生的节拍的总数,启动时,内核将这个变量初始化为0;在每次的时钟中断处理程序都会增加该变量的值,jiffies一秒内增加的值就是HZ,系统运行时间以秒为单位计算,则为系统运行了jiffies/HZ秒。

在如下定义(2.6内核):


extern u64 __jiffy_data jiffies_64;
extern unsigned long volatile __jiffy_data jiffies;

在2.6的内核中它的变量类型从无符号长整型变为了u64,也就是说,即使是32位的机器,也使用无符号的64位整型表示jiffies。大多数的代码只涉及jiffies的低32位,访问jiffies的代码中会读到jiffies_64的低32位,可以通过get_ jiffies_64()函数读取整个64位。在64位的体系结构中jiffies_64和jiffies指的同一个变量,代码可以既可以通过jiffies也可以通过get_ jiffies_64()读取。

3、为了避免jiffies值的回绕(溢出),内核中提供了几个API函数来比较节拍计数。定义在中。


/* time_is_before_jiffies(a) return true if a is before jiffies */
#define time_is_before_jiffies(a) time_after(jiffies, a)
/* time_is_after_jiffies(a) return true if a is after jiffies */
#define time_is_after_jiffies(a) time_before(jiffies, a) 
/* time_is_before_eq_jiffies(a) return true if a is before or equal to jiffies*/
#define time_is_before_eq_jiffies(a) time_after_eq(jiffies, a)
/* time_is_after_eq_jiffies(a) return true if a is after or equal to jiffies*/
#define time_is_after_eq_jiffies(a) time_before_eq(jiffies, a)

例:

可以利用jiffies设置超时等,譬如:


unsigned long timeout = jiffies + HZ * 2; // 2秒钟后超时
       if(time_before(jiffies, timeout){
       // 还没有超时
    }
    else{
       // 已经超时
    }

4、时间类型

Linux下常用的时间类型有4个:time_t,struct timeval,struct timespec,struct tm。一般用于实际时间(墙上时间),定义在文件中。

(1)time_t是一个长整型,一般用来表示用1970年以来的秒数,格林威治时间。

(2)Struct timeval有两个成员,一个是秒,一个是微妙。


struct timeval{
            long tv_sec; /**//* seconds */
            long tv_usec; /**//* microseconds */
 };

(3)struct timespec有两个成员,一个是秒,一个是纳秒。


struct timespec{

         time_t tv_sec; /**//* seconds */格林威治时间
         long tv_nsec; /**//* nanoseconds */
 };

(4)struct tm是直观意义上的时间表示方法:

struct tm {
        int tm_sec; /**//* seconds */
        int tm_min; /**//* minutes */
        int tm_hour; /**//* hours */
        int tm_mday; /**//* day of the month */
        int tm_mon; /**//* month */
        int tm_year; /**//* year */
        int tm_yday; /**//* day in the year */
        int tm_wday; /**//* day of the week */
        int tm_isdst; /**//* daylight saving time */
 };

:一般使用int gettimeofday(struct timeval *tv, struct timezone *tz);得到墙上时间,墙上时间一般会在用户空间里使用,而在内核空间里,大多数情况下只要获取相对时间就OK了,也就是说用jiffies搞定。

5、内核定时器

内核定时器可以理解为一个软件定时器,它可以被动态的创建、更改和销毁等,而且运行次数没有限制。

定时器同结构time_list表示,此结构体定义在中。


struct timer_list {
       struct list_head entry; //包含定时器的链表
       unsigned long expires; //以jiffies为单位的定时值
       void (*function)(unsigned long); //定时时间到的处理函数
       unsigned long data; //传给片处理函数的参数,可以用在共用一个处理函数的情况。
       struct tvec_base *base; //内部值,用户呼略是安全的
#ifdef CONFIG_TIMER_STATS
       void *start_site;
       char start_comm[16];
       int start_pid;
#endif
#ifdef CONFIG_LOCKDEP
       struct lockdep_map lockdep_map;
#endif
};

在使用定时器时,没有必要去深入了解这个数据结构体的成员。可以使用在中的API来操作。

增加定时器

void add_timer(struct timer_list * timer);
删除定时器

int del_timer(struct timer_list * timer);
修改定时器的expire

int mod_timer(struct timer_list *timer, unsigned long expires);
使用定时器的一般流程为:
(1)定义timer、编写function;
(2)为timer的expires、data、function赋值;
(3)调用add_timer将timer加入列表;
(4)在定时器到期时,function被执行;
(5)在程序中涉及timer控制的地方适当地调用del_timer、mod_timer删除timer或修改timer的expires。
补充:一般应该使用del_timer_sync()来代替del_timer(),在俺的板子上就同没有必要了,因为俺的板子是单处理器的,在SMP中一定要代替使用。

6、短延迟

前面所讲到的时间都是很长的了啦,在内核中提供了二个更短延迟的函数来供开发人员使用

udelay(unsigned long usecs);   //微秒
mdelay(unsigned long msecs);  //毫秒
前者用软件循环指定的微妙数,后者调用前者达到延迟毫秒级。udelay 函数只能用于获取较短的时间延迟,因为loops_per_second值的精度只有8位,所以,当计算更长的延迟时会积累出相当大的误差。尽管最大能允 许的延迟将近1秒(因为更长的延迟就要溢出),推荐的 udelay 函数的参数的最大值是取1000微秒(1毫秒)。延迟大于 11 毫秒时可以使用函数mdelay。
:要特别注意的是 udelay 是个忙等待函数(所以 mdelay 也是),在延迟的时间段内无法运行其他的任务,因此要十分小心,尤其是 mdelay,除非别无他法,要尽量避免使用。

时间: 2024-10-07 13:28:44

Linux之定时器与时间管理 【转】的相关文章

Linux内核——定时器和时间管理

定时器和时间管理 系统定时器是一种可编程硬件芯片.它能以固定频率产生中断.该中断就是所谓的定时器中断.它所相应的中断处理程序负责更新系统时间,还负责执行须要周期性执行的任务. 系统定时器和时钟中断处理程序是Linux系统内核管理机制中的中枢. 另外一个关注的焦点是动态定时器--一种用来推迟运行程序的工具. 比方说.假设软驱马达在一定时间内都未活动,那么软盘驱动程序会使用动态定时器关闭软驱马达. 内核能够动态创建或销毁动态定时器. 内核中的时间观念 内核在硬件的帮助下计算和管理时间. 硬件为内核提

《Linux内核设计与实现》读书笔记(十一)- 定时器和时间管理

系统中有很多与时间相关的程序(比如定期执行的任务,某一时间执行的任务,推迟一段时间执行的任务),因此,时间的管理对于linux来说非常重要. 主要内容: 系统时间 定时器 定时器相关概念 定时器执行流程 实现程序延迟的方法 定时器和延迟的例子 1. 系统时间 系统中管理的时间有2种:实际时间和定时器. 1.1  实际时间 实际时间就是现实中钟表上显示的时间,其实内核中并不常用这个时间,主要是用户空间的程序有时需要获取当前时间, 所以内核中也管理着这个时间. 实际时间的获取是在开机后,内核初始化时

(笔记)Linux内核学习(八)之定时器和时间管理

一 内核中的时间观念 内核在硬件的帮助下计算和管理时间.硬件为内核提供一个系统定时器用以计算流逝的时间.系 统定时器以某种频率自行触发,产生时钟中断,进入内核时钟中断处理程序中进行处理. 墙上时间和系统运行时间根据时钟间隔来计算. 利用时间中断周期执行的工作: 更新系统运行时间: 更新实际时间: 在smp系统上,均衡调度程序中各处理器上运行队列: 检查当前进程是否用尽了时间片,重新进行调度: 运行超时的动态定时器: 更新资源消耗和处理器时间的统计值: 二 节拍率 系统定时器的频率:通过静态预处理

向下之旅(十四):定时器和时间管理

相对于事件驱动而言,内核中有大量的函数都是基于时间驱动的.有些函数是周期执行的,有些操作是需要等待一个相对的时间后才运行.除了上述两类函数需要内核提供时间外,内核还必须管理系统的运行时间以及当前日期和时间. 其中相对时间和绝对时间是不同的,若某个事件在5秒后被调度执行,那么系统所需要的是——相对时间(相对现在起5秒).如果涉及到日期和时间,内核不但要计算流逝的时间还要计算绝对时间. 周期性产生的事件和推迟执行的时间之间的差别:前者比如每10毫秒一次——都是由系统定时器驱动的.系统定时器是一种可编

《Linux设计与实现》学习笔记——定时器和时间管理

基本概念 系统定时器 一种可编程硬件芯片,能够以固定频率产生中断,定时器中断,他的中断处理程序负责更新系统时间,负责执行需要周期性运行的任务. 触发中断的频率即节拍率,可以编程预定. 动态定时器 一种用来推迟执行程序的工具 HZ 系统定时器的频率(节拍率)是静态预处理定义的. HZ数大小确定应考虑的因素: HZ越高: 优势:提高时间驱动事件的准确度:调度的粒度更细,进程抢占更准确: 内核定时器以更高的频度和准确度执行. 劣势:节拍率越高,时钟中断程序占用处理器时间越多:并且频繁打乱处理器cach

定时器和时间管理

一.jiffies 自系统启动以来产生的节拍的总数. extern unsigned long volatile jiffies; extern u64 jiffies_64; 比较时间的几个宏: time_after(unknown, known) //unknown after known ? true : false; time_before(unknown, known) //unknown before known ? true : false; time_after_eq(unkno

linux下的时间管理概述

2017/6/21 时间这一概念在生活中至关重要,而在操作系统中也同样重要,其在系统中的功能绝不仅仅是给用户提供时间这么简单,内核的许多机制都依赖于时间子系统.但凡是要在某个精确的时间执行某个事件,必然会用到时间子系统.而内核中定时器的发展可以以2.6版本的内核作为一个分界点.之前主要是低分辨率定时器,2.6开发期间便引入了高分辨率定时器.分辨率的高低意味着时间粒度的大小. 本节重点介绍下涉及到的几个重点概念 时钟设备 周期时钟/动态时钟 高分辨率/低分辨率 定时器 时钟设备时钟设备算是最低层的

Linux时间管理涉及数据结构和传统低分辨率时钟的实现

上篇文章大致描述了Linux时间管理的基本情况,看了一些大牛们的博客感觉自己写的内容很匮乏,但是没办法,只能通过这种方式提升自己--闲话不说,本节介绍下时间管理下重要的数据结构 设备相关数据结构 //时钟源结构 struct clocksource{} //时钟设备结构 struct tick_device { struct clock_event_device *evtdev; enum tick_device_mode mode;//记录对应时钟事件设备的模式 }; enum tick_de

解析Linux内核的基本的模块管理与时间管理操作---超时处理【转】

转自:http://www.jb51.net/article/79960.htm 这篇文章主要介绍了Linux内核的基本的模块管理与时间管理操作,包括模块加载卸载函数的使用和定时器的用法等知识,需要的朋友可以参考下 内核模块管理Linux设备驱动会以内核模块的形式出现,因此学会编写Linux内核模块编程是学习linux设备驱动的先决条件. Linux内核的整体结构非常庞大,其包含的组件非常多.我们把需要的功能都编译到linux内核,以模块方式扩展内核功能. 先来看下最简单的内核模块 ? 1 2