訂閱
糾錯
加入自媒體

實(shí)例講解Linux內(nèi)核信號SIGIO的使用

一、信號

1. 基本概念

信號是在軟件層次上對中斷機(jī)制的一種模擬,在原理上,一個進(jìn)程收到一個信號與處理器收到一個中斷請求可以說是一樣的。信號是異步的,一個進(jìn)程不必通過任何操作來等待信號的到達(dá),事實(shí)上,進(jìn)程也不知道信號到底什么時候到達(dá)。

例如鍵盤輸入中斷按鍵(^C),它的發(fā)生在程序執(zhí)行過程中是不可預(yù)測的。

信號是進(jìn)程間通信機(jī)制中唯一的異步通信機(jī)制,可以看作是異步通知,通知接收信號的進(jìn)程有哪些事情發(fā)生了。

硬件異常也能產(chǎn)生信號,例如被零除、無效內(nèi)存引用(test里產(chǎn)生的就是這種錯誤)等。這些條件通常先由內(nèi)核硬件檢測到,然后通知內(nèi)核。內(nèi)核將決定產(chǎn)生什么樣的信號。

同一個信號的額外發(fā)生通常不會被排隊(duì)。如果信號在被阻塞時發(fā)生了5次,當(dāng)我們反阻塞這個信號時,這個信號的信號處理函數(shù)通常只被調(diào)用一次。

同一時刻只能處理一個信號,在信號處理函數(shù)發(fā)信號給自己時,該信號會被pending。

信號的數(shù)值越小,則優(yōu)先級越高。當(dāng)進(jìn)程收到多個待處理信號時,總是先處理優(yōu)先級別高的信號。

信號處理函數(shù)的?梢允褂帽恢袛嗟囊部梢允褂锚(dú)立的,具體可以通過系統(tǒng)調(diào)用設(shè)置。

信號機(jī)制經(jīng)過POSIX實(shí)時擴(kuò)展后,功能更加強(qiáng)大,除了基本通知功能外,還可以傳遞附加信息。

2. 處理方式

忽略:接收到信號后不做任何反應(yīng)。捕獲:用自定義的信號處理函數(shù)來執(zhí)行特定的動作。默認(rèn):接收到信號后按系統(tǒng)默認(rèn)的行為處理該信號。這是多數(shù)應(yīng)用采取的處理方式。

二、Linux下的信號類型

使用kill -l就會顯示出linux支持的信號列表。

其中列表中,編號為1 ~ 31的信號為傳統(tǒng)UNIX支持的信號,是不可靠信號(非實(shí)時的),編號為32 ~ 63的信號是后來擴(kuò)充的,稱做可靠信號(實(shí)時信號)。不可靠信號和可靠信號的區(qū)別在于前者不支持排隊(duì),可能會造成信號丟失,而后者不會。

下面我們對編號小于SIGRTMIN的信號進(jìn)行討論(下面的編號 依次對應(yīng)信號 的數(shù)值為1 - 31)。

1) SIGHUP

本信號在用戶終端連接(正;蚍钦)結(jié)束時發(fā)出, 通常是在終端的控制進(jìn)程結(jié)束時, 通知同一session內(nèi)的各個作業(yè), 這時它們與控制終端不再關(guān)聯(lián)。

登錄Linux時,系統(tǒng)會分配給登錄用戶一個終端(Session)。在這個終端運(yùn)行的所有程序,包括前臺進(jìn)程組和后臺進(jìn)程組,一般都 屬于這個 Session。當(dāng)用戶退出Linux登錄時,前臺進(jìn)程組和后臺有對終端輸出的進(jìn)程將會收到SIGHUP信號。這個信號的默認(rèn)操作為終止進(jìn)程,因此前臺進(jìn) 程組和后臺有終端輸出的進(jìn)程就會中止。不過可以捕獲這個信號,比如wget能捕獲SIGHUP信號,并忽略它,這樣就算退出了Linux登錄,wget也 能繼續(xù)下載。

此外,對于與終端脫離關(guān)系的守護(hù)進(jìn)程,這個信號用于通知它重新讀取配置文件。

2) SIGINT

