2019-08-25 17:49:17 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
2006-07-10 19:45:13 +08:00
|
|
|
/*
|
2007-10-16 16:27:00 +08:00
|
|
|
* Copyright (C) 2000 - 2007 Jeff Dike (jdike@{addtoit,linux.intel}.com)
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|
|
|
|
|
2008-02-05 14:31:14 +08:00
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/ptrace.h>
|
|
|
|
#include <linux/sched.h>
|
2019-08-23 19:16:23 +08:00
|
|
|
#include <linux/ftrace.h>
|
2008-02-05 14:31:14 +08:00
|
|
|
#include <asm/siginfo.h>
|
|
|
|
#include <asm/signal.h>
|
|
|
|
#include <asm/unistd.h>
|
2012-10-08 10:27:32 +08:00
|
|
|
#include <frame_kern.h>
|
|
|
|
#include <kern_util.h>
|
2019-08-23 19:16:23 +08:00
|
|
|
#include <os.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
EXPORT_SYMBOL(block_signals);
|
|
|
|
EXPORT_SYMBOL(unblock_signals);
|
|
|
|
|
2019-08-23 19:16:23 +08:00
|
|
|
void block_signals_trace(void)
|
|
|
|
{
|
|
|
|
block_signals();
|
|
|
|
if (current_thread_info())
|
|
|
|
trace_hardirqs_off();
|
|
|
|
}
|
|
|
|
|
|
|
|
void unblock_signals_trace(void)
|
|
|
|
{
|
|
|
|
if (current_thread_info())
|
|
|
|
trace_hardirqs_on();
|
|
|
|
unblock_signals();
|
|
|
|
}
|
|
|
|
|
|
|
|
void um_trace_signals_on(void)
|
|
|
|
{
|
|
|
|
if (current_thread_info())
|
|
|
|
trace_hardirqs_on();
|
|
|
|
}
|
|
|
|
|
|
|
|
void um_trace_signals_off(void)
|
|
|
|
{
|
|
|
|
if (current_thread_info())
|
|
|
|
trace_hardirqs_off();
|
|
|
|
}
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* OK, we're invoking a handler
|
2006-07-10 19:45:13 +08:00
|
|
|
*/
|
2013-10-07 03:57:10 +08:00
|
|
|
static void handle_signal(struct ksignal *ksig, struct pt_regs *regs)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2012-05-02 21:59:21 +08:00
|
|
|
sigset_t *oldset = sigmask_to_save();
|
2012-09-07 01:39:47 +08:00
|
|
|
int singlestep = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
unsigned long sp;
|
|
|
|
int err;
|
|
|
|
|
2022-04-27 05:30:17 +08:00
|
|
|
if (test_thread_flag(TIF_SINGLESTEP) && (current->ptrace & PT_PTRACED))
|
2012-09-07 01:39:47 +08:00
|
|
|
singlestep = 1;
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/* Did we come from a system call? */
|
2007-10-16 16:27:00 +08:00
|
|
|
if (PT_REGS_SYSCALL_NR(regs) >= 0) {
|
2005-04-17 06:20:36 +08:00
|
|
|
/* If so, check system call restarting.. */
|
2008-02-05 14:31:14 +08:00
|
|
|
switch (PT_REGS_SYSCALL_RET(regs)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
case -ERESTART_RESTARTBLOCK:
|
|
|
|
case -ERESTARTNOHAND:
|
|
|
|
PT_REGS_SYSCALL_RET(regs) = -EINTR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case -ERESTARTSYS:
|
2013-10-07 03:57:10 +08:00
|
|
|
if (!(ksig->ka.sa.sa_flags & SA_RESTART)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
PT_REGS_SYSCALL_RET(regs) = -EINTR;
|
|
|
|
break;
|
|
|
|
}
|
2020-08-24 06:36:59 +08:00
|
|
|
fallthrough;
|
2005-04-17 06:20:36 +08:00
|
|
|
case -ERESTARTNOINTR:
|
|
|
|
PT_REGS_RESTART_SYSCALL(regs);
|
|
|
|
PT_REGS_ORIG_SYSCALL(regs) = PT_REGS_SYSCALL_NR(regs);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sp = PT_REGS_SP(regs);
|
2013-10-07 03:57:10 +08:00
|
|
|
if ((ksig->ka.sa.sa_flags & SA_ONSTACK) && (sas_ss_flags(sp) == 0))
|
2005-04-17 06:20:36 +08:00
|
|
|
sp = current->sas_ss_sp + current->sas_ss_size;
|
|
|
|
|
|
|
|
#ifdef CONFIG_ARCH_HAS_SC_SIGNALS
|
2013-10-07 03:57:10 +08:00
|
|
|
if (!(ksig->ka.sa.sa_flags & SA_SIGINFO))
|
|
|
|
err = setup_signal_stack_sc(sp, ksig, regs, oldset);
|
2005-04-17 06:20:36 +08:00
|
|
|
else
|
|
|
|
#endif
|
2013-10-07 03:57:10 +08:00
|
|
|
err = setup_signal_stack_si(sp, ksig, regs, oldset);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2013-10-07 03:57:10 +08:00
|
|
|
signal_setup_done(err, ksig, singlestep);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2015-07-04 03:44:20 +08:00
|
|
|
void do_signal(struct pt_regs *regs)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2013-10-07 03:57:10 +08:00
|
|
|
struct ksignal ksig;
|
|
|
|
int handled_sig = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2016-01-26 06:33:30 +08:00
|
|
|
while (get_signal(&ksig)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
handled_sig = 1;
|
|
|
|
/* Whee! Actually deliver the signal. */
|
2013-10-07 03:57:10 +08:00
|
|
|
handle_signal(&ksig, regs);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Did we come from a system call? */
|
2007-10-16 16:27:00 +08:00
|
|
|
if (!handled_sig && (PT_REGS_SYSCALL_NR(regs) >= 0)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
/* Restart the system call - no handlers present */
|
2008-02-05 14:31:14 +08:00
|
|
|
switch (PT_REGS_SYSCALL_RET(regs)) {
|
2006-01-19 09:44:02 +08:00
|
|
|
case -ERESTARTNOHAND:
|
|
|
|
case -ERESTARTSYS:
|
|
|
|
case -ERESTARTNOINTR:
|
2005-04-17 06:20:36 +08:00
|
|
|
PT_REGS_ORIG_SYSCALL(regs) = PT_REGS_SYSCALL_NR(regs);
|
|
|
|
PT_REGS_RESTART_SYSCALL(regs);
|
2006-01-19 09:44:02 +08:00
|
|
|
break;
|
|
|
|
case -ERESTART_RESTARTBLOCK:
|
2006-07-10 19:45:13 +08:00
|
|
|
PT_REGS_ORIG_SYSCALL(regs) = __NR_restart_syscall;
|
2005-04-17 06:20:36 +08:00
|
|
|
PT_REGS_RESTART_SYSCALL(regs);
|
2006-01-19 09:44:02 +08:00
|
|
|
break;
|
2007-10-16 16:27:00 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2007-10-16 16:27:00 +08:00
|
|
|
/*
|
|
|
|
* if there's no signal to deliver, we just put the saved sigmask
|
|
|
|
* back
|
|
|
|
*/
|
2012-05-22 11:33:55 +08:00
|
|
|
if (!handled_sig)
|
|
|
|
restore_saved_sigmask();
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|