Linux时间子系统之一:clock source(时钟源).doc

上传人:白大夫 文档编号:3255367 上传时间:2019-08-06 格式:DOC 页数:6 大小:27.50KB
返回 下载 相关 举报
Linux时间子系统之一:clock source(时钟源).doc_第1页
第1页 / 共6页
亲,该文档总共6页,到这儿已超出免费预览范围,如果喜欢就下载吧!
资源描述

《Linux时间子系统之一:clock source(时钟源).doc》由会员分享,可在线阅读,更多相关《Linux时间子系统之一:clock source(时钟源).doc(6页珍藏版)》请在三一文库上搜索。

1、Linux时间子系统之一:clock source(时钟源)clock source用于为linux内核提供一个时间基线,如果你用linux的date命令获取当前时间,内核会读取当前的clock source,转换并返回合适的时间单位给用户空间。在硬件层,它通常实现为一个由固定时钟频率驱动的计数器,计数器只能单调地增加,直到溢出为止。时钟源是内核计时的基础,系统启动时,内核通过硬件RTC获得当前时间,在这以后,在大多数情况下,内核通过选定的时钟源更新实时时间信息(墙上时间),而不再读取RTC的时间。本节的内核代码树基于V3.4.10。1. struct clocksource结构内核用一个cl

2、ocksource结构对真实的时钟源进行软件抽象,现在我们从clock source的数据结构开始,它的定义如下:cppview plaincopystructclocksource/*Hotpathdata,fitsinasinglecachelinewhenthe*clocksourceitselfiscachelinealigned.*/cycle_t(*read)(structclocksource*cs);cycle_tcycle_last;cycle_tmask;u32mult;u32shift;u64max_idle_ns;u32maxadj;#ifdefCONFIG_ARCH_

3、CLOCKSOURCE_DATAstructarch_clocksource_dataarchdata;#endifconstchar*name;structlist_headlist;intrating;int(*enable)(structclocksource*cs);void(*disable)(structclocksource*cs);unsignedlongflags;void(*suspend)(structclocksource*cs);void(*resume)(structclocksource*cs);/*private:*/#ifdefCONFIG_CLOCKSOUR

4、CE_WATCHDOG/*Watchdogrelateddata,usedbytheframework*/structlist_headwd_list;cycle_tcs_last;cycle_twd_last;#endif_cacheline_aligned;我们只关注clocksource中的几个重要的字段。1.1 raTIng:时钟源的精度同一个设备下,可以有多个时钟源,每个时钟源的精度由驱动它的时钟频率决定,比如一个由10MHz时钟驱动的时钟源,他的精度就是100nS。clocksource结构中有一个raTIng字段,代表着该时钟源的精度范围,它的取值范围如下:1-99: 不适合于用

5、作实际的时钟源,只用于启动过程或用于测试;100-199:基本可用,可用作真实的时钟源,但不推荐;200-299:精度较好,可用作真实的时钟源;300-399:很好,精确的时钟源;400-499:理想的时钟源,如有可能就必须选择它作为时钟源;1.2 read回调函数时钟源本身不会产生中断,要获得时钟源的当前计数,只能通过主动调用它的read回调函数来获得当前的计数值,注意这里只能获得计数值,也就是所谓的cycle,要获得相应的时间,必须要借助clocksource的mult和shift字段进行转换计算。1.3 mult和shift字段因为从clocksource中读到的值是一个cycle计数值

