arm64: kernel: restore HW breakpoint registers in cpu_suspend
When a CPU resumes from low-power, it restores HW breakpoint and watchpoint slots through a CPU PM notifier. Since we want to enable debugging as early as possible in the resume path, the mdscr content is restored along the general purpose registers in the cpu_suspend API and debug exceptions are reenabled when cpu_suspend returns. Since the CPU PM notifier is run after a CPU has been resumed, we cannot expect HW breakpoint registers to contain sane values till the notifier is run, since the HW breakpoints registers content is unknown at reset; this means that the CPU might run with debug exceptions enabled, mdscr restored but HW breakpoint registers containing junk values that can trigger spurious debug exceptions. This patch fixes current HW breakpoints restore by moving the HW breakpoints registers restoration to the cpu_suspend API, before the debug exceptions are enabled. This way, as soon as the cpu_suspend function returns the kernel can resume debugging with sane values in HW breakpoint registers. Signed-off-by: Lorenzo Pieralisi <lorenzo.pieralisi@arm.com> Acked-by: Will Deacon <will.deacon@arm.com> Signed-off-by: Catalin Marinas <catalin.marinas@arm.com>
This commit is contained in:
parent
f4be8433fc
commit
65c021bb49
|
@ -894,29 +894,10 @@ static struct notifier_block hw_breakpoint_reset_nb = {
|
||||||
.notifier_call = hw_breakpoint_reset_notify,
|
.notifier_call = hw_breakpoint_reset_notify,
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifdef CONFIG_CPU_PM
|
#ifdef CONFIG_ARM64_CPU_SUSPEND
|
||||||
static int hw_breakpoint_cpu_pm_notify(struct notifier_block *self,
|
extern void cpu_suspend_set_dbg_restorer(void (*hw_bp_restore)(void *));
|
||||||
unsigned long action,
|
|
||||||
void *v)
|
|
||||||
{
|
|
||||||
if (action == CPU_PM_EXIT) {
|
|
||||||
hw_breakpoint_reset(NULL);
|
|
||||||
return NOTIFY_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
return NOTIFY_DONE;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct notifier_block hw_breakpoint_cpu_pm_nb = {
|
|
||||||
.notifier_call = hw_breakpoint_cpu_pm_notify,
|
|
||||||
};
|
|
||||||
|
|
||||||
static void __init hw_breakpoint_pm_init(void)
|
|
||||||
{
|
|
||||||
cpu_pm_register_notifier(&hw_breakpoint_cpu_pm_nb);
|
|
||||||
}
|
|
||||||
#else
|
#else
|
||||||
static inline void hw_breakpoint_pm_init(void)
|
static inline void cpu_suspend_set_dbg_restorer(void (*hw_bp_restore)(void *))
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -947,7 +928,8 @@ static int __init arch_hw_breakpoint_init(void)
|
||||||
|
|
||||||
/* Register hotplug notifier. */
|
/* Register hotplug notifier. */
|
||||||
register_cpu_notifier(&hw_breakpoint_reset_nb);
|
register_cpu_notifier(&hw_breakpoint_reset_nb);
|
||||||
hw_breakpoint_pm_init();
|
/* Register cpu_suspend hw breakpoint restore hook */
|
||||||
|
cpu_suspend_set_dbg_restorer(hw_breakpoint_reset);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -38,6 +38,22 @@ int __cpu_suspend_finisher(unsigned long arg, struct cpu_suspend_ctx *ptr,
|
||||||
return cpu_ops[cpu]->cpu_suspend(arg);
|
return cpu_ops[cpu]->cpu_suspend(arg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This hook is provided so that cpu_suspend code can restore HW
|
||||||
|
* breakpoints as early as possible in the resume path, before reenabling
|
||||||
|
* debug exceptions. Code cannot be run from a CPU PM notifier since by the
|
||||||
|
* time the notifier runs debug exceptions might have been enabled already,
|
||||||
|
* with HW breakpoints registers content still in an unknown state.
|
||||||
|
*/
|
||||||
|
void (*hw_breakpoint_restore)(void *);
|
||||||
|
void __init cpu_suspend_set_dbg_restorer(void (*hw_bp_restore)(void *))
|
||||||
|
{
|
||||||
|
/* Prevent multiple restore hook initializations */
|
||||||
|
if (WARN_ON(hw_breakpoint_restore))
|
||||||
|
return;
|
||||||
|
hw_breakpoint_restore = hw_bp_restore;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* cpu_suspend
|
* cpu_suspend
|
||||||
*
|
*
|
||||||
|
@ -73,6 +89,13 @@ int cpu_suspend(unsigned long arg)
|
||||||
if (ret == 0) {
|
if (ret == 0) {
|
||||||
cpu_switch_mm(mm->pgd, mm);
|
cpu_switch_mm(mm->pgd, mm);
|
||||||
flush_tlb_all();
|
flush_tlb_all();
|
||||||
|
/*
|
||||||
|
* Restore HW breakpoint registers to sane values
|
||||||
|
* before debug exceptions are possibly reenabled
|
||||||
|
* through local_dbg_restore.
|
||||||
|
*/
|
||||||
|
if (hw_breakpoint_restore)
|
||||||
|
hw_breakpoint_restore(NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
Loading…
Reference in New Issue