在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

定時器原理以及一般定時器實現的方式

開關電源芯片 ? 來源:Linux內核那些事 ? 作者:Linux內核那些事 ? 2021-08-14 11:15 ? 次閱讀

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

基于排序鏈表方式:

通過排序鏈表來保存定時器,由于鏈表是排序好的,所以獲取最小(最早到期)的定時器的時間復雜度為 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內核是怎么實現時間輪算法的。

定義五個等級的數組

#define TVN_BITS 6#define TVR_BITS 8#define TVN_SIZE (1 《《 TVN_BITS) // 64#define TVR_SIZE (1 《《 TVR_BITS) // 256#define TVN_MASK (TVN_SIZE - 1)#define TVR_MASK (TVR_SIZE - 1)struct timer_vec {

int index;

struct list_head vec[TVN_SIZE];

};

struct timer_vec_root {

int index;

struct list_head vec[TVR_SIZE];

};

static struct timer_vec tv5;static struct timer_vec tv4;static struct timer_vec tv3;static struct timer_vec tv2;static struct timer_vec_root tv1;void init_timervecs (void)

{

int i;

for (i = 0; i 《 TVN_SIZE; i++) {

INIT_LIST_HEAD(tv5.vec + i);

INIT_LIST_HEAD(tv4.vec + i);

INIT_LIST_HEAD(tv3.vec + i);

INIT_LIST_HEAD(tv2.vec + i);

}

for (i = 0; i 《 TVR_SIZE; i++)

INIT_LIST_HEAD(tv1.vec + i);

}

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

插入定時器

static inline void internal_add_timer(struct timer_list *timer)

{

/*

* must be cli-ed when calling this

*/

unsigned long expires = timer-》expires;

unsigned long idx = expires - timer_jiffies;

struct list_head * vec;

if (idx 《 TVR_SIZE) { // 0 ~ 255

int i = expires & TVR_MASK;

vec = tv1.vec + i;

} else if (idx 《 1 《《 (TVR_BITS + TVN_BITS)) { // 256 ~ 16191

int i = (expires 》》 TVR_BITS) & TVN_MASK;

vec = tv2.vec + i;

} else if (idx 《 1 《《 (TVR_BITS + 2 * TVN_BITS)) {

int i = (expires 》》 (TVR_BITS + TVN_BITS)) & TVN_MASK;

vec = tv3.vec + i;

} else if (idx 《 1 《《 (TVR_BITS + 3 * TVN_BITS)) {

int i = (expires 》》 (TVR_BITS + 2 * TVN_BITS)) & TVN_MASK;

vec = tv4.vec + i;

} else if ((signed long) idx 《 0) {

/* can happen if you add a timer with expires == jiffies,

* or you set a timer to go off in the past

*/

vec = tv1.vec + tv1.index;

} else if (idx 《= 0xffffffffUL) {

int i = (expires 》》 (TVR_BITS + 3 * TVN_BITS)) & TVN_MASK;

vec = tv5.vec + i;

} else {

/* Can only get here on architectures with 64-bit jiffies */

INIT_LIST_HEAD(&timer-》list);

return;

}

/*

* 添加到鏈表中

*/

list_add(&timer-》list, vec-》prev);

}

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

執行到期的定時器

static inline void cascade_timers(struct timer_vec *tv)

{

/* cascade all the timers from tv up one level */

struct list_head *head, *curr, *next;

head = tv-》vec + tv-》index;

curr = head-》next;

/*

* We are removing _all_ timers from the list, so we don‘t have to

* detach them individually, just clear the list afterwards.

*/

while (curr != head) {

struct timer_list *tmp;

tmp = list_entry(curr, struct timer_list, list);

next = curr-》next;

list_del(curr);

internal_add_timer(tmp);

curr = next;

}

INIT_LIST_HEAD(head);

tv-》index = (tv-》index + 1) & TVN_MASK;

}

static inline void run_timer_list(void)

