女人被狂躁到高潮视频免费无遮挡,内射人妻骚骚骚,免费人成小说在线观看网站,九九影院午夜理论片少妇,免费av永久免费网址

當(dāng)前位置:首頁 > 嵌入式 > 嵌入式軟件
[導(dǎo)讀]在Linux中,僅等待CPU時(shí)間的進(jìn)程稱為就緒進(jìn)程,它們被放置在一個(gè)運(yùn)行隊(duì)列中,一個(gè)就緒進(jìn)程的狀 態(tài)標(biāo)志位為TASK_RUNNING。一旦一個(gè)運(yùn)行中的進(jìn)程時(shí)間片用完, Linux 內(nèi)核的調(diào)度器會(huì)剝奪這個(gè)進(jìn)程對(duì)CPU的控制權(quán),并且從運(yùn)行隊(duì)列中選擇一個(gè)合適的進(jìn)程投入運(yùn)行。

1 Linux進(jìn)程的睡眠和喚醒

在Linux中,僅等待CPU時(shí)間的進(jìn)程稱為就緒進(jìn)程,它們被放置在一個(gè)運(yùn)行隊(duì)列中,一個(gè)就緒進(jìn)程的狀 態(tài)標(biāo)志位為TASK_RUNNING。一旦一個(gè)運(yùn)行中的進(jìn)程時(shí)間片用完, Linux 內(nèi)核的調(diào)度器會(huì)剝奪這個(gè)進(jìn)程對(duì)CPU的控制權(quán),并且從運(yùn)行隊(duì)列中選擇一個(gè)合適的進(jìn)程投入運(yùn)行。

當(dāng)然,一個(gè)進(jìn)程也可以主動(dòng)釋放CPU的控制權(quán)。函數(shù) schedule()是一個(gè)調(diào)度函數(shù),它可以被一個(gè)進(jìn)程主動(dòng)調(diào)用,從而調(diào)度其它進(jìn)程占用CPU。一旦這個(gè)主動(dòng)放棄CPU的進(jìn)程被重新調(diào)度占用 CPU,那么它將從上次停止執(zhí)行的位置開始執(zhí)行,也就是說它將從調(diào)用schedule()的下一行代碼處開始執(zhí)行。

有時(shí)候,進(jìn)程需要等待直到某個(gè)特定的事件發(fā)生,例如設(shè)備初始化完成、I/O 操作完成或定時(shí)器到時(shí)等。在這種情況下,進(jìn)程則必須從運(yùn)行隊(duì)列移出,加入到一個(gè)等待隊(duì)列中,這個(gè)時(shí)候進(jìn)程就進(jìn)入了睡眠狀態(tài)。

Linux 中的進(jìn)程睡眠狀態(tài)有兩種:一種是可中斷的睡眠狀態(tài),其狀態(tài)標(biāo)志位TASK_INTERRUPTIBLE;

另一種是不可中斷 的睡眠狀態(tài),其狀態(tài)標(biāo)志位為TASK_UNINTERRUPTIBLE??芍袛嗟乃郀顟B(tài)的進(jìn)程會(huì)睡眠直到某個(gè)條件變?yōu)檎?,比如說產(chǎn)生一個(gè)硬件中斷、釋放 進(jìn)程正在等待的系統(tǒng)資源或是傳遞一個(gè)信號(hào)都可以是喚醒進(jìn)程的條件。不可中斷睡眠狀態(tài)與可中斷睡眠狀態(tài)類似,但是它有一個(gè)例外,那就是把信號(hào)傳遞到這種睡眠 狀態(tài)的進(jìn)程不能改變它的狀態(tài),也就是說它不響應(yīng)信號(hào)的喚醒。不可中斷睡眠狀態(tài)一般較少用到,但在一些特定情況下這種狀態(tài)還是很有用的,比如說:進(jìn)程必須等 待,不能被中斷,直到某個(gè)特定的事件發(fā)生。

在現(xiàn)代的Linux操作系統(tǒng)中,進(jìn)程一般都是用調(diào)用schedule()的方法進(jìn)入睡眠狀態(tài)的,下面的代碼演

