日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > linux >内容正文

linux

OS / Linux / 系统阻塞在系统调用中时如果收到信号,系统如何处理?

發布時間:2024/10/14 linux 78 豆豆
生活随笔 收集整理的這篇文章主要介紹了 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 / 系统阻塞在系统调用中时如果收到信号,系统如何处理?的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。