一个进程的信号屏蔽字规定了当前堵塞而不能递送给该进程的信号集调用函数sigprocmask能够检測或更改其信号屏蔽字,或者在一个步骤中同一时候运行这两个操作。

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

首先,若oset是非空指针,那么进程的当前信号屏蔽字通过oset返回。

其次,若set是一个非空指针,则參数how指示怎样改动当前信号屏蔽字。

表10-4说明了how可选用的值。注意,不能堵塞SIGKILL和SIGSTOP信号

表10-4 用sigprocmask更改当前信号屏蔽字的方法

  how

  说明

  SIG_BLOCK   该进程新的信号屏蔽字是其当前信号屏蔽字和set指向信号集的并集。set包括了我们希望堵塞的附加信号
  SIG_UNBLOCK   该进程新的信号屏蔽字是其当前信号屏蔽字和set所指向信号集补集的交集。set包括了我希望解除堵塞的信号
  SIG_SETMASK   该进程新的信号屏蔽字将被set指向的信号集的值取代

假设set是空指针,则不改变该进程的信号屏蔽字,how的值也无意义

在调用sigprocmask后假设有不论什么未决的、不再堵塞的信号,则在sigprocmask返回前,至少会将当中一个信号递送给该进程。

1、有时候不希望在接到信号时就马上停止当前运行,去处理信号,同一时候也不希望忽略该信号,而是延时一段时间去调用信号处理函数。这样的情况是通过堵塞信号实现的。

2、信号堵塞和忽略信号的差别。

堵塞的概念和忽略信号是不同的。操作系统在信号被进程解除堵塞之前不会讲信号传递出去,被堵塞的信号也不会影响进程的行为,信号仅仅是临时被阻止传递。当进程忽略一个信号时,信号会被传递出去但进程会将信号丢弃。

1)头文件:#include <signal.h>

2)一个保护临界区代码的错误实例:(sigprocmask()和pause()实现)

#include <unistd.h>

#include <signal.h>

#include <stdio.h>

void handler(intsig)   //信号处理函数的实现

{

printf("SIGINT sig");

}

int main()

{

sigset_tnew,old;

structsigaction act;

act.sa_handler = handler;  //信号处理函数handler

sigemptyset(&act.sa_mask);

act.sa_flags = 0;

sigaction(SIGINT, &act, 0); //准备捕捉SIGINT信号

sigemptyset(&new);

sigaddset(&new, SIGINT);

sigprocmask(SIG_BLOCK, &new,&old); //将SIGINT信号堵塞,同一时候保存当前信号集

printf("Blocked");

sigprocmask(SIG_SETMASK, &old,NULL);  //取消堵塞

pause();

return0;

}

上面实例的问题是:本来期望pause()之后,来SIGINT信号,能够结束程序;但是,假设当“取消堵塞”和“pause”之间,正好来了SIGINT信号,结果程序由于pause的原因会一直挂起。。。

解决的方式,当然是sigsuspend()函数了。

3)使用sigsuspend()的程序

#include <unistd.h>

#include <signal.h>

#include <stdio.h>

void handler(int sig)  //信号处理程序

{

if(sig == SIGINT)

printf("SIGINT sig");

else if(sig == SIGQUIT)

printf("SIGQUIT sig");

else

printf("SIGUSR1 sig");

}

int main()

{

sigset_tnew,old,wait;   //三个信号集

structsigaction act;

act.sa_handler = handler;

sigemptyset(&act.sa_mask);

act.sa_flags = 0;

sigaction(SIGINT, &act,0);   //能够捕捉下面三个信号:SIGINT/SIGQUIT/SIGUSR1

sigaction(SIGQUIT, &act, 0);

sigaction(SIGUSR1, &act, 0);

sigemptyset(&new);

sigaddset(&new,SIGINT);  //SIGINT信号增加到new信号集中

sigemptyset(&wait);

sigaddset(&wait, SIGUSR1); //SIGUSR1信号增加wait

sigprocmask(SIG_BLOCK, &new,&old);      //将SIGINT堵塞,保存当前信号集到old中

//临界区代码运行

if(sigsuspend(&wait) != -1) //程序在此处挂起;用wait信号集替换new信号集。即:过来SIGUSR1信 号,堵塞掉,程序继续挂起;过来其它信号,比如SIGINT,则会唤醒程序。运行sigsuspend的原子操作。注意:假设“sigaddset(&wait,SIGUSR1);”这句没有,则此处不会堵塞不论什么信号,即过来不论什么信号均会唤醒程序。

printf("sigsuspend error");

printf("Aftersigsuspend");

sigprocmask(SIG_SETMASK, &old, NULL);

return0;

}