程序終止(interrupt)信號, 在用戶鍵入INTR字符(通常是Ctrl-C)時發(fā)出,用于通知前臺進(jìn)程組終止進(jìn)程。

3) SIGQUIT

和SIGINT類似, 但由QUIT字符(通常是Ctrl-)來控制. 進(jìn)程在因收到SIGQUIT退出時會產(chǎn)生core文件, 在這個意義上類似于一個程序錯誤信號。

4) SIGILL

執(zhí)行了非法指令. 通常是因?yàn)榭蓤?zhí)行文件本身出現(xiàn)錯誤, 或者試圖執(zhí)行數(shù)據(jù)段. 堆棧溢出時也有可能產(chǎn)生這個信號。

5) SIGTRAP

由斷點(diǎn)指令或其它trap指令產(chǎn)生. 由debugger使用。

6) SIGABRT

調(diào)用abort函數(shù)生成的信號。

7) SIGBUS

非法地址, 包括內(nèi)存地址對齊(alignment)出錯。比如訪問一個四個字長的整數(shù), 但其地址不是4的倍數(shù)。它與SIGSEGV的區(qū)別在于后者是由于對合法存儲地址的非法訪問觸發(fā)的(如訪問不屬于自己存儲空間或只讀存儲空間)。

8) SIGFPE

在發(fā)生致命的算術(shù)運(yùn)算錯誤時發(fā)出. 不僅包括浮點(diǎn)運(yùn)算錯誤, 還包括溢出及除數(shù)為0等其它所有的算術(shù)的錯誤。

9) SIGKILL

用來立即結(jié)束程序的運(yùn)行. 本信號不能被阻塞、處理和忽略。如果管理員發(fā)現(xiàn)某個進(jìn)程終止不了,可嘗試發(fā)送這個信號。

10) SIGUSR1

留給用戶使用

11) SIGSEGV

試圖訪問未分配給自己的內(nèi)存, 或試圖往沒有寫權(quán)限的內(nèi)存地址寫數(shù)據(jù).

信號 11,即表示程序中可能存在特定條件下的非法內(nèi)存訪問。

12) SIGUSR2

留給用戶使用

13) SIGPIPE

管道破裂。這個信號通常在進(jìn)程間通信產(chǎn)生,比如采用FIFO(管道)通信的兩個進(jìn)程,讀管道沒打開或者意外終止就往管道寫,寫進(jìn)程會收到SIGPIPE信號。此外用Socket通信的兩個進(jìn)程,寫進(jìn)程在寫Socket的時候,讀進(jìn)程已經(jīng)終止。

14) SIGALRM

時鐘定時信號, 計(jì)算的是實(shí)際的時間或時鐘時間. alarm函數(shù)使用該信號.

15) SIGTERM

程序結(jié)束(terminate)信號, 與SIGKILL不同的是該信號可以被阻塞和處理。通常用來要求程序自己正常退出,shell命令kill缺省產(chǎn)生這個信號。如果進(jìn)程終止不了,我們才會嘗試SIGKILL。

17) SIGCHLD

子進(jìn)程結(jié)束時, 父進(jìn)程會收到這個信號。

如果父進(jìn)程沒有處理這個信號,也沒有等待(wait)子進(jìn)程,子進(jìn)程雖然終止,但是還會在內(nèi)核進(jìn)程表中占有表項(xiàng),這時的子進(jìn)程稱為僵尸 進(jìn)程。這種情 況我們應(yīng)該避免(父進(jìn)程或者忽略SIGCHILD信號,或者捕捉它,或者wait它派生的子進(jìn)程,或者父進(jìn)程先終止,這時子進(jìn)程的終止自動由init進(jìn)程 來接管)。

18) SIGCONT

讓一個停止(stopped)的進(jìn)程繼續(xù)執(zhí)行. 本信號不能被阻塞. 可以用一個handler來讓程序在由stopped狀態(tài)變?yōu)槔^續(xù)執(zhí)行時完成特定的工作. 例如, 重新顯示提示符

19) SIGSTOP

停止(stopped)進(jìn)程的執(zhí)行. 注意它和terminate以及interrupt的區(qū)別:該進(jìn)程還未結(jié)束, 只是暫停執(zhí)行. 本信號不能被阻塞, 處理或忽略.

