日B视频 亚洲,啪啪啪网站一区二区,91色情精品久久,日日噜狠狠色综合久,超碰人妻少妇97在线,999青青视频,亚洲一区二卡,让本一区二区视频,日韩网站推荐

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

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

3天內(nèi)不再提示

定時器原理以及一般定時器實現(xiàn)的方式

開關電源芯片 ? 來源:Linux內(nèi)核那些事 ? 作者:Linux內(nèi)核那些事 ? 2021-08-14 11:15 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

定時器原理一般定時器實現(xiàn)的方式有以下幾種:

基于排序鏈表方式:

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

基于最小堆方式:

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

基于平衡二叉樹方式:

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

時間輪:

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

時間輪 類似于日常生活的時鐘。

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

一般的定時器范圍為一個32位整型的大小,也就是 0 ~ 4294967295,如果通過一個數(shù)組來存儲的話,就需要一個元素個數(shù)為4294967296的數(shù)組,非常浪費內(nèi)存。這個時候就可以通過類似于時鐘的方式:通過多級數(shù)組來存儲。

時鐘通過時分秒來進行分級,當然我們也可以這樣,但對于計算機來說,時分秒的分級不太友好,所以Linux內(nèi)核中,對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秒。

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

每級數(shù)組上面都有一個指針,指向當前要執(zhí)行的定時器。每當時間走一秒,Linux首先會移動第一級的指針,然后執(zhí)行當前位置上的定時器。當指針變?yōu)?時,會移動下一級的指針,并把該位置上的定時器重新計算一次并且插入到時間輪中,其他級如此類推。

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

Linux時間輪的實現(xiàn)那么接下來我們看看Linux內(nèi)核是怎么實現(xiàn)時間輪算法的。

定義五個等級的數(shù)組

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

}

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

插入定時器

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() 函數(shù)的主要工作是計算定時器到期時間所屬的等級范圍,然后把定時器添加到鏈表中。

執(zhí)行到期的定時器

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

}

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

每循環(huán)一次對第一級數(shù)組指針進行加一操作,當?shù)谝患墧?shù)組指針變?yōu)?(即所有定時器都執(zhí)行完),那么就移動下一個等級的指針,并把該位置上的定時器重新計算插入到時間輪中,重新計算定時器通過 cascade_timers() 函數(shù)實現(xiàn)。

編輯:jq

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

    關注

    19

    文章

    7841

    瀏覽量

    93493
  • 定時器
    +關注

    關注

    23

    文章

    3375

    瀏覽量

    124655
  • TCP協(xié)議
    +關注

    關注

    1

    文章

    101

    瀏覽量

    12825

