Chinaunix

标题: 有关sigsuspend函数的一个测试,疑惑? [打印本页]

作者: flikelinux    时间: 2012-03-30 10:24
标题: 有关sigsuspend函数的一个测试,疑惑?
这个测试例子,是APUE p270页的一个例子,但我对执行结果不太理解,代码如下
  1. #include<stdio.h>
  2. #include<unistd.h>
  3. #include<signal.h>
  4. #include<error.h>

  5. void pr_mask(const char* str)
  6. {
  7.         sigset_t sigset;
  8.         int errno_save;
  9.         if(sigprocmask(SIG_BLOCK,NULL,&sigset)<0)
  10.            printf("sigprocmask error\n");
  11.         printf("%s:",str);
  12.         if(sigismember(&sigset,SIGINT))
  13.            printf("SIGINT ");
  14.         if(sigismember(&sigset,SIGQUIT))
  15.            printf("SIGQUIT ");
  16.         if(sigismember(&sigset,SIGUSR1))
  17.            printf("SIGUSR1 ");
  18.         if(sigismember(&sigset,SIGALRM))
  19.            printf("SIGALRM ");
  20.         printf("\n");
  21. }

  22. //
  23. void sig_int(int signo)
  24. {
  25.         pr_mask("\nin sig_int");
  26. }

  27. int main()
  28. {
  29.         sigset_t newmask,oldmask,waitmask;
  30.         if(signal(SIGINT,sig_int)==SIG_ERR)
  31.             printf("error 1");
  32.         sigemptyset(&waitmask);
  33.         sigaddset(&waitmask,SIGUSR1);
  34.         sigemptyset(&newmask);
  35.         sigaddset(&newmask,SIGINT);
  36.        
  37.         if(sigprocmask(SIG_BLOCK,&newmask,&oldmask)<0)
  38.            printf("error 2");
  39.         pr_mask("in critial region");
  40.         if(sigsuspend(&waitmask)!=-1)
  41.            printf("error 3");
  42.         pr_mask("after return from sigsuspend");
  43.         if(sigprocmask(SIG_SETMASK,&oldmask,NULL)<0)
  44.                 printf("error 4");
  45.     pr_mask("program exit");
  46.     return 0;
  47. }
复制代码
执行结果是:
in critial region:SIGINT
^C
in sig_int:SIGINT SIGUSR1
after return from sigsuspend:SIGINT
program exit:

网上说sigsuspend执行的步骤是:
(1) 设置新的mask阻塞当前进程;
(2) 收到信号,恢复原先mask;
(3) 调用该进程设置的信号处理函数;
(4) 待信号处理函数返回后,sigsuspend返回。

那么根据第二条,我觉得在in sig_int:不应该有SIGUSR1,因为已经恢复以前的mask了,且以前的mask并不阻塞SIGUSR1。有人能给我指点一下吗?
作者: flikelinux    时间: 2012-03-30 16:21
没人知道吗?求指点!
作者: kinfinger    时间: 2012-03-30 21:42
http://blog.chinaunix.net/uid-22334392-id-3069347.html\"\"可能会对你有帮助
作者: suqin0802    时间: 2012-04-01 16:54
本帖最后由 suqin0802 于 2012-04-01 17:14 编辑

我也被这个问题困扰了~~
我将网上下载的源代码拷贝了一份建立文件10-15.c,编译执行后的结果是没有SIGINT的

这个是可以想得通的!
但是以前在网上下载了源代码包整体编译后得出的可执行文件/apue.2e/signals/suspend1,运行这个得出结果是跟书上的一致,实在是费解啊,求解释!!!我真的只是复制粘贴了一下代码而已,为什么结果会不同呢?
代码如下:
  1. #include "apue.h"

  2. static void        sig_int(int);

  3. int
  4. main(void)
  5. {
  6.         sigset_t        newmask, oldmask, waitmask;

  7.         pr_mask("program start: ");

  8.         if (signal(SIGINT, sig_int) == SIG_ERR)
  9.                 err_sys("signal(SIGINT) error");
  10.         sigemptyset(&waitmask);
  11.         sigaddset(&waitmask, SIGUSR1);
  12.         sigemptyset(&newmask);
  13.         sigaddset(&newmask, SIGINT);

  14.         /*
  15.          * Block SIGINT and save current signal mask.
  16.          */
  17.         if (sigprocmask(SIG_BLOCK, &newmask, &oldmask) < 0)
  18.                 err_sys("SIG_BLOCK error");

  19.         /*
  20.          * Critical region of code.
  21.          */
  22.         pr_mask("in critical region: ");
  23.        
  24.         /*
  25.          * Pause, allowing all signals except SIGUSR1.
  26.          */
  27.         if (sigsuspend(&waitmask) != -1)
  28.                 err_sys("sigsuspend error");

  29.         pr_mask("after return from sigsuspend: ");

  30.         /*
  31.          * Reset signal mask which unblocks SIGINT.
  32.          */
  33.         if (sigprocmask(SIG_SETMASK, &oldmask, NULL) < 0)
  34.                 err_sys("SIG_SETMASK error");

  35.         /*
  36.          * And continue processing ...
  37.          */
  38.         pr_mask("program exit: ");

  39.         exit(0);
  40. }

  41. void pr_mask(const char *str)
  42. {
  43.         sigset_t sigset;
  44.         int errno_save;

  45.         errno_save = errno;
  46.         if(sigprocmask(0,NULL,&sigset) <0) err_sys("sigprocmask error");

  47.         printf("%s",str);
  48.         if(sigismember(&sigset,SIGINT)) printf("SIGINT ");
  49.         if(sigismember(&sigset,SIGQUIT)) printf("SIGQUIT ");
  50.         if(sigismember(&sigset,SIGUSR1)) printf("SIGUSR1 ");
  51.         if(sigismember(&sigset,SIGALRM)) printf("SIGALRM ");
  52.                
  53.         printf("\n");
  54.         errno = errno_save;
  55. }

  56. static void
  57. sig_int(int signo)
  58. {
  59.         pr_mask("\nin sig_int: ");
  60. }
复制代码

作者: flikelinux    时间: 2012-04-02 21:31
回复 4# suqin0802


    嗯,我也比较费解。不懂啊。。。




欢迎光临 Chinaunix (http://bbs.chinaunix.net/) Powered by Discuz! X3.2