|
sigaction(查询或设置信号处理方式) |
相关函数
|
signal,sigprocmask,sigpending,sigsuspend |
表头文件
|
#include<signal.h> |
定义函数
|
int sigaction(int signum,const struct sigaction *act ,struct sigaction *oldact); |
函数说明
|
sigaction()会依参数signum指定的信号编号来设置该信号的处理函数。参数signum可以指定SIGKILL和SIGSTOP以外的所有信号。
|
返回值
|
执行成功则返回0,如果有错误则返回-1。 |
错误代码
|
EINVAL 参数signum 不合法, 或是企图拦截SIGKILL/SIGSTOPSIGKILL信号 EFAULT 参数act,oldact指针地址无法存取。 EINTR 此调用被中断 |
范例
|
#include <stdio.h> #include <signal.h> #include <unistd.h> void show_handler(int sig) { printf("I got signal %d\n", sig); int i; for(i = 0; i < 5; i++) { printf("i = %d\n", i); sleep(1); } } int main(void) { int i = 0; struct sigaction act, oldact; act.sa_handler = show_handler; sigaddset(&act.sa_mask, SIGQUIT); //见注(1) act.sa_flags = SA_RESETHAND | SA_NODEFER; //见注(2) //act.sa_flags = 0; //见注(3) sigaction(SIGINT, &act, &oldact); while(1) { sleep(1); printf("sleeping %d\n", i); i++; } } |
sigaddset(增加一个信号至信号集)
相关函数
|
sigemptyset,sigfillset,sigdelset,sigismember |
表头文件
|
#include<signal.h> |
定义函数
|
int sigaddset(sigset_t *set,int signum); |
函数说明
|
sigaddset()用来将参数signum 代表的信号加入至参数set 信号集里。 |
返回值
|
执行成功则返回0,如果有错误则返回-1。 |
错误代码
|
EFAULT 参数set指针地址无法存取 EINVAL 参数signum非合法的信号编号 |
sigprocmask(查询或设置信号遮罩)
相关函数
|
signal,sigaction,sigpending,sigsuspend |
表头文件
|
#include<signal.h> |
定义函数
|
int sigprocmask(int how,const sigset_t *set,sigset_t * oldset); |
函数说明
|
sigprocmask()可以用来改变目前的信号遮罩,其操作依参数how来决定 SIG_BLOCK 新的信号遮罩由目前的信号遮罩和参数set 指定的信号遮罩作联集 SIG_UNBLOCK 将目前的信号遮罩删除掉参数set指定的信号遮罩 SIG_SETMASK 将目前的信号遮罩设成参数set指定的信号遮罩。 如果参数oldset不是NULL指针,那么目前的信号遮罩会由此指针返回。 |
返回值
|
执行成功则返回0,如果有错误则返回-1。 |
错误代码
|
EFAULT 参数set,oldset指针地址无法存取。 EINTR 此调用被中断 |
实例:
#include <signal.h> #include <stdio.h> #include <string.h> #include <errno.h> #include <unistd.h> #define PROMPT "你想终止程序吗?" char *prompt=PROMPT; void ctrl_c_op(int signo) { write(STDERR_FILENO,prompt,strlen(prompt)); } int main() { struct sigaction act; act.sa_handler=ctrl_c_op; sigemptyset(&act.sa_mask); act.sa_flags=0; if(sigaction(SIGINT,&act,NULL)<0) { fprintf(stderr,"Install Signal Action Error:%s\n\a",strerror(errno)); exit(1); } while(1); }
#include <unistd.h> #include <stdio.h> #include <errno.h> #include <fcntl.h> #include <signal.h> #include <string.h> #include <pwd.h> #include <sys/types.h> #include <sys/stat.h> /* Linux 的默任个人的邮箱地址是 /var/spool/mail/ */ #define MAIL_DIR "/var/spool/mail/" /* 睡眠10 秒钟 */ #define SLEEP_TIME 10 #define MAX_FILENAME 255 unsigned char notifyflag=1; long get_file_size(const char *filename){ struct stat buf; if(stat(filename,&buf)==-1){ if(errno==ENOENT)return 0; else return -1; } return (long)buf.st_size; } void send_mail_notify(void){ fprintf(stderr,"New mail has arrived\007\n"); } void turn_on_notify(int signo){ notifyflag=1; } void turn_off_notify(int signo){ notifyflag=0; } int check_mail(const char *filename){ long old_mail_size,new_mail_size; sigset_t blockset,emptyset; sigemptyset(&blockset); sigemptyset(&emptyset); sigaddset(&blockset,SIGUSR1); sigaddset(&blockset,SIGUSR2); old_mail_size=get_file_size(filename); if(old_mail_size<0)return 1; if(old_mail_size>0) send_mail_notify(); sleep(SLEEP_TIME); while(1){ if(sigprocmask(SIG_BLOCK,&blockset,NULL)<0) return 1; while(notifyflag==0)sigsuspend(&emptyset); if(sigprocmask(SIG_SETMASK,&emptyset,NULL)<0) return 1; new_mail_size=get_file_size(filename); if(new_mail_size>old_mail_size)send_mail_notify; old_mail_size=new_mail_size; sleep(SLEEP_TIME); } } int main(void){ char mailfile[MAX_FILENAME]; struct sigaction newact; struct passwd *pw; if((pw=getpwuid(getuid()))==NULL){ fprintf(stderr,"Get Login Name Error:%s\n\a",strerror(errno)); exit(1); } strcpy(mailfile,MAIL_DIR); strcat(mailfile,pw->pw_name); newact.sa_handler=turn_on_notify; newact.sa_flags=0; sigemptyset(&newact.sa_mask); sigaddset(&newact.sa_mask,SIGUSR1); sigaddset(&newact.sa_mask,SIGUSR2); if(sigaction(SIGUSR1,&newact,NULL)<0) fprintf(stderr,"Turn On Error:%s\n\a",strerror(errno)); newact.sa_handler=turn_off_notify; if(sigaction(SIGUSR1,&newact,NULL)<0) fprintf(stderr,"Turn Off Error:%s\n\a",strerror(errno)); check_mail(mailfile); exit(0); }这个程序会在也可以检查用户的邮件.不过提供了一个开关,如果用 户不想程序提示有新的邮件到来,可以向程序发送SIGUSR2 信号,如果想程序提供提示可以 发送SIGUSR1 信号.