]> git.baikalelectronics.ru Git - kernel.git/commitdiff
powerpc/pseries, ps3: panic flush kernel messages before halting system
authorNicholas Piggin <npiggin@gmail.com>
Sat, 23 Dec 2017 16:49:23 +0000 (02:49 +1000)
committerMichael Ellerman <mpe@ellerman.id.au>
Mon, 22 Jan 2018 00:44:24 +0000 (11:44 +1100)
Platforms with a panic handler that halts the system can have problems
getting kernel messages out, because the panic notifiers are called
before kernel/panic.c does its flushing of printk buffers an console
etc.

This was attempted to be solved with commit a3b2cb30f252 ("powerpc: Do
not call ppc_md.panic in fadump panic notifier"), but that wasn't the
right approach and caused other problems, and was reverted by commit
ab9dbf771ff9.

Instead, the powernv shutdown paths have already had a similar
problem, fixed by taking the message flushing sequence from
kernel/panic.c. That's a little bit ugly, but while we have the code
duplicated, it will work for this case as well. So have ppc panic
handlers do the same flushing before they terminate.

Without this patch, a qemu pseries_le_defconfig guest stops silently
when issued the nmi command when xmon is off and no crash dumpers
enabled. Afterwards, an oops is printed by each CPU as expected.

Fixes: ab9dbf771ff9 ("Revert "powerpc: Do not call ppc_md.panic in fadump panic notifier"")
Signed-off-by: Nicholas Piggin <npiggin@gmail.com>
Reviewed-by: David Gibson <david@gibson.dropbear.id.au>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
arch/powerpc/include/asm/bug.h
arch/powerpc/kernel/traps.c
arch/powerpc/platforms/powernv/opal.c
arch/powerpc/platforms/ps3/setup.c
arch/powerpc/platforms/pseries/setup.c

index 97c38472b92487093a976b9bc128554166d37645..fd06dbe7d7d3218c3608385c6690ddaece7780da 100644 (file)
@@ -136,7 +136,8 @@ extern void _exception(int, struct pt_regs *, int, unsigned long);
 extern void _exception_pkey(int, struct pt_regs *, int, unsigned long, int);
 extern void die(const char *, struct pt_regs *, long);
 extern bool die_will_crash(void);
-
+extern void panic_flush_kmsg_start(void);
+extern void panic_flush_kmsg_end(void);
 #endif /* !__ASSEMBLY__ */
 
 #endif /* __KERNEL__ */
index cb07d65197836ec2f3cf0a113a60cd61b285a717..4260c73461a3cf77f6be33455c8185d1271655b9 100644 (file)
@@ -39,6 +39,8 @@
 #include <linux/ratelimit.h>
 #include <linux/context_tracking.h>
 #include <linux/smp.h>
+#include <linux/console.h>
+#include <linux/kmsg_dump.h>
 
 #include <asm/emulated_ops.h>
 #include <asm/pgtable.h>
@@ -143,6 +145,28 @@ static int die_owner = -1;
 static unsigned int die_nest_count;
 static int die_counter;
 
+extern void panic_flush_kmsg_start(void)
+{
+       /*
+        * These are mostly taken from kernel/panic.c, but tries to do
+        * relatively minimal work. Don't use delay functions (TB may
+        * be broken), don't crash dump (need to set a firmware log),
+        * don't run notifiers. We do want to get some information to
+        * Linux console.
+        */
+       console_verbose();
+       bust_spinlocks(1);
+}
+
+extern void panic_flush_kmsg_end(void)
+{
+       printk_safe_flush_on_panic();
+       kmsg_dump(KMSG_DUMP_PANIC);
+       bust_spinlocks(0);
+       debug_locks_off();
+       console_flush_on_panic();
+}
+
 static unsigned long oops_begin(struct pt_regs *regs)
 {
        int cpu;
index 69b5263fc9e3c8d7cc85cb0a329ac0742ab29baa..c15182765ff59a4809ac5777a8bdeb4fae0b89e9 100644 (file)
@@ -461,24 +461,14 @@ static int opal_recover_mce(struct pt_regs *regs,
 
 void pnv_platform_error_reboot(struct pt_regs *regs, const char *msg)
 {
-       /*
-        * This is mostly taken from kernel/panic.c, but tries to do
-        * relatively minimal work. Don't use delay functions (TB may
-        * be broken), don't crash dump (need to set a firmware log),
-        * don't run notifiers. We do want to get some information to
-        * Linux console.
-        */
-       console_verbose();
-       bust_spinlocks(1);
+       panic_flush_kmsg_start();
+
        pr_emerg("Hardware platform error: %s\n", msg);
        if (regs)
                show_regs(regs);
        smp_send_stop();
-       printk_safe_flush_on_panic();
-       kmsg_dump(KMSG_DUMP_PANIC);
-       bust_spinlocks(0);
-       debug_locks_off();
-       console_flush_on_panic();
+
+       panic_flush_kmsg_end();
 
        /*
         * Don't bother to shut things down because this will
index 6244bc849469e33af7dcc5a4ac2b21dd970ac1c6..77a37520068ded6cd49b4e5d60d0982ae69a5c38 100644 (file)
@@ -113,6 +113,7 @@ static void ps3_panic(char *str)
        printk("   System does not reboot automatically.\n");
        printk("   Please press POWER button.\n");
        printk("\n");
+       panic_flush_kmsg_end();
 
        while(1)
                lv1_pause(1);
index db76963e693d735f1cc6264020e4057abab60ffa..42e9cf0b31804ceeb9a7d17eaeccc19718f0257a 100644 (file)
@@ -533,6 +533,12 @@ static void __init pSeries_setup_arch(void)
        ppc_md.pcibios_root_bridge_prepare = pseries_root_bridge_prepare;
 }
 
+static void pseries_panic(char *str)
+{
+       panic_flush_kmsg_end();
+       rtas_os_term(str);
+}
+
 static int __init pSeries_init_panel(void)
 {
        /* Manually leave the kernel version on the panel. */
@@ -761,7 +767,7 @@ define_machine(pseries) {
        .pcibios_fixup          = pSeries_final_fixup,
        .restart                = rtas_restart,
        .halt                   = rtas_halt,
-       .panic                  = rtas_os_term,
+       .panic                  = pseries_panic,
        .get_boot_time          = rtas_get_boot_time,
        .get_rtc_time           = rtas_get_rtc_time,
        .set_rtc_time           = rtas_set_rtc_time,