Linux时间子系统之一:认识timer_list和timer_stats和使用
内核版本:v3.4.xxx
一、前言
内核提供了方便查看当前系统TickDevice、活动的Timer列表以及Timer使用的统计信息。
内核分别用两个节点来表示TimerList和Timer统计信息。
一个是/proc/timer_list,打印per_cpu的hrtimer_bases信息以及基于此的timer列表,包括三种时钟MONOTONIC/REALTIME/BOOTTIME;以及Broadcast Tick Device和Per CPU Tick Device信息。
另一个是/proc/timer_stats,需要echo 1 > /proc/timer_stats打开,echo 0 /proc/timer_stats关闭。cat /proc/timer_stats可以获取统计信息。
二、背景介绍
Per CPU TickDevice是维护当前CPU Tick和hrtimer的基础,只有工作在OneShot模式下才能实现hrtimer。在CPU进入cpuidle后,Per CPU TickDevice可能会被关闭,这时候可以打开Broadcast Tick作为唤醒源。超时后,就会将系统从cpuidle退出,进入正常工作模式打开Per CPU TickDevice。
hrtimer有三种时钟基准,这三种时钟基准对应不同的时间获取函数,但主要是基于TimeKeeping的统计。
三、认识timer_list
kernel/time/timer_list.c中init_timer_list_procfs创建/proc/timer_list节点。
|
static int timer_list_show(struct seq_file *m, void *v) SEQ_printf(m, "Timer List Version: v0.6\n");------------------------------------------------------------------(1) for_each_online_cpu(cpu)----------------------------------------------------------------------------------------(2) SEQ_printf(m, "\n"); return 0; |
(1)打印概况信息,hrtimer基准时钟个数和系统MONOTONIC时间。
(2)这里是per_cpu信息,按照hrtimer_cpu_base->hrtimer_clock_base->hrtimer层级关系打印。
(3)打印Broadcast TickDevice和per_cpu的TickDevice设备信息。
下面根据一个示例来分析:
|
Timer List Version: v0.6 cpu: 0-----------------------print_cpuCPU0的信息如下,从clock0到clock2 --依次是:序列号、hrtimer、timer状态、hrtimer超时函数、进程名/进程号 --超时绝对时间和相对时间。 -------------------------------------------------------------------hrtimer_cpu_base部分,参照结构体解释---------------------------------------------------------------------- ---------------------------------------------------------------------tick_sched部分,参照结构体解释----------------------------------------------------------------------------- Tick Device: mode: 1-----------------------Broadcast TickDevice Tick Device: mode: 1---------------------per_cpu TickDevice,详细信息参照clock_event_device结构体解释。 |
四、认识timer_stats
kernel/time/timer_stats.c中init_tstas_procfs创建了/proc/timer_stats节点。统计细心
在每次hrtimer超时函数中都会进行统计,必须打开CONFIG_TIMER_STATS才有效。
| __hrtimer_start_range_ns-->raise_softirq_irqoff-->HRTIMER_SOFTIRQ-->run_hrtimer_softirq-->hrtimer_interrupt-->__run_hrtimer-->timer_stats_account_hrtimer |
timer_stats_account_hrtimer是执行统计信息的主体,这里面有一个timer_stats_active开关。是通过echo [0|1] > /proc/timer_stats进行开关。
|
void timer_stats_update_stats(void *timer, pid_t pid, void *startf, if (likely(!timer_stats_active))----------------------------------------是否打开开关 lock = &per_cpu(tstats_lookup_lock, raw_smp_processor_id()); input.timer = timer;---------------------------------------------------填充entry结构体 raw_spin_lock_irqsave(lock, flags); entry = tstat_lookup(&input, comm);-------------------------------在tstat_hash_table中查找input,找到count递增;没找到新增一个entry插入hash表。 out_unlock: |
显示的地方在tstats_show,所有的统计信息存放在static struct entry entries[MAX_ENTRIES]结构体数组中,当前entry数目通过nr_entries来记录。
|
static int tstats_show(struct seq_file *m, void *v) mutex_lock(&show_mutex); time = ktime_sub(time_stop, time_start); period = ktime_to_timespec(time); seq_puts(m, "Timer Stats Version: v0.2\n"); for (i = 0; i < nr_entries; i++) {----------------------遍历entries数组,数组内容在每次hrtimer中断处理中进行更新。 print_name_offset(m, (unsigned long)entry->start_func); events += entry->count; ms += period.tv_sec * 1000; if (events && period.tv_sec) mutex_unlock(&show_mutex); return 0; |
一个实例分析:
|
Timer Stats Version: v0.2 --依次是:timer count、进程号、进程名称、超时函数、启动timer函数 |
五、使用
1.通过timer_stats可以知道系统的timer使用程度,一定程度上反映了进程的活跃状态。
2.timer_list可以知道系统TickDevice相关设备信息。
3.还可以知道timer状态,以及即将发生的Timer。
4.系统相关信息尤其是tick_sched(系统调度jiffies、idle、iowait等)和hrtimer_cpu_base(retries、hang)部分。
Linux时间子系统之一:认识timer_list和timer_stats和使用的更多相关文章
- Linux时间子系统专题汇总
关于Linux时间子系统有两个系列文章讲的非常好,分别是WowoTech和DroidPhone. 还有两本书分别是介绍: Linux用户空间时间子系统<Linux/UNIX系统编程手册>的 ...
- Linux时间子系统之(二):软件架构
专题文档汇总目录 Notes:从框架上讲解了时间子系统,从底向上包括CPU Local TImer.Global Counter.Clock Souce/Clock Events模块管理.Tick D ...
- Linux时间子系统之六:高精度定时器(HRTIMER)的原理和实现
转自:http://blog.csdn.net/droidphone/article/details/8074892 上一篇文章,我介绍了传统的低分辨率定时器的实现原理.而随着内核的不断演进,大牛们已 ...
- Linux时间子系统之五:低分辨率定时器的原理和实现
专题文档汇总目录 Notes:低精度timer在内核中的数据结构以及API接口:低精度timer精巧高效的分组,使用cascade进行定时器移位,组内Timer FIFO:低精度Timer的初始化流程 ...
- Linux时间子系统之(一):时间的基本概念
专题文档汇总目录 Notes:Linux时间基准点:Linux时间和broken-down time(struct tm):不同精度的时间表示time_t.timeval.timespec. 原文地址 ...
- Linux时间子系统之(三):用户空间接口函数
专题文档汇总目录 Notes:用户空间时间相关接口函数: 类型 API 精度 说明 时间 time stime time_t 精度为秒级 逐渐要被淘汰.需要定义__ARCH_WANT_SYS_TIME ...
- Linux时间子系统之(四):timekeeping
专题文档汇总目录 Notes:timekeeping模块的狠心数据结构是timekeeper,它维护了系统不同类型时钟的时间值,并且介绍了获取不同类型时钟时间的函数. clocksource切换通过c ...
- Linux时间子系统之(五):POSIX Clock
专题文档汇总目录 Notes: 本章主要介绍了若干种类的静态时钟,这些时钟都可以通过k_clock表示,注册到posix_clocks中.这些都是静态时钟,可以分为三大类:各种REALTIME时钟.带 ...
- Linux时间子系统之(六):POSIX timer
专题文档汇总目录 Notes:首先讲解了POSIX timer的标识(唯一识别).POSIX Timer的组织(管理POSIX Timer).内核中如何抽象POSIX Timer:然后分析了POSIX ...
- Linux时间子系统之(十二):periodic tick
专题文档汇总目录 Notes:TickDevice模式,以及clocckevent设备.TickDevice设备的初始化,TickDevice是如何加入到系统中的.周期性Tick的产生. 原文地址:L ...
随机推荐
- Media Player Classic - HC 源代码分析 7:详细信息选项卡(CPPageFileInfoDetails)
===================================================== Media Player Classic - HC 源代码分析系列文章列表: Media P ...
- 苹果新的编程语言 Swift 语言进阶(十六)--泛型
泛型允许你定义一个宽松.可重用的函数或者类型,使用泛型能够避免代码的重复,也能以更清楚和抽象的方式来表达程序的意图. 泛型是Swift语言提供的强大功能之一,Swift提供的许多标准库都使用了泛型来创 ...
- C++之默认参数
C++可以为不指定参数提供默认值.一旦给一个参数赋了默认值,后面的所有参数,也都必须为默认值,并且默认值的类型也必须正确,默认值可以在原型或者函数定义中给出,但是不能两个位置同时给出. 接下来我们上代 ...
- CRF资料
与最大熵模型相似,条件随机场(Conditional random fields,CRFs)是一种机器学习模型,在自然语言处理的许多领域(如词性标注.中文分词.命名实体识别等)都有比较好的应用效果.条 ...
- LeetCode(64)- Min Stack
题目: Design a stack that supports push, pop, top, and retrieving the minimum element in constant time ...
- DTN学习,theONE模拟器网络相关资料整理
下面是一个百度空间的: http://hi.baidu.com/jensenliao 博客园的一篇博客:theONE模拟器简介(主要讲述,软件配置,软件结构) http://www.cnblogs.c ...
- CALayer简介
一.什么是CALayer * 在iOS系统中,你能看得见摸得着的东西基本上都是UIView,比如一个按钮.一个文本标签.一个文本输入框.一个图标等等,这些都是UIView. * 其实UIView之所以 ...
- javascript随机一个1-9的数字
window.onload=function(){ var oTxt=document.getElementById('txt'); for(i=1;i<=200;i ...
- 《深入理解JAVA虚拟机》笔记1
java程序运行时的内存空间,按照虚拟机规范有下面几项: )程序计数器 指示下条命令执行地址.当然是线程私有,不然线程怎么能并行的起来. 不重要,占内存很小,忽略不计. )方法区 这个名字很让我迷惑. ...
- java中内存的使用
一个java运行起来执行代码,主要的内存消耗有这几块: 1.堆 2.栈 :栈是每个线程一个的,是以消耗的内存是内存大小*线程数,当线程数特多时候需要小心 . 3.直接内存:主要是通道时候的缓存,在内存 ...