提交 b10089a1 编写于 作者: P Peter Maydell 提交者: Laurent Vivier

linux-user: Don't call gdb_handlesig() before queue_signal()

The CPU main-loop routines for linux-user generally
call gdb_handlesig() when they're about to queue a
SIGTRAP signal. This is wrong, because queue_signal()
will cause us to pend a signal, and process_pending_signals()
will then call gdb_handlesig() itself. So the effect is that
we notify gdb of the SIGTRAP, and then if gdb says "OK,
continue with signal X" we will incorrectly notify
gdb of the signal X as well. We don't do this double-notify
for anything else, only SIGTRAP.

Remove this unnecessary and incorrect code from all
the targets except for nios2 (whose main loop is
doing something different and broken, and will be handled
in a separate patch).

This bug only manifests if the user responds to the reported
SIGTRAP using "signal SIGFOO" rather than "continue"; since
the latter is the overwhelmingly common thing to do after a
breakpoint most people won't have hit this.
Signed-off-by: NPeter Maydell <peter.maydell@linaro.org>
Reviewed-by: NRichard Henderson <richard.henderson@linaro.org>
Message-Id: <20181019174958.26616-2-peter.maydell@linaro.org>
Signed-off-by: NLaurent Vivier <laurent@vivier.eu>
上级 e285977e
...@@ -73,7 +73,7 @@ ...@@ -73,7 +73,7 @@
void cpu_loop(CPUARMState *env) void cpu_loop(CPUARMState *env)
{ {
CPUState *cs = CPU(arm_env_get_cpu(env)); CPUState *cs = CPU(arm_env_get_cpu(env));
int trapnr, sig; int trapnr;
abi_long ret; abi_long ret;
target_siginfo_t info; target_siginfo_t info;
...@@ -121,13 +121,10 @@ void cpu_loop(CPUARMState *env) ...@@ -121,13 +121,10 @@ void cpu_loop(CPUARMState *env)
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
case EXCP_BKPT: case EXCP_BKPT:
sig = gdb_handlesig(cs, TARGET_SIGTRAP); info.si_signo = TARGET_SIGTRAP;
if (sig) { info.si_errno = 0;
info.si_signo = sig; info.si_code = TARGET_TRAP_BRKPT;
info.si_errno = 0; queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
info.si_code = TARGET_TRAP_BRKPT;
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
}
break; break;
case EXCP_SEMIHOST: case EXCP_SEMIHOST:
env->xregs[0] = do_arm_semihosting(env); env->xregs[0] = do_arm_semihosting(env);
......
...@@ -179,14 +179,10 @@ void cpu_loop(CPUAlphaState *env) ...@@ -179,14 +179,10 @@ void cpu_loop(CPUAlphaState *env)
} }
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
info.si_signo = gdb_handlesig(cs, TARGET_SIGTRAP); info.si_signo = TARGET_SIGTRAP;
if (info.si_signo) { info.si_errno = 0;
info.si_errno = 0; info.si_code = TARGET_TRAP_BRKPT;
info.si_code = TARGET_TRAP_BRKPT; queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
} else {
arch_interrupt = false;
}
break; break;
case EXCP_INTERRUPT: case EXCP_INTERRUPT:
/* Just indicate that signals should be handled asap. */ /* Just indicate that signals should be handled asap. */
......
...@@ -397,18 +397,10 @@ void cpu_loop(CPUARMState *env) ...@@ -397,18 +397,10 @@ void cpu_loop(CPUARMState *env)
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
excp_debug: excp_debug:
{ info.si_signo = TARGET_SIGTRAP;
int sig; info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
sig = gdb_handlesig(cs, TARGET_SIGTRAP); queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
if (sig)
{
info.si_signo = sig;
info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
}
}
break; break;
case EXCP_KERNEL_TRAP: case EXCP_KERNEL_TRAP:
if (do_kernel_trap(env)) if (do_kernel_trap(env))
......
...@@ -64,18 +64,10 @@ void cpu_loop(CPUCRISState *env) ...@@ -64,18 +64,10 @@ void cpu_loop(CPUCRISState *env)
} }
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
{ info.si_signo = TARGET_SIGTRAP;
int sig; info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
sig = gdb_handlesig(cs, TARGET_SIGTRAP); queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
if (sig)
{
info.si_signo = sig;
info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
}
}
break; break;
case EXCP_ATOMIC: case EXCP_ATOMIC:
cpu_exec_step_atomic(cs); cpu_exec_step_atomic(cs);
......
...@@ -182,13 +182,10 @@ void cpu_loop(CPUHPPAState *env) ...@@ -182,13 +182,10 @@ void cpu_loop(CPUHPPAState *env)
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info); queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
trapnr = gdb_handlesig(cs, TARGET_SIGTRAP); info.si_signo = TARGET_SIGTRAP;
if (trapnr) { info.si_errno = 0;
info.si_signo = trapnr; info.si_code = TARGET_TRAP_BRKPT;
info.si_errno = 0; queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
info.si_code = TARGET_TRAP_BRKPT;
queue_signal(env, trapnr, QEMU_SI_FAULT, &info);
}
break; break;
case EXCP_INTERRUPT: case EXCP_INTERRUPT:
/* just indicate that signals should be handled asap */ /* just indicate that signals should be handled asap */
......
...@@ -225,18 +225,10 @@ void cpu_loop(CPUX86State *env) ...@@ -225,18 +225,10 @@ void cpu_loop(CPUX86State *env)
/* just indicate that signals should be handled asap */ /* just indicate that signals should be handled asap */
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
{ info.si_signo = TARGET_SIGTRAP;
int sig; info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
sig = gdb_handlesig(cs, TARGET_SIGTRAP); queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
if (sig)
{
info.si_signo = sig;
info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
}
}
break; break;
case EXCP_ATOMIC: case EXCP_ATOMIC:
cpu_exec_step_atomic(cs); cpu_exec_step_atomic(cs);
......
...@@ -112,18 +112,10 @@ void cpu_loop(CPUM68KState *env) ...@@ -112,18 +112,10 @@ void cpu_loop(CPUM68KState *env)
} }
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
{ info.si_signo = TARGET_SIGTRAP;
int sig; info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
sig = gdb_handlesig(cs, TARGET_SIGTRAP); queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
if (sig)
{
info.si_signo = sig;
info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
}
}
break; break;
case EXCP_ATOMIC: case EXCP_ATOMIC:
cpu_exec_step_atomic(cs); cpu_exec_step_atomic(cs);
......
...@@ -113,18 +113,10 @@ void cpu_loop(CPUMBState *env) ...@@ -113,18 +113,10 @@ void cpu_loop(CPUMBState *env)
} }
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
{ info.si_signo = TARGET_SIGTRAP;
int sig; info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
sig = gdb_handlesig(cs, TARGET_SIGTRAP); queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
if (sig)
{
info.si_signo = sig;
info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
}
}
break; break;
case EXCP_ATOMIC: case EXCP_ATOMIC:
cpu_exec_step_atomic(cs); cpu_exec_step_atomic(cs);
......
...@@ -592,18 +592,10 @@ done_syscall: ...@@ -592,18 +592,10 @@ done_syscall:
/* just indicate that signals should be handled asap */ /* just indicate that signals should be handled asap */
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
{ info.si_signo = TARGET_SIGTRAP;
int sig; info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
sig = gdb_handlesig(cs, TARGET_SIGTRAP); queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
if (sig)
{
info.si_signo = sig;
info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
}
}
break; break;
case EXCP_SC: case EXCP_SC:
if (do_store_exclusive(env)) { if (do_store_exclusive(env)) {
......
...@@ -85,13 +85,10 @@ void cpu_loop(CPUOpenRISCState *env) ...@@ -85,13 +85,10 @@ void cpu_loop(CPUOpenRISCState *env)
/* We processed the pending cpu work above. */ /* We processed the pending cpu work above. */
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
trapnr = gdb_handlesig(cs, TARGET_SIGTRAP); info.si_signo = TARGET_SIGTRAP;
if (trapnr) { info.si_errno = 0;
info.si_signo = trapnr; info.si_code = TARGET_TRAP_BRKPT;
info.si_errno = 0; queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
info.si_code = TARGET_TRAP_BRKPT;
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
}
break; break;
case EXCP_ATOMIC: case EXCP_ATOMIC:
cpu_exec_step_atomic(cs); cpu_exec_step_atomic(cs);
......
...@@ -69,7 +69,7 @@ void cpu_loop(CPUPPCState *env) ...@@ -69,7 +69,7 @@ void cpu_loop(CPUPPCState *env)
{ {
CPUState *cs = CPU(ppc_env_get_cpu(env)); CPUState *cs = CPU(ppc_env_get_cpu(env));
target_siginfo_t info; target_siginfo_t info;
int trapnr, sig; int trapnr;
target_ulong ret; target_ulong ret;
for(;;) { for(;;) {
...@@ -449,15 +449,10 @@ void cpu_loop(CPUPPCState *env) ...@@ -449,15 +449,10 @@ void cpu_loop(CPUPPCState *env)
env->gpr[3] = ret; env->gpr[3] = ret;
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
sig = gdb_handlesig(cs, TARGET_SIGTRAP); info.si_signo = TARGET_SIGTRAP;
if (sig) { info.si_errno = 0;
info.si_signo = sig; info.si_code = TARGET_TRAP_BRKPT;
info.si_errno = 0; queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
info.si_code = TARGET_TRAP_BRKPT;
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
} else {
arch_interrupt = false;
}
break; break;
case EXCP_INTERRUPT: case EXCP_INTERRUPT:
/* just indicate that signals should be handled asap */ /* just indicate that signals should be handled asap */
......
...@@ -88,7 +88,7 @@ void cpu_loop(CPURISCVState *env) ...@@ -88,7 +88,7 @@ void cpu_loop(CPURISCVState *env)
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
gdbstep: gdbstep:
signum = gdb_handlesig(cs, TARGET_SIGTRAP); signum = TARGET_SIGTRAP;
sigcode = TARGET_TRAP_BRKPT; sigcode = TARGET_TRAP_BRKPT;
break; break;
default: default:
......
...@@ -61,12 +61,9 @@ void cpu_loop(CPUS390XState *env) ...@@ -61,12 +61,9 @@ void cpu_loop(CPUS390XState *env)
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
sig = gdb_handlesig(cs, TARGET_SIGTRAP); sig = TARGET_SIGTRAP;
if (sig) { n = TARGET_TRAP_BRKPT;
n = TARGET_TRAP_BRKPT; goto do_signal_pc;
goto do_signal_pc;
}
break;
case EXCP_PGM: case EXCP_PGM:
n = env->int_pgm_code; n = env->int_pgm_code;
switch (n) { switch (n) {
......
...@@ -57,19 +57,10 @@ void cpu_loop(CPUSH4State *env) ...@@ -57,19 +57,10 @@ void cpu_loop(CPUSH4State *env)
/* just indicate that signals should be handled asap */ /* just indicate that signals should be handled asap */
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
{ info.si_signo = TARGET_SIGTRAP;
int sig; info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
sig = gdb_handlesig(cs, TARGET_SIGTRAP); queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
if (sig) {
info.si_signo = sig;
info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
} else {
arch_interrupt = false;
}
}
break; break;
case 0xa0: case 0xa0:
case 0xc0: case 0xc0:
......
...@@ -268,18 +268,10 @@ void cpu_loop (CPUSPARCState *env) ...@@ -268,18 +268,10 @@ void cpu_loop (CPUSPARCState *env)
} }
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
{ info.si_signo = TARGET_SIGTRAP;
int sig; info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
sig = gdb_handlesig(cs, TARGET_SIGTRAP); queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
if (sig)
{
info.si_signo = sig;
info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
}
}
break; break;
case EXCP_ATOMIC: case EXCP_ATOMIC:
cpu_exec_step_atomic(cs); cpu_exec_step_atomic(cs);
......
...@@ -239,13 +239,10 @@ void cpu_loop(CPUXtensaState *env) ...@@ -239,13 +239,10 @@ void cpu_loop(CPUXtensaState *env)
} }
break; break;
case EXCP_DEBUG: case EXCP_DEBUG:
trapnr = gdb_handlesig(cs, TARGET_SIGTRAP); info.si_signo = TARGET_SIGTRAP;
if (trapnr) { info.si_errno = 0;
info.si_signo = trapnr; info.si_code = TARGET_TRAP_BRKPT;
info.si_errno = 0; queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
info.si_code = TARGET_TRAP_BRKPT;
queue_signal(env, trapnr, QEMU_SI_FAULT, &info);
}
break; break;
case EXC_DEBUG: case EXC_DEBUG:
default: default:
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册