示了如何讓正在運(yùn)行的進(jìn)程進(jìn)入睡眠狀態(tài)。

sleeping_task = current;

set_current_state(TASK_INTERRUPTIBLE);

schedule();

func1();

/* Rest of the code ... */

在第一個(gè)語句中,程序存儲(chǔ)了一份進(jìn)程結(jié)構(gòu)指針sleeping_task,current 是一個(gè)宏,它指向正在執(zhí)行

的進(jìn)程結(jié)構(gòu)。set_current_state()將該進(jìn)程的狀態(tài)從執(zhí)行狀態(tài)TASK_RUNNING 變成睡眠狀態(tài)

TASK_INTERRUPTIBLE。 如果schedule()是被一個(gè)狀態(tài)為TASK_RUNNING 的進(jìn)程調(diào)度,那么schedule()將調(diào)度另外一個(gè)進(jìn)程占用CPU;如果schedule()是被一個(gè)狀態(tài)為TASK_INTERRUPTIBLE 或TASK_UNINTERRUPTIBLE 的進(jìn)程調(diào)度,那么還有一個(gè)附加的步驟將被執(zhí)行:當(dāng)前執(zhí)行的進(jìn)程在另外一個(gè)進(jìn)程被調(diào)度之前會(huì)被從運(yùn)行隊(duì)列中移出,這將導(dǎo)致正在運(yùn)行的那個(gè)進(jìn)程進(jìn)入睡眠,因?yàn)?它已經(jīng)不在運(yùn)行隊(duì)列中了。

我們可以使用下面的這個(gè)函數(shù)將剛才那個(gè)進(jìn)入睡眠的進(jìn)程喚醒。

wake_up_process(sleeping_task);

在調(diào)用了wake_up_process()以后,這個(gè)睡眠進(jìn)程的狀態(tài)會(huì)被設(shè)置為TASK_RUNNING,而且調(diào)度器

會(huì)把它加入到運(yùn)行隊(duì)列中去。當(dāng)然,這個(gè)進(jìn)程只有在下次被調(diào)度器調(diào)度到的時(shí)候才能真正地投入運(yùn)行。

2 無效喚醒

幾乎在所有的情況下,進(jìn)程都會(huì)在檢查了某些條件之后,發(fā)現(xiàn)條件不滿足才進(jìn)入睡眠??墒怯械臅r(shí)候

進(jìn)程卻會(huì)在 判定條件為真后開始睡眠,如果這樣的話進(jìn)程就會(huì)無限期地休眠下去,這就是所謂的無效喚醒問題。在操作系統(tǒng)中,當(dāng)多個(gè)進(jìn)程都企圖對(duì)共享數(shù)據(jù)進(jìn)行某種處理,而 最后的結(jié)果又取決于進(jìn)程運(yùn)行的順序時(shí),就會(huì)發(fā)生競(jìng)爭條件,這是操作系統(tǒng)中一個(gè)典型的問題,無效喚醒恰恰就是由于競(jìng)爭條件導(dǎo)致的。

設(shè)想有兩個(gè)進(jìn)程A 和B,A 進(jìn)程正在處理一個(gè)鏈表,它需要檢查這個(gè)鏈表是否為空,如果不空就對(duì)鏈

表里面的數(shù)據(jù)進(jìn)行一些操作,同時(shí)B進(jìn)程也在往這個(gè)鏈表添加節(jié)點(diǎn)。當(dāng)這個(gè)鏈表是空的時(shí)候,由于無數(shù)據(jù)可操作,這時(shí)A進(jìn)程就進(jìn)入睡眠,當(dāng)B進(jìn)程向鏈表里面添加了節(jié)點(diǎn)之后它就喚醒A 進(jìn)程,其代碼如下:

A進(jìn)程:

1 spin_lock(&list_lock);

2 if(list_empty(&list_head)) {

3 spin_unlock(&list_lock);

4 set_current_state(TASK_INTERRUPTIBLE);

5 schedule();

6 spin_lock(&list_lock);

7 }

8

9 /* Rest of the code ... */

10 spin_unlock(&list_lock);

