成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

一文讀懂 Linux 定時器實現

系統 Linux
本文介紹了Linux定時器的相關內容,希望能幫到你。一起來看一下吧。

[[416836]]

定時器原理

一般定時器實現的方式有以下幾種:

基于排序鏈表方式:

通過排序鏈表來保存定時器,由于鏈表是排序好的,所以獲取最小(最早到期)的定時器的時間復雜度為 O(1)。但插入需要遍歷整個鏈表,所以時間復雜度為 O(n)。如下圖:

圖片

基于最小堆方式:

通過最小堆來保存定時器,在最小堆中獲取最小定時器的時間復雜度為 O(1),但插入一個定時器的時間復雜度為 O(log n)。如下圖:

圖片

基于平衡二叉樹方式:

使用平衡二叉樹(如紅黑樹)保存定時器,在平衡二叉樹中獲取最小定時器的時間復雜度為 O(log n)(也可以通過緩存最小值的方法來達到 O(1)),而插入一個定時器的時間復雜度為 O(log n)。如下圖:

圖片

時間輪:

但對于Linux這種對定時器依賴性比較高(網絡子模塊的TCP協議使用了大量的定時器)的操作系統來說,以上的數據結構都是不能滿足要求的。所以Linux使用了效率更高的定時器算法:時間輪。

時間輪 類似于日常生活的時鐘,如下圖:

圖片

日常生活的時鐘,每當秒針轉一圈時,分針就會走一格,而分針走一圈時,時針就會走一格。而時間輪的實現方式與時鐘類似,就是把到期時間當成一個輪,然后把定時器掛在這個輪子上面,每當時間走一秒就移動時針,并且執行那個時針上的定時器,如下圖:

圖片

一般的定時器范圍為一個32位整型的大小,也就是 0 ~ 4294967295,如果通過一個數組來存儲的話,就需要一個元素個數為4294967296的數組,非常浪費內存。這個時候就可以通過類似于時鐘的方式:通過多級數組來存儲。時鐘通過時分秒來進行分級,當然我們也可以這樣,但對于計算機來說,時分秒的分級不太友好,所以Linux內核中,對32位整型分為5個級別,第一個等級存儲0 ~ 255秒 的定時器,第二個等級為 256秒 ~ 256*64秒,第三個等級為 256*64秒 ~ 256*64*64秒,第四個等級為 256*64*64秒 ~ 256*64*64*64秒,第五個等級為 256*64*64*64秒 ~ 256*64*64*64*64秒。如下圖:

圖片

注意:第二級至第五級數組的第一個槽是不掛任何定時器的。

每級數組上面都有一個指針,指向當前要執行的定時器。每當時間走一秒,Linux首先會移動第一級的指針,然后執行當前位置上的定時器。當指針變為0時,會移動下一級的指針,并把該位置上的定時器重新計算一次并且插入到時間輪中,其他級如此類推。如下圖所示:

圖片

當要執行到期的定時器只需要移動第一級數組上的指針并且執行該位置上的定時器列表即可,所以時間復雜度為 O(1),而插入一個定時器也很簡單,先計算定時器的過期時間范圍在哪一級數組上,并且連接到該位置上的鏈表即可,時間復雜度也是 O(1)。

Linux時間輪的實現

那么接下來我們看看Linux內核是怎么實現時間輪算法的。

定義五個等級的數組 

  1. #define TVN_BITS 6  
  2. #define TVR_BITS 8  
  3. #define TVN_SIZE (1 << TVN_BITS)  // 64  
  4. #define TVR_SIZE (1 << TVR_BITS)  // 256  
  5. #define TVN_MASK (TVN_SIZE - 1)  
  6. #define TVR_MASK (TVR_SIZE - 1)  
  7. struct timer_vec {  
  8.     int index;  
  9.     struct list_head vec[TVN_SIZE];  
  10. };  
  11. struct timer_vec_root {  
  12.     int index;  
  13.     struct list_head vec[TVR_SIZE];  
  14. };  
  15. static struct timer_vec tv5; 
  16. static struct timer_vec tv4;  
  17. static struct timer_vec tv3;  
  18. static struct timer_vec tv2;  
  19. static struct timer_vec_root tv1;  
  20. void init_timervecs (void)  
  21.  
  22.     int i;  
  23.     for (i = 0; i < TVN_SIZE; i++) {  
  24.         INIT_LIST_HEAD(tv5.vec + i);  
  25.         INIT_LIST_HEAD(tv4.vec + i);  
  26.         INIT_LIST_HEAD(tv3.vec + i);  
  27.         INIT_LIST_HEAD(tv2.vec + i);  
  28.     }  
  29.     for (i = 0; i < TVR_SIZE; i++)  
  30.         INIT_LIST_HEAD(tv1.vec + i);  

