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

在信號處理程序插入到用戶堆棧之前,do_signal()函數(shù)會(huì)首先把用戶程序的堆棧指針向下擴(kuò)展longs個(gè)長字,然后將相應(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)指向了 信號處理函數(shù),而且現(xiàn)在新的esp也已經(jīng)指向了 新的位置。