精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

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

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

3天內不再提示

Linux驅動開發-內核定時器

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

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

1. 內核定時器介紹

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

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

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

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

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

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

2. 內核定時器相關API函數

2.1 修改定時器超時時間

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

2.2 初始化定時器

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

2.3 關閉定時器

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

2.4 關閉定時器

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

2.5 轉換時間(微妙單位)

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

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

函數原型 unsigned long msecs_to_jiffies(const unsigned int m)
函數功能 轉換時間(毫秒為單位),用于填充定時器結構體,設置超時時間
函數參數 m:要轉換的時間值(毫秒為單位)
函數返回值 成功返回轉換成功的時間。用于填充定時器結構體,設置超時時間
函數定義文件 \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宏調用)

#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。

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

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

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

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

(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; /*單位是節拍*/
	timer.function=timer_function;
	timer.data=666;
	
	/*1. 初始化定時器*/
	init_timer(&timer);
	/*2. 添加定時器到內核*/
	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);    /*驅動入口--安裝驅動的時候執行*/
module_exit(tiny4412_linux_timer_cleanup); /*驅動出口--卸載驅動的時候執行*/

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

4. 內核提供的延時函數

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

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

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

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

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

示例: 計算經過的一段時間

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

    關注

    3

    文章

    1363

    瀏覽量

    40228
  • 定時器
    +關注

    關注

    23

    文章

    3237

    瀏覽量

    114467
  • 函數
    +關注

    關注

    3

    文章

    4304

    瀏覽量

    62429
收藏 人收藏

    評論

    相關推薦

    linux驅動程序如何加載進內核

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

    Linux 驅動開發與應用開發,你知道多少?

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

    定時器的工作方式介紹

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

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

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

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

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

    如何實現一個軟件定時器

    Linux,uC/OS,FreeRTOS等操作系統中,都帶有軟件定時器,原理大同小異。典型的實現方法是:通過一個硬件定時器產生固定的時鐘節拍,每次硬件定時器中斷到,就對一個全局的時間
    的頭像 發表于 04-29 11:00 ?580次閱讀

    s7200定時器的五種故障介紹

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

    ?PLC定時器介紹

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

    使用555定時器的可調雙定時器電路

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

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

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

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

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

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

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

    單片機定時器的用法

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

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

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

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

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