20) SIGTSTP

停止進(jìn)程的運(yùn)行, 但該信號可以被處理和忽略. 用戶鍵入SUSP字符時(通常是Ctrl-Z)發(fā)出這個信號

21) SIGTTIN

當(dāng)后臺作業(yè)要從用戶終端讀數(shù)據(jù)時, 該作業(yè)中的所有進(jìn)程會收到SIGTTIN信號. 缺省時這些進(jìn)程會停止執(zhí)行.

22) SIGTTOU

類似于SIGTTIN, 但在寫終端(或修改終端模式)時收到.

23) SIGURG

有"緊急"數(shù)據(jù)或out-of-band數(shù)據(jù)到達(dá)socket時產(chǎn)生.

24) SIGXCPU

超過CPU時間資源限制. 這個限制可以由getrlimit/setrlimit來讀取/改變。

25) SIGXFSZ

當(dāng)進(jìn)程企圖擴(kuò)大文件以至于超過文件大小資源限制。

26) SIGVTALRM

虛擬時鐘信號. 類似于SIGALRM, 但是計(jì)算的是該進(jìn)程占用的CPU時間.

27) SIGPROF

類似于SIGALRM/SIGVTALRM, 但包括該進(jìn)程用的CPU時間以及系統(tǒng)調(diào)用的時間.

28) SIGWINCH

窗口大小改變時發(fā)出.

29) SIGIO

文件描述符準(zhǔn)備就緒, 可以開始進(jìn)行輸入/輸出操作.

30) SIGPWR

Power failure

31) SIGSYS

非法的系統(tǒng)調(diào)用。

三、 信號行為說明

不通的信號在不同的標(biāo)準(zhǔn)下,功能有所差別,下面列出主要的信號的默認(rèn)行為和說明:

名稱數(shù)字標(biāo)準(zhǔn)默認(rèn)行為說明SIGILL4ANSI終止+coredump執(zhí)行了非法指令. 通常是因?yàn)榭蓤?zhí)行文件本身出現(xiàn)錯誤, 或者試圖執(zhí)行數(shù)據(jù)段. 堆棧溢出時也有可能產(chǎn)生這個信號SIGABRT6ANSI終止+coredump調(diào)用abort函數(shù)生成的信號SIGBUS74.2 BSD終止+coredump非法地址, 包括內(nèi)存地址對齊(alignment)出錯。比如訪問一個四個字長的整數(shù), 但其地址不是4的倍數(shù)。它與SIGSEGV的區(qū)別在于后者是由于對合法存儲地址的非法訪問觸發(fā)的(如訪問不屬于自己存儲空間或只讀存儲空間)SIGFPE8ANSI終止+coredump在發(fā)生致命的算術(shù)運(yùn)算錯誤時發(fā)出. 不僅包括浮點(diǎn)運(yùn)算錯誤, 還包括溢出及除數(shù)為0等其它所有的算術(shù)的錯誤SIGSEGV11ANSI終止+coredump試圖訪問未分配給自己的內(nèi)存, 或試圖往沒有寫權(quán)限的內(nèi)存地址寫數(shù)據(jù)。訪問空指針,野指針基本都產(chǎn)生這個信號,也是最常見的信號SIGSTKFLT16N/A終止堆棧錯誤SIGPIPE13POSIX終止管道破裂。這個信號通常在進(jìn)程間通信產(chǎn)生,比如采用FIFO(管道)通信的兩個進(jìn)程,讀管道沒打開或者意外終止就往管道寫,寫進(jìn)程會收到SIGPIPE信號。