sigsuspend的原子操作是:

(1)设置新的mask堵塞当前进程(上面是用wait替换new,即堵塞SIGUSR1信号)

(2)收到SIGUSR1信号,堵塞,程序继续挂起;收到其它信号,恢复原先的mask(即包括SIGINT信号的)。

(3)调用该进程设置的信号处理函数(程序中假设先来SIGUSR1信号,然后过来SIGINT信号,则信号处理函数会调用两次,打印不同的内容。第一次打印SIGINT,第二次打印SIGUSR1,由于SIGUSR1是前面堵塞的)

(4)待信号处理函数返回,sigsuspend返回了。(sigsuspend将捕捉信号和信号处理函数集成到一起了)

总结:

在nginx源代码中,ngx_master_process_cycle函数中,首先调用了sigprocmask()函数堵塞了部分信号,在for循环中调用了sigsuspend函数 ,可是sigsuspend函数中的set为空,也就是说,在for循环之前,假设有set集合中的信号到来就堵塞,在for循环之内,不论什么信号到来都进程处理。函数返回之后同一时候也恢复了原来的信号掩码!

sigsuspend的整个原子操作过程为:

(1) 设置新的mask堵塞当前进程;

(2) 收到信号,恢复原先mask;

(3) 调用该进程设置的信号处理函数;

(4) 待信号处理函数返回后,sigsuspend返回。

大致就是上面这个过程,噢,原来signal handler是原子操作的一部分,并且是在恢复屏蔽字后运行的,

int sigsuspend(const sigset_t *sigmask);

此函数用于进程的挂起,sigmask指向一个信号集。当此函数被调用时,sigmask所指向的信号集中的信号将赋值给信号掩码。之后进程挂起。直到进程捕捉到信号,并调用处理函数返回时,函数sigsuspend返回。信号掩码恢复为信号调用前的值,同一时候将errno设为EINTR。进程结束信号可将其马上停止。

    #include <stdio.h>
#include <signal.h> void checkset();
void func();
void main()
{
sigset_tblockset,oldblockset,zeroset,pendmask;
printf("pid:%ld\n",(long)getpid());
signal(SIGINT,func); sigemptyset(&blockset);
sigemptyset(&zeroset);
sigaddset(&blockset,SIGINT); sigprocmask(SIG_SETMASK,&blockset,&oldblockset);
checkset();
sigpending(&pendmask); if(sigismember(&pendmask,SIGINT))
printf("SIGINTpending\n");
//不堵塞不论什么信号 不论什么信号到来都会激活进程
if(sigsuspend(&zeroset)!= -1)
{
printf("sigsuspenderror\n");
exit(0);
} printf("afterreturn\n");
sigprocmask(SIG_SETMASK,&oldblockset,NULL); printf("SIGINTunblocked\n");
} void checkset()
{ sigset_tset;
printf("checksetstart:\n");
if(sigprocmask(0,NULL,&set)<0)
{
printf("checksetsigprocmask error!!\n");
exit(0);
} if(sigismember(&set,SIGINT))
printf("sigint\n"); if(sigismember(&set,SIGTSTP))
printf("sigtstp\n"); if(sigismember(&set,SIGTERM))
printf("sigterm\n");
printf("checksetend\n");
} void func()
{
printf("hellofunc\n");
}

结果:

pid:5474
checksetstart:
sigint
checksetend
^Chellofunc
afterreturn
checksetstart:
checksetend
SIGINTunblocked

从执行的结果 以及和上面的样例进行比較较就能够知道这两个函数的函数。

