免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 1019 | 回复: 0
打印 上一主题 下一主题

完美搞清LINUX TIMER问题! [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2007-05-24 00:49 |只看该作者 |倒序浏览

代码实现: 完美搞清LINUX TIMER问题!
QUOTE: /***
* timer_pending - is a timer pending?
* @timer: the timer in question
*
* timer_pending will tell whether a given timer is currently pending,
* or not. Callers must ensure serialization wrt. other operations done
* to this timer, eg. interrupt contexts, or other CPUs on SMP.
*
* return value: 1 if the timer is pending, 0 if not.
*/
static inline int timer_pending(const struct timer_list * timer) {
        return timer->entry.next != NULL;
}
QUOTE: struct timer_list {
        struct list_head entry;
        unsigned long expires;
        void (*function)(unsigned long);
        unsigned long data;
        struct timer_base_s *base;
};
上面TIMER中entry字段用于将该TIMER连接到相应的TIMER队列。借用ULK3中的图片说明一下,


如图,比如连接到图中的TV2双链中。
那么这里有两个问题
  • timer挂起为什么可以通过 QUOTE: return timer->entry.next != NULL;
    来判断,是不是TIMER在过期后执行前首先要从双链中解除下来。所以上面的判断可以这样进行。
  • QUOTE: typedef struct tvec_t_base_s {
            spinlock_t lock;
            unsigned long timer_jiffies;
            struct timer_list *running_timer;
            tvec_root_t tv1;
            tvec_t tv2;
            tvec_t tv3;
            tvec_t tv4;
            tvec_t tv5;
        } tvec_base_t;
    中timer_jiffies如何更新. 该字段是用于从 QUOTE: unsigned long idx = expires - base->timer_jiffies;
    算式中得出idx ,利用该idx 将TIMER插入相应的双链表。即
    QUOTE: tvec_root_t tv1;
            tvec_t tv2;
            tvec_t tv3;
            tvec_t tv4;
            tvec_t tv5;
    中的一个。
    qtdszws (异次元空间)发表于 2007-5-17 15:12
    »1.timer挂起为什么可以通过return timer->entry.next != NULL; 来判断,是不是TIMER在过期后执行前首先要从双链中解除下来。所以上面的判断可以这样进行。

    »2.timer_jiffies如何更新
    在run_timer_list中更新
    [Copy to clipboard]CODE: v2.4.0 static inline void
    run_timer_list(void) {
            spin_lock_irq(&timerlist_lock);
            while ((long)(jiffies - timer_jiffies) >= 0) {//定时器到时,可能还错过了几个jiffies,补上
                    struct list_head *head, *curr;
                    if (!tv1.index) {// 最近要执行的定时器队列已被遍历一遍
                            int n = 1;
                            do {
                                    cascade_timers(tvecs[n]);
                            } while (tvecs[n]->index == 1 && ++n next;
                    if (curr != head) {
                            struct timer_list *timer;
                            void (*fn)(unsigned long);
                            unsigned long data;
                            timer = list_entry(curr, struct timer_list, list);
                            fn = timer->function;
                            data= timer->data;
                            detach_timer(timer);
                            timer->list.next = timer->list.prev = NULL;
                            //一次只运行一个定时器,run_timer_list是在timer_bh中运行的,
                            //所以一次只有一个CPU执行run_timer_list
                            timer_enter(timer);
                            spin_unlock_irq(&timerlist_lock);
                            fn(data);//执行定时任务
                            spin_lock_irq(&timerlist_lock);
                            timer_exit();
                            goto repeat;
                    }
                    ++timer_jiffies; //更新
                    tv1.index = (tv1.index + 1) & TVR_MASK;
            }
            spin_unlock_irq(&timerlist_lock);
    }



    本文来自ChinaUnix博客,如果查看原文请点:http://blog.chinaunix.net/u1/35189/showart_308264.html
  • 您需要登录后才可以回帖 登录 | 注册

    本版积分规则 发表回复

      

    北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
    未成年举报专区
    中国互联网协会会员  联系我们:huangweiwei@itpub.net
    感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

    清除 Cookies - ChinaUnix - Archiver - WAP - TOP