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

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

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

Linux驅動開發(fā)-內(nèi)核定時器

DS小龍哥-嵌入式技術 ? 2022-09-17 15:06 ? 次閱讀

【摘要】 內(nèi)核定時器是內(nèi)核用來控制在未來某個時間點(基于jiffies(節(jié)拍總數(shù)))調(diào)度執(zhí)行某個函數(shù)的一種機制,相關函數(shù)位于 和 kernel/timer.c 文件中。 當內(nèi)核定時器定時時間到達時,會進入用戶指定的函數(shù),相當于軟中斷。內(nèi)核定時器注冊開啟后,運行一次就不會再運行(相當于自動注銷),我們可以重新設置定時器的超時時間,讓定時器重復運行。

1. 內(nèi)核定時器介紹

內(nèi)核定時器是內(nèi)核用來控制在未來某個時間點(基于jiffies(節(jié)拍總數(shù)))調(diào)度執(zhí)行某個函數(shù)的一種機制,相關函數(shù)位于 和 kernel/timer.c 文件中。

當內(nèi)核定時器定時時間到達時,會進入用戶指定的函數(shù),相當于軟中斷。內(nèi)核定時器注冊開啟后,運行一次就不會再運行(相當于自動注銷),我們可以重新設置定時器的超時時間,讓定時器重復運行。

每當時鐘中斷發(fā)生時,全局變量jiffies(一個32位的unsigned long 變量)就加1,因此jiffies記錄了linux系統(tǒng)啟動后時鐘中斷發(fā)生的次數(shù),驅動程序常利用jiffies來計算不同事件間的時間間隔。內(nèi)核每秒鐘將jiffies變量增加HZ次。因此,對于HZ值為100的系統(tǒng),jiffy+1等于隔了10ms,而對于HZ為1000的系統(tǒng),jiffy+1僅為1ms。

內(nèi)核定時器結構體:
下面列出了需要關心的成員

struct timer_list {
	unsigned long expires;         //設置超時時間,用jiffies作為基準值
	void (*function)(unsigned long); //類似中斷服務函數(shù),設置定時器到時后處理的函數(shù) 
	unsigned long data;           //中斷服務函數(shù)的參數(shù)
}

expires設置:以當前時間為基準加上延時時間,時間基準用jiffies變量表示,延時時間可以使用以下兩個宏轉換成jiffies單位。

2. 內(nèi)核定時器相關API函數(shù)

2.1 修改定時器超時時間

函數(shù)原型 *int mod_timer(struct timer_list timer, unsigned long expires)
函數(shù)功能 修改定時器超時時間
函數(shù)參數(shù) timer:對應的定時器結構體 expires:超時時間
函數(shù)返回值 成功返回 :修改成功的時間值
函數(shù)定義文件 \linux-3.5\kernel\timer.c

2.2 初始化定時器

函數(shù)原型 #define init_timer(timer)\
函數(shù)功能 初始化定時器結構
函數(shù)參數(shù) timer:對應的定時器結構體
函數(shù)定義文件 \linux-3.5\include\linux\timer.h

2.3 關閉定時器

函數(shù)原型 int del_timer(struct timer_list *timer)
函數(shù)功能 關閉定時器,停用一個定時器。
函數(shù)參數(shù) timer:對應的定 時器結構體
函數(shù)返回值 返回0:成功
函數(shù)定義文件 \linux-3.5\include\linux\timer.h

2.4 關閉定時器

函數(shù)原型 int del_timer_sync(struct timer_list *timer)
函數(shù)功能 關閉定時器,停用一個定時器,多處理器使用。如果編內(nèi)核時不支持 SMP(多處理器), del_timer_sync()和 del_timer()等價
函數(shù)參數(shù) timer:對應的定時器結構體
函數(shù)返回值 返回0:成功
函數(shù)定義文件 \linux-3.5\include\linux\timer.h

2.5 轉換時間(微妙單位)

函數(shù)原型 unsigned long usecs_to_jiffies(const unsigned int m)
函數(shù)功能 轉換時間(微妙單位),用于填充定時器結構體,設置超時時間
函數(shù)參數(shù) m:要轉換的時間值(微妙為單位)
函數(shù)返回值 成功返回轉換成功的時間。用于填充定時器結構體,設置超時時間
函數(shù)定義文件 \linux-3.5\kernel\timer.c

2.6 轉換時間(毫秒為單位)

函數(shù)原型 unsigned long msecs_to_jiffies(const unsigned int m)
函數(shù)功能 轉換時間(毫秒為單位),用于填充定時器結構體,設置超時時間
函數(shù)參數(shù) m:要轉換的時間值(毫秒為單位)
函數(shù)返回值 成功返回轉換成功的時間。用于填充定時器結構體,設置超時時間
函數(shù)定義文件 \linux-3.5\kernel\timer.c

將jiffies單位轉為struct timespec結構體表示:

Void jiffies_to_timespec(const unsigned long jiffies, struct timespec *value);

示例:
jiffies_to_timespec(jiffies,&value);
printk("value.ts_sec=%d\n",value.tv_sec);
printk("value.tv_nsec=%d\n",value.tv_nsec);

2.7 初始化定時器的結構體成員

TIMER_INITIALIZER( _function, _expires, _data) 宏用于賦值定時器結構體的function、 expires、 data 和 base 成員, 這個宏的定義如下所示:(被DEFINE_TIMER宏調(diào)用)

#define TIMER_INITIALIZER(_function, _expires, _data) {		\
		.entry = { .prev = TIMER_ENTRY_STATIC },	\
		.function = (_function),			\
		.expires = (_expires),				\
		.data = (_data),				\
		.base = &boot_tvec_bases,			\
		.slack = -1,					\
		__TIMER_LOCKDEP_MAP_INITIALIZER(		\
			__FILE__ ":" __stringify(__LINE__))	\
	}

2.8 初始化定時器并且賦值

DEFINE_TIMER( _na me , _functi o n, _e x pires, _data) 宏是定義并初始化定時器成員的“快捷方式”, 這個宏定義如下所示:

/*初始化定時器,并進行賦值*/
#define DEFINE_TIMER(_name, _function, _expires, _data)		\
	struct timer_list _name =				\
		TIMER_INITIALIZER(_function, _expires, _data)

2.9 定時器初始化賦值

setup_timer()也可用于初始化定時器并賦值其成員, 其源代碼如下:

//初始化定時器并進行賦值
#define setup_timer(timer, fn, data)					\
	do {								\
		static struct lock_class_key __key;			\
		setup_timer_key((timer), #timer, &__key, (fn), (data));\
	} while (0)

static inline void setup_timer_key(struct timer_list * timer,
				const char *name,
				struct lock_class_key *key,
				void (*function)(unsigned long),
				unsigned long data)
{
	timer->function = function;
	timer->data = data;
	init_timer_key(timer, name, key);
}

3. 使用定時器的步驟

(1) 定義定時器結構體timer_list。

/*定義一個內(nèi)核定時器配置結構體*/
static struct timer_list mytimer ; 

(2) 設置超時時間,定義定時器處理函數(shù)和傳參。

mytimer.expires=jiffies+ msecs_to_jiffies(1000); /*設置定時器的超時時間,1000毫秒*/
//或者
//mytimer.expires=jiffies+HZ; /*設置定時器的超時時間,1000毫秒*/

mytimer.function = time_fun;	              /*定時器超時的回調(diào)函數(shù),類似中斷服務函數(shù)*/
mytimer.data = 12;                       /*傳給定時器服務函數(shù)的參數(shù)*/

(3) 開啟定時器。

init_timer(&mytimer);          /*初始化定時器*/
add_timer(&mytimer);	        /*啟動定時器*/

完整示例代碼:

#include 
#include 
#include 

static struct timer_list timer;

static void timer_function(unsigned long data)
{
	printk("data=%ld\n",data);
	mod_timer(&timer,msecs_to_jiffies(3000)+jiffies);
}

static int __init tiny4412_linux_timer_init(void)
{
	timer.expires=HZ*3+jiffies; /*單位是節(jié)拍*/
	timer.function=timer_function;
	timer.data=666;
	
	/*1. 初始化定時器*/
	init_timer(&timer);
	/*2. 添加定時器到內(nèi)核*/
	add_timer(&timer);
    printk("驅動測試: 驅動安裝成功\n");
    return 0;
}

static void __exit tiny4412_linux_timer_cleanup(void)
{
	/*3. 刪除定時器*/
	del_timer_sync(&timer);
    printk("驅動測試: 驅動卸載成功\n");
}

module_init(tiny4412_linux_timer_init);    /*驅動入口--安裝驅動的時候執(zhí)行*/
module_exit(tiny4412_linux_timer_cleanup); /*驅動出口--卸載驅動的時候執(zhí)行*/

MODULE_LICENSE("GPL");  /*設置模塊的許可證--GPL*/

4. 內(nèi)核提供的延時函數(shù)

Linux 內(nèi)核中提供了進行納秒、微秒和毫秒延遲。
void ndelay(unsigned long nsecs) ;
void udelay(unsigned long usecs) ;
void mdelay(unsigned long msecs) ;
上述延遲的實現(xiàn)原理本質(zhì)上是忙等待,根據(jù) CPU 頻率進行一定次數(shù)的循環(huán)。在內(nèi)核中,最好不要直接使用mdelay()函數(shù), 這將無謂地耗費CPU資源。
void msleep(unsigned int millisecs) ;
unsigned long msleep_interruptible(unsigned int millisecs) ;
void ssleep(unsigned int seconds) ;
上述函數(shù)將使得調(diào)用它的進程睡眠參數(shù)指定的時間, msleep()、 ssleep()不能被打斷,而 msleep_interruptible()則可以被打斷。

5. 精度較高的時間獲取方式

高精度定時器通常用ktime作為計時單位。
獲取內(nèi)核高精度時間單位: ktime_t ktime_get(void)

下面是一些時間輔助函數(shù)用于計算和轉換:

ktime_t ktime_set(const long secs, const unsigned long nsecs);   
ktime_t ktime_sub(const ktime_t lhs, const ktime_t rhs);   
ktime_t ktime_add(const ktime_t add1, const ktime_t add2);   
ktime_t ktime_add_ns(const ktime_t kt, u64 nsec);   
ktime_t ktime_sub_ns(const ktime_t kt, u64 nsec);   
ktime_t timespec_to_ktime(const struct timespec ts);   
ktime_t timeval_to_ktime(const struct timeval tv);   
struct timespec ktime_to_timespec(const ktime_t kt);   //轉換的時間通過timespec結構體保存
struct timeval ktime_to_timeval(const ktime_t kt);     //轉換的時間通過timeval結構體保存
s64 ktime_to_ns(const ktime_t kt);     //轉換為ns單位
int ktime_equal(const ktime_t cmp1, const ktime_t cmp2);   
s64 ktime_to_us(const ktime_t kt);    //轉換為us單位
s64 ktime_to_ms(const ktime_t kt);    //轉換為ms單位
ktime_t ns_to_ktime(u64 ns);

示例: 計算經(jīng)過的一段時間

static int hello_init(void)
{
	ktime_t my_time,my_time2;
	unsigned int i,j;
	unsigned int time_cnt=0;
	my_time=ktime_get();    		//獲取當前時間
	i=ktime_to_us(my_time); 		//轉us
	
	udelay(600);  				//延時一段時間
	
	my_time2=ktime_get();  	  	//獲取當前時間
	j=ktime_to_us(my_time2);  		//轉us
	
	printk("time_cnt=%ld\n",j-i); 	//得出之間差值,正確值為: 600
	return 0;
}
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 內(nèi)核
    +關注

    關注

    3

    文章

    1361

    瀏覽量

    40185
  • 定時器
    +關注

    關注

    23

    文章

    3232

    瀏覽量

    114331
  • 函數(shù)
    +關注

    關注

    3

    文章

    4284

    瀏覽量

    62325
收藏 人收藏

    評論

    相關推薦

    linux驅動程序如何加載進內(nèi)核

    Linux系統(tǒng)中,驅動程序是內(nèi)核與硬件設備之間的橋梁。它們允許內(nèi)核與硬件設備進行通信,從而實現(xiàn)對硬件設備的控制和管理。 驅動程序的編寫
    的頭像 發(fā)表于 08-30 15:02 ?347次閱讀

    Linux 驅動開發(fā)與應用開發(fā),你知道多少?

    一、Linux驅動開發(fā)與應用開發(fā)的區(qū)別開發(fā)層次不同:Linux
    的頭像 發(fā)表于 08-30 12:16 ?520次閱讀
    <b class='flag-5'>Linux</b> <b class='flag-5'>驅動</b><b class='flag-5'>開發(fā)</b>與應用<b class='flag-5'>開發(fā)</b>,你知道多少?

    定時器的工作方式介紹

    定時器是計算機和嵌入式系統(tǒng)中常見的一種硬件模塊,用于實現(xiàn)定時和計數(shù)功能。定時器的工作方式通常由一組寄存來控制,這些寄存定義了
    的頭像 發(fā)表于 07-12 10:29 ?654次閱讀

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

    在微控制編程中,定時器是一種非常常見的功能模塊,用于實現(xiàn)各種定時和計數(shù)功能。定時器的工作原理是通過內(nèi)部的計數(shù)來跟蹤時間的流逝,當計數(shù)
    的頭像 發(fā)表于 07-12 10:25 ?682次閱讀

    鴻蒙開發(fā)系統(tǒng)基礎能力:Timer定時器

    設置一個定時器,該定時器定時器到期后執(zhí)行一個函數(shù)。
    的頭像 發(fā)表于 06-28 11:33 ?788次閱讀
    鴻蒙<b class='flag-5'>開發(fā)</b>系統(tǒng)基礎能力:Timer<b class='flag-5'>定時器</b>

    如何實現(xiàn)一個軟件定時器?

    Linux,uC/OS,F(xiàn)reeRTOS等操作系統(tǒng)中,都帶有軟件定時器,原理大同小異。典型的實現(xiàn)方法是:通過一個硬件定時器產(chǎn)生固定的時鐘節(jié)拍,每次硬件定時器中斷到,就對一個全局的時間
    的頭像 發(fā)表于 04-29 11:00 ?568次閱讀

    s7200定時器的五種故障介紹

    定時器或CPU故障:如果定時器本身或PLC的CPU出現(xiàn)故障,也可能導致定時器無法復位。此時,需要檢查定時器和CPU的工作狀態(tài),確保其正常運行。
    的頭像 發(fā)表于 04-03 17:08 ?2098次閱讀

    ?PLC定時器介紹

    定時器是PLC中重要的編程元件,是累計時間增量的內(nèi)部器件。大部分自動控制領域都需要定時器進行延時控制,靈活地使用定時器可以編制出復雜的控制程序。
    發(fā)表于 03-22 12:36 ?2075次閱讀
    ?PLC<b class='flag-5'>定時器</b>介紹

    使用555定時器的可調(diào)雙定時器電路

    定時器 IC 555 是最通用和最常用的 IC 之一,因為它的應用范圍更廣,如 PWM放大器、延遲定時器、開關電路、占空比選擇、時鐘脈沖發(fā)生等。這也可用于各種應用,如精確
    的頭像 發(fā)表于 02-25 15:16 ?1951次閱讀
    使用555<b class='flag-5'>定時器</b>的可調(diào)雙<b class='flag-5'>定時器</b>電路

    定時器原理能控制馬達嗎為什么

    定時器原理可以用于控制馬達。馬達是一種將電能轉換為機械能的設備,通常由電動機和傳動裝置組成。定時器是一種電子設備,用來生成和計時精確而穩(wěn)定的時間信號。通過將定時器與馬達控制電路相連,可以實現(xiàn)對馬達
    的頭像 發(fā)表于 01-23 15:21 ?583次閱讀

    555定時器的基本功能 555定時器的工作原理及其應用

    555定時器是一種非常常見和常用的集成電路,它具有廣泛的應用領域,例如計時、頻率分頻、脈沖寬度調(diào)制等。本文將詳細介紹555定時器的基本功能、工作原理以及應用。 一、555定時器的基本功能 555
    的頭像 發(fā)表于 01-18 11:12 ?1.3w次閱讀

    AWTK 開源串口屏開發(fā)(6) - 定時器的用法

    定時器是個常用的功能,AWTK串口屏提供了豐富的定時器函數(shù),用于定時器的啟動、停止、暫停、恢復、修改和重置等功能,本文以計時的例子來介紹定時器
    的頭像 發(fā)表于 01-13 08:24 ?519次閱讀
    AWTK 開源串口屏<b class='flag-5'>開發(fā)</b>(6) - <b class='flag-5'>定時器</b>的用法

    單片機定時器的用法

    本章以CW32通用定時器為例介紹單片機定時器的用法。
    的頭像 發(fā)表于 01-04 10:37 ?1310次閱讀
    單片機<b class='flag-5'>定時器</b>的用法

    定時器會阻塞線程嗎 定時器指令有哪幾種

    定時器會阻塞線程嗎 定時器指令有哪幾種? 定時器一般不會阻塞線程,但具體是否會阻塞取決于所使用的定時器實現(xiàn)方式和使用方式。 定時器指令可以分
    的頭像 發(fā)表于 12-19 14:03 ?866次閱讀

    CKS32F4xx系列MCU SysTick定時器的原理及使用方法

    本課將為大家講解CKS32F4xx系列產(chǎn)品的SysTick定時器原理及使用方法。SysTick定時器也叫SysTick滴答定時器,屬于Cortex-M4內(nèi)核外設。SysTick
    的頭像 發(fā)表于 12-18 09:21 ?1208次閱讀
    CKS32F4xx系列MCU SysTick<b class='flag-5'>定時器</b>的原理及使用方法