B進(jìn)程:

100 spin_lock(&list_lock);

101 list_add_tail(&list_head, new_node);

102 spin_unlock(&list_lock);

103 wake_up_process(processa_task);

這里會(huì)出現(xiàn)一個(gè)問題,假如當(dāng)A進(jìn)程執(zhí)行到第3行后第4行前的時(shí)候,B進(jìn)程被另外一個(gè)處理器調(diào)度

投 入運(yùn)行。在這個(gè)時(shí)間片內(nèi),B進(jìn)程執(zhí)行完了它所有的指令,因此它試圖喚醒A進(jìn)程,而此時(shí)的A進(jìn)程還沒有進(jìn)入睡眠,所以喚醒操作無效。在這之后,A 進(jìn)程繼續(xù)執(zhí)行,它會(huì)錯(cuò)誤地認(rèn)為這個(gè)時(shí)候鏈表仍然是空的,于是將自己的狀態(tài)設(shè)置為TASK_INTERRUPTIBLE然后調(diào)用schedule()進(jìn)入睡 眠。由于錯(cuò)過了B進(jìn)程喚醒,它將會(huì)無限期的睡眠下去,這就是無效喚醒問題,因?yàn)榧词规湵碇杏袛?shù)據(jù)需要處理,A 進(jìn)程也還是睡眠了。

3 避免無效喚醒

如何避免無效喚醒問題呢?我們發(fā)現(xiàn)無效喚醒主要發(fā)生在檢查條件之后和進(jìn)程狀態(tài)被設(shè)置為睡眠狀

態(tài)之前, 本來B進(jìn)程的wake_up_process()提供了一次將A進(jìn)程狀態(tài)置為TASK_RUNNING 的機(jī)會(huì),可惜這個(gè)時(shí)候A進(jìn)程的狀態(tài)仍然是TASK_RUNNING,所以wake_up_process()將A進(jìn)程狀態(tài)從睡眠狀態(tài)轉(zhuǎn)變?yōu)檫\(yùn)行狀態(tài)的努力 沒有起到預(yù)期的作用。要解決這個(gè)問題,必須使用一種保障機(jī)制使得判斷鏈表為空和設(shè)置進(jìn)程狀態(tài)為睡眠狀態(tài)成為一個(gè)不可分割的步驟才行,也就是必須消除競(jìng)爭條 件產(chǎn)生的根源,這樣在這之后出現(xiàn)的wake_up_process ()就可以起到喚醒狀態(tài)是睡眠狀態(tài)的進(jìn)程的作用了。

找到了原因后,重新設(shè)計(jì)一下A進(jìn)程的代碼結(jié)構(gòu),就可以避免上面例子中的無效喚醒問題了。

A進(jìn)程:

1 set_current_state(TASK_INTERRUPTIBLE);

2 spin_lock(&list_lock);

3 if(list_empty(&list_head)) {

4 spin_unlock(&list_lock);

5 schedule();

6 spin_lock(&list_lock);

7 }

8 set_current_state(TASK_RUNNING);

9

10 /* Rest of the code ... */

11 spin_unlock(&list_lock);

可以看到,這段代碼在測(cè)試條件之前就將當(dāng)前執(zhí)行進(jìn)程狀態(tài)轉(zhuǎn)設(shè)置成TASK_INTERRUPTIBLE了,并且在鏈表不為空的情況下又將自己置為TASK_RUNNING狀態(tài)。這樣一來如果B進(jìn)程在A進(jìn)程進(jìn)程檢查

了鏈表為空以后調(diào)用wake_up_process(),那么A進(jìn)程的狀態(tài)就會(huì)自動(dòng)由原來TASK_INTERRUPTIBLE

變成TASK_RUNNING,此后即使進(jìn)程又調(diào)用了schedule(),由于它現(xiàn)在的狀態(tài)是TASK_RUNNING,所以仍然不會(huì)被從運(yùn)行隊(duì)列中移出,因而不會(huì)錯(cuò)誤的進(jìn)入睡眠,當(dāng)然也就避免了無效喚醒問題。

4 Linux內(nèi)核的例子