sigsuspend sigprocmask函数的用法的更多相关文章

  1. 有关日期的函数操作用法总结,to_date(),trunc(),add_months();

    相关知识链接: Oracle trunc()函数的用法 oracle add_months函数 Oracle日期格式转换,tochar(),todate() №2:取得当前日期是一个星期中的第几天,注 ...

  2. Oracle to_date()函数的用法

    Oracle to_date()函数的用法 to_date()是Oracle数据库函数的代表函数之一,下文对Oracle to_date()函数的几种用法作了详细的介绍说明,供您参考学习. 在Orac ...

  3. js中bind、call、apply函数的用法

    最近一直在用 js 写游戏服务器,我也接触 js 时间不长,大学的时候用 js 做过一个 H3C 的 web的项目,然后在腾讯实习的时候用 js 写过一些奇怪的程序,自己也用 js 写过几个的网站.但 ...

  4. Oracle trunc()函数的用法

    Oracle trunc()函数的用法 /**************日期********************/1.select trunc(sysdate) from dual --2013-0 ...

  5. freemarker内置函数和用法

    原文链接:http://www.iteye.com/topic/908500 在我们应用Freemarker 过程中,经常会操作例如字符串,数字,集合等,却不清楚Freemrker 有没有类似于Jav ...

  6. matlab中patch函数的用法

    http://blog.sina.com.cn/s/blog_707b64550100z1nz.html matlab中patch函数的用法——emily (2011-11-18 17:20:33) ...

  7. JavaScript中常见的数组操作函数及用法

    JavaScript中常见的数组操作函数及用法 昨天写了个帖子,汇总了下常见的JavaScript中的字符串操作函数及用法.今天正好有时间,也去把JavaScript中常见的数组操作函数及用法总结一下 ...

  8. JavaScript中常见的字符串操作函数及用法

    JavaScript中常见的字符串操作函数及用法 最近几次参加前端实习生招聘的笔试,发现很多笔试题都会考到字符串的处理,比方说去哪儿网笔试题.淘宝的笔试题等.如果你经常参加笔试或者也是一个过来人,相信 ...

  9. oracle的substr函数的用法

    oracle的substr函数的用法 取得字符串中指定起始位置和长度的字符串   substr( string, start_position, [ length ] ) 如:     substr( ...

随机推荐

  1. Cocos2d-x使用Javascript开发js绑定C++&lt;代码演示样例&gt;

    class IOSiAPDelegate{ public: virtual ~IOSiAPDelegate() {} }; class IOSAlipay{ public: IOSAlipay(); ...

  2. strace跟踪线程调用

    方法一:strace -fp pid , 可以跟踪所有线程, 进程的系统调用. [root@xxxx]strace -p 24091 Process xxx attached - interrupt ...

  3. ios开发之级联菜单(两个tableView实现)

    一:在ios项目实际开发中经常会看到级联菜单的效果:如图:点击左侧菜单,右侧菜单刷新数据.此篇用两个tableView来实现如图效果: 二:代码: 1:构造数据模型:利用kvc快速构建数据模型 #im ...

  4. HTTP协议和HTTPS协议初探

    概况 HTTP是hypertext transfer protocol(超文本传输协议)的简写.它是TCP/IP协议的一个应用层协议,用于定义WEB浏览器与WEBserver之间交换数据的过程. HT ...

  5. 最全面的iOS和Mac开源项目和第三方库汇总

    标签: UI 下拉刷新 EGOTableViewPullRefresh – 最早的下拉刷新控件. SVPullToRefresh – 下拉刷新控件. MJRefresh – 仅需一行代码就可以为UIT ...

  6. DATAGUARD在做SWITCHOVER切换时遇到问题总结

    1.主库在进行物理主备库角色转换的时候遇到ORA-01093错误 SQL> select switchover_status from v$database;   SWITCHOVER_STAT ...

  7. CRT(C Runtime Library)—— C/C++运行时库

    C runtime library(part of the C standard library) 任何一个 C 程序,它的背后都有一套庞大的代码来进行支撑,使得该程序得以运行在更高级别上,而不必担心 ...

  8. Android onKeyDown监听返回键无效

    当我们的Activity继承了TabActivity,在该类中重写onKeyDown是监听不到返回键的, 具体解决方法如下: 重写dispatchKeyEvent /** * 退出 */ @Overr ...

  9. HPE Comware Lab - Simulator

    http://h20565.www2.hpe.com/hpsc/swd/public/readIndex?sp4ts.oid=7107838&ac.admitted=1405352934644 ...

  10. VBA Code for Word Navigation Pane 【failed】 view-showheading-method-word

    https://msdn.microsoft.com/VBA/Word-VBA/articles/view-showheading-method-word View.ShowHeading Metho ...