軟件中斷
。異步事件
的方法。SIG
開頭。在頭文件signal.h(其中include的bits/signum.h)中,信號名都被定義為正整數(shù)常量,不存在編號為0的信號。kill函數(shù)對信號編號0有特殊的應用。
很多條件可以產(chǎn)生信號:
- 用戶按下某些終端鍵時:Ctrl+C、Ctrl+/、Ctrl+Z
- 硬件異常產(chǎn)生信號:除數(shù)為0、無效的內(nèi)存引用
- 進程調(diào)用kill函數(shù)可將任意信號發(fā)送給另一個進程或進程組
- 當檢測到某些軟件條件已經(jīng)發(fā)生,并應將其通知有關進程時產(chǎn)生信號。如:SIGURG(網(wǎng)絡連接上傳來帶外數(shù)據(jù))、SIGPipE(在管道的讀進程已經(jīng)終止后,一個進程寫此管道)、SIGALRM(進程所設置的定時器超時)
信號是異步事件的經(jīng)典實例。產(chǎn)生信號的事件對進程而言是隨機出現(xiàn)的。進程不能簡單地測試一個變量(如errno)來判斷是否發(fā)生了一個信號,而是必須告訴內(nèi)核“在此信號發(fā)生時,請執(zhí)行下列操作”。
當某個信號出現(xiàn)時,可以告訴內(nèi)核按下列3種方式之一進行處理,稱之為信號的處理
- 忽略此信號。SIG_IGN。只有兩種信號不能被忽略:SIGKILL和SIGSTOP。原因是:它們向內(nèi)核和超級用戶提供了使進程終止或停止的可靠方法。另外,如果忽略某些由硬件異常產(chǎn)生的信號(如非法內(nèi)存引用或除以0),則進程的運行行為是未定義的。
- 捕捉信號。即通知內(nèi)核在某種信號發(fā)生后,調(diào)用一個用戶函數(shù)。
- 執(zhí)行系統(tǒng)默認動作。對大多數(shù)信號的默認動作是終止該進程。
終止+core。大多數(shù)Unix系統(tǒng)調(diào)試程序都使用core文件檢查進程終止時的狀態(tài)。
在下列條件下不產(chǎn)生core文件:
- 進程是設置用戶ID的,而且當前用戶并非程序文件的所有者
- 進程是設置組ID的,而且當前用戶并非程序文件的組所有者
- 用戶沒有寫當前工作目錄的權限
- 文件已存在,而且用戶對該文件沒有寫權限
4個平臺對各種signal的支持及默認處理方式
- SIGABRT。調(diào)用abort函數(shù)時產(chǎn)生此信號。
- SIGALRM。當用alarm函數(shù)設置的定時器超時時,產(chǎn)生此信號。
- SIGCHLD。在一個進程終止或停止時,該信號被送給其父進程。按系統(tǒng)默認,將忽略此信號。
- SIGFPE。表示算術運算異常,如除以0、浮點溢出等。
- SIGHUP。如果終端接口檢測到一個連接斷開,則將此信號送給與該終端相關的控制進程(會話首進程)。通常使用此信號通知守護進程再次讀取它們的配置文件。選用此信號的理由是:守護進程不會有控制終端,通常決不會接收到這種信號。
- SIGILL。表示進程執(zhí)行一條非法硬件指令。
- SIGINT。當用戶按下中斷鍵Ctrl+C時,終端驅(qū)動程序產(chǎn)生此信號并發(fā)送至前臺進程組的每一個進程。
- SIGIO。指示一個異步I/O事件。
- SIGTERM。由kill命令發(fā)送的系統(tǒng)默認終止信號。
- SIGKILL。不能捕獲或忽略。它向管理員提供了一紅殺死任一進程的可靠方法。
- SIGPIPE。如果在管道的讀進程已終止時寫管道,則產(chǎn)生此信號。
- SIGQUIT。當用戶在終端上按下退出鍵Ctrl+/時,終端驅(qū)動程序產(chǎn)生此信號并發(fā)送給前臺進程組中的所有進程。此信號除了終止前臺進程組(和SIGINT一樣),同時產(chǎn)生一個core文件。
- SIGSEGV。指示進程進行了一次無效的內(nèi)存引用。
- SIGTSTP。交互停止信號。當用戶在終端上按下掛起鍵Ctrl+Z時,終端驅(qū)動程序產(chǎn)生此信號,并發(fā)送至前臺進程組的所有進程。
- SIGSTOP。類似于交互停止信號(SIGTSTP),但它不能被捕獲或忽略。
- SIGCONT。此作業(yè)控制信號發(fā)送給需要繼續(xù)運行,但當前處于停止狀態(tài)的進程。
- SIGTTIN。當一個后臺進程組進程試圖讀其控制終端時,終端驅(qū)動程序產(chǎn)生此信號。下列情況例外:1. 讀進程忽略或阻塞此信號;2. 讀進程所屬的進程組是孤兒進程組,此時讀操作返回出錯,errno設置為EIO
- SIGTTOU。當一個后臺進程組進程試圖寫其控制終端時,終端驅(qū)動程序產(chǎn)生此信號。
- SIGURG。通知進程已經(jīng)發(fā)生一個緊急情況。如帶外數(shù)據(jù)到達。
- SIGUSR1、SIGUSR2。用戶定義的信號,可用于應用程序。
#include <signal.h>
void (*signal(int signo, void (*func)(int)))(int);
Returns: PRevious disposition of signal (see following) if OK, SIG_ERR on error
typedef void Sigfunc(int);
Sigfunc* signal(int, Sigfunc*);
#define SIG_ERR (void (*)())-1
#define SIG_DFL (void (*)())0
#define SIG_IGN (void (*)())1
exec,程序啟動
當exec執(zhí)行一個程序時,所有信號都被設置為它們的默認動作,除非調(diào)用exec的進程忽略該信號(則繼續(xù)保持忽略)。也就是說,exec函數(shù)將原先設置為要捕獲的信號都更改為默認動作,其他保持不變。因為當exec一個新程序時,信號處理程序的地址很可能在新程序中已無意義。
fork,進程創(chuàng)建
當一個進程調(diào)用fork時,其子進程繼承父進程的信號處理方式。因為信號處理程序的地址在子進程中是有意義的。
again:
if ((n = read(fd, buf, BUFFSIZE)) < 0) {
if (errno == EINTR)
goto again; /* just an interrupted system call */
/* handle other errors */
}
- 如果進程正在執(zhí)行malloc,而在信號處理程序中又再次調(diào)用malloc,這時會?
- 如果進程正在執(zhí)行getpwnam,這是將其結果存放在靜態(tài)存儲單元中的函數(shù),而在信號處理程序中又再次調(diào)用getpwnam,這時會?
可重入的
,并被稱為異步信號安全
的。不可重入的
,因為:
- 它們使用靜態(tài)數(shù)據(jù)結構
- 它們調(diào)用malloc或free
- 它們是標準的I/O函數(shù)。標準I/O庫的很多實現(xiàn)都以不可重入方式使用全局數(shù)據(jù)結構。
遞送
了一個信號。在信號產(chǎn)生和遞送之間的時間間隔內(nèi),稱信號是未決
的。信號屏蔽字
,它規(guī)定了當前要阻塞遞送到該進程的信號集。進程可以調(diào)用sigprocmask函數(shù)
來檢測和更改其當前信號屏蔽字。sigpending函數(shù)
來判定哪些信號是設置為阻塞并處于未決狀態(tài)的。排隊
。除非支持POSIX.1實時擴展,否則大多數(shù)Unix并不對信號排隊,而只遞送一次。
#include <signal.h>
int kill(pid_t pid, int signo);
int raise(int signo);
Both return: 0 if OK, −1 on error
- pid > 0,發(fā)送給進程ID為pid的進程
- pid == 0,發(fā)送給與發(fā)送進程屬于同一進程組的所有進程
- pid < 0,發(fā)送給其進程組ID等于pid絕對值,而且發(fā)送進程具有權限向其發(fā)送信號的所有進程
- pid == -1,發(fā)送給發(fā)送進程具有權限向它們發(fā)送信號的所有進程
關于發(fā)送信號的權限
- 超級用戶可將信號發(fā)送給任一進程。
- 非超級用戶,其基本規(guī)則是發(fā)
新聞熱點
疑難解答