{

spin_lock_irq(&timerlist_lock);

while ((long)(jiffies - timer_jiffies) 》= 0) {

struct list_head *head, *curr;

if (!tv1.index) { // 完成了一個輪回, 移動下一個單位的定時器

int n = 1;

do {

cascade_timers(tvecs[n]);

} while (tvecs[n]-》index == 1 && ++n 《 NOOF_TVECS);

}

repeat:

head = tv1.vec + tv1.index;

curr = head-》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;

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);

}

執行到期的定時器主要通過 run_timer_list() 函數完成,該函數首先比較當前時間與最后一次運行 run_timer_list() 函數時間的差值,然后循環這個差值的次數,并執行當前指針位置上的定時器。

每循環一次對第一級數組指針進行加一操作,當第一級數組指針變為0(即所有定時器都執行完),那么就移動下一個等級的指針,并把該位置上的定時器重新計算插入到時間輪中,重新計算定時器通過 cascade_timers() 函數實現。

編輯:jq

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 計算機
    +關注

    關注

    19

    文章

    7637

    瀏覽量

    90327
  • 定時器
    +關注

    關注

    23

    文章

    3290

    瀏覽量

    117422
  • TCP協議
    +關注

    關注

    1

    文章

    101

    瀏覽量

    12372

原文標題:一文讀懂:Linux定時器實現

