前言:
首先我们要知道,在Linux内核中的进程控制块PCB是一个结构体——task_struct,除了包含进程id,状态,工作目录,用户id,组id,文件描述符,还包含了信号相关的信息,主要指阻塞信号集(信号屏蔽字)block和未决信号集pending。 block状态字,pending状态都是64bit,分别代表Linux系统中的64个信号。
阻塞信号集(信号屏蔽字)
含义:就是将某些信号加入集合,对他们设置屏蔽,当屏蔽某个信号后,再接受这个信号,该信号的处理就会“阻塞”,当解除屏蔽后就会处理该信号。
未决信号集
含义:
- 信号产生,未决信号集描述该信号的位会立刻翻转为1,表示信号处于未决状态,当信号被处理对应位会翻转回为0,但是这一刻是非常短暂的
- 信号产生后由于某些原因(主要是阻塞)不能抵达,这类信号的集合称之为未决信号集,在屏蔽解除前,信号一直处于未决状态。
信号在内核中的表示
执行信号的处理动作成为信号递达(Delivery),信号从产生到递达之间的状态称为信号未决(Pending)。进程可以选择阻塞(Block)某个信号。
被阻塞的信号产生时将保持在未决状态,直到进程解除对此信号的阻塞,才执行递达的动作
注意:阻塞和忽略是不同的,只要信号被阻塞就不会递达,而忽略是在递达之后可选的一种处理动作
信号不会丢失,如果信号被阻塞,只会保持信号未决,但是信号不丢失
只要取消阻塞 信号依然会到来
对应的图如下:
未决和阻塞标志可以用相同的数据类型sigset_t来存储,sigset_t称为信号集,这个类型可以表示每个信号的“有效”或“无效”状态,在阻塞信号集中“有效”和“无效”的含义是该信号是否被阻塞,而在未决信号集中“有效”和“无效”的含义是该信号是否处于未决状态。
信号集操作函数
上个图快速了解一下
sigemptyset
#include <signal.h>
int sigemptyset(sigset_t *set);
sigemptyset是初始化set所指向的信号集,让其中所有的信号的对应的比特位清零,表示该信号集不包含任何有效信号。
返回值:调用成功则返回0,出错则返回-1
sigfillset
#include <signal.h>
int sigfillset(sigset_t *set);
sigfillset初始化set所指向的信号集,使其中所有信号的对应bit置位,表示该信号集的有效信号包括系统支持的所有信号。
返回值:调用成功则返回0,出错则返回-1
sigaddset
#include <signal.h>
int sigaddset(sigset_t *set, int signo);
sigaddset把某个特定信号加入到信号集上。
返回值:调用成功则返回0,出错则返回-1
sigdelset
#include <signal.h>
int sigdelset(sigset_t *set, int signo);
sigdelset把某个特定信号去掉
返回值:调用成功则返回0,出错则返回-1
sigismember
#include <signal.h>
int sigismember(const sigset_t *set, int signo);
sigismember是为了判断某个信号是否在该信号集中。
返回1代表在该信号集合里。
sigprocmask
#include <signal.h>
int sigprocmask(int how, const sigset_t *set, sigset_t *oset);
how参数 表示意义
SIG_BLOCK set包含了我们希望添加到当前信号屏蔽字的信号(往里加) //相当于block = block | set;
SIG_UNBLOCK set包含了我们希望从当前信号屏蔽字中解除阻塞的信号(往外减)
SIG_SETMASK 设置当前信号屏蔽字为set所指向的值(重新设置)
如果调用sigprocmask解除了对当前若干个未决信号的阻塞,则在sigprocmask返回前,至少将其中一个信号递达。
sigprocmask可以读取或更改进程的信号屏蔽字(阻塞信号集)。
返回值:若成功则为0,若出错则为-1
如果oset是非空指针,则读取进程的当前信号屏蔽字通过oset参数传出。如果set是非空指针,则 更改进程的信号屏蔽字,参数how指示如何更改。如果oset和set都是非空指针,则先将原来的信号 屏蔽字备份到oset里,然后根据set和how参数更改信号屏蔽字。假设当前的信号屏蔽字为mask,
sigpending
#include <signal.h>
int sigpending(sigset_t *set);
sigpending:读取当前进程的未决信号集,通过set参数传出。(输出型参数)
返回值:调用成功则返回0,出错则返回-1