一、signalfd 是什么
信號(hào)(signal)本質(zhì)是 Linux 進(jìn)程間通信的一種機(jī)制,也叫軟中斷信號(hào)。signalfd 是一個(gè)跟信號(hào)關(guān)聯(lián)的文件描述符,能夠以 io 的行為獲取到系統(tǒng)信號(hào),屬性上來講 signalfd 也是一個(gè)匿名 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 的一種;句柄關(guān)聯(lián)的重要信息就是 sigmask,通過?/proc/${pid}/fdinfo/3?能看到這個(gè)值其實(shí)信號(hào)是很講究的,甚至有信號(hào)編程一說,Linux 的 signalfd 為信號(hào)的處理提供了一種新的方法,統(tǒng)一到文件的 io 模式,契合一切接文件的理念。
系統(tǒng)調(diào)用:
#include
int signalfd(int fd, const sigset_t *mask, int flags);
該系統(tǒng)調(diào)用返回一個(gè)整數(shù)類型 signalfd,這個(gè)句柄跟信號(hào)行為綁定,當(dāng)發(fā)生信號(hào)的時(shí)候,句柄觸發(fā)可讀事件。
名列前茅個(gè)參數(shù)也可以傳入一個(gè)有效的信號(hào) fd 的句柄,如果傳入的是 -1 ,那么內(nèi)核會(huì)自動(dòng)創(chuàng)建一個(gè)新的 fd 。
完整的代碼例子,在 Linux 機(jī)器上,通過 man signalfd 就可以獲取到。
信號(hào)能夠像文件一樣 read 出來,這種優(yōu)雅的信號(hào)處理方式得益于 signalfd 的封裝;信號(hào)是掛在在進(jìn)程 task_struct 結(jié)構(gòu)體上的,信號(hào)隊(duì)列非空的時(shí)候 signalfd 句柄可讀;和 epoll 池的配合同樣還是老套路,epoll_ctl 注冊的時(shí)候調(diào)用?.poll?接口掛載 epoll 的 wait entry 到?sighand->signalfd_wqh?之上,信號(hào)發(fā)送時(shí)()喚醒 epoll ;signalfd 也是一種匿名 fd 類型。延伸閱讀:
二、信號(hào)的使用示例
我們通過一段代碼實(shí)例來看一下信號(hào)量的使用吧。
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信號(hào)量的處理。程序運(yùn)行起來后,當(dāng)按下ctrl + c時(shí),IntHandler信號(hào)處理函數(shù)被觸發(fā)。