sigaction(查询或设置信号处理方式) | |
相关函数 | signal,sigprocmask,sigpending,sigsuspend |
表头文件 | #include<signal.h> |
定义函数 | int sigaction(int signum,const struct sigaction *act ,struct sigaction *oldact); |
函数说明 | sigaction()会依参数signum指定的信号编号来设置该信号的处理函数。参数signum可以指定SIGKILL和SIGSTOP以外的所有信号。 如参数结构sigaction定义如下 struct sigaction { void (*sa_handler) (int); sigset_t sa_mask; int sa_flags; void (*sa_restorer) (void); } sa_handler此参数和signal()的参数handler相同,代表新的信号处理函数,其他意义请参考signal()。 sa_mask 用来设置在处理该信号时暂时将sa_mask 指定的信号搁置。 sa_restorer 此参数没有使用。 sa_flags 用来设置信号处理的其他相关操作,下列的数值可用。 OR 运算(|)组合 A_NOCLDSTOP : 如果参数signum为SIGCHLD,则当子进程暂停时并不会通知父进程 SA_ONESHOT/SA_RESETHAND:当调用新的信号处理函数前,将此信号处理方式改为系统预设的方式。 SA_RESTART:被信号中断的系统调用会自行重启 SA_NOMASK/SA_NODEFER:在处理此信号未结束前不理会此信号的再次到来。 如果参数oldact不是NULL指针,则原来的信号处理方式会由此结构sigaction 返回。 |
返回值 | 执行成功则返回0,如果有错误则返回-1。 |
错误代码 | EINVAL 参数signum 不合法, 或是企图拦截SIGKILL/SIGSTOPSIGKILL信号 EFAULT 参数act,oldact指针地址无法存取。 EINTR 此调用被中断 |
范例 | |
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 信号.