OS / Linux / 系统阻塞在系统调用中时如果收到信号,系统如何处理?
生活随笔
收集整理的這篇文章主要介紹了
OS / Linux / 系统阻塞在系统调用中时如果收到信号,系统如何处理?
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
今天在看《深入 Linux 內核架構》這本書的時候,看到如題的場景。
之前的處理方式是檢測系統調用函數的返回值,如果返回值為 -1 并且 errno 是 EINTR ,那么就知道了該系統調用被中斷打斷了,需要重新調用該函數。
栗子:
#include <signal.h> #include <stdio.h> #include <unistd.h> #include <errno.h> #include <string.h>volatile int signaled = 0;void handler(int) {printf("signaled called\n");signaled = 1; }int main(int argc, char *argv[]) {char ch;struct sigaction sigact;sigact.sa_handler = handler;//sigact.sa_flags = SA_RESTART;sigaction(SIGINT, &sigact, NULL);while (read(STDIN_FILENO, &ch, 1) == -1)printf("signaled call err info = %s\n", strerror(errno));return 0; }運行之后不斷地執行 Ctrl + C 時,會不斷的顯示如下內容:
signaled call err info = Interrupted system call
因為在阻塞 read 函數的時候收到了中斷信號,導致系統調用函數提前返回。
上述是傳統的做法,今天看書中還有另外一種做法,是 linux 參考 BSD 方案,大致流程是當系統阻塞在 read 函數中時,如果系統收到了中斷信號,系統調用不會提前返回,內核會在信號處理程序執行結束之后自動重啟該調用。
啟動上述功能的代碼是:
sigact.sa_flags = SA_RESTART;栗子:
#include <signal.h> #include <stdio.h> #include <unistd.h> #include <errno.h> #include <string.h>volatile int signaled = 0;void handler(int) {printf("signaled called\n");signaled = 1; }int main(int argc, char *argv[]) {char ch;struct sigaction sigact;sigact.sa_handler = handler;sigact.sa_flags = SA_RESTART;sigaction(SIGINT, &sigact, NULL);while (read(STDIN_FILENO, &ch, 1) != 1 && !signaled);return 0; }運行之后不斷地執行 Ctrl + C 時,會不斷的顯示如下內容:
signaled called?因為程序阻塞在 read 函數,收到中斷信號之后,由于內核自動重啟了 read 函數,導致程序還是阻塞在 read 函數中,始終無法響應 signaled 的變化,導致程序無法正常退出。
拓展
查看 errno 值含義:
/usr/include/asm-generic/errno-base.h???? 或者??? errno.h
?
(SAW:Game Over!)
?
總結
以上是生活随笔為你收集整理的OS / Linux / 系统阻塞在系统调用中时如果收到信号,系统如何处理?的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: OS / Linux / Select
- 下一篇: OS / Linux / clone、f