sigaction函数注册信号处理函数
sigaction函数 q 包含头文件<signal.h> q 功能:sigaction函数用于改变进程接收到特定信号后的行为。 q 原型: int sigaction(int signum,const struct sigaction *act,const struct sigaction *old); q 参数 q 该函数的第一个参数为信号的值,可以为除SIGKILL及SIGSTOP外的任何一 个特定有效的信号(为这两个信号定义自己的处理函数,将导致信号安装错误) q 第二个参数是指向结构sigaction的一个实例的指针,在结构 sigaction的实例中,指定了对特定信号的处理,可以为空,进程会以缺省方式对信号处理 q 第三个参数oldact指向的对象用来保存原来对相应信号的处理,可指定oldact为NULL。 q 返回值:函数成功返回0,失败返回-1 signal(num., handle) |
sigaction结构体 q 第二个参数最为重要,其中包含了对指定信号的处理、信号所传递的信息、信号处理函数执行过程中应屏蔽掉哪些函数等等 struct sigaction { void (*sa_handler)(int); //信号处理程序 不接受额外数据 void (*sa_sigaction)(int, siginfo_t *, void *); //信号处理程序 能接受额外数据,和sigqueue配合使用 sigset_t sa_mask; // int sa_flags; //影响信号的行为 SA_SIGINFO表示能接受数据 void (*sa_restorer)(void); //废弃 }; 注意1:回调函数句柄sa_handler、sa_sigaction只能任选其一。 注意2:The sigaction structure is defined as something like 思考如何测试? |
会查找、会用man手册,是通往高手的必经之路。 The siginfo_t parameter to sa_sigaction is a struct with the following elements siginfo_t { int si_signo; /* Signal number */ int si_errno; /* An errno value */ int si_code; /* Signal code */ 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 */ void * si_addr; /* Memory location which caused fault */ int si_band; /* Band event */ int si_fd; /* File descriptor */ } |
实验1:sigaction的函数注册信号,基本用法 void handler(int sig) { printf("recv a sig=%d ", sig); } __sighandler_t my_signal(int sig, __sighandler_t handler) { struct sigaction act; struct sigaction oldact; act.sa_handler = handler; sigemptyset(&act.sa_mask); act.sa_flags = 0; if (sigaction(sig, &act, &oldact) < 0) return SIG_ERR; return oldact.sa_handler; } int main(int argc, char *argv[]) { struct sigaction act; sigset_t sa_mask;
act.sa_handler = handler; act.sa_flags = 0; sigemptyset(&act.sa_mask); //测试信号安装函数 //sigaction(SIGINT, &act, NULL);
//模拟signal函数 my_signal(SIGINT, handler); for (;;) { pause(); } return 0; }
|
实验2:测试sigaction结构体第三个参数sigset_t sa_mask的作用 /* struct sigaction { void (*sa_handler)(int); void (*sa_sigaction)(int, siginfo_t *, void *); sigset_t sa_mask; int sa_flags; void (*sa_restorer)(void); } */ //测试sigaction结构体第三个参数sigset_t sa_mask的作用 //作用 sigaddset(&act.sa_mask, SIGQUIT); 加入到sa_mask中的信号,被阻塞(信号处理函数执行的过程中被阻塞)。 //注意:SIGQUIT信号最终还会抵达 int main(int argc, char *argv[]) { struct sigaction act; act.sa_handler = handler;
sigemptyset(&act.sa_mask); sigaddset(&act.sa_mask, SIGQUIT); act.sa_flags = 0; if (sigaction(SIGINT, &act, NULL) < 0) ERR_EXIT("sigaction error"); for (;;) pause(); return 0; } void handler(int sig) { printf("recv a sig=%d 信号处理函数执行的时候,阻塞sa_mask中的信号 ", sig); sleep(5); } |
sigqueue新的信号发送函数
sigqueue函数 q 功能:新的发送信号系统调用,主要是针对实时信号提出的支持信号带有参数,与函数sigaction()配合使用。 q 注意:和kill函数相比Int kill(pid_t pid, int siq)多了参数 q 原型: int sigqueue(pid_t pid, int sig, const union sigval value); q 参数 q sigqueue的第1个参数是指定接收信号的进程id,第2个参数确定即将发送的信号,第3个参数是一个联合数据结构union sigval,指定了信号传递的参数,即通常所说的4字节值。 q 返回值成功返回0,失败返回-1 |
q sigqueue()比kill()传递了更多的附加信息,但sigqueue()只能向一个进程发送信号,而不能发送信号给一个进程组。 q sigval联合体 typedef union sigval { int sival_int; void *sival_ptr; }sigval_t; |