6、,要转换为时间,我们必须要知道驱动clocksource的时钟频率F,一个简单的计算就可以完成:t = cycle/F;可是clocksource并没有保存时钟的频率F,因为使用上面的公式进行计算,需要使用浮点运算,这在内核中是不允许的,因此,内核使用了另外一个变通的办法,根据时钟的频率和期望的精度,事先计算出两个辅助常数mult和shift,然后使用以下公式进行cycle和t的转换:t = (cycle * mult) shift;只要我们保证:F = (1 shift;从转换精度考虑,mult的值是越大越好,但是为了计算过程不发生溢出,mult的值又不能取得过大。为此内核假设cycle计数

7、值被转换后的最大时间值:10分钟(600秒),主要的考虑是CPU进入IDLE状态后,时间信息不会被更新,只要在10分钟内退出IDLE,clocksource的cycle计数值就可以被正确地转换为相应的时间,然后系统的时间信息可以被正确地更新。当然最后的结果不一定是10分钟,它由clocksource_max_deferment进行计算,并保存max_idle_ns字段中,TIckless的代码要考虑这个值,以防止在NO_HZ配置环境下,系统保持IDLE状态的时间过长。在这样,由10分钟这个假设的时间值,我们可以推算出合适的mult和shift值。2. clocksource的注册和初始化通常,

8、clocksource要在初始化阶段通过clocksource_register_hz函数通知内核它的工作时钟的频率,调用的过程如下:由上图可见,最终大部分工作会转由_clocksource_register_scale完成,该函数首先完成对mult和shift值的计算,然后根据mult和shift值,最终通过clocksource_max_deferment获得该clocksource可接受的最大IDLE时间,并记录在clocksource的max_idle_ns字段中。clocksource_enqueue函数负责按clocksource的rating的大小,把该clocksource按顺

9、序挂在全局链表clocksource_list上,rating值越大,在链表上的位置越靠前。每次新的clocksource注册进来,都会触发clocksource_select函数被调用,它按照rating值选择最好的clocksource,并记录在全局变量curr_clocksource中,然后通过timekeeping_notify函数通知timekeeping,当前clocksource已经变更,关于timekeeping,我将会在后续的博文中阐述。3. clocksource watchdog系统中可能同时会注册对个clocksource,各个clocksource的精度和稳定性各不相

10、同,为了筛选这些注册的clocksource,内核启用了一个定时器用于监控这些clocksource的性能,定时器的周期设为0.5秒:cppview plaincopy#defineWATCHDOG_INTERVAL(HZ1)#defineWATCHDOG_THRESHOLD(NSEC_PER_SEC4)当有新的clocksource被注册时,除了会挂在全局链表clocksource_list外,还会同时挂在一个watchdog链表上:watchdog_list。定时器周期性地(0.5秒)检查watchdog_list上的clocksource,WATCHDOG_THRESHOLD的值定义为0

11、.0625秒,如果在0.5秒内,clocksource的偏差大于这个值就表示这个clocksource是不稳定的,定时器的回调函数通过clocksource_watchdog_kthread线程标记该clocksource,并把它的rate修改为0,表示精度极差。4. 建立clocksource的简要过程在系统的启动阶段,内核注册了一个基于jiffies的clocksource,代码位于kernel/time/jiffies.c:cppview plaincopystructclocksourceclocksource_jiffies=.name=jiffies,.rating=1,/*low

12、estvalidrating*/.read=jiffies_read,.mask=0xffffffff,/*32bits*/.mult=NSEC_PER_JIFFYJIFFIES_SHIFT,/*detailsabove*/.shift=JIFFIES_SHIFT,;.staticint_initinit_jiffies_clocksource(void)returnclocksource_register(core_initcall(init_jiffies_clocksource);它的精度只有1/HZ秒,rating值为1,如果平台的代码没有提供定制的clocksource_defaul

13、t_clock函数,它将返回该clocksource:cppview plaincopystructclocksource*_init_weakclocksource_default_clock(void)return然后,在初始化的后段,clocksource的代码会把全局变量curr_clocksource设置为上述的clocksource:cppview plaincopystaticint_initclocksource_done_booting(void).curr_clocksource=clocksource_default_clock();.finished_booting=1;.clocksource_select();.return0;fs_initcall(clocksource_done_booting);当然,如果平台级的代码在初始化时也会注册真正的硬件clocksource,所以经过clocksource_select()函数后,curr_clocksource将会被设为最合适的clocksource。如果clocksource_select函数认为需要切换更好的时钟源,它会通过timekeeping_notify通知timekeeping系统,使用新的clocksource进行时间计数和更新操作。

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 其他


经营许可证编号:宁ICP备18001539号-1