免费注册 查看新帖 |

Chinaunix

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

希望在该贴完美搞清LINUX TIMER问题! [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2007-05-17 11:38 |只看该作者 |倒序浏览
嗨,各位,
/***
* 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;
}
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双链中。

那么这里我有两个问题想请教,
1.timer挂起为什么可以通过
return timer->entry.next != NULL;
来判断,
是不是TIMER在过期后执行前首先要从双链中解除下来。所以上面的判断可以这样进行。
2.
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如何更新,请高手详细解释下。该字段用于

unsigned long idx = expires - base->timer_jiffies;

算式中得出idx ,利用该idx 将TIMER插入相应的双链表。即

tvec_root_t tv1;
        tvec_t tv2;
        tvec_t tv3;
        tvec_t tv4;
        tvec_t tv5;

中的一个。

timer.JPG (28.84 KB, 下载次数: 19)

timer.JPG

论坛徽章:
0
2 [报告]
发表于 2007-05-17 15:12 |只看该作者
>>1.timer挂起为什么可以通过return timer->entry.next != NULL;
来判断,是不是TIMER在过期后执行前首先要从双链中解除下来。所以上面的判断可以这样进行。



>>2.timer_jiffies如何更新

在run_timer_list中更新


  1. v2.4.0
  2. static inline void run_timer_list(void)
  3. {
  4.         spin_lock_irq(&timerlist_lock);
  5.         while ((long)(jiffies - timer_jiffies) >= 0) {//定时器到时,可能还错过了几个jiffies,补上
  6.                 struct list_head *head, *curr;
  7.                 if (!tv1.index) {// 最近要执行的定时器队列已被遍历一遍
  8.                         int n = 1;
  9.                         do {
  10.                                 cascade_timers(tvecs[n]);
  11.                         } while (tvecs[n]->index == 1 && ++n < NOOF_TVECS);
  12.                 }
  13. repeat:
  14.                 head = tv1.vec + tv1.index;
  15.                 curr = head->next;
  16.                 if (curr != head) {
  17.                         struct timer_list *timer;
  18.                         void (*fn)(unsigned long);
  19.                         unsigned long data;

  20.                         timer = list_entry(curr, struct timer_list, list);
  21.                         fn = timer->function;
  22.                         data= timer->data;

  23.                         detach_timer(timer);
  24.                         timer->list.next = timer->list.prev = NULL;
  25.                         //一次只运行一个定时器,run_timer_list是在timer_bh中运行的,
  26.                         //所以一次只有一个CPU执行run_timer_list
  27.                         timer_enter(timer);
  28.                         spin_unlock_irq(&timerlist_lock);
  29.                         fn(data);//执行定时任务
  30.                         spin_lock_irq(&timerlist_lock);
  31.                         timer_exit();
  32.                         goto repeat;
  33.                 }
  34.                 ++timer_jiffies; //更新
  35.                 tv1.index = (tv1.index + 1) & TVR_MASK;
  36.         }
  37.         spin_unlock_irq(&timerlist_lock);
  38. }
复制代码
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP