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

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

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

linux內(nèi)核中對信號量的DOWN操作方式

書生途 ? 來源:書生途 ? 作者:書生途 ? 2022-05-11 15:41 ? 次閱讀

DOWN操作:linux內(nèi)核中,對信號量的DOWN操作有如下幾種:

void down(struct semaphore *sem); //不可中斷
int down_interruptible(struct semaphore *sem);//可中斷
int down_killable(struct semaphore *sem);//睡眠的進(jìn)程可以因為受到致命信號而被喚醒,中斷獲取信號量的操作。
int down_trylock(struct semaphore *sem);//試圖獲取信號量,若無法獲得則直接返回1而不睡眠。返回0則 表示獲取到了信號量
int down_timeout(struct semaphore *sem,long jiffies);//表示睡眠時間是有限制的,如果在jiffies指明的時間到期時仍然無法獲得信號量,則將返回錯誤碼。

在以上四種函數(shù)中,驅(qū)動程序使用的最頻繁的就是down_interruptible函數(shù),以下將對該函數(shù)進(jìn)行分析。

down_interruptible函數(shù)的定義如下:

int down_interruptible(struct semaphore *sem)
{
       unsigned long flags;
       int result = 0;
       spin_lock_irqsave(&sem->lock,flags);
       if (likely(sem->count> 0))
              sem->count--;
       else
              result =__down_interruptible(sem);
       spin_unlock_irqrestore(&sem->lock,flags);
       return result;
}

函數(shù)分析:函數(shù)首先通過spin_lock_irqsave的調(diào)用來保證對sem->count操作的原子性。如果count>0,表示當(dāng)前進(jìn)程可以獲得信號量,將count的值減1然后退出。如果count不大于0,表明當(dāng)前進(jìn)程無法獲取信號量,則調(diào)用__down_interruptible,后者會繼續(xù)調(diào)用__down_common。

__down_common 函數(shù)定義如下:

static inline int __sched __down_common(struct semaphore *sem, longstate,
                                                        longtimeout)
{
       struct task_struct *task= current;
       struct semaphore_waiterwaiter;
       list_add_tail(&waiter.list,&sem->wait_list);
       waiter.task = task;
       waiter.up = 0; 
       for (;;) {
              if(signal_pending_state(state, task))
                     gotointerrupted;
              if (timeout <=0)
                     gototimed_out;
              __set_task_state(task,state);
              spin_unlock_irq(&sem->lock);
              timeout =schedule_timeout(timeout);
              spin_lock_irq(&sem->lock);
              if (waiter.up)
                     return 0;
       } 
 timed_out:
       list_del(&waiter.list);
       return -ETIME;
 interrupted:
       list_del(&waiter.list);
       return -EINTR;
}

函數(shù)分析:在__down_common函數(shù)數(shù)執(zhí)行了以下操作。

(1)將當(dāng)前進(jìn)程放到信號量成員變量wait_list所管理的隊列中。

(2)在一個for循環(huán)中把當(dāng)前的進(jìn)程狀態(tài)這是為TASK_INTERRUPTIBLE,在調(diào)用schedule_timeout使當(dāng)前進(jìn)程進(jìn)入睡眠狀態(tài),函數(shù)將停留在schedule_timeout調(diào)用上,知道再次被調(diào)度執(zhí)行。

(3) 當(dāng)該進(jìn)程再一次被調(diào)度時,按原因執(zhí)行相應(yīng)的操作:如果waiter.up不為0說明進(jìn)程被該信號量的up操作所喚醒,進(jìn)程可以獲得信號量。如果進(jìn)程是因為被用戶空間的信號所中斷或超時信號所引起的喚醒,則返回相應(yīng)的錯誤代碼。

UP操作:LINUX內(nèi)核只提供了一個up函數(shù)

up函數(shù)定義如下:

void up(struct semaphore *sem)
{
       unsigned long flags;
 
       spin_lock_irqsave(&sem->lock,flags);
       if(likely(list_empty(&sem->wait_list)))
              sem->count++;
       else
              __up(sem);
       spin_unlock_irqrestore(&sem->lock,flags);
}

函數(shù)分析:如果sem的wait_list隊列為空,則表明沒有其他進(jìn)程正在等待該信號量,那么只需要把sem的count加1即可。如果wait_list隊列不為空,則說明有其他進(jìn)程正睡眠在wait_list上等待該信號,此時調(diào)用__up(sem)來喚醒進(jìn)程:

__up()函數(shù)定義如下:

static noinline void __sched __up(struct semaphore *sem)
{
       struct semaphore_waiter*waiter = list_first_entry(&sem->wait_list,
                                          structsemaphore_waiter, list);
       list_del(&waiter->list);
       waiter->up = 1;
       wake_up_process(waiter->task);
}

函數(shù)分析:在函數(shù)中,調(diào)用了wake_up_process來喚醒進(jìn)程,這樣進(jìn)程就從之前的__down_interruptible調(diào)用中的timeout=schedule_timeout(timeout)處醒來,wait-up=1, __down_interruptible返回0,進(jìn)程獲得了信號量。

up()與down()函數(shù)之間的聯(lián)系:由上面對兩個函數(shù)的分析可以知道,__down_common函數(shù)中timeout=schedule_timeout(timeout) 有著很重要的作用。

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

    關(guān)注

    87

    文章

    11329

    瀏覽量

    209969
  • down
    +關(guān)注

    關(guān)注

    0

    文章

    5

    瀏覽量

    9875
  • 信號量
    +關(guān)注

    關(guān)注

    0

    文章

    53

    瀏覽量

    8364