原文標題:一文讀懂:Linux定時器實現(xiàn)

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

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    瑞薩RA系列FSP庫開發(fā)實戰(zhàn)指南之AGT低功耗定時器簡介和結(jié)構框圖

    瑞薩RA MCU有兩種定時器外設:GPT(General PWM Timer)定時器和AGT(Asynchronous General Purpose Timer)定時器。
    的頭像 發(fā)表于 04-30 16:47 ?3696次閱讀
    瑞薩RA系列FSP庫開發(fā)實戰(zhàn)指南之AGT低功耗<b class='flag-5'>定時器</b>簡介和結(jié)構框圖

    深入剖析NE555定時器:特性、參數(shù)與應用

    ,我們就來深入了解下NE555定時器的相關特性、參數(shù)以及典型應用。 文件下載: NE555M/TR.pdf 產(chǎn)品簡介 NE555定時器內(nèi)部由閾值比較
    的頭像 發(fā)表于 04-28 15:40 ?100次閱讀

    LM555QML定時器:功能特性、應用及設計要點

    了解下它。 文件下載: lm555qml.pdf 、LM555QML簡介 LM555QML是款可直接替代SE555/NE555的定時器,能實現(xiàn)
    的頭像 發(fā)表于 02-10 15:40 ?344次閱讀

    深入解析xx555系列精密定時器:功能、應用與設計要點

    深入解析xx555系列精密定時器:功能、應用與設計要點 在電子工程師的工具箱中,定時器種至關重要的組件,它廣泛應用于各種電子設備中,用于實現(xiàn)精確的時間控制。今天,我們將深入探討xx
    的頭像 發(fā)表于 02-10 15:40 ?995次閱讀

    深入解析 LM555 定時器:特性、應用與設計要點

    深入解析 LM555 定時器:特性、應用與設計要點 、引言 在電子工程師的工具箱中,定時器芯片是常用的基礎元件之。而 TI 公司的 LM555
    的頭像 發(fā)表于 02-10 15:35 ?585次閱讀

    rt-thread軟件定時器大家一般怎么用?

    請教各位,rt-thread軟件定時器大家一般怎么用 ? 按文檔說明 軟定時器是在個單獨的任務里運行,不能在定時器里做會導致延時的操作,所
    發(fā)表于 01-12 09:39

    LAT1173高精度定時器的同步功能應用筆記

    STM32G474 所含的高精度定時器(HRTIMER)其實包含了多個定時器,多個定時器之間可以單獨工作,也可以進行同步,且高精度定時器還能與片上的其他
    發(fā)表于 01-11 17:32 ?0次下載

    LAT1183+高精度定時器中 single-shot 計數(shù)模式不工作應用筆記

    PWM 輸出,在調(diào)試模式下發(fā)現(xiàn)該子定時器的計數(shù)直為 0,即計數(shù)直沒有啟動,但如果將計數(shù)方式
    發(fā)表于 01-11 17:28 ?0次下載

    實現(xiàn)個嵌入式的軟件定時器

    ,一般可分為兩種:數(shù)組結(jié)構和鏈表結(jié)構。什么意思呢?這是(多個)軟件定時器在內(nèi)存中的存儲方式,可以用數(shù)組來存,也可以用鏈表來存。 兩者的優(yōu)劣之分就是兩種數(shù)據(jù)結(jié)構的特性之分:數(shù)組方式
    發(fā)表于 12-10 08:29

    單片機定時器中斷

    定時器/計數(shù)的工作方式寄存,確定工作方式和功能;TCON是控制寄存,控制T0,T1的啟動
    發(fā)表于 11-24 06:22

    PWM、定時器、SysTick 區(qū)別及應用場景

    。下面我們來梳理清楚。、基本概念定時器(Timer)MCU內(nèi)最基礎的計數(shù)外設,通過計數(shù)時鐘周期實現(xiàn)定時、計數(shù)功能。多數(shù)MCU內(nèi)部有多個通用定時器
    的頭像 發(fā)表于 11-17 10:53 ?720次閱讀
    PWM、<b class='flag-5'>定時器</b>、SysTick 區(qū)別及應用場景

    ?TLC551 LinCMOS? 定時器芯片技術文檔總結(jié)

    TLC551 是使用 TI LinCMOS 制造的單片定時電路^TM的^過程。這定時器與 CMOS、TTL 和 MOS 邏輯完全兼容,工作頻率高達 2 MHz。與 NE555 定時器相比,該器件由于輸入阻抗高,因此使用更小的
    的頭像 發(fā)表于 09-24 09:16 ?1056次閱讀
    ?TLC551 LinCMOS? <b class='flag-5'>定時器</b>芯片技術文檔總結(jié)

    SysTick系統(tǒng)滴答定時器簡介

    SysTick—系統(tǒng)定時器是屬于CM33內(nèi)核中的個外設,內(nèi)嵌在NVIC中。系統(tǒng)定時器個24bit的向下遞減的計數(shù),計數(shù)
    的頭像 發(fā)表于 09-23 09:50 ?1847次閱讀
    SysTick系統(tǒng)滴答<b class='flag-5'>定時器</b>簡介

    大彩講堂:VisualHMI-LUA教程-定時器的使用指南

    定時器的使用
    的頭像 發(fā)表于 08-31 16:59 ?1396次閱讀
    大彩講堂:VisualHMI-LUA教程-<b class='flag-5'>定時器</b>的使用指南

    第十二章 SysTick——系統(tǒng)定時器

    本章介紹了W55MH32的SysTick系統(tǒng)定時器,它是24位遞減計數(shù),含4個寄存,可配置定時、中斷,用于產(chǎn)生時基 等。
    的頭像 發(fā)表于 05-22 17:16 ?1267次閱讀
    第十二章 SysTick——系統(tǒng)<b class='flag-5'>定時器</b>
    印江| 山东省| 安西县| 大理市| 康乐县| 泰来县| 泾阳县| 新田县| 镇平县| 北票市| 嘉义市| 星子县| 绵竹市| 铁岭市| 建瓯市| 仁怀市| 阿图什市| 穆棱市| 万宁市| 淮安市| 上林县| 海南省| 邻水| 仁布县| 华亭县| 祁东县| 云南省| 蕲春县| 兰考县| 祁门县| 博野县| 广南县| 高尔夫| 巧家县| 隆安县| 托克托县| 麻阳| 焦作市| 台中市| 龙川县| 嘉黎县|