文章出處:【微信號:gh_3980db2283cd,微信公眾號:開關電源芯片】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦
    熱點推薦

    MCU定時器/計數

    RISC-V核低功耗MCU通過靈活的定時器架構、低功耗模式適配及硬件級中斷優化,在工業控制、智能家居等場景中實現高精度計時與能耗控制的協同設計,滿足復雜任務調度與實時響應的雙重需求?。 、?硬件
    的頭像 發表于 04-27 13:54 ?205次閱讀

    詳解CKS32F107xx系列的定時器同步功能

    CKS32F107xx系列部分定時器在內部是相連的,可用于定時器同步或鏈接,方便用戶配置不同的同步模式,以便在電機控制、數據采集和PWM信號生成等應用中,實現復雜的時間序列和多通道的同步操作。本節課
    的頭像 發表于 11-26 17:51 ?982次閱讀
    詳解CKS32F107xx系列的<b class='flag-5'>定時器</b>同步功能

    定時器自動控制開關怎么設置

    定時器自動控制開關是種常見的自動化設備,廣泛應用于家庭、工業、農業等多個領域。通過定時器,用戶可以預設時間,讓設備在特定時間自動開啟或關閉,從而
    的頭像 發表于 09-19 16:19 ?3703次閱讀

    定時器的基本組成和工作模式

    定時器是計算機或電子設備中常見的個硬件或軟件組件,其主要功能是測量和控制時間的流逝。它在各種應用中起著至關重要的作用,如操作系統調度、多媒體播放、網絡通信、工業自動化控制以及家電設備的定時
    的頭像 發表于 08-19 18:28 ?2575次閱讀

    定時器的工作方式介紹

    定時器是計算機和嵌入式系統中常見的種硬件模塊,用于實現定時和計數功能。定時器的工作方式通常由
    的頭像 發表于 07-12 10:29 ?1808次閱讀

    定時器相關的寄存有哪些類型

    在微控制編程中,定時器種非常常見的功能模塊,用于實現各種定時和計數功能。定時器的工作原理是
    的頭像 發表于 07-12 10:25 ?1603次閱讀

    鴻蒙開發系統基礎能力:Timer定時器

    設置定時器,該定時器定時器到期后執行個函數。
    的頭像 發表于 06-28 11:33 ?1392次閱讀
    鴻蒙開發系統基礎能力:Timer<b class='flag-5'>定時器</b>

    長持續時間定時器電路圖 時間定時器的工作原理和功能

    時間定時器種用于計時和調度任務的工具。它允許我們在特定的時間間隔內執行某個任務,或者在特定的時間點執行某個操作。定時器在計算機系統中的應用非常廣泛,從操作系統的任務調度、網絡傳輸的控制到實時系統
    的頭像 發表于 06-24 17:34 ?4290次閱讀
    長持續時間<b class='flag-5'>定時器</b>電路圖 時間<b class='flag-5'>定時器</b>的工作原理和功能

    使用Arduino的可變定時器繼電器設計

    定時器繼電器,也被稱為時間繼電器,是種在設定的時間間隔后觸發特定操作的電子元件。它在工業自動化、家庭生活、交通管理等領域具有廣泛的應用。定時器繼電器的主要功能是通過設定特定的時間間隔,來控制電路的通斷,
    的頭像 發表于 06-24 16:52 ?1524次閱讀
    使用Arduino的可變<b class='flag-5'>定時器</b>繼電器設計

    定時器繼電器的工作原理和分類

    在電氣控制和自動化系統中,定時器繼電器是種關鍵的電氣控制元件。它通過設定特定的時間間隔,來控制電路的通斷,實現設備的自動化操作。本文將對定時器繼電器的定義、特點、工作原理、分類、應用
    的頭像 發表于 06-21 18:06 ?1665次閱讀

    三菱PLC編程實現讀出時間定時器

    的功能,而定時器實現時間控制的關鍵組件。本文將詳細介紹如何使用三菱PLC編程實現讀出時間定時器。 1. 定時器的基本概念
    的頭像 發表于 06-20 11:11 ?4399次閱讀

    三菱plc如何顯示定時器時間

    三菱PLC(Programmable Logic Controller,可編程邏輯控制)是種廣泛應用于工業自動化領域的設備。定時器是PLC中的種基本功能,用于
    的頭像 發表于 06-20 11:10 ?3198次閱讀

    三菱plc定時器范圍怎么設置

    三菱PLC(Programmable Logic Controller,可編程邏輯控制)是種廣泛應用于工業自動化領域的控制設備。在三菱PLC中,定時器種重要的編程元件,用于
    的頭像 發表于 06-20 11:04 ?4498次閱讀

    三菱plc定時器最大設定值

    定時器種非常重要的功能,用于實現時間控制和延時控制。 、三菱PLC定時器的基本概念 定時器
    的頭像 發表于 06-20 11:03 ?3319次閱讀

    三菱plc編程定時器編程咋輸入

    三菱PLC(Programmable Logic Controller,可編程邏輯控制)是種廣泛應用于工業自動化控制領域的設備。在三菱PLC中,定時器實現時間控制功能的重要組件。
    的頭像 發表于 06-20 11:01 ?2751次閱讀
    主站蜘蛛池模板: 亚洲精品在线免费观看视频 | 国语对白老女人8av 孩交精品xxxx视频视频 | 天天操 夜夜操 | 日本三浦理惠子中文字幕 | 日本不卡免费高清一级视频 | 国产午夜a理论毛片在线影院 | 男人的网址| 亚洲欧美日韩在线观看你懂的 | 成人久久久精品乱码一区二区三区 | 欧美成人伊人十综合色 | 免费人成在线观看网站品爱网日本 | 天堂视频网 | 色多多免费观看 | 91网址在线播放 | 久久国产乱子伦精品免 | 国产午夜精品福利久久 | 国产精品性 | 亚偷熟乱区视频在线观看 | 五月六月婷婷 | 午夜精品久久久久久久四虎 | 午夜精品一区二区三区在线观看 | 四虎4hu永久免费国产精品 | 91md天美精东蜜桃传媒在线 | 视频 在线| 日本成人福利视频 | 日本一卡二卡3卡四卡网站精品 | 成人午夜大片免费7777 | 五月在线观看 | 在线观看亚洲免费视频 | 天天爽天天干天天操 | 国产亚洲高清在线精品不卡 | 国产精品久久在线观看 | 日本黄色片www | 久久久久久久久女黄 | 国产一区二区三区不卡观 | 天天射天天干 | 欧美人与禽交 | 四虎国产精品永久地址51 | 免费黄色福利视频 | 天天色啪| gay超刺激污文 |