Linux Signal (6): 信号屏蔽字
2007-09-25 15:46
218 查看
1. 概念:
信号屏蔽字就是进程中被阻塞的信号集, 这些信号不能发送给该进程, 它们在该进程中被"屏蔽"了. 后面我们会提到, 实际上它们是被阻塞了.
2. 信号屏蔽函数:
#include <signal.h>
int sigprocmask(int how, const sigset_t *restrict set, sigset_t *restrict oset);
成功则返回0, 出错则返回-1.
sigprocmask函数有3个参数:
how: 修改信号屏蔽字的方式.
set: 把这个信号集设为新的当前信号屏蔽字. 如果为NULL则不改变.
oset: 保存进程旧的信号屏蔽字. 如果为NULL则不保存.
参数中的how可以取3个值:
sigprocmask中的how参数
另外要说的是, sigprocmask只为单线程定义的, 在多线程中要使用pthread_sigmask.
3. 未处理的信号:
在调用信号屏蔽的相关函数后, 被屏蔽的信号对于调用进程是阻塞的, 不能发送给调用进程, 因此是未决的. 取得这些阻塞的信号集, 可以通过调用sigpending函数.
#include <signal.h>
int sigpending(sigset_t *set);
成功则返回0, 出错则返回-1.
4. 实例:
下面通过一个简单的实例来说明这篇文章中所讲到的两个函数.
#include <signal.h>
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
/* SIGQUIT handler */
static void sig_quit(int signo)
{
printf("SIGQUIT is caught ");
}
int main()
{
sigset_t new, old, pend;
/* Handle SIGQUIT */
if (signal(SIGQUIT, sig_quit) == SIG_ERR)
{
perror("signal");
exit(1);
}
/* Add SIGQUIT to sigset */
if (sigemptyset(&new) < 0)
perror("sigemptyset");
if (sigaddset(&new, SIGQUIT) < 0)
perror("sigaddset");
/* Mask SIGQUIT */
if (sigprocmask(SIG_SETMASK, &new, &old) < 0)
{
perror("sigprocmask");
exit(1);
}
printf("SIGQUIT is blocked ");
printf("Now try Ctrl / ");
sleep(5); /* SIGQUIT will pending */
/* Get pending */
if (sigpending(&pend) < 0)
perror("sigpending");
if (sigismember(&pend, SIGQUIT))
printf("SIGQUIT pending ");
/* Restore signal mask */
if (sigprocmask(SIG_SETMASK, &old, NULL) < 0)
{
perror("sigprocmask");
exit(1);
}
printf(" SIGQUIT unblocked ");
printf("Now try Ctrl / ");
sleep(5);
return 0;
}
这个程序在开始的时候用sigprocmask屏蔽了SIGQUIT(ctrl+/触发), 在5秒内触发的该信号将可以从sigpending中获得; 然后程序把SIGQUIT解除屏蔽(恢复以前的屏蔽字), 此时再触发该信号将调用sig_quit信号处理函数.
信号屏蔽字就是进程中被阻塞的信号集, 这些信号不能发送给该进程, 它们在该进程中被"屏蔽"了. 后面我们会提到, 实际上它们是被阻塞了.
2. 信号屏蔽函数:
#include <signal.h>
int sigprocmask(int how, const sigset_t *restrict set, sigset_t *restrict oset);
成功则返回0, 出错则返回-1.
sigprocmask函数有3个参数:
how: 修改信号屏蔽字的方式.
set: 把这个信号集设为新的当前信号屏蔽字. 如果为NULL则不改变.
oset: 保存进程旧的信号屏蔽字. 如果为NULL则不保存.
参数中的how可以取3个值:
how | 说明 |
SIG_BLOCK | 修改后, 该进程新的信号屏蔽字是其当前屏蔽字和set指向的信号集的并集. |
SIG_UNBLOCK | 修改后, 该进程新的信号屏蔽字是其当前屏蔽字和set指向的信号集的补集的交集. |
SIG_SETMASK | 修改后, 该进程新的信号屏蔽字将被set指向的信号集的值代替 |
3. 未处理的信号:
在调用信号屏蔽的相关函数后, 被屏蔽的信号对于调用进程是阻塞的, 不能发送给调用进程, 因此是未决的. 取得这些阻塞的信号集, 可以通过调用sigpending函数.
#include <signal.h>
int sigpending(sigset_t *set);
成功则返回0, 出错则返回-1.
4. 实例:
下面通过一个简单的实例来说明这篇文章中所讲到的两个函数.
#include <signal.h>
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
/* SIGQUIT handler */
static void sig_quit(int signo)
{
printf("SIGQUIT is caught ");
}
int main()
{
sigset_t new, old, pend;
/* Handle SIGQUIT */
if (signal(SIGQUIT, sig_quit) == SIG_ERR)
{
perror("signal");
exit(1);
}
/* Add SIGQUIT to sigset */
if (sigemptyset(&new) < 0)
perror("sigemptyset");
if (sigaddset(&new, SIGQUIT) < 0)
perror("sigaddset");
/* Mask SIGQUIT */
if (sigprocmask(SIG_SETMASK, &new, &old) < 0)
{
perror("sigprocmask");
exit(1);
}
printf("SIGQUIT is blocked ");
printf("Now try Ctrl / ");
sleep(5); /* SIGQUIT will pending */
/* Get pending */
if (sigpending(&pend) < 0)
perror("sigpending");
if (sigismember(&pend, SIGQUIT))
printf("SIGQUIT pending ");
/* Restore signal mask */
if (sigprocmask(SIG_SETMASK, &old, NULL) < 0)
{
perror("sigprocmask");
exit(1);
}
printf(" SIGQUIT unblocked ");
printf("Now try Ctrl / ");
sleep(5);
return 0;
}
这个程序在开始的时候用sigprocmask屏蔽了SIGQUIT(ctrl+/触发), 在5秒内触发的该信号将可以从sigpending中获得; 然后程序把SIGQUIT解除屏蔽(恢复以前的屏蔽字), 此时再触发该信号将调用sig_quit信号处理函数.
相关文章推荐
- Linux Signal (5): 信号屏蔽字
- Linux Signal (6): 信号屏蔽字
- Linux 信号signal处理机制
- Linux 进程间通信 --- 信号通信 --- signal --- signal(SIGINT, my_func); --- 按键驱动异步通知
- linux学习---信号(signal,sigaction,kill,sigqueue,sigprocmask,sigpending,sigsuspend)
- linux信号机制之sigaction结构体浅析,signal 函数,信号捕捉
- Linux 进程间通信 --- 信号通信 --- signal --- signal(SIGINT, my_func); --- 按键驱动异步通知
- Linux信号(signal) 机制分析
- linux下 signal信号机制的透彻分析与各种实例讲解 3
- Linux中信号Signal的学习与理解
- Linux 信号signal处理机制
- Linux信号(signal) 机制分析
- Linux编程 SIGNAL 信号表
- linux信号signal处理机制(四)
- Linux c/c++后端编程,信号量,屏蔽和不屏蔽,信号捕获;
- Linux--信号处理:在某个信号发生时屏蔽其他的信号
- linux信号屏蔽字
- linux 信号屏蔽
- Linux信号(signal) 机制分析
- Linux信号(signal) 机制分析