一、signalfd 是什么
信號(signal)本質是 Linux 進程間通信的一種機制,也叫軟中斷信號。signalfd 是一個跟信號關聯的文件描述符,能夠以 io 的行為獲取到系統信號,屬性上來講 signalfd 也是一個匿名 fd 類型:
root@ubuntu:~# ll /proc/15445/fd
lrwx—— 1 root root 64 Aug 24 16:42 3 -> anon_inode:[signalfd]
root@ubuntu:~# cat /proc/15445/fdinfo/3
pos: 0
flags: 02
mnt_id: 11
sigmask: 0000000000000006
從這里可以得到簡單的信息:
signal 用的匿名 inode ,signalfd 屬于匿名 fd 的一種;句柄關聯的重要信息就是 sigmask,通過?/proc/${pid}/fdinfo/3?能看到這個值其實信號是很講究的,甚至有信號編程一說,Linux 的 signalfd 為信號的處理提供了一種新的方法,統一到文件的 io 模式,契合一切接文件的理念。
系統調用:
#include
int signalfd(int fd, const sigset_t *mask, int flags);
該系統調用返回一個整數類型 signalfd,這個句柄跟信號行為綁定,當發生信號的時候,句柄觸發可讀事件。
名列前茅個參數也可以傳入一個有效的信號 fd 的句柄,如果傳入的是 -1 ,那么內核會自動創建一個新的 fd 。
完整的代碼例子,在 Linux 機器上,通過 man signalfd 就可以獲取到。
信號能夠像文件一樣 read 出來,這種優雅的信號處理方式得益于 signalfd 的封裝;信號是掛在在進程 task_struct 結構體上的,信號隊列非空的時候 signalfd 句柄可讀;和 epoll 池的配合同樣還是老套路,epoll_ctl 注冊的時候調用?.poll?接口掛載 epoll 的 wait entry 到?sighand->signalfd_wqh?之上,信號發送時()喚醒 epoll ;signalfd 也是一種匿名 fd 類型。延伸閱讀:
二、信號的使用示例
我們通過一段代碼實例來看一下信號量的使用吧。
void IntHandler(int signum) {
? std::cout << time(NULL) << ” Got a int signal” << signum << std::endl;
? std::this_thread::sleep_for(5s);
? std::cout << time(NULL) << ” Fininsh int signal” << signum << std::endl;
}
int main(int argc, char* argv[]) {
? signal(SIGINT, IntHandler);
? while(true) {
??? std::this_thread::sleep_for(10s);
??? std::cout << “.” << std::endl;
? }
? std::cout << std::endl;
? return 0;
}
上面這段代碼,我們通過signal(SIGINT, IntHandler);自定義了SIGINT信號量的處理。程序運行起來后,當按下ctrl + c時,IntHandler信號處理函數被觸發。