提交 cbf2ba95 编写于 作者: N Nicholas Piggin 提交者: Michael Ellerman

powerpc/64s: system reset interrupt preserve HSRRs

Code that uses HSRR registers is not required to clear MSR[RI] by
convention, however the system reset NMI itself may use HSRR
registers (e.g., to call OPAL) and clobber them.

Rather than introduce the requirement to clear RI in order to use
HSRRs, have system reset interrupt save and restore HSRRs.
Signed-off-by: NNicholas Piggin <npiggin@gmail.com>
Signed-off-by: NMichael Ellerman <mpe@ellerman.id.au>
上级 ccd47702
...@@ -435,14 +435,32 @@ void hv_nmi_check_nonrecoverable(struct pt_regs *regs) ...@@ -435,14 +435,32 @@ void hv_nmi_check_nonrecoverable(struct pt_regs *regs)
void system_reset_exception(struct pt_regs *regs) void system_reset_exception(struct pt_regs *regs)
{ {
unsigned long hsrr0, hsrr1;
bool nested = in_nmi();
bool saved_hsrrs = false;
/* /*
* Avoid crashes in case of nested NMI exceptions. Recoverability * Avoid crashes in case of nested NMI exceptions. Recoverability
* is determined by RI and in_nmi * is determined by RI and in_nmi
*/ */
bool nested = in_nmi();
if (!nested) if (!nested)
nmi_enter(); nmi_enter();
/*
* System reset can interrupt code where HSRRs are live and MSR[RI]=1.
* The system reset interrupt itself may clobber HSRRs (e.g., to call
* OPAL), so save them here and restore them before returning.
*
* Machine checks don't need to save HSRRs, as the real mode handler
* is careful to avoid them, and the regular handler is not delivered
* as an NMI.
*/
if (cpu_has_feature(CPU_FTR_HVMODE)) {
hsrr0 = mfspr(SPRN_HSRR0);
hsrr1 = mfspr(SPRN_HSRR1);
saved_hsrrs = true;
}
hv_nmi_check_nonrecoverable(regs); hv_nmi_check_nonrecoverable(regs);
__this_cpu_inc(irq_stat.sreset_irqs); __this_cpu_inc(irq_stat.sreset_irqs);
...@@ -492,6 +510,11 @@ void system_reset_exception(struct pt_regs *regs) ...@@ -492,6 +510,11 @@ void system_reset_exception(struct pt_regs *regs)
if (!(regs->msr & MSR_RI)) if (!(regs->msr & MSR_RI))
nmi_panic(regs, "Unrecoverable System Reset"); nmi_panic(regs, "Unrecoverable System Reset");
if (saved_hsrrs) {
mtspr(SPRN_HSRR0, hsrr0);
mtspr(SPRN_HSRR1, hsrr1);
}
if (!nested) if (!nested)
nmi_exit(); nmi_exit();
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册