Linux被中斷的系統(tǒng)如何調(diào)用詳解
前言
慢系統(tǒng)調(diào)用,指的是可能永遠無法返回,從而使進程永遠阻塞的系統(tǒng)調(diào)用,比如無客戶連接時的accept、無輸入時的read都屬于慢速系統(tǒng)調(diào)用。
在Linux中,當阻塞于某個慢系統(tǒng)調(diào)用的進程捕獲一個信號,則該系統(tǒng)調(diào)用就會被中斷,轉(zhuǎn)而執(zhí)行信號處理函數(shù),這就是被中斷的系統(tǒng)調(diào)用。
然而,當信號處理函數(shù)返回時,有可能發(fā)生以下的情況:
- 如果信號處理函數(shù)是用signal注冊的,系統(tǒng)調(diào)用會自動重啟,函數(shù)不會返回
- 如果信號處理函數(shù)是用sigaction注冊的
- 默認情況下,系統(tǒng)調(diào)用不會自動重啟,函數(shù)將返回失敗,同時errno被置為EINTR
- 只有中斷信號的SA_RESTART標志有效時,系統(tǒng)調(diào)用才會自動重啟
下面我們編寫代碼,分別驗證上述幾種情形,其中系統(tǒng)調(diào)用選擇read,中斷信號選擇SIGALRM,中斷信號由alarm產(chǎn)生。
使用signal
#include <stdio.h> #include <signal.h> #include <unistd.h> #include <errno.h> void handler(int s) { printf("read is interrupt by signal handler\n"); return; } int main() { char buf[10]; int nread = 0; signal(SIGALRM, handler); alarm(2); printf("read start\n"); nread = read(STDIN_FILENO, buf, sizeof(buf)); printf("read return\n"); if ((nread < 0) && (errno == EINTR)) { printf("read return failed, errno is EINTR\n"); } return 0; }
使用sigaction + 默認情況
#include <stdio.h> #include <signal.h> #include <unistd.h> #include <errno.h> void handler(int s) { printf("read is interrupt by signal handler\n"); return; } int main() { char buf[10]; int nread = 0; struct sigaction act; sigemptyset(&act.sa_mask); act.sa_handler = handler; act.sa_flags = 0; //不給SIGALRM信號設置SA_RESTART標志,使用sigaction的默認處理方式 //act.sa_flag |= SA_INTERRUPT; //SA_INTERRUPT是sigaction的默認處理方式,即不自動重啟被中斷的系統(tǒng)調(diào)用 //實際上,不管act.sa_flags值為多少,只要不設置SA_RESTART,sigaction都是按SA_INTERRUPT處理的 sigaction(SIGALRM, &act, NULL); alarm(2); printf("read start\n"); nread = read(STDIN_FILENO, buf, sizeof(buf)); printf("read return\n"); if ((nread < 0) && (errno == EINTR)) { printf("read return failed, errno is EINTR\n"); } return 0; }
使用sigaction + 指定SA_RESTART標志
#include <stdio.h> #include <signal.h> #include <unistd.h> #include <errno.h> void handler(int s) { printf("read is interrupt by signal handler\n"); return; } int main() { char buf[10]; int nread = 0; struct sigaction act; sigemptyset(&act.sa_mask); act.sa_handler = handler; act.sa_flags = 0; act.sa_flags |= SA_RESTART; //給SIGALRM信號設置SA_RESTART標志 sigaction(SIGALRM, &act, NULL); alarm(2); printf("read start\n"); nread = read(STDIN_FILENO, buf, sizeof(buf)); printf("read return\n"); if ((nread < 0) && (errno == EINTR)) { printf("read return failed, errno is EINTR\n"); } return 0; }
由于對被中斷系統(tǒng)調(diào)用處理方式的差異性,因此對應用程序來說,與被中斷的系統(tǒng)調(diào)用相關的問題是:
- 應用程序無法保證總是知道信號處理函數(shù)的注冊方式,以及是否設置了SA_RESTART標志
- 可移植的代碼必須顯式處理關鍵函數(shù)的出錯返回,當函數(shù)出錯且errno等于EINTR時,可以根據(jù)實際需求進行相應處理,比如重啟該函數(shù)
int nread = read(fd, buf, 1024); if (nread < 0) { if (errno == EINTR) { //read被中斷,其實不應該算作失敗,可以根據(jù)實際需求進行處理,比如重寫調(diào)用read,也可以忽略它 } else { //read真正的讀錯誤 } }
總結
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對腳本之家的支持。
相關文章
linux修改root密碼和linux忘記root密碼后找回密碼的方法
這篇文章主要介紹了linux修改root密碼和linux忘記root密碼后找回密碼的方法,需要的朋友可以參考下2014-01-01Linux下使用使用socket實現(xiàn)TCP服務端的示例代碼
套接字(socket)是 Linux 下的一種進程間通信機制(socket IPC),它不僅支持同一主機的不同進程間通信,還支持跨網(wǎng)絡的不同主機的進程間通信,本文介紹了 Linux 下使用 socket 接口實現(xiàn) TCP 服務端的示例程序,需要的朋友可以參考下2024-03-03Linux系統(tǒng)下使用rpm方式安裝最新mysql5.7.17完整步驟
這篇文章主要介紹了Linux系統(tǒng)下使用rpm方式安裝最新mysql5.7.17完整步驟,需要的朋友可以參考下2017-01-01