麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁 > 系統 > Unix > 正文

《Unix環境高級編程》讀書筆記 第10章-信號

2024-06-28 13:24:24
字體:
來源:轉載
供稿:網友
《Unix環境高級編程》讀書筆記 第10章-信號1.引言
  • 信號是軟件中斷
  • 信號提供了一種處理異步事件的方法。
2. 信號概念
  • 信號的名字都是以3個字符SIG開頭。
  • linux3.2.0支持31種信號。FreeBSD、Linux和Solaris作為實時擴展都支持另外的應用程序定義的信號。
  • 在頭文件signal.h(其中include的bits/signum.h)中,信號名都被定義為正整數常量,不存在編號為0的信號。kill函數對信號編號0有特殊的應用。

  • 很多條件可以產生信號:

    1. 用戶按下某些終端鍵時:Ctrl+C、Ctrl+/、Ctrl+Z
    2. 硬件異常產生信號:除數為0、無效的內存引用
    3. 進程調用kill函數可將任意信號發送給另一個進程或進程組
    4. 當檢測到某些軟件條件已經發生,并應將其通知有關進程時產生信號。如:SIGURG(網絡連接上傳來帶外數據)、SIGPipE(在管道的讀進程已經終止后,一個進程寫此管道)、SIGALRM(進程所設置的定時器超時)
  • 信號是異步事件的經典實例。產生信號的事件對進程而言是隨機出現的。進程不能簡單地測試一個變量(如errno)來判斷是否發生了一個信號,而是必須告訴內核“在此信號發生時,請執行下列操作”。

  • 當某個信號出現時,可以告訴內核按下列3種方式之一進行處理,稱之為信號的處理

    1. 忽略此信號。SIG_IGN。只有兩種信號不能被忽略:SIGKILL和SIGSTOP。原因是:它們向內核和超級用戶提供了使進程終止或停止的可靠方法。另外,如果忽略某些由硬件異常產生的信號(如非法內存引用或除以0),則進程的運行行為是未定義的。
    2. 捕捉信號。即通知內核在某種信號發生后,調用一個用戶函數。
    3. 執行系統默認動作。對大多數信號的默認動作是終止該進程。
  • 終止+core。大多數Unix系統調試程序都使用core文件檢查進程終止時的狀態。

  • 在下列條件下不產生core文件:

    1. 進程是設置用戶ID的,而且當前用戶并非程序文件的所有者
    2. 進程是設置組ID的,而且當前用戶并非程序文件的組所有者
    3. 用戶沒有寫當前工作目錄的權限
    4. 文件已存在,而且用戶對該文件沒有寫權限
  • 4個平臺對各種signal的支持及默認處理方式

  • 主要信號簡要說明:
    • SIGABRT。調用abort函數時產生此信號。
    • SIGALRM。當用alarm函數設置的定時器超時時,產生此信號。
    • SIGCHLD。在一個進程終止或停止時,該信號被送給其父進程。按系統默認,將忽略此信號。
    • SIGFPE。表示算術運算異常,如除以0、浮點溢出等。
    • SIGHUP。如果終端接口檢測到一個連接斷開,則將此信號送給與該終端相關的控制進程(會話首進程)。通常使用此信號通知守護進程再次讀取它們的配置文件。選用此信號的理由是:守護進程不會有控制終端,通常決不會接收到這種信號。
    • SIGILL。表示進程執行一條非法硬件指令。
    • SIGINT。當用戶按下中斷鍵Ctrl+C時,終端驅動程序產生此信號并發送至前臺進程組的每一個進程。
    • SIGIO。指示一個異步I/O事件。
    • SIGTERM。由kill命令發送的系統默認終止信號。
    • SIGKILL。不能捕獲或忽略。它向管理員提供了一紅殺死任一進程的可靠方法。
    • SIGPIPE。如果在管道的讀進程已終止時寫管道,則產生此信號。
    • SIGQUIT。當用戶在終端上按下退出鍵Ctrl+/時,終端驅動程序產生此信號并發送給前臺進程組中的所有進程。此信號除了終止前臺進程組(和SIGINT一樣),同時產生一個core文件。
    • SIGSEGV。指示進程進行了一次無效的內存引用。
    • SIGTSTP。交互停止信號。當用戶在終端上按下掛起鍵Ctrl+Z時,終端驅動程序產生此信號,并發送至前臺進程組的所有進程。
    • SIGSTOP。類似于交互停止信號(SIGTSTP),但它不能被捕獲或忽略。
    • SIGCONT。此作業控制信號發送給需要繼續運行,但當前處于停止狀態的進程。
    • SIGTTIN。當一個后臺進程組進程試圖讀其控制終端時,終端驅動程序產生此信號。下列情況例外:1. 讀進程忽略或阻塞此信號;2. 讀進程所屬的進程組是孤兒進程組,此時讀操作返回出錯,errno設置為EIO
    • SIGTTOU。當一個后臺進程組進程試圖寫其控制終端時,終端驅動程序產生此信號。
    • SIGURG。通知進程已經發生一個緊急情況。如帶外數據到達。
    • SIGUSR1、SIGUSR2。用戶定義的信號,可用于應用程序。
