信号在内核中的表示
实际执行信号的处理动作称为信号递达(Delivery),信号从产生到递达之间的状态,称为信号未决(Pending)。进程可以选择阻塞(Block)某个信号。被阻塞的信号产生时将保持在未决状态,直到进程解除对此信号的阻塞,才执行递达的动作。注意,阻塞和忽略是不同的,只要信号被阻塞就不会递达,而忽略是在递达之后可选的一种处理动作。信号在内核中的表示可以看作是这样的:
1)block集(阻塞集、屏蔽集):一个进程所要屏蔽的信号,在对应要屏蔽的信号位置1
2)pending集(未决信号集):如果某个信号在进程的阻塞集中,则也在未决集中对应位置1,表示该信号不能被递达,不会被处理3)handler(信号处理函数集):表示每个信号所对应的信号处理函数,当信号不在未决集中时,将被调用。
4)block状态字、pending状态字均64位(bit);
5)block状态字用户可以读写,pending状态字用户只能读;这是信号设计机制。
那么我们该如何对信号的屏蔽字状态进行改变和读取呢?接下来我们介绍一组信号集操作函数:
#include <signal.h>
int sigemptyset(sigset_t *set); //把信号集清零;(64bit/8=8字节)
int sigfillset(sigset_t *set); //把信号集64bit全部置为1
int sigaddset(sigset_t *set, int signo); //根据signo,把信号集中的对应位置成1
int sigdelset(sigset_t *set, int signo); //根据signo,把信号集中的对应位置成0
int sigismember(const sigset_t *set, int signo); //判断signo是否在信号集中
sigprocmask 功能:读取或者更改进程的信号屏蔽字(Block)
int sigprocmask(int how, const sigset_t *set, sigset_t *oset);
返回值:若成功则为0,若出错则为-1
读取:如果oset是非空指针,则读取进程的当前信号屏蔽字通过oset参数传出。
更改:如果set是非空指针,则更改进程的信号屏蔽字,参数how指示如何更改。如果oset和set都是非空指针,则先将原来的信号屏蔽字备份到oset里,然后根据set和how参数更改信号屏蔽字。假设当前的信号屏蔽字为mask,下表说明了how参数的可选值。
sigpending获取信号未决状态字(pending)信息,保存至set态,NSIG信号的最大值=64。
#include <signal.h>
int sigpending(sigset_t *set);
sigismember函数