1.Jiffies
Jiffies为Linux核心变数(unsigned long),它被用来记录系统自开机以来,已经过了多少tick。每发生一次timer interrupt,Jiffies变数会被加一。值得注意的是,Jiffies于系统开机时,并非初始化成零,而是被设为-300*HZ (arch/i386/kernel/time.c),即代表系统于开机五分钟后,jiffies便会溢位。那溢位怎么办?事实上,Linux核心定义几个macro(timer_after、time_after_eq、time_before与time_before_eq),即便是溢位,也能借由这几个macro正确地取得jiffies的内容。
另外,80x86架构定义一个与jiffies相关的变数jiffies_64 ,此变数64位元,要等到此变数溢位可能要好几百万年。因此要等到溢位这刻发生应该很难吧。
2. HZ
Linux核心每隔固定周期会发出timer interrupt (IRQ 0),HZ是用来定义每一秒有几次timer interrupts。举例来说,HZ为1000,代表每秒有1000次timer interrupts。 HZ可在编译核心时设定,如下所示(以核心版本2.6.20-15为例):
adrian@adrian-desktop:~$ cd /usr/src/linux
adrian@adrian-desktop:/usr/src/linux$ make menuconfig
Processor type and features —> Timer frequency (250 HZ) —>
其中HZ可设定100、250、300或1000。
3. Tick
Tick是HZ的倒数,意即timer interrupt每发生一次中断的时间。如HZ为250时,tick为4毫(millisecond)。
4. 时间耗时统计
有时候,设备驱动程序可能需要将用jiffies表达的时间间隔转化成毫秒ms或者是微秒us的形式,这种情况大多出现在需要将时钟滴答这种形式转化成人类易于理解的ms或者是us这样的时间形式下,比如为了在驱动程序中打印出一次DMA传输所花费的时间,在DMA传输开始前记录start_jiffies = jiffies,然后进行DMA传输,在DMA传输结束后记录下end_jiffies = jiffies,这样本次的DMA传输所花费的时间将为time = jiffies_to_msecs(end_jiffies - start_jiffies),这里time的单位将会是ms。
unsigned long start= jiffies;
process();
unsigned long end= jiffies;
//unsigned long used_time = (start - end) *1000/HZ;
unsigned int used_time = jiffies_to_msecs(end - start);
printk("used: %u ms\n",used_time);
内核中提供转换的函数。
<include/linux/jiffies.h>
unsigned int jiffies_to_msecs(const unsigned long j);
unsigned int jiffies_to_usecs(const unsigned long j);
unsigned long msecs_to_jiffies(const unsigned int m);
unsigned long usecs_to_jiffies(const unsigned int u);
以上就是用于在内核进行时间统计的方法。