上面的代碼定義第一級數組為 timer_vec_root 類型,其 index 成員是當前要執行的定時器指針(對應 vec 成員的下標),而 vec 成員是一個鏈表數組,數組元素個數為256,每個元素上保存了該秒到期的定時器列表,其他等級的數組類似。

插入定時器 

  1. static inline void internal_add_timer(struct timer_list *timer)  
  2.  
  3.     /*  
  4.      * must be cli-ed when calling this  
  5.      */  
  6.     unsigned long expires = timer->expires;  
  7.     unsigned long idx = expires - timer_jiffies;  
  8.     struct list_head * vec;  
  9.     if (idx < TVR_SIZE) { // 0 ~ 255  
  10.         int i = expires & TVR_MASK;  
  11.         vec = tv1.vec + i;  
  12.     } else if (idx < 1 << (TVR_BITS + TVN_BITS)) { // 256 ~ 16191  
  13.         int i = (expires >> TVR_BITS) & TVN_MASK;  
  14.         vec = tv2.vec + i;  
  15.     } else if (idx < 1 << (TVR_BITS + 2 * TVN_BITS)) {  
  16.         int i = (expires >> (TVR_BITS + TVN_BITS)) & TVN_MASK;  
  17.         vec =  tv3.vec + i;  
  18.     } else if (idx < 1 << (TVR_BITS + 3 * TVN_BITS)) {  
  19.         int i = (expires >> (TVR_BITS + 2 * TVN_BITS)) & TVN_MASK;  
  20.         vec = tv4.vec + i;  
  21.     } else if ((signed long) idx < 0) { 
  22.         /* can happen if you add a timer with expires == jiffies,  
  23.          * or you set a timer to go off in the past  
  24.          */  
  25.         vec = tv1.vec + tv1.index;  
  26.     } else if (idx <= 0xffffffffUL) {  
  27.         int i = (expires >> (TVR_BITS + 3 * TVN_BITS)) & TVN_MASK;  
  28.         vec = tv5.vec + i;  
  29.     } else {  
  30.         /* Can only get here on architectures with 64-bit jiffies */  
  31.         INIT_LIST_HEAD(&timer->list);  
  32.         return;  
  33.     }  
  34.     /*  
  35.      * 添加到鏈表中  
  36.      */  
  37.     list_add(&timer->list, vec->prev);  

internal_add_timer() 函數的主要工作是計算定時器到期時間所屬的等級范圍,然后把定時器添加到鏈表中。

執行到期的定時器 

  1. static inline void cascade_timers(struct timer_vec *tv)  
  2.  
  3.     /* cascade all the timers from tv up one level */  
  4.     struct list_head *head, *curr, *next;  
  5.     head = tv->vec + tv->index;  
  6.     curr = head->next;  
  7.     /*  
  8.      * We are removing _all_ timers from the list, so we don't  have to  
  9.      * detach them individually, just clear the list afterwards.  
  10.      */  
  11.     while (curr != head) {  
  12.         struct timer_list *tmp;  
  13.         tmp = list_entry(curr, struct timer_list, list);  
  14.         next = curr->next;  
  15.         list_del(curr);  
  16.         internal_add_timer(tmp);  
  17.         curr = next 
  18.     }  
  19.     INIT_LIST_HEAD(head);  
  20.     tv->index = (tv->index + 1) & TVN_MASK;  
  21.  
  22. static inline void run_timer_list(void)  
  23.  
  24.     spin_lock_irq(&timerlist_lock);  
  25.     while ((long)(jiffies - timer_jiffies) >= 0) {  
  26.         struct list_head *head, *curr;  
  27.         if (!tv1.index) { // 完成了一個輪回, 移動下一個單位的定時器  
  28.             int n = 1 
  29.             do {  
  30.                 cascade_timers(tvecs[n]);  
  31.             } while (tvecs[n]->index == 1 && ++n < NOOF_TVECS);  
  32.         }  
  33. repeat:  
  34.         head = tv1.vec + tv1.index;  
  35.         curr = head->next;  
  36.         if (curr != head) {  
  37.             struct timer_list *timer;  
  38.             void (*fn)(unsigned long);  
  39.             unsigned long data;   
  40.             timer = list_entry(curr, struct timer_list, list);  
  41.             fn = timer->function;  
  42.             datatimer->data;  
  43.             detach_timer(timer);  
  44.             timer->list.next = timer->list.prev = NULL 
  45.             timer_enter(timer);  
  46.             spin_unlock_irq(&timerlist_lock);  
  47.             fn(data); 
  48.             spin_lock_irq(&timerlist_lock);  
  49.             timer_exit();  
  50.             goto repeat;  
  51.         }  
  52.         ++timer_jiffies;  
  53.         tv1.index = (tv1.index + 1) & TVR_MASK;  
  54.     }  
  55.     spin_unlock_irq(&timerlist_lock);  

執行到期的定時器主要通過 run_timer_list() 函數完成,該函數首先比較當前時間與最后一次運行 run_timer_list() 函數時間的差值,然后循環這個差值的次數,并執行當前指針位置上的定時器。每循環一次對第一級數組指針進行加一操作,當第一級數組指針變為0(即所有定時器都執行完),那么就移動下一個等級的指針,并把該位置上的定時器重新計算插入到時間輪中,重新計算定時器通過 cascade_timers() 函數實現。 

責任編輯:龐桂玉 來源: 良許Linux
相關推薦

2023-02-28 18:09:53

Javascript定時器

2022-09-21 09:04:07

Python裝飾器

2023-12-22 19:59:15

2021-08-04 16:06:45

DataOps智領云

2021-11-02 10:53:56

Linux機制CPU

2020-12-29 09:56:29

瀏覽器緩存HTTP

2023-03-03 08:26:32

負載均衡算法服務

2018-09-28 14:06:25

前端緩存后端

2022-09-22 09:00:46

CSS單位

2022-11-06 21:14:02

數據驅動架構數據

2025-04-03 10:56:47

2023-11-27 17:35:48

ComponentWeb外層

2023-05-20 17:58:31

低代碼軟件

2022-07-05 06:30:54

云網絡網絡云原生

2022-07-26 00:00:03

語言模型人工智能

2022-12-01 17:23:45

2021-12-29 18:00:19

無損網絡網絡通信網絡

2022-10-20 08:01:23

2021-10-20 07:18:51

Linux延時隊列

2020-12-30 09:05:24

架構微內核系統
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 精品一区二区三区免费毛片 | 久久国产成人 | 激情在线视频网站 | 日韩一区二区免费视频 | 成人免费黄视频 | 亚洲欧美日韩成人在线 | 国产精品久久久久久久久久久免费看 | 久久久精品网站 | 一级做a爰片性色毛片16美国 | 日韩欧美成人精品 | 91视视频在线观看入口直接观看 | 欧美日韩国产中文 | 亚洲综合精品 | 国产精品久久久久久久久久免费看 | 国产精品久久亚洲7777 | 欧美成人一区二免费视频软件 | 久久久久久久久久一区 | 一区二区三区亚洲视频 | 久久99网 | 五月激情综合 | 国产精品欧美一区二区三区不卡 | 精国产品一区二区三区四季综 | 欧美黑人体内she精在线观看 | 99精品久久久久久中文字幕 | 精品国产欧美在线 | 欧美一级大片免费看 | 久久亚洲一区二区三区四区 | 四虎永久免费黄色影片 | 极品粉嫩国产48尤物在线播放 | 日本三级黄视频 | 日本a视频| 欧美日韩综合精品 | 欧美a级成人淫片免费看 | 欧美偷偷操 | 亚洲精品久久久久久久不卡四虎 | 国内av在线 | 一区在线播放 | 国产色在线 | 国产在线精品一区二区三区 | 日韩亚洲视频在线 | 精品伦精品一区二区三区视频 |