3. 函數signal
  1. #include <signal.h>
  2. void (*signal(int signo, void (*func)(int)))(int);
  3. Returns: PRevious disposition of signal (see following) if OK, SIG_ERR on error
  • signal函數由ISO C定義。不涉及多進程、進程組以及終端I/O等,所以它對信號的定義非常含糊,以致于對Unix系統而言幾乎毫無用處。
  • 因為signal的語義與實現有關,所以最好使用sigaction函數代替signal函數。
  • 本書中的所有實例均使用圖10-18中給出的signal函數,該函數使用sigaction函數是一個平臺無關、語義一致的實現。
  • signo參數是上面的信號名。func參數可以是常量SIG_IGN、SIG_DFL或接收到該信號后要調用的函數的地址,即信號處理程序的地址。signal函數的返回值是指向在此之前的信號處理程序的指針。
  1. typedef void Sigfunc(int);
  2. Sigfunc* signal(int, Sigfunc*);
  3. #define SIG_ERR (void (*)())-1
  4. #define SIG_DFL (void (*)())0
  5. #define SIG_IGN (void (*)())1
  • exec,程序啟動

    當exec執行一個程序時,所有信號都被設置為它們的默認動作,除非調用exec的進程忽略該信號(則繼續保持忽略)。也就是說,exec函數將原先設置為要捕獲的信號都更改為默認動作,其他保持不變。因為當exec一個新程序時,信號處理程序的地址很可能在新程序中已無意義。

  • fork,進程創建

    當一個進程調用fork時,其子進程繼承父進程的信號處理方式。因為信號處理程序的地址在子進程中是有意義的。

4. 不可靠的信號
  • 早期的Unix版本中,信號是不可靠的。不可靠指的是,信號可能會丟失:一個信號發生了,當進程卻可能一直不知道。同時,進程對信號的控制能力很差,它能捕獲或忽略它,但不能阻塞。
  • 早期版本的另一個問題是:在進程每次接到信號對其進行處理時,隨即將該信號動作重置為默認值。故需要再次建立對該信號的捕獲,但在此期間有一個時間窗口。
  • 早期版本的另一個問題是:在進程不希望某種信號發生時,它不能關閉該信號,只能忽略它。
5. 中斷的系統調用
  • 早期Unix系統的一個特性是:如果進程在執行一個低速系統調用而阻塞期間捕捉到一個信號,則該系統調用就被中斷不再繼續運行。該系統調用返回出錯,其errno設置為EINTR。
  • 為了支持這種特性,將系統調用分為兩類:低速系統調用和其他系統調用。
  • 低速系統調用是可能會使進程永遠阻塞的一類系統調用。
  • 與被中斷的系統調用相關的問題是必須顯式地處理出錯返回。典型的代碼序列如下:
  1. again:
  2. if ((n = read(fd, buf, BUFFSIZE)) < 0) {
  3. if (errno == EINTR)
  4. goto again; /* just an interrupted system call */
  5. /* handle other errors */
  6. }
  • 4.2 BSD引進了某些被中斷系統調用的自動重啟動,包括ioctl、read、readv、write、writev、wait、waitpid。但是這種自動重啟動的處理方式也會帶來問題,某些應用程序并不希望這些函數被中斷后重啟動。為此,4.3 BSD運行進程基于每個信號禁用此功能。
  • POSIX.1要求只有中斷信號的SA_RESTART標志有效時,實現才重啟動系統調用。
  • 歷史上,使用signal函數建立信號處理程序時,對于如何處理被中斷的系統調用,各種實現的做法各不相同。
