【Linux】信号的处理以及信号集操作函数

首先来再次看看这张图:

sigset_t

从上图来看,每个信号只有一个bit的未决标志,非0即1,不记录该信号产生了多少次,阻塞标志也是这样表示的。因此,未决和阻塞标志可以用相同的数据类型sigset_t来存储,sigset_t称为信号集,这个类型可以表示每个信号的“有效”或“无效”状态,在阻塞信号集中“有效”和“无效”的含义是该信号是否被阻塞,而在未决信号集中“有效”和“无效”的含义是该信号是否处于未决状态。

阻塞信号集也叫做当前进程的信号屏蔽字(Signal Mask),这里的“屏蔽”应该理解为阻塞而不是忽略。不能对sigset_t进行位操作,去修改里面的内容,因为我们不知道他的底层里面都存放的是什么,它只是帮助我们拿到操作系统底层的数据,因此我们要对位图进行操作时必须要使用操作系统提供的信号集操作函数。

信号集操作函数

sigset_t类型对于每种信号用一个bit表示“有效”或“无效”状态,至于这个类型内部如何存储这些bit则依赖于系统
实现,从使用者的角度是不必关心的,使用者只能调用以下函数来操作sigset_ t变量,而不应该对它的内部数据做
任何解释,比如用printf直接打印sigset_t变量是没有意义的。

#include <signal.h>
1.int sigemptyset(sigset_t *set); //初始化信号集、清空
2.int sigfillset(sigset_t *set); //填满信号集
3.int sigaddset (sigset_t *set, int signo); //把某个信号添加到信号集
4.int sigdelset(sigset_t *set, int signo); //删除信号集中的某个信号
5.int sigismember(const sigset_t *set, int signo); //判断一个信号是否在信号集中
  • 函数sigemptyset初始化set所指向的信号集,使其中所有信号的对应bit清零,表示该信号集不包含 任何有效信号。
  • 函数sigfillset初始化set所指向的信号集,使其中所有信号的对应bit置位,表示 该信号集的有效信号包括系统支持的所有信号。
  • 注意,在使用sigset_ t类型的变量之前,一定要调 用sigemptyset或sigfillset做初始化,使信号集处于确定的状态。初始化sigset_t变量之后就可以在调用sigaddset和sigdelset在该信号集中添加或删除某种有效信号。

这四个函数都是成功返回0,出错返回-1。sigismember是一个布尔函数,用于判断一个信号集的有效信号中是否包含
某种 信号,若包含则返回1,不包含则返回0,出错返回-1。

sigprocmask

调用函数sigprocmask可以读取或更改进程的信号屏蔽字(阻塞信号集)。

#include <signal.h>
int sigprocmask(int how, const sigset_t *set, sigset_t *oset);
返回值:若成功则为0,若出错则为-1

如果oset是非空指针,则读取进程的当前信号屏蔽字通过oset参数传出。如果set是非空指针,则更改进程的信号屏蔽字,参数how指示如何更改。如果oset和set都是非空指针,则先将原来的信号屏蔽字备份到oset里,然后根据set和how参数更改信号屏蔽字。假设当前的信号屏蔽字为mask,下表说明了how参数的可选值。

SIG_BLOCK set包含了我们希望添加到当前信号屏蔽字的信号,相当于mask=mask|set
SIG_UNBLOCK set包含了我们希望从当前信号屏蔽字中解除阻塞的信号,相当于mask=mask&~set
SIG_SETMASK 设置当前信号屏蔽字为set所指向的值,相当于mask=set

如果调用sigprocmask解除了对当前若干个未决信号的阻塞,则在sigprocmask返回前,至少将其中一个信号递达。

sigpending

读取当前进程的未决信号集,通过set参数传出。

#include <signal.h>
int sigpending(sigset_t *set);

 

代码实例

输出结果

具体代码:

  1 #include <stdio.h>
  2 #include <unistd.h>
  3 #include <signal.h>
  4 void handler(int signo){
  5     printf("Get a Signal No. is%d\n",signo);
  6 }
  7 void show(sigset_t *pending){
  8     int i = 1;
  9     for( ;i<= 31;i++){
 10         if(sigismember(pending, i)){
 11             printf("1");
 12         }else{
 13             printf("0");
 14         }
 15     }
 16     printf("\n");
 17 }
 18 int main(){
 19     sigset_t set, oset;
 20     sigemptyset(&set);
 21     sigemptyset(&oset);
 22     sigaddset(&set , 2);
 23     sigprocmask(SIG_SETMASK, &set, &oset);
 24     signal(2, handler);
 25     sigset_t pending;
 26     int i = 15;
 27     while(1) {
 28         sigpending(&pending);
 29         show(&pending);
 31         if(i-- == 0){
 32             sigprocmask(SIG_SETMASK, &oset, NULL);
 33         }
 34     }
 35     return 0;
 36 }

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章