linux內(nèi)核信號(hào)機(jī)制
signal.c 涉及到內(nèi)核中所有有關(guān)信號(hào)處理的函數(shù),在linux系統(tǒng)中,信號(hào)是一種“軟件中斷”處理機(jī)制。
有許多較為復(fù)雜的程序使用到了信號(hào),信號(hào)機(jī)制提供了一種處理異步事件的方法。
例如當(dāng)發(fā)生硬件異常時(shí),系統(tǒng)也會(huì)向正在執(zhí)行的進(jìn)程發(fā)送相應(yīng)的信號(hào)。
另外一個(gè)進(jìn)程也可以向另一個(gè)進(jìn)程發(fā)送信號(hào),例如使用kill()函數(shù)向同組的子進(jìn)程發(fā)送終止信號(hào)。
在內(nèi)核代碼中通常使用一個(gè)無(wú)符號(hào)長(zhǎng)整數(shù)(32位)中的比特位來(lái)表示各種不同的信號(hào),因此最多可以表示
32個(gè)不同的信號(hào)。
在本版linux內(nèi)核中定義了22種不同的信號(hào),
2 對(duì)進(jìn)程來(lái)說(shuō) ,當(dāng)收到一個(gè)信號(hào)的時(shí)候,可以有三種不同的處理方式
(1) 忽略該信號(hào)。 大多數(shù)信號(hào)都可以被進(jìn)程忽略,但是有兩個(gè)信號(hào)忽略不掉:SIGKILL 和 SIGSTOP 。
其原因是為了給超級(jí)用戶提供一個(gè)確定的方法來(lái)終止或停止 指定的任何進(jìn)程。
(2 捕捉該信號(hào)。為了進(jìn)行該操作,我們必須首先告訴內(nèi)核在指定的信號(hào)發(fā)生時(shí)調(diào)用我們自定義的信號(hào)處理函數(shù)。
在該處理函數(shù)中,我們可以做任何操作,也可以什么都不做。自定義信號(hào)處理函數(shù)來(lái)捕獲SIGTERM 信號(hào),
并在該函數(shù)中做一些清理臨時(shí)文件的工作。SIGTERM 信號(hào)是KILL命令發(fā)送的信號(hào)。
(3) 執(zhí)行默認(rèn)操作。 內(nèi)核為每種信號(hào)都提供一種默認(rèn)操作。通常這些默認(rèn)操作就是終止進(jìn)程的執(zhí)行。
3 下面主要是對(duì)signal.c中的 do_signal()函數(shù)的解釋,因?yàn)檫@個(gè)函數(shù)比較難理解
do_signal()函數(shù)是內(nèi)核系統(tǒng)調(diào)用(int 0x80)中斷處理程序中對(duì)信號(hào)的預(yù)處理程序,在進(jìn)程每次調(diào)用系統(tǒng)調(diào)用的時(shí)候,
若進(jìn)程已經(jīng)接受到信號(hào),則該函數(shù)會(huì)把信號(hào)的處理句柄插入到用戶程序堆棧中去。 這樣,在當(dāng)前系統(tǒng)調(diào)用結(jié)束返回后就會(huì)
立刻執(zhí)行信號(hào)句柄程序。

在信號(hào)處理程序插入到用戶堆棧之前,do_signal()函數(shù)會(huì)首先把用戶程序的堆棧指針向下擴(kuò)展longs個(gè)長(zhǎng)字,然后將相應(yīng)的參數(shù)添加進(jìn)其中。
(2) 當(dāng)用戶程序調(diào)用系統(tǒng)調(diào)用進(jìn)入到內(nèi)核堆棧的時(shí)候,內(nèi)核堆棧如下所示:
內(nèi)核態(tài)堆棧
原SS
原ESP
EFLAGS
CS
EIP

如上圖所示,eip已經(jīng)指向了 信號(hào)處理函數(shù),而且現(xiàn)在新的esp也已經(jīng)指向了 新的位置。