6. 可重入函數
  • 進程捕捉到信號并對其進行處理時,進程正在執行的正常指令序列就被信號處理程序臨時中斷,它首先執行該信號處理程序中的指令。但是,在信號處理程序中,不能判斷捕捉到信號時進程執行到何處:
    1. 如果進程正在執行malloc,而在信號處理程序中又再次調用malloc,這時會?
    2. 如果進程正在執行getpwnam,這是將其結果存放在靜態存儲單元中的函數,而在信號處理程序中又再次調用getpwnam,這時會?
  • SUS說明了在信號處理程序中保證調用安全的函數。這些函數是可重入的,并被稱為異步信號安全的。
  • 沒有列入上圖的大多數函數是不可重入的,因為:
    1. 它們使用靜態數據結構
    2. 它們調用malloc或free
    3. 它們是標準的I/O函數。標準I/O庫的很多實現都以不可重入方式使用全局數據結構。
  • 應當了解,即使信號處理程序調用的是上圖中的函數,但是由于每個線程只有一個errno變量,所以信號處理程序可能會修改其原先值。故作為一個通用的規則,先保存,后恢復。
7. SIGCLD語義8. 可靠信號術語和語義
  • 首先,當造成信號的事件發生時,向進程發送一個信號。
  • 當對信號采取了某種動作時,我們說向進程遞送了一個信號。在信號產生和遞送之間的時間間隔內,稱信號是未決的。
  • 進程可以選用“阻塞信號遞送”。內核在遞送一個原來被阻塞的信號給進程時(而不是在產生該信號時),才決定對它的處理方式。因此,進程在信號遞送給它之前仍可改變對該信號的動作。
  • 每個進程都有一個信號屏蔽字,它規定了當前要阻塞遞送到該進程的信號集。進程可以調用sigprocmask函數來檢測和更改其當前信號屏蔽字。
  • 進程調用sigpending函數來判定哪些信號是設置為阻塞并處于未決狀態的。
  • 如果在進程解除對某個信號的阻塞之前,該信號發生了多次,那么?如果遞送該信號多次,則稱這些信號進行了排隊。除非支持POSIX.1實時擴展,否則大多數Unix并不對信號排隊,而只遞送一次。
9. 函數kill和raise
  • kill函數將信號發送給進程或進程組
  • raise函數則允許進程向自身發送信號。
  1. #include <signal.h>
  2. int kill(pid_t pid, int signo);
  3. int raise(int signo);
  4. Both return: 0 if OK, −1 on error
  • raise(signo); 等價于 kill(getpid(), signo);
  • kill的pid參數有以下4種情況:
    • pid > 0,發送給進程ID為pid的進程
    • pid == 0,發送給與發送進程屬于同一進程組的所有進程
    • pid < 0,發送給其進程組ID等于pid絕對值,而且發送進程具有權限向其發送信號的所有進程
    • pid == -1,發送給發送進程具有權限向它們發送信號的所有進程
  • 關于發送信號的權限

    1. 超級用戶可將信號發送給任一進程。
    2. 非超級用戶,其基本規則是發
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 国产精品一区二区三区在线看 | 国产在线观看免费视频软件 | 成人免费一区 | 欧美一级一区二区三区 | 啪啪激情| 9999久久| 午夜视频在线看 | 深夜视频在线观看 | 高清视频一区二区 | 中文字幕国产亚洲 | 国产精品久久久久久久久久东京 | 黄色影院av | 无遮挡一级毛片视频 | 国产精品手机在线亚洲 | 国产一区二区三区手机在线 | 久久亚色 | 日本在线国产 | 亚洲第一综合色 | 黑人三级毛片 | 中文字幕在线视频网站 | 免费黄色在线 | 久久视频精品 | 欧美日韩国产一区二区三区在线观看 | 欧美乱论| 国产91片 | 欧美中文字幕一区二区三区亚洲 | 日韩视频在线观看免费视频 | 毛片在线视频观看 | 日韩欧美视频一区二区三区 | 欧美性生活免费视频 | 黄片一级毛片 | 一区二区高清视频在线观看 | 黄色片免费看网站 | 日本在线视频一区二区三区 | 本站只有精品 | 亚洲午夜在线视频 | 色婷婷a v | 久国久产久精永久网页 | sese综合| www.91sp| 美女网站黄在线观看 |