让我们先回顾一下未决信号集是什么。当信号从产生到抵达目的地时,这个过程称为信号递达。而信号从产生到递达的中间状态,则被称为信号的未决状态。产生未决状态的原因可能是信号被阻塞,也就是信号屏蔽字(或阻塞信号集)中的对应位被置为1。阻塞信号集和未决信号集都是由内核维护的。整个过程如下图所示:

我们有时需要屏蔽某个信号,就需要去修改阻塞信号集。那么,我们该如何修改阻塞信号集?系统提供的一个方法是,我们先创建一个跟阻塞信号集一样的集合,再利用它去修改阻塞信号集。

系统提供了一系列的信号集设定函数。这些函数如下所示:

sigset_t  set;

信号集数据类型,本质是typedef unsigned long sigset_t; 

int sigemptyset(sigset_t *set);

将某个信号集清0

int sigfillset(sigset_t *set);

将某个信号集置1

int sigaddset(sigset_t *set, int signum);

将某个信号加入信号集

int sigdelset(sigset_t *set, int signum);

将某个信号清出信号集

以上几个函数返回值均是:成功:0;失败:-1

int sigismember(const sigset_t *set, int signum);

判断某个信号是否在信号集中

返回值:在集合:1;不在:0;出错:-1
登录后复制

使用以上这些函数创建完信号集后,要如何去改变阻塞信号集呢?系统又提供了一个函数:sigprocmask函数。sigprocmask函数可以用来屏蔽信号,也可以用来解除屏蔽信号,其本质就是利用我们创建的信号集去改变阻塞信号集。

函数原型:

int sigprocmask(int how, const sigset_t *set, sigset_t *oldset);
登录后复制

返回值:

成功:0;失败:-1,设置errno

参数解释:

set:传入参数,是一个位图,set中哪位置1,就表示当前进程屏蔽哪个信号。

oldset:传出参数,保存旧的信号屏蔽集。这个与setitimer有点相似。

how参数取值:

假设当前的信号屏蔽字为mask

1.SIG_BLOCK:当how设置为此值,set表示需要屏蔽的信号。相当于 mask = mask | set

2.SIG_UNBLOCK:当how设置为此,set表示需要解除屏蔽的信号。相当于 mask = mask & ~set

3.SIG_SETMASK:当how设置为此,set表示用于替代原始屏蔽及的新屏蔽集。相当于 mask = set若,调用sigprocmask解除了对当前若干个信号的阻塞,则在sigprocmask返回前,至少将其中一个信号递达。

我们如何读取未决信号集?系统提供了sigpending函数。

函数原型:

int sigpending(sigset_t *set);
登录后复制

参数说明:

set传出参数。

返回值:

返回值:成功:0;失败:-1,设置errno

例:把所有常规信号的未决状态打印至屏幕。

#include 
#include 
#include 

void printPending(sigset_t *set)
{
    int i = 0;

    for (i = 0; i if (sigismember(set, i) == 1)
            printf("1");
        else
            printf("0");
    }
    printf("
");
}

int main()
{
    sigset_t set, oldset, pendset;
    sigemptyset(&set);
    sigaddset(&set, SIGQUIT);   // ctrl +  将产生SIGQUIT信号
    sigprocmask(SIG_BLOCK, &set, &oldset);
    while (1) {
        sigpending(&pendset);
        printPending(&pendset);     // 写一个函数打印未决信号集
        sleep(1);
    }
}
登录后复制

以上就是详解Linux信号集操作函数的详细内容,更多请关注慧达安全导航其它相关文章!

点赞(0)

评论列表 共有 0 条评论

暂无评论
立即
投稿
发表
评论
返回
顶部