此外用Socket通信的兩個進(jìn)程,寫進(jìn)程在寫Socket的時候,讀進(jìn)程已經(jīng)終止SIGTRAP5POSIX終止+coredump由斷點(diǎn)指令或其它trap指令產(chǎn)生. 由debugger使用SIGHUP1POSIX終止用戶終端連接(正常或非正常)結(jié)束時發(fā)出, 通常是在終端的控制進(jìn)程結(jié)束時, 通知同一session內(nèi)的各個作業(yè), 這時它們與控制終端不再關(guān)聯(lián)SIGINT2ANSI終止程序終止(interrupt)信號, 在用戶鍵入INTR字符(通常是Ctrl-C)時發(fā)出,用于通知前臺進(jìn)程組終止進(jìn)程SIGQUIT3POSIX終止+coredump和SIGINT類似, 但由QUIT字符(通常是Ctrl-)來控制. 進(jìn)程在因收到SIGQUIT退出時會產(chǎn)生core文件, 在這個意義上類似于一個程序錯誤信號SIGKILL9POSIX終止用來立即結(jié)束程序的運(yùn)行. 本信號不能被阻塞、捕獲和忽略。如果管理員發(fā)現(xiàn)某個進(jìn)程終止不了,可嘗試發(fā)送這個信號SIGCHLD17POSIX忽略子進(jìn)程結(jié)束時, 父進(jìn)程會收到這個信號。如果父進(jìn)程沒有處理這個信號,也沒有等待(wait)子進(jìn)程,子進(jìn)程雖然終止,但是還會在內(nèi)核進(jìn)程表中占有表項(xiàng),這時的子進(jìn)程稱為僵尸進(jìn)程。

這種情 況我們應(yīng)該避免(父進(jìn)程或者忽略SIGCHILD信號,或者捕捉它,或者wait它派生的子進(jìn)程,或者父進(jìn)程先終止,這時子進(jìn)程的終止自動由init進(jìn)程來接管)SIGCONT18POSIX繼續(xù)/忽略讓一個停止(stopped)的進(jìn)程繼續(xù)執(zhí)行. 本信號不能被阻塞 . 可以用一個handler來讓程序在由stopped狀態(tài)變?yōu)槔^續(xù)執(zhí)行時完成特定的工作. 例如, 重新顯示提示符..在進(jìn)程掛起時是繼續(xù),否則是忽略SIGSTOP19POSIX暫停暫停進(jìn)程的執(zhí)行. 注意它和terminate以及interrupt的區(qū)別:該進(jìn)程還未結(jié)束, 只是暫停執(zhí)行. 本信號不能被阻塞、捕獲或忽略SIGALRM14POSIX終止時鐘定時信號, 計(jì)算的是實(shí)際的時間或時鐘時間. alarm函數(shù)使用該信號四、信號分類

在以上列出的信號中,程序不可捕獲、阻塞或忽略的信號有:

SIGKILL,SIGSTOP

不能恢復(fù)至默認(rèn)動作的信號有:

SIGILL,SIGTRAP

默認(rèn)會導(dǎo)致進(jìn)程流產(chǎn)的信,有:

SIGABRT,SIGBUS,SIGFPE,SIGILL,SIGIOT,SIGQUIT,SIGSEGV,SIGTRAP,SIGXCPU,SIGXFSZ

默認(rèn)會導(dǎo)致進(jìn)程退出的信號有:

SIGALRM,SIGHUP,SIGINT,SIGKILL,SIGPIPE,SIGPOLL,SIGPROF,SIGSYS,SIGTERM,SIGUSR1,SIGUSR2,SIGVTALRM

默認(rèn)會導(dǎo)致進(jìn)程停止的信號有:

SIGSTOP,SIGTSTP,SIGTTIN,SIGTTOU

默認(rèn)進(jìn)程忽略的信號有:

SIGCHLD,SIGPWR,SIGURG,SIGWINCH

此外,SIGIO在SVR4是退出,在4.3BSD中是忽略;

SIGCONT在進(jìn)程掛起時是繼續(xù),否則是忽略,不能被阻塞

終止程序的時候在不得已的情況下不能用SIGKILL,因?yàn)镾IGKILL不會對子進(jìn)程進(jìn)行處理,只是把對自己進(jìn)行處理。

五、信號驅(qū)動IO-SIGIO-29

下面我們主要講SIGIO-29的使用。

參考上圖:

