Commit d6a00b35 authored by Oleg Nesterov's avatar Oleg Nesterov

uprobes/x86: Fix arch_uprobe_disable_step() && UTASK_SSTEP_TRAPPED interaction

arch_uprobe_disable_step() should also take UTASK_SSTEP_TRAPPED into
account. In this case the probed insn was not executed, we need to
clear X86_EFLAGS_TF if it was set by us and that is all.

Again, this code will look more clean when we move it into
arch_uprobe_post_xol() and arch_uprobe_abort_xol().
Signed-off-by: default avatarOleg Nesterov <oleg@redhat.com>
Acked-by: default avatarSrikar Dronamraju <srikar@linux.vnet.ibm.com>
parent 3a4664aa
...@@ -706,14 +706,20 @@ void arch_uprobe_disable_step(struct arch_uprobe *auprobe) ...@@ -706,14 +706,20 @@ void arch_uprobe_disable_step(struct arch_uprobe *auprobe)
{ {
struct task_struct *task = current; struct task_struct *task = current;
struct arch_uprobe_task *autask = &task->utask->autask; struct arch_uprobe_task *autask = &task->utask->autask;
bool trapped = (task->utask->state == UTASK_SSTEP_TRAPPED);
struct pt_regs *regs = task_pt_regs(task); struct pt_regs *regs = task_pt_regs(task);
/* /*
* The state of TIF_BLOCKSTEP was not saved so we can get an extra * The state of TIF_BLOCKSTEP was not saved so we can get an extra
* SIGTRAP if we do not clear TF. We need to examine the opcode to * SIGTRAP if we do not clear TF. We need to examine the opcode to
* make it right. * make it right.
*/ */
if (autask->saved_tf) if (unlikely(trapped)) {
send_sig(SIGTRAP, task, 0); if (!autask->saved_tf)
else if (!(auprobe->fixups & UPROBE_FIX_SETF)) regs->flags &= ~X86_EFLAGS_TF;
regs->flags &= ~X86_EFLAGS_TF; } else {
if (autask->saved_tf)
send_sig(SIGTRAP, task, 0);
else if (!(auprobe->fixups & UPROBE_FIX_SETF))
regs->flags &= ~X86_EFLAGS_TF;
}
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment