Replace all setjmp()/longjmp() with sigsetjmp()/siglongjmp()

The setjmp() function doesn't specify whether signal masks are saved and
restored; on Linux they are not, but on BSD (including MacOSX) they are.
We want to have consistent behaviour across platforms, so we should
always use "don't save/restore signal mask" (this is also generally
going to be faster). This also works around a bug in MacOSX where the
signal-restoration on longjmp() affects the signal mask for a completely
different thread, not just the mask for the thread which did the longjmp.
The most visible effect of this was that ctrl-C was ignored on MacOSX
because the CPU thread did a longjmp which resulted in its signal mask
being applied to every thread, so that all threads had SIGINT and SIGTERM
blocked.

The POSIX-sanctioned portable way to do a jump without affecting signal
masks is to siglongjmp() to a sigjmp_buf which was created by calling
sigsetjmp() with a zero savemask parameter, so change all uses of
setjmp()/longjmp() accordingly. [Technically POSIX allows sigsetjmp(buf, 0)
to save the signal mask; however the following siglongjmp() must not
restore the signal mask, so the pair can be effectively considered as
"sigjmp/longjmp which don't touch the mask".]

For Windows we provide a trivial sigsetjmp/siglongjmp in terms of
setjmp/longjmp -- this is OK because no user will ever pass a non-zero
savemask.

The setjmp() uses in tests/tcg/test-i386.c and tests/tcg/linux-test.c
are left untouched because these are self-contained singlethreaded
test programs intended to be run under QEMU's Linux emulation, so they
have neither the portability nor the multithreading issues to deal with.

Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Richard Henderson <rth@twiddle.net>
Tested-by: Stefan Weil <sw@weilnetz.de>
Reviewed-by: Laszlo Ersek <lersek@redhat.com>
Signed-off-by: Blue Swirl <blauwirbel@gmail.com>
This commit is contained in:
Peter Maydell 2013-02-20 15:21:09 +00:00 committed by Blue Swirl
parent d1c36ba707
commit 6ab7e5465a
9 changed files with 52 additions and 42 deletions

View File

@ -45,7 +45,7 @@ static unsigned int pool_size;
typedef struct { typedef struct {
Coroutine base; Coroutine base;
void *stack; void *stack;
jmp_buf env; sigjmp_buf env;
} CoroutineUContext; } CoroutineUContext;
/** /**
@ -59,7 +59,7 @@ typedef struct {
CoroutineUContext leader; CoroutineUContext leader;
/** Information for the signal handler (trampoline) */ /** Information for the signal handler (trampoline) */
jmp_buf tr_reenter; sigjmp_buf tr_reenter;
volatile sig_atomic_t tr_called; volatile sig_atomic_t tr_called;
void *tr_handler; void *tr_handler;
} CoroutineThreadState; } CoroutineThreadState;
@ -115,8 +115,8 @@ static void __attribute__((constructor)) coroutine_init(void)
static void coroutine_bootstrap(CoroutineUContext *self, Coroutine *co) static void coroutine_bootstrap(CoroutineUContext *self, Coroutine *co)
{ {
/* Initialize longjmp environment and switch back the caller */ /* Initialize longjmp environment and switch back the caller */
if (!setjmp(self->env)) { if (!sigsetjmp(self->env, 0)) {
longjmp(*(jmp_buf *)co->entry_arg, 1); siglongjmp(*(sigjmp_buf *)co->entry_arg, 1);
} }
while (true) { while (true) {
@ -145,14 +145,14 @@ static void coroutine_trampoline(int signal)
/* /*
* Here we have to do a bit of a ping pong between the caller, given that * Here we have to do a bit of a ping pong between the caller, given that
* this is a signal handler and we have to do a return "soon". Then the * this is a signal handler and we have to do a return "soon". Then the
* caller can reestablish everything and do a longjmp here again. * caller can reestablish everything and do a siglongjmp here again.
*/ */
if (!setjmp(coTS->tr_reenter)) { if (!sigsetjmp(coTS->tr_reenter, 0)) {
return; return;
} }
/* /*
* Ok, the caller has longjmp'ed back to us, so now prepare * Ok, the caller has siglongjmp'ed back to us, so now prepare
* us for the real machine state switching. We have to jump * us for the real machine state switching. We have to jump
* into another function here to get a new stack context for * into another function here to get a new stack context for
* the auto variables (which have to be auto-variables * the auto variables (which have to be auto-variables
@ -179,7 +179,7 @@ static Coroutine *coroutine_new(void)
/* The way to manipulate stack is with the sigaltstack function. We /* The way to manipulate stack is with the sigaltstack function. We
* prepare a stack, with it delivering a signal to ourselves and then * prepare a stack, with it delivering a signal to ourselves and then
* put setjmp/longjmp where needed. * put sigsetjmp/siglongjmp where needed.
* This has been done keeping coroutine-ucontext as a model and with the * This has been done keeping coroutine-ucontext as a model and with the
* pth ideas (GNU Portable Threads). See coroutine-ucontext for the basics * pth ideas (GNU Portable Threads). See coroutine-ucontext for the basics
* of the coroutines and see pth_mctx.c (from the pth project) for the * of the coroutines and see pth_mctx.c (from the pth project) for the
@ -220,7 +220,7 @@ static Coroutine *coroutine_new(void)
/* /*
* Now transfer control onto the signal stack and set it up. * Now transfer control onto the signal stack and set it up.
* It will return immediately via "return" after the setjmp() * It will return immediately via "return" after the sigsetjmp()
* was performed. Be careful here with race conditions. The * was performed. Be careful here with race conditions. The
* signal can be delivered the first time sigsuspend() is * signal can be delivered the first time sigsuspend() is
* called. * called.
@ -261,8 +261,8 @@ static Coroutine *coroutine_new(void)
* type-conversion warnings related to the `volatile' qualifier and * type-conversion warnings related to the `volatile' qualifier and
* the fact that `jmp_buf' usually is an array type. * the fact that `jmp_buf' usually is an array type.
*/ */
if (!setjmp(old_env)) { if (!sigsetjmp(old_env, 0)) {
longjmp(coTS->tr_reenter, 1); siglongjmp(coTS->tr_reenter, 1);
} }
/* /*
@ -311,9 +311,9 @@ CoroutineAction qemu_coroutine_switch(Coroutine *from_, Coroutine *to_,
s->current = to_; s->current = to_;
ret = setjmp(from->env); ret = sigsetjmp(from->env, 0);
if (ret == 0) { if (ret == 0) {
longjmp(to->env, action); siglongjmp(to->env, action);
} }
return ret; return ret;
} }

View File

@ -46,7 +46,7 @@ static unsigned int pool_size;
typedef struct { typedef struct {
Coroutine base; Coroutine base;
void *stack; void *stack;
jmp_buf env; sigjmp_buf env;
#ifdef CONFIG_VALGRIND_H #ifdef CONFIG_VALGRIND_H
unsigned int valgrind_stack_id; unsigned int valgrind_stack_id;
@ -130,8 +130,8 @@ static void coroutine_trampoline(int i0, int i1)
co = &self->base; co = &self->base;
/* Initialize longjmp environment and switch back the caller */ /* Initialize longjmp environment and switch back the caller */
if (!setjmp(self->env)) { if (!sigsetjmp(self->env, 0)) {
longjmp(*(jmp_buf *)co->entry_arg, 1); siglongjmp(*(sigjmp_buf *)co->entry_arg, 1);
} }
while (true) { while (true) {
@ -145,14 +145,15 @@ static Coroutine *coroutine_new(void)
const size_t stack_size = 1 << 20; const size_t stack_size = 1 << 20;
CoroutineUContext *co; CoroutineUContext *co;
ucontext_t old_uc, uc; ucontext_t old_uc, uc;
jmp_buf old_env; sigjmp_buf old_env;
union cc_arg arg = {0}; union cc_arg arg = {0};
/* The ucontext functions preserve signal masks which incurs a system call /* The ucontext functions preserve signal masks which incurs a
* overhead. setjmp()/longjmp() does not preserve signal masks but only * system call overhead. sigsetjmp(buf, 0)/siglongjmp() does not
* works on the current stack. Since we need a way to create and switch to * preserve signal masks but only works on the current stack.
* a new stack, use the ucontext functions for that but setjmp()/longjmp() * Since we need a way to create and switch to a new stack, use
* for everything else. * the ucontext functions for that but sigsetjmp()/siglongjmp() for
* everything else.
*/ */
if (getcontext(&uc) == -1) { if (getcontext(&uc) == -1) {
@ -178,8 +179,8 @@ static Coroutine *coroutine_new(void)
makecontext(&uc, (void (*)(void))coroutine_trampoline, makecontext(&uc, (void (*)(void))coroutine_trampoline,
2, arg.i[0], arg.i[1]); 2, arg.i[0], arg.i[1]);
/* swapcontext() in, longjmp() back out */ /* swapcontext() in, siglongjmp() back out */
if (!setjmp(old_env)) { if (!sigsetjmp(old_env, 0)) {
swapcontext(&old_uc, &uc); swapcontext(&old_uc, &uc);
} }
return &co->base; return &co->base;
@ -242,9 +243,9 @@ CoroutineAction qemu_coroutine_switch(Coroutine *from_, Coroutine *to_,
s->current = to_; s->current = to_;
ret = setjmp(from->env); ret = sigsetjmp(from->env, 0);
if (ret == 0) { if (ret == 0) {
longjmp(to->env, action); siglongjmp(to->env, action);
} }
return ret; return ret;
} }

View File

@ -35,7 +35,7 @@ void cpu_loop_exit(CPUArchState *env)
CPUState *cpu = ENV_GET_CPU(env); CPUState *cpu = ENV_GET_CPU(env);
cpu->current_tb = NULL; cpu->current_tb = NULL;
longjmp(env->jmp_env, 1); siglongjmp(env->jmp_env, 1);
} }
/* exit the current TB from a signal handler. The host registers are /* exit the current TB from a signal handler. The host registers are
@ -47,7 +47,7 @@ void cpu_resume_from_signal(CPUArchState *env, void *puc)
/* XXX: restore cpu registers saved in host registers */ /* XXX: restore cpu registers saved in host registers */
env->exception_index = -1; env->exception_index = -1;
longjmp(env->jmp_env, 1); siglongjmp(env->jmp_env, 1);
} }
#endif #endif
@ -234,7 +234,7 @@ int cpu_exec(CPUArchState *env)
/* prepare setjmp context for exception handling */ /* prepare setjmp context for exception handling */
for(;;) { for(;;) {
if (setjmp(env->jmp_env) == 0) { if (sigsetjmp(env->jmp_env, 0) == 0) {
/* if an exception is pending, we execute it here */ /* if an exception is pending, we execute it here */
if (env->exception_index >= 0) { if (env->exception_index >= 0) {
if (env->exception_index >= EXCP_INTERRUPT) { if (env->exception_index >= EXCP_INTERRUPT) {

View File

@ -226,7 +226,7 @@ struct dis_private {
bfd_byte the_buffer[MAX_MNEM_SIZE]; bfd_byte the_buffer[MAX_MNEM_SIZE];
bfd_vma insn_start; bfd_vma insn_start;
int orig_sizeflag; int orig_sizeflag;
jmp_buf bailout; sigjmp_buf bailout;
}; };
enum address_mode enum address_mode
@ -303,7 +303,7 @@ fetch_data2(struct disassemble_info *info, bfd_byte *addr)
STATUS. */ STATUS. */
if (priv->max_fetched == priv->the_buffer) if (priv->max_fetched == priv->the_buffer)
(*info->memory_error_func) (status, start, info); (*info->memory_error_func) (status, start, info);
longjmp (priv->bailout, 1); siglongjmp(priv->bailout, 1);
} }
else else
priv->max_fetched = addr; priv->max_fetched = addr;
@ -3661,7 +3661,7 @@ print_insn (bfd_vma pc, disassemble_info *info)
start_codep = priv.the_buffer; start_codep = priv.the_buffer;
codep = priv.the_buffer; codep = priv.the_buffer;
if (setjmp (priv.bailout) != 0) if (sigsetjmp(priv.bailout, 0) != 0)
{ {
const char *name; const char *name;

View File

@ -624,7 +624,7 @@ struct private
bfd_byte *max_fetched; bfd_byte *max_fetched;
bfd_byte the_buffer[MAXLEN]; bfd_byte the_buffer[MAXLEN];
bfd_vma insn_start; bfd_vma insn_start;
jmp_buf bailout; sigjmp_buf bailout;
}; };
/* Make sure that bytes from INFO->PRIVATE_DATA->BUFFER (inclusive) /* Make sure that bytes from INFO->PRIVATE_DATA->BUFFER (inclusive)
@ -644,7 +644,7 @@ fetch_data2(struct disassemble_info *info, bfd_byte *addr)
if (status != 0) if (status != 0)
{ {
(*info->memory_error_func) (status, start, info); (*info->memory_error_func) (status, start, info);
longjmp (priv->bailout, 1); siglongjmp(priv->bailout, 1);
} }
else else
priv->max_fetched = addr; priv->max_fetched = addr;
@ -1912,9 +1912,10 @@ print_insn_m68k (bfd_vma memaddr, disassemble_info *info)
priv.max_fetched = priv.the_buffer; priv.max_fetched = priv.the_buffer;
priv.insn_start = memaddr; priv.insn_start = memaddr;
if (setjmp (priv.bailout) != 0) if (sigsetjmp(priv.bailout, 0) != 0) {
/* Error return. */ /* Error return. */
return -1; return -1;
}
switch (info->mach) switch (info->mach)
{ {

View File

@ -184,7 +184,7 @@ typedef struct CPUWatchpoint {
struct GDBRegisterState *gdb_regs; \ struct GDBRegisterState *gdb_regs; \
\ \
/* Core interrupt code */ \ /* Core interrupt code */ \
jmp_buf jmp_env; \ sigjmp_buf jmp_env; \
int exception_index; \ int exception_index; \
\ \
CPUArchState *next_cpu; /* next CPU sharing TB cache */ \ CPUArchState *next_cpu; /* next CPU sharing TB cache */ \

View File

@ -63,6 +63,14 @@
# undef setjmp # undef setjmp
# define setjmp(env) _setjmp(env, NULL) # define setjmp(env) _setjmp(env, NULL)
#endif #endif
/* QEMU uses sigsetjmp()/siglongjmp() as the portable way to specify
* "longjmp and don't touch the signal masks". Since we know that the
* savemask parameter will always be zero we can safely define these
* in terms of setjmp/longjmp on Win32.
*/
#define sigjmp_buf jmp_buf
#define sigsetjmp(env, savemask) setjmp(env)
#define siglongjmp(env, val) longjmp(env, val)
/* Declaration of ffs() is missing in MinGW's strings.h. */ /* Declaration of ffs() is missing in MinGW's strings.h. */
int ffs(int i); int ffs(int i);

View File

@ -2740,7 +2740,7 @@ static const mon_cmd_t qmp_cmds[] = {
/*******************************************************************/ /*******************************************************************/
static const char *pch; static const char *pch;
static jmp_buf expr_env; static sigjmp_buf expr_env;
#define MD_TLONG 0 #define MD_TLONG 0
#define MD_I32 1 #define MD_I32 1
@ -3135,7 +3135,7 @@ static const MonitorDef monitor_defs[] = {
static void expr_error(Monitor *mon, const char *msg) static void expr_error(Monitor *mon, const char *msg)
{ {
monitor_printf(mon, "%s\n", msg); monitor_printf(mon, "%s\n", msg);
longjmp(expr_env, 1); siglongjmp(expr_env, 1);
} }
/* return 0 if OK, -1 if not found */ /* return 0 if OK, -1 if not found */
@ -3345,7 +3345,7 @@ static int64_t expr_sum(Monitor *mon)
static int get_expr(Monitor *mon, int64_t *pval, const char **pp) static int get_expr(Monitor *mon, int64_t *pval, const char **pp)
{ {
pch = *pp; pch = *pp;
if (setjmp(expr_env)) { if (sigsetjmp(expr_env, 0)) {
*pp = pch; *pp = pch;
return -1; return -1;
} }

View File

@ -70,7 +70,7 @@ void cpu_resume_from_signal(CPUArchState *env1, void *puc)
#endif #endif
} }
env1->exception_index = -1; env1->exception_index = -1;
longjmp(env1->jmp_env, 1); siglongjmp(env1->jmp_env, 1);
} }
/* 'pc' is the host PC at which the exception was raised. 'address' is /* 'pc' is the host PC at which the exception was raised. 'address' is