高级字符驱动程序操作之休眠(理论篇)
1. 休眠的原則
?
第一條: "永遠(yuǎn)不要在原子上下文中進(jìn)入休眠" (LDD3 p149)
第二條: 當(dāng)線程被喚醒時(shí),應(yīng)當(dāng)檢查等待的條件是否為真
第三條: 確保即將休眠的線程一定會(huì)被喚醒
?
2. 簡單休眠
?
wait_event(queue, condition) // 不可中斷的休眠,不符合休眠的原則(第一條)不推薦
wait_event_interruptible(queue, condition) // 接受終端的休眠,推薦
wait_event_timeout(queue, condition, timeout) // 不可中斷,定時(shí)喚醒
wait_event_interruptible_timeout(queue, condition, timeout) // 接受中斷,定時(shí)喚醒
?
約定: 對于不可中斷的休眠,使用wait_up喚醒,對于接受中斷的休眠,使用wait_up_interruptible喚醒
?
3. 阻塞和非阻塞標(biāo)志
?
file文件結(jié)構(gòu)指針filp的成員f_flags可以設(shè)定進(jìn)程的阻塞和非阻塞標(biāo)志,如果filp_flags & O_NONBLOCK為真,進(jìn)程被標(biāo)志為非阻塞的,反之進(jìn)程被標(biāo)志為可阻塞的
?
4. 高級(jí)休眠
?
第一步: 初始化wait_queue_t結(jié)構(gòu)
???????????? 靜態(tài)初始化 DEFINE_WAIT(my_wait);
???????????? 動(dòng)態(tài)初始化 wait_queue_t my_wait;
????????????????????????????? init_wait(&my_wait);
?
第二步: 設(shè)置進(jìn)程狀態(tài)
????????????? void prepare_to_wait(wait_queue_head_t *queue,
????????????????????????????????????????????????? wait_queue_t *wait,
????????????????????????????????????????????????? int state);
????????????? state就是要設(shè)置的新狀態(tài),它可以是TASK_INTERRUPTIBLE(可中斷休眠狀態(tài))和TASK_UNINTERRUPTIBLE(不可中斷休眠)
?
第三步:??讓出處理器,開始休眠
????????????? if (!condition)
????????????????????? schedule();
?
第四步: 如果condition為true,表示進(jìn)程不需要休眠了,要完成一些清理工作
????????????? void finish_wait(wait_queue_head_t *queue, wait_queue_t *wait);
?
5. 獨(dú)占等待
?
當(dāng)調(diào)用wake_up時(shí),所有在等待隊(duì)列中的線程都被喚醒,當(dāng)?shù)却?duì)列中的線程很多的時(shí)候,這樣喚醒的代價(jià)就會(huì)顯得很"瘋狂"
為了防止這種情況,驅(qū)動(dòng)程序員可以考慮在等待隊(duì)列入口為休眠線程設(shè)置WQ_FLAG_EXCLUSIVE標(biāo)志,這樣在wake_up的時(shí)候,操作系統(tǒng)喚醒第一個(gè)具有WQ_FLAG_EXCLUSIVE標(biāo)志的線程后就停止喚醒操作
?
6. 緩沖區(qū)操作
?
這里的緩沖區(qū)操作是僅僅是策略問題,而非機(jī)制問題,為了看懂scullpipe,對緩沖區(qū)操作應(yīng)該有清晰的認(rèn)識(shí)
?
需要注意的是: 讀寫指針都不會(huì)讀寫緩沖區(qū)的結(jié)尾地址,緩沖區(qū)的結(jié)尾地址僅僅用于回卷;
??????????????????????因?yàn)樽x指針始終在追趕寫指針,是寫指針讓讀寫指針能夠相遇,也就是寫指針不可能超過讀指針,除非讀指針讀完所有數(shù)據(jù),與寫指針重逢;
?????????????????????? 當(dāng)讀寫指針相遇,表示緩沖區(qū)為空。
總結(jié)
以上是生活随笔為你收集整理的高级字符驱动程序操作之休眠(理论篇)的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 蒙特 卡罗方法matlab,蒙特·卡罗方
- 下一篇: Flask 多线程