天天看点

linux c学习笔记----信号(sigaction,sigaddset,sigprocmask)

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 此调用被中断

范例
#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 信号.      
上一篇: Codeforces 1136
下一篇: Luogu 1071