信号屏蔽pending

xiaoxiao2021-02-28  130

讲到pending,就不得不提一下阻塞信号方面的知识:

实际执行信号的处理动作称为信号递达(Delivery),信号从产生到递达之间的状态,称为信号未(Pending)。进程可以选择阻塞(Block )某个信号。被阻塞的信号产生时将保持在未决状态,直到进程解除对此信号的阻塞,才 执行递达的动作。 注意: 阻塞和忽略是不同的,只要信号被阻塞就不会递达,而忽略是在递达之后可选的一种处理动作

信号间三张表的关系(block->pending->handler)

那么信号是怎么产生的呢?

1、键盘组合键(Ctrl+C、Ctrl+\、Ctrl+Z……) 2、软硬件产生(指针异常,内存溢出) 3、kill 信号(Linux下kill -l可对信号进行查看)

那收到信号了,又该怎么处理应对呢?

处理方式,一般有三种: 1.忽略 2.默认动作 3.用户自定义

信号阻塞:

信号阻塞,这PCB中,block所对应的比特位是“1”,表示信号阻塞,pending未决状态位可为“1”或者“0”,如果信号一旦递达,则进行信号处理(handler) 信号集操作函数 sigset_t类型对于每种信号用⼀一个bit表示“有效”或“无效”状态 int sigemptyset(sigset_t *set);//初始化set所指向的信号集,使其中所有信号的对应bit清零,表示该信号集不包含任何有效信号
 int sigfillset(sigset_t *set);//t初始化set所指向的信号集,使其中所有信号的对应bit置位有效,表示该信号集的有效信号包括系统支持的所有信号
 int sigaddset(sigset_t *set, int signo);//信号集中添加某种有效信号 int sigdelset(sigset_t *set, int signo);//信号集中删除某种有效信号
 int sigismember(const sigset_t *set, int signo); //用于判断一个信号集的有效信号中是否包含某种信号

调用函数sigprocmask可以读取或更改进程的信号屏蔽字(阻塞信号集)。 
 int sigprocmask(int how, const sigset_t *set, sigset_t *oset); 返回值:若成功则为0,若出错则为-1 how参数: 1)SIG_BLOCK添加到当前信号屏蔽字的信号 2)SIG_UNBLOCK从当前信号屏蔽字中解除的信号 3)SIG_SETMASK设置当前信号屏蔽字为set所指向的值

#include<stdio.h> #include<signal.h> #include<unistd.h> void printsigset(sigset_t *set) { int i = 0; for(;i<32;i++) { if(sigismember(set,i)) { putchar('1'); } else { putchar('0'); } } puts(""); } int main() { sigset_t s,p; sigemptyset(&s); sigaddset(&s,SIGINT); sigprocmask(SIG_BLOCK,&s,NULL); while(1) { sigpending(&p); printsigset(&p); sleep(1); } return 0; }

Ctrl+C前后 前: 后:

转载请注明原文地址: https://www.6miu.com/read-61238.html

最新回复(0)