信号集给我们提供了一个能表示多个信号的是数据类型(sigset_t), 它将在sigprocmask, sigpending, sigsuspend之类的函数中用到, 这些函数我会在以后的文章中介绍.
1. 信号集相关函数:
#include
<
signal.h
>
int
sigemptyset(sigset_t
*
set
);
成功则返回0, 出错则返回
-
1
.
这个函数用作初始化set指向的信号集, 清空其中的所有信号.
#include
<
signal.h
>
int
sigfillset(sigset_t
*
set
);
成功则返回0, 出错则返回
-
1
.
这个函数用作初始化set指向的信号集, 填充其中的所有信号.
#include
<
signal.h
>
int
sigaddset(sigset_t
*
set
,
int
signo);
int
sigdelset(sigset_t
*
set
,
int
signo);
成功则返回0, 出错则返回
-
1
.
这两个函数用作向set指向的信号集中, 增加/删除一个signo代表的信号.
#include
<
signal.h
>
int
sigismember(
const
sigset_t
*
set
,
int
signo);
真则返回1, 假则返回0, 出错则返回
-
1
.
这个函数用作判断signo信号是否在set指向的信号集中.
2. 宏:
在signal.h中有两个宏:
#define
sigemptyset(ptr) (*(ptr) = 0)
#define
sigfillset(ptr) (*ptr = ~(sigset_t)0, 0)
这两个宏分别定义了sigemptyset和sigfillset两个函数的行为.
- sigemptyset: 把ptr指向的地址的内容设为0.
- sigfillset: 这是一个逗号表达式, 把0转换为sigset_t类型, 然后按位取反, 并返回0.
通过以上两个宏, 我们可以确切地说, sigset_t是用多位(比信号总数更多的位数)来表示信号集概念的.
因此,
sigaddset: 将某一位设置为1.
sigdelset: 将某一位设置为0.
sigismember: 测试某一个指定位.
下面我们来实现这些函数.
3. 实例:
#include
<
signal.h
>
#include
<
errno.h
>
/*
NSIG defined in <signal.h>
*/
#define
SIGBAD(signo) ((signo) <= 0 || (signo) >= NSIG)
int
sigaddset(sigset_t
*
set
,
int
signo)
{
if
(SIGBAD(signo))
{
errno
=
EINVAL;
return
(
-
1
);
}
*
set
|=
1
<<
(signo
-
1
);
return
0
;
}
int
sigdelset(sigset_t
*
set
,
int
signo)
{
if
(SIGBAD(signo))
{
errno
=
EINVAL;
return
(
-
1
);
}
*
set
&=
~
(
1
<<
(signo
-
1
));
return
0
;
}
int
sigismember(
const
sigset_t
*
set
,
int
signo)
{
if
(SIGBAD(signo))
{
errno
=
EINVAL;
return
(
-
1
);
}
return
((
*
set
&
(
1
<<
(signo
-
1
)))
!=
0
);
}
说明一下里面的几个细节:
signo - 1: 因为不存在编号为0的信号, 也就是第0位与编号为1的信号是对应的, 所以减1.
SIGBAD: 小于等于0, 或者大于最大信号编号NSIG.
1 << : 1的左移位操作, 右边补0, 所以执行或操作时直接实现添加; 执行与操作时需要先取反, 以保证本位为0, 其他位不变, 这样来实现删除.
sigismember: 最后一个return语句看上去有些复杂. 想起来很难想到, 但看起来应该不难. 执行移位后的与操作, 当存在时使本位保持不变, 其他位清零; 当不存在时, 全部清零. 然后判断是否为0, 这样来实现判断存在性.