在Linux操作系統(tǒng)中,內(nèi)核的穩(wěn)定性至關(guān)重要,為了避免在Linux操作系統(tǒng)內(nèi)核中出現(xiàn)無效喚醒問題,

Linux內(nèi)核在需要進(jìn)程睡眠的時(shí)候應(yīng)該使用類似如下的操作:

/* ‘q’是我們希望睡眠的等待隊(duì)列 */

DECLARE_WAITQUEUE(wait,current);

add_wait_queue(q, &wait);

set_current_state(TASK_INTERRUPTIBLE);

/* 或TASK_INTERRUPTIBLE */

while(!condition) /* ‘condition’ 是等待的條件*/

schedule();

set_current_state(TASK_RUNNING);

remove_wait_queue(q, &wait);

上面的操作,使得進(jìn)程通過下面的一系列步驟安全地將自己加入到一個(gè)等待隊(duì)列中進(jìn)行睡眠:首先調(diào)

用DECLARE_WAITQUEUE ()創(chuàng)建一個(gè)等待隊(duì)列的項(xiàng),然后調(diào)用add_wait_queue()把自己加入到等待隊(duì)列中,并且將進(jìn)程的狀態(tài)設(shè)置為 TASK_INTERRUPTIBLE 或者TASK_INTERRUPTIBLE。然后循環(huán)檢查條件是否為真:如果是的話就沒有必要睡眠,如果條件不為真,就調(diào)用schedule()。當(dāng)進(jìn)程 檢查的條件滿足后,進(jìn)程又將自己設(shè)置為TASK_RUNNING 并調(diào)用remove_wait_queue()將自己移出等待隊(duì)列。

從上面可以看到,Linux的內(nèi)核代碼維護(hù)者也是在進(jìn)程檢查條件之前就設(shè)置進(jìn)程的狀態(tài)為睡眠狀態(tài),

然后才循環(huán)檢查條件。如果在進(jìn)程開始睡眠之前條件就已經(jīng)達(dá)成了,那么循環(huán)會(huì)退出并用set_current_state()將自己的狀態(tài)設(shè)置為就緒,這樣同樣保證了進(jìn)程不會(huì)存在錯(cuò)誤的進(jìn)入睡眠的傾向,當(dāng)然也就不會(huì)導(dǎo)致出現(xiàn)無效喚醒問題。

下面讓我們用linux 內(nèi)核中的實(shí)例來看看Linux 內(nèi)核是如何避免無效睡眠的,這段代碼出自Linux2.6的內(nèi)核(linux-2.6.11/kernel/sched.c: 4254):

4253 /* Wait for kthread_stop */

4254 set_current_state(TASK_INTERRUPTIBLE);

4255 while (!kthread_should_stop()) {

4256 schedule();

4257 set_current_state(TASK_INTERRUPTIBLE);

4258 }

4259 __set_current_state(TASK_RUNNING);

4260 return 0;

上面的這些代碼屬于遷移服務(wù)線程migration_thread,這個(gè)線程不斷地檢查kthread_should_stop(),

直 到kthread_should_stop()返回1它才可以退出循環(huán),也就是說只要kthread_should_stop()返回0該進(jìn)程就會(huì)一直睡 眠。從代碼中我們可以看出,檢查kthread_should_stop()確實(shí)是在進(jìn)程的狀態(tài)被置為TASK_INTERRUPTIBLE后才開始執(zhí)行 的。因此,如果在條件檢查之后但是在schedule()之前有其他進(jìn)程試圖喚醒它,那么該進(jìn)程的喚醒操作不會(huì)失效。

小結(jié)

通過上面的討論,可以發(fā)現(xiàn)在Linux 中避免進(jìn)程的無效喚醒的關(guān)鍵是在進(jìn)程檢查條件之前就將進(jìn)程的

狀態(tài)置為TASK_INTERRUPTIBLE或TASK_UNINTERRUPTIBLE,并且如果檢查的條件滿足的話就應(yīng)該

將其狀態(tài)重新設(shè)置為TASK_RUNNING。這樣無論進(jìn)程等待的條件是否滿足, 進(jìn)程都不會(huì)因?yàn)楸灰瞥鼍途w隊(duì)列而錯(cuò)誤地進(jìn)入睡眠狀態(tài),從而避免了無效喚醒問題。

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

Arm 控股有限公司(納斯達(dá)克股票代碼:ARM,以下簡稱 Arm)今日宣布推出全新 Arm? Lumex? 計(jì)算子系統(tǒng) (Compute Subsystem, CSS) 平臺(tái),這是一套專為旗艦級(jí)智能手機(jī)及下一代個(gè)人電腦加...

關(guān)鍵字: CPU AI 消費(fèi)電子

Arm 控股有限公司(納斯達(dá)克股票代碼:ARM,以下簡稱 Arm)今日宣布推出全新 Arm? Lumex?計(jì)算子系統(tǒng) (Compute Subsystem, CSS) 平臺(tái),這是一套專為旗艦級(jí)智能手機(jī)及下一代個(gè)人電腦加速...

關(guān)鍵字: 消費(fèi)電子 CPU AI

8位單片機(jī)在嵌入式設(shè)計(jì)領(lǐng)域已經(jīng)成為半個(gè)多世紀(jì)以來的主流選擇。盡管嵌入式系統(tǒng)市場(chǎng)日益復(fù)雜,8位單片機(jī)依然不斷發(fā)展,積極應(yīng)對(duì)新的挑戰(zhàn)和系統(tǒng)需求。如今,Microchip推出的8位PIC?和AVR?單片機(jī)系列,配備了先進(jìn)的獨(dú)立...

關(guān)鍵字: 單片機(jī) 嵌入式 CPU

CPU親和度通過限制進(jìn)程或線程可以運(yùn)行的CPU核心集合,使得它們只能在指定的CPU核心上執(zhí)行。這可以減少CPU緩存的失效次數(shù),提高緩存命中率,從而提升系統(tǒng)性能。

關(guān)鍵字: Linux 嵌入式

上海2025年8月15日 /美通社/ -- 在數(shù)字化轉(zhuǎn)型浪潮與數(shù)據(jù)安全需求的雙重驅(qū)動(dòng)下,瀾起科技今日重磅推出第六代津逮? 性能核 CPU (以下簡稱 C6P )。這款融合突破性架構(gòu)、全棧兼容性與芯片級(jí)安全防護(hù)的高性能服...

關(guān)鍵字: CPU BSP 數(shù)字化 AI

在Linux系統(tǒng)性能優(yōu)化中,內(nèi)存管理與網(wǎng)絡(luò)連接處理是兩大核心領(lǐng)域。vm.swappiness與net.core.somaxconn作為關(guān)鍵內(nèi)核參數(shù),直接影響系統(tǒng)在高負(fù)載場(chǎng)景下的穩(wěn)定性與響應(yīng)速度。本文通過實(shí)戰(zhàn)案例解析這兩個(gè)...

關(guān)鍵字: Linux 內(nèi)存管理

采用分離式架構(gòu),充分利用主機(jī) CPU 和 PCIe? 基礎(chǔ)設(shè)施,克服傳統(tǒng)存儲(chǔ)瓶頸

關(guān)鍵字: CPU 數(shù)據(jù)中心 服務(wù)器

受生成式 AI 驅(qū)動(dòng), RISC-V 芯片市場(chǎng)快速發(fā)展。預(yù)計(jì)到2030年,RISC-V SoC出貨量將達(dá)到1618.1億顆,營收將達(dá)到927億美元。其中,用于AI加速器的RISC-V SoC出貨量將達(dá)到41億顆,營收將達(dá)...

關(guān)鍵字: RISC-V CPU AI CUDA ARM 推理

對(duì)于LLM,我使用b谷歌Gemini的免費(fèi)層,所以唯一的成本是n8n托管。在使用了n8n Cloud的免費(fèi)積分后,我決定將其托管在Railway上(5美元/月)。然而,由于n8n是開源的,您可以在自己的服務(wù)器上托管它,而...

關(guān)鍵字: 人工智能 n8n Linux
關(guān)閉