#include <sys/types.h>
#include <err.h>
#include <setjmp.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#if (TEST_SETJMP + TEST_U_SETJMP + TEST_SIGSETJMP) != 1
#error one of TEST_SETJMP, TEST_U_SETJMP, or TEST_SIGSETJMP must be defined
#endif
#ifdef TEST_SETJMP
#define BUF jmp_buf
#define SET(b, m) setjmp(b)
#define JMP(b, v) longjmp(b, v)
#endif
#ifdef TEST_U_SETJMP
#define BUF jmp_buf
#define SET(b, m) _setjmp(b)
#define JMP(b, v) _longjmp(b, v)
#endif
#ifdef TEST_SIGSETJMP
#define BUF sigjmp_buf
#define SET(b, m) sigsetjmp(b, m)
#define JMP(b, v) siglongjmp(b, v)
#endif
int expectsignal;
static void
aborthandler(int signo)
{
if (expectsignal)
_exit(0);
else {
warnx("kill(SIGABRT) succeeded");
_exit(1);
}
}
int
main(int argc, char *argv[])
{
struct sigaction sa;
BUF jb;
sigset_t ss;
int i, x;
i = getpid();
#ifdef TEST_SETJMP
expectsignal = 0;
#endif
#ifdef TEST_U_SETJMP
expectsignal = 1;
#endif
#ifdef TEST_SIGSETJMP
if (argc != 2 ||
(strcmp(argv[1], "save") && strcmp(argv[1], "nosave"))) {
fprintf(stderr, "usage: %s [save|nosave]\n", argv[0]);
exit(1);
}
expectsignal = (strcmp(argv[1], "save") != 0);
#endif
sa.sa_handler = aborthandler;
sigemptyset(&sa.sa_mask);
sa.sa_flags = 0;
if (sigaction(SIGABRT, &sa, NULL) == -1)
err(1, "sigaction failed");
if (sigemptyset(&ss) == -1)
err(1, "sigemptyset failed");
if (sigaddset(&ss, SIGABRT) == -1)
err(1, "sigaddset failed");
if (sigprocmask(SIG_BLOCK, &ss, NULL) == -1)
err(1, "sigprocmask (1) failed");
x = SET(jb, !expectsignal);
if (x != 0) {
if (x != i)
errx(1, "setjmp returned wrong value");
kill(i, SIGABRT);
if (expectsignal)
errx(1, "kill(SIGABRT) failed");
else
exit(0);
}
if (sigprocmask(SIG_UNBLOCK, &ss, NULL) == -1)
err(1, "sigprocmask (2) failed");
JMP(jb, i);
errx(1, "jmp failed");
}