您的位置:首页 > 运维架构 > Linux

Linux信号机制之sigaction结构体浅析

2017-08-14 14:11 519 查看

http://www.cnblogs.com/gogly/articles/2416989.html

信号安装函数sigaction(int signum,const struct sigaction *act,struct sigaction *oldact)的第二个参数是一个指向sigaction结构的指针(结构体名称与函数名一样,千万别弄混淆了)。在结构sigaction的实例中,指定了对特定信号的处理,信号所传递的信息,信号处理函数执行过程中应屏蔽掉哪些函数等。当然,此指针也可以为NULL,进程会以默认方式处理信号。以下就简单介绍一下sigaction结构以及一般的用法。

  对于内核头文件而言,struct sigaction 结构体定义在kernel/include/asm/signal.h,此头文件又被kernel/include/linux/signal.h包含。

  对于用户空间的头文件而言,struct sigaction定义在 /usr/include/bits/sigaction.h,此头文件又被/usr/include/signal.h包含,所以应用程序中如果用到此结构,只要#include <signal.h>即可。注意内核中的定义和应用程序中的定义是不一样的,内核空间的sigaction结构只支持函数类型为__sighandler_t的信号处理函数,不能处理信号传递的额外信息。具体定义如下:

  /* Type of a signal handler.   */

  typedef void (*__sighandler_t)(int);

  #ifdef __KERNEL__

  struct old_sigaction {

          __sighandler_t sa_handler;

         old_sigset_t sa_mask;

         unsigned long sa_flags;

         void (*sa_restorer)(void);

  };

  struct sigaction {

         __sighandler_t sa_handler;

        unsigned long sa_flags;

        void (*sa_restorer)(void);

        sigset_t sa_mask;   /* mask last for extensibility */

  };

  struct k_sigaction {

        struct sigaction sa;

  };

  #else

  /* Here we must cater to libcs that poke about in kernel headers.   */

  struct sigaction {

          union {

                  __sighandler_t _sa_handler;

                  void (*_sa_sigaction)(int, struct siginfo *, void *);

          } _u;

          sigset_t sa_mask;

          unsigned long sa_flags;

          void (*sa_restorer)(void);

  };

  #define sa_handler   _u._sa_handler

  #define sa_sigaction _u._sa_sigaction

  #endif /* __KERNEL__ */

  sa_handler的原型是一个参数为int,返回类型为void的函数指针。参数即为信号值,所以信号不能传递除信号值之外的任何信息;

  sa_sigaction的原型是一个带三个参数,类型分别为int,struct siginfo *,void *,返回类型为void的函数指针。第一个参数为信号值;第二个参数是一个指向struct siginfo结构的指针,此结构中包含信号携带的数据值;第三个参数没有使用。

  sa_mask指定在信号处理程序执行过程中,哪些信号应当被阻塞。默认当前信号本身被阻塞。

  sa_flags包含了许多标志位,比较重要的一个是SA_SIGINFO,当设定了该标志位时,表示信号附带的参数可以传递到信号处理函数中。即使sa_sigaction指定信号处理函数,如果不设置SA_SIGINFO,信号处理函数同样不能得到信号传递过来的数据,在信号处理函数中对这些信息的访问都将导致段错误。

  sa_restorer已过时,POSIX不支持它,不应再使用。

  因此,当你的信号需要接收附加信息的时候,你必须给sa_sigaction赋信号处理函数指针,同时还要给sa_flags赋SA_SIGINFO,类似下面的代码:

     #include <signal.h>

     ……

     void sig_handler_with_arg(int sig,siginfo_t *sig_info,void *unused){……}

    

     int main(int argc,char **argv)

     {

              struct sigaction sig_act;

              ……

              sigemptyset(&sig_act.sa_mask);

              sig_act.sa_sigaction=sig_handler_with_arg;

              sig_act.sa_flags=SA_SIGINFO;

  

               ……

     }

  如果你的应用程序只需要接收信号,而不需要接收额外信息,那你需要的设置的是sa_handler,而不是sa_sigaction,你的程序可能类似下面的代码:

     #include <signal.h>

     ……

     void sig_handler(int sig){……}

    

     int main(int argc,char **argv)

     {

              struct sigaction sig_act;

              ……

              sigemptyset(&sig_act.sa_mask);

              sig_act.sa_handler=sig_handler;

              sig_act.sa_flags=0; 

               ……

      }

 

  例:

  #include<sys/types.h>

  #include<unistd.h>

  #include<signal.h>

  #include<stdio.h>

  #include<stdlib.h>

  int main(void)

  {

    sigset_t set,pendset;

    struct sigaction action;

    //清空信号集

    sigemptyset(&set);        

    //加入SIGTERM呢个信号      

    sigaddset(&set,SIGTERM);           

    //设为阻塞

    sigprocmask(SIG_BLOCK,&set,NULL);     

    //因为阻塞所以就算用kill发送信号都无反应

    kill(getpid(),SIGTERM); 

    //查下有什么阻塞信号,装入&pendset里面

    sigpending(&pendset);  

    //看下面有什么信号在里面

    if(sigismember(&pendset,SIGTERM))     

    {

      printf("yes,the SIGTERM is here\n");

      //清空阻塞信号集

      sigemptyset(&action.sa_mask); 

      //信号涵数处理为ignore(忽略)       

      action.sa_handler=SIG_IGN;        

       //启动同signal功能类的信号涵数 

      sigaction(SIGTERM,&action,NULL);     

    }

    //解除之前的信号阻塞

    sigprocmask(SIG_UNBLOCK,&set,NULL);      

    exit(EXIT_SUCCESS);

  }

  附:Linux C常用函数http://man.chinaunix.net/develop/c&c++/linux_c/default.htm
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: