15信号sigaction

信号处理

信号值小于 SIGRTMIN 的信号 (1~31) 都是不可靠信号

某些unix版本中,调用信号函数处理后会自动恢复默认信号处理,所以在信号处理函数中还需要继续调用signal函数设置信号处理

这个问题在linux中并不存在

linux 信号处理使用 sigaction函数,

 

 

sigaction 与 signal

主要优点: sigaction 支持参数传递

 

简单运用,作用与signal相同

void simFun(int sig)

{   

      printf("receive sig=%d\n",sig);

}

 

void testSigaction()

{

      printf("pid=%d\n",getpid());

      struct sigaction act,old;

      act.sa_handler=simFun;

      sigaction(SIGUSR1,&act,&old);

     

      while(1);

}

 

发出信号SIGUSR1,自动响应simFun处理函数。

 

 

信号处理 

int sigaction(int sig,  struct sigaction *act, struct sigaction *oldact)

sig :指定捕获的信号

act :指定捕获信号后的处理方式

oldact:  上一次指定的处理方式

 

 

 

信号处理

struct sigaction

{

#ifdef __USE_POSIX199309

    union {

              __sighandler_t sa_handler;        /* void(*)(int)  -- if SA_SIGINFO is not set.  */

              void (*sa_sigaction) (int sig, siginfo_t *, void *); /* Used if SA_SIGINFO is set.  */

          }__sigaction_handler;

# define sa_handler     __sigaction_handler.sa_handler

# define sa_sigaction   __sigaction_handler.sa_sigaction

__sigset_t     sa_mask;          /* 信号处理时,那些信号被阻塞 (默认为当前信号阻塞)*/

int            sa_flags;            /* 响应信号的其他行为,其中 SA_SIGINFO 标志位指定信号处理方法*/

void             (*sa_restorer) (void);             /*过时,不再使用.  */

};

 

结构体参数:

sa_mask

sigaddset(&xx.sa_mask,  SIGINT)   增加阻塞中断信号

 

sa_flags

SA_NOCLDSTOP     (1)              子进程暂停时不发送SIGCHLD信号通知父进程,SIGSTOP  SIGTSTP  SIGTTIN  SIGTTOU                     

SA_NOCLDWAIT     (2)         子进程结束时,不变为僵尸进程,针对SIGCHLD信号

SA_SIGINFO       (4)            使用sa_sigaction方法处理信号

SA_RESTART       (0x10000000)   被信号中断的系统处理会自动重启

SA_NODEFER       (0x40000000)   处理某信号时,这个信号不再阻塞, 可重入

SA_RESETHAND     (0x80000000)   捕获信号后,这个信号恢复缺省处理

 

 

siginfo_t结构体:

siginfo_t {

               int      si_signo;    /* Signal number */

               int      si_errno;    /* An errno value */

               int      si_code;     /* Signal code */

               int      si_trapno;   /* Trap number that caused

                                        hardware-generated signal

                                        (unused on most architectures) */

               pid_t    si_pid;      /* Sending process ID */

               uid_t    si_uid;      /* Real user ID of sending process */

               int      si_status;   /* Exit value or signal */

               clock_t  si_utime;    /* User time consumed */

               clock_t  si_stime;    /* System time consumed */

               sigval_t si_value;    /* Signal value */

               int      si_int;      /* POSIX.1b signal */

               void    *si_ptr;      /* POSIX.1b signal */

               int      si_overrun;  /* Timer overrun count; POSIX.1b timers */

               int      si_timerid;  /* Timer ID; POSIX.1b timers */

               void    *si_addr;     /* Memory location which caused fault */

               long     si_band;     /* Band event (was int in

                                        glibc 2.3.2 and earlier) */

                 int      si_fd;       /* File descriptor */

               short    si_addr_lsb; /* Least significant bit of address

                                        (since Linux 2.6.32) */

           }

 

 

int sigqueue(pid, sig, union sigval sgv)

发送信号给进程,使用同 kill效果一样

不同于kill 的地方,多了一个信号参数,  只能向某个进程发送信号,不能向进程组发送信号

如:sigqueue(getpid(), SIGUSR1, sgv);给自己发SIGUSR1。

union sigval

{

      int    sival_int;

      void*  sival_ptr;

}

 

 

 

带参数的例子:

void complexFun(int sig,siginfo_t *info,void *ex)

{

      sleep(3);

      printf("receive sig=%d\n",sig);

      printf("Sending process ID:%d\n",info->si_pid);

      printf("Real user ID of sending process:%d\n",info->si_uid);

      printf("extrc value:%d\n",info->si_value.sival_int);     

}

 

void testSigaction()

{

      printf("pid=%d\n",getpid());

      struct sigaction act,old;

      //act.sa_handler=simFun;

      //sigaction(SIGUSR1,&act,&old);

     

      sigaddset(&act.sa_mask,SIGINT);          

     

      act.sa_flags|=SA_SIGINFO;

      act.sa_flags|=SA_NODEFER;

      act.sa_sigaction=complexFun;

      sigaction(SIGUSR1,&act,&old);   

 

      union sigval sgv;

      sgv.sival_int=123;

      sigqueue(getpid(), SIGUSR1, sgv);

     

      while(1);

}

 

 

SA_NODEFER       (0x40000000)   处理某信号时,这个信号不再阻塞, 可重入

比如:同一个按钮连续按下10次,那么该按钮信号可能只响应5,6次,达不到10。设置该属性后,就可以达到10.达到阻塞后推迟响应的效果。

 

 

sigqueue(getpid(), SIGUSR1, sgv);

给某一进程发送信号,如例子中的给自己发出SIGUSR1信号,并传入额外的参数。

 

posted @ 2018-06-22 23:50  gd_沐辰  阅读(264)  评论(0编辑  收藏  举报