sigsetjmp和siglongjmp用于在信号处理程序中进行非局部跳,而不应该使用setjmp和longjmp
函数原型:
NAME
setjmp, sigsetjmp - save stack context for non-local goto
SYNOPSIS
#include <setjmp.h>
int setjmp(jmp_buf env);
int sigsetjmp(sigjmp_buf env, int savesigs);
NAME
longjmp, siglongjmp - non-local jump to a saved stack context
SYNOPSIS
#include <setjmp.h>
void longjmp(jmp_buf env, int val);
void siglongjmp(sigjmp_buf env, int val);
#include <stdio.h>
#include <errno.h>
#include <unistd.h>
#include <signal.h>
#include <setjmp.h>
#include <time.h>
extern int errno;
static void pr_mask(const char *ident)
{
sigset_t sigset;
int errno_save;
errno_save = errno;
struct sigaction oact;
if(sigprocmask(0, NULL, &sigset) < 0)
{
perror("sigprocmask");
return;
}
printf("%s", ident);
if(sigismember(&sigset, SIGINT))
printf("SIGINT ");
if(sigismember(&sigset, SIGQUIT))
printf("SIGQUIT ");
if(sigismember(&sigset, SIGUSR1))
printf("SIGUSR1 ");
if(sigismember(&sigset, SIGALRM))
printf("SIGALRM ");
printf("\n");
errno = errno_save;
}
static void sig_usr1(int), sig_alrm(int);
static sigjmp_buf jmpbuf;
static volatile sig_atomic_t canjump;
int main(int argc, char **argv)
{
if(signal(SIGUSR1, sig_usr1) == SIG_ERR)
perror("SIGUSR1");
if(signal(SIGALRM, sig_alrm) == SIG_ERR)
perror("SIGALRM");
pr_mask("starting main: ");
if(sigsetjmp(jmpbuf, 0))
{
pr_mask("ending main: ");
return -1;
}
canjump = 1;
while(1)
{
pause();
}
return 0;
}
static void sig_usr1(int signo)
{
time_t starttime;
if(canjump == 0)
{
return;
}
pr_mask("staring sig_usr1: ");
alarm(3);
starttime = time(NULL);
while(1)
{
if(time(NULL) > starttime + 5)
break;
}
pr_mask("finish sig_usr1: ");
canjump = 0;
siglongjmp(jmpbuf, 1);
}
static void sig_alrm(int signo)
{
pr_mask("in sig_alrm: ");
}