Skip to content

Commit f898817

Browse files
Steven Rostedtrostedt
authored andcommitted
x86: Allow nesting of the debug stack IDT setting
When the NMI handler runs, it checks if it preempted a debug handler and if that handler is using the debug stack. If it is, it changes the IDT table not to update the stack, otherwise it will reset the debug stack and corrupt the debug handler it preempted. Now that ftrace uses breakpoints to change functions from nops to callers, many more places may hit a breakpoint. Unfortunately this includes some of the calls that lockdep performs. Which causes issues with the debug stack. It too needs to change the debug stack before tracing (if called from the debug handler). Allow the debug_stack_set_zero() and debug_stack_reset() to be nested so that the debug handlers can take advantage of them too. [ Used this_cpu_*() over __get_cpu_var() as suggested by H. Peter Anvin ] Signed-off-by: Steven Rostedt <rostedt@goodmis.org>
1 parent c0525a6 commit f898817

File tree

1 file changed

+7
-1
lines changed

1 file changed

+7
-1
lines changed

arch/x86/kernel/cpu/common.c

Lines changed: 7 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1101,14 +1101,20 @@ int is_debug_stack(unsigned long addr)
11011101
addr > (__get_cpu_var(debug_stack_addr) - DEBUG_STKSZ));
11021102
}
11031103

1104+
static DEFINE_PER_CPU(u32, debug_stack_use_ctr);
1105+
11041106
void debug_stack_set_zero(void)
11051107
{
1108+
this_cpu_inc(debug_stack_use_ctr);
11061109
load_idt((const struct desc_ptr *)&nmi_idt_descr);
11071110
}
11081111

11091112
void debug_stack_reset(void)
11101113
{
1111-
load_idt((const struct desc_ptr *)&idt_descr);
1114+
if (WARN_ON(!this_cpu_read(debug_stack_use_ctr)))
1115+
return;
1116+
if (this_cpu_dec_return(debug_stack_use_ctr) == 0)
1117+
load_idt((const struct desc_ptr *)&idt_descr);
11121118
}
11131119

11141120
#else /* CONFIG_X86_64 */

0 commit comments

Comments
 (0)