收藏 人收藏

    評論

    相關(guān)推薦

    Linux下進(jìn)程通訊之信號量

    ?信號量集,就是由多個信號量組成的一個數(shù)組。 作為一個整體, 信號量集中所有的信號量使用同一個等待隊列。 Linux
    的頭像 發(fā)表于 08-19 19:55 ?2074次閱讀
    <b class='flag-5'>Linux</b>下進(jìn)程通訊之<b class='flag-5'>信號量</b>集

    FreeRTOS信號量使用教程

    信號量操作系統(tǒng)重要的一部分,信號量一般用來進(jìn)行資源管理和任務(wù)同步, FreeRTOS中信號量又分為二值
    的頭像 發(fā)表于 12-19 09:22 ?3263次閱讀
    FreeRTOS<b class='flag-5'>信號量</b>使用教程

    芯靈思SinlinxA33開發(fā)板的Linux內(nèi)核信號量學(xué)習(xí)

    被喚醒,轉(zhuǎn)入步驟(1)。    (4) 當(dāng)進(jìn)程不再使用一個信號量控制的資源時,信號量值加1。如果此時有進(jìn)程正在睡眠等待此信號量,則喚醒此進(jìn)程。     維護(hù)信號量狀態(tài)的是
    發(fā)表于 02-20 15:50

    芯靈思SinlinxA64開發(fā)板 Linux內(nèi)核信號量學(xué)習(xí)

    等待此信號量,則喚醒此進(jìn)程。     維護(hù)信號量狀態(tài)的是Linux內(nèi)核操作系統(tǒng)而不是用戶進(jìn)程。我們可以從頭文件/usr/src/
    發(fā)表于 03-15 16:10

    LINUX內(nèi)核學(xué)習(xí)指南:構(gòu)建系統(tǒng)、信號量設(shè)計、GPIO操作函數(shù)

    控制路徑可以睡眠。我們從 LINUX內(nèi)核信號量最直觀的設(shè)計/實現(xiàn)出發(fā),通過一步步改進(jìn),揭示完整的信號量設(shè)計/實現(xiàn),然后探討在不同平臺上通用的信號量
    發(fā)表于 07-08 14:45

    Linux操作系統(tǒng)信號量機(jī)制的實時化改造

    為了提高Linux操作系統(tǒng)的實時性,研究了Linux操作系統(tǒng)System V信號量機(jī)制在內(nèi)核
    發(fā)表于 06-25 16:41 ?18次下載

    你了解Linux 各類信號量?

    內(nèi)核信號量與用戶信號量,用戶信號量分為POXIS信號量和SYSTEMV信號量,POXIS
    發(fā)表于 05-04 17:19 ?2531次閱讀
    你了解<b class='flag-5'>Linux</b> 各類<b class='flag-5'>信號量</b>?

    信號量和自旋鎖

    。??? Linux 使用的同步機(jī)制可以說從2.0到2.6以來不斷發(fā)展完善。從最初的原子操作,到后來的信號量,從大內(nèi)核鎖到今天的自旋鎖。這些同步機(jī)制的發(fā)展伴隨
    發(fā)表于 04-02 14:43 ?813次閱讀

    Linux IPC System V 信號量

    ?(Linux-specific)返回和IPC_INFO一樣的信息,除了以下方面:semusz成員返回當(dāng)前系統(tǒng)存在的信號量集的數(shù)目,semaem返回系統(tǒng)中所有信號量集中的
    發(fā)表于 04-02 14:46 ?334次閱讀

    Linux 多線程信號量同步

    直到系統(tǒng)將資源分配給該進(jìn)程(進(jìn)入等待隊列,一直等到資源輪到該進(jìn)程)。V操作:如果在該信號量的等待隊列中有進(jìn)程在等待資源,則喚醒一個阻塞進(jìn)程;如果沒有進(jìn)程等待它,則釋放一個資源(即信號量值加1)。模型
    發(fā)表于 04-02 14:47 ?408次閱讀

    華為物聯(lián)網(wǎng)操作系統(tǒng) LiteOS內(nèi)核教程04-信號量

    1. LiteOS內(nèi)核信號量 1.1.信號量 在多任務(wù)操作系統(tǒng),不同的任務(wù)之間需要同步運行,信號量
    發(fā)表于 03-12 17:06 ?1738次閱讀

    Linux信號量(2):POSIX 信號量

    存儲)置于可選部分。在 SUSv4 之前,POSIX 信號量接口已經(jīng)被包含在信號量選項。在 SUSv4 ,這些接口被移至了基本規(guī)范,而
    的頭像 發(fā)表于 10-29 17:34 ?733次閱讀

    LINUX內(nèi)核信號量設(shè)計與實現(xiàn)

    控制路徑可以睡眠。我們從 LINUX內(nèi)核信號量最直觀的設(shè)計/實現(xiàn)出發(fā),通過一步步改進(jìn),揭示在x86平臺上完整的信號量設(shè)計/實現(xiàn),然后探討在不同平臺上通用的
    發(fā)表于 01-14 16:55 ?18次下載

    LINUX內(nèi)核信號量設(shè)計與實現(xiàn)

    控制路徑可以睡眠。我們從 LINUX內(nèi)核信號量最直觀的設(shè)計/實現(xiàn)出發(fā),通過一步步改進(jìn),揭示在x86平臺上完整的信號量設(shè)計/實現(xiàn),然后探討在不同平臺上通用的
    發(fā)表于 01-14 16:55 ?5次下載

    FreeRTOS的二值信號量

    FreeRTOS信號量是一種任務(wù)間通信的方式,信號量包括:二值信號量、互斥信號量、計數(shù)
    的頭像 發(fā)表于 02-10 15:07 ?1527次閱讀