時刻1 通過sigaction系統(tǒng)調(diào)用建立信號SIGIO的信號處理函數(shù),該函數(shù)壺立即返回,注意,對應(yīng)的驅(qū)動必須支持方法.fastnc時刻2 數(shù)據(jù)此時沒有準(zhǔn)備好,應(yīng)進(jìn)程會繼續(xù)執(zhí)行,而內(nèi)核會繼續(xù)等待數(shù)據(jù),也就是說等待數(shù)據(jù)階段應(yīng)用進(jìn)程是非阻塞的。時刻3 內(nèi)核準(zhǔn)備好了數(shù)據(jù),要向應(yīng)用進(jìn)程復(fù)制數(shù)據(jù),通過函數(shù)kill_fasync()向應(yīng)用程序遞交SIGIO信號,二應(yīng)用程序的信號處理程序會被調(diào)用到,在該函數(shù)中我們可以通過read等系統(tǒng)調(diào)用從內(nèi)核賦值程序到進(jìn)程時刻4 在賦值數(shù)據(jù)期間,進(jìn)程阻塞時刻5 數(shù)據(jù)復(fù)制完成,會返回成功的指示,應(yīng)用程序可以繼續(xù)處理數(shù)據(jù)

信號驅(qū)動 I/O 的 CPU 利用率很高,因?yàn)樵趫D中,等待數(shù)據(jù)的那段時間2,應(yīng)用程序可以繼續(xù)執(zhí)行其他操作。

六、程序?qū)崿F(xiàn)

1. 信號注冊函數(shù)signal()#include

功能:

給信號signum注冊處理函數(shù),函數(shù)原型是void (*sighandler_t)(int)
當(dāng)收到信號signum后,就會調(diào)用注冊的函數(shù)

參數(shù):

int signum  信號值
sighandler_t handler  信號處理函數(shù)
2.內(nèi)核函數(shù)void kill_fasync(struct fasync_struct **fp, int sig, int band)

功能:

發(fā)送信號sig給進(jìn)程,通知進(jìn)程是可讀還是可寫,由band給出
POLLIN    :可讀
POLLOUT:可寫

通用字符設(shè)備的.fasync方法,一般都是固定的寫法,我們暫時可以不用關(guān)心他的原理,會用即可,具體寫法如下:

static ssize_t hello_write (struct file *filep, const char __user *buf, size_t size, loff_t *pos)
{
int error;
…………
kill_fasync(&hello_fasync,SIGIO,POLLIN);
return size;
}
static struct file_operations hello_ops =
{
…………
.fasync = hello_fasync_func,
};
2. 源程序

驅(qū)動程序:hello.c

 
*公眾號:一口Linux
*2021.6.21
*version: 1.0.0
#include

write.c

 
*一口Linux
*2021.6.21
*version: 1.0.0
#include

test.c

 
*公眾號:一口Linux
*2021.6.21
*version: 1.0.0
#include

編譯

make
gcc test.c -o run
gcc write.c -o run

執(zhí)行:

insmod hello.ko

先開啟一個終端 ,執(zhí)行

./run

再開啟一個終端 ,執(zhí)行

./w

執(zhí)行結(jié)果如下:

可以看到,寫入數(shù)據(jù)后,信號處理程序被調(diào)用到,并且打印出信號的值29,同時從驅(qū)動力讀取出數(shù)據(jù)。

本例以字符設(shè)備為基礎(chǔ)來實(shí)現(xiàn),詳細(xì)原理,請參考博主其他文章。

聲明: 本文由入駐維科號的作者撰寫,觀點(diǎn)僅代表作者本人,不代表OFweek立場。如有侵權(quán)或其他問題,請聯(lián)系舉報(bào)。

發(fā)表評論

0條評論,0人參與

請輸入評論內(nèi)容...

請輸入評論/評論長度6~500個字

您提交的評論過于頻繁,請輸入驗(yàn)證碼繼續(xù)

  • 看不清,點(diǎn)擊換一張  刷新

暫無評論

暫無評論

    掃碼關(guān)注公眾號
    OFweek人工智能網(wǎng)
    獲取更多精彩內(nèi)容
    文章糾錯
    x
    *文字標(biāo)題:
    *糾錯內(nèi)容:
    聯(lián)系郵箱:
    *驗(yàn) 證 碼:

    粵公網(wǎng)安備 44030502002758號