diff options
Diffstat (limited to 'arch/powerpc/xmon/xmon.c')
-rw-r--r-- | arch/powerpc/xmon/xmon.c | 84 |
1 files changed, 34 insertions, 50 deletions
diff --git a/arch/powerpc/xmon/xmon.c b/arch/powerpc/xmon/xmon.c index eab3492a45c5..9b49c65ee7a4 100644 --- a/arch/powerpc/xmon/xmon.c +++ b/arch/powerpc/xmon/xmon.c | |||
@@ -17,6 +17,7 @@ | |||
17 | #include <linux/reboot.h> | 17 | #include <linux/reboot.h> |
18 | #include <linux/delay.h> | 18 | #include <linux/delay.h> |
19 | #include <linux/kallsyms.h> | 19 | #include <linux/kallsyms.h> |
20 | #include <linux/kmsg_dump.h> | ||
20 | #include <linux/cpumask.h> | 21 | #include <linux/cpumask.h> |
21 | #include <linux/export.h> | 22 | #include <linux/export.h> |
22 | #include <linux/sysrq.h> | 23 | #include <linux/sysrq.h> |
@@ -894,13 +895,13 @@ cmds(struct pt_regs *excp) | |||
894 | #endif | 895 | #endif |
895 | default: | 896 | default: |
896 | printf("Unrecognized command: "); | 897 | printf("Unrecognized command: "); |
897 | do { | 898 | do { |
898 | if (' ' < cmd && cmd <= '~') | 899 | if (' ' < cmd && cmd <= '~') |
899 | putchar(cmd); | 900 | putchar(cmd); |
900 | else | 901 | else |
901 | printf("\\x%x", cmd); | 902 | printf("\\x%x", cmd); |
902 | cmd = inchar(); | 903 | cmd = inchar(); |
903 | } while (cmd != '\n'); | 904 | } while (cmd != '\n'); |
904 | printf(" (type ? for help)\n"); | 905 | printf(" (type ? for help)\n"); |
905 | break; | 906 | break; |
906 | } | 907 | } |
@@ -1097,7 +1098,7 @@ static long check_bp_loc(unsigned long addr) | |||
1097 | return 1; | 1098 | return 1; |
1098 | } | 1099 | } |
1099 | 1100 | ||
1100 | static char *breakpoint_help_string = | 1101 | static char *breakpoint_help_string = |
1101 | "Breakpoint command usage:\n" | 1102 | "Breakpoint command usage:\n" |
1102 | "b show breakpoints\n" | 1103 | "b show breakpoints\n" |
1103 | "b <addr> [cnt] set breakpoint at given instr addr\n" | 1104 | "b <addr> [cnt] set breakpoint at given instr addr\n" |
@@ -1193,7 +1194,7 @@ bpt_cmds(void) | |||
1193 | 1194 | ||
1194 | default: | 1195 | default: |
1195 | termch = cmd; | 1196 | termch = cmd; |
1196 | cmd = skipbl(); | 1197 | cmd = skipbl(); |
1197 | if (cmd == '?') { | 1198 | if (cmd == '?') { |
1198 | printf(breakpoint_help_string); | 1199 | printf(breakpoint_help_string); |
1199 | break; | 1200 | break; |
@@ -1359,7 +1360,7 @@ static void xmon_show_stack(unsigned long sp, unsigned long lr, | |||
1359 | sp + REGS_OFFSET); | 1360 | sp + REGS_OFFSET); |
1360 | break; | 1361 | break; |
1361 | } | 1362 | } |
1362 | printf("--- Exception: %lx %s at ", regs.trap, | 1363 | printf("--- Exception: %lx %s at ", regs.trap, |
1363 | getvecname(TRAP(®s))); | 1364 | getvecname(TRAP(®s))); |
1364 | pc = regs.nip; | 1365 | pc = regs.nip; |
1365 | lr = regs.link; | 1366 | lr = regs.link; |
@@ -1623,14 +1624,14 @@ static void super_regs(void) | |||
1623 | 1624 | ||
1624 | cmd = skipbl(); | 1625 | cmd = skipbl(); |
1625 | if (cmd == '\n') { | 1626 | if (cmd == '\n') { |
1626 | unsigned long sp, toc; | 1627 | unsigned long sp, toc; |
1627 | asm("mr %0,1" : "=r" (sp) :); | 1628 | asm("mr %0,1" : "=r" (sp) :); |
1628 | asm("mr %0,2" : "=r" (toc) :); | 1629 | asm("mr %0,2" : "=r" (toc) :); |
1629 | 1630 | ||
1630 | printf("msr = "REG" sprg0= "REG"\n", | 1631 | printf("msr = "REG" sprg0= "REG"\n", |
1631 | mfmsr(), mfspr(SPRN_SPRG0)); | 1632 | mfmsr(), mfspr(SPRN_SPRG0)); |
1632 | printf("pvr = "REG" sprg1= "REG"\n", | 1633 | printf("pvr = "REG" sprg1= "REG"\n", |
1633 | mfspr(SPRN_PVR), mfspr(SPRN_SPRG1)); | 1634 | mfspr(SPRN_PVR), mfspr(SPRN_SPRG1)); |
1634 | printf("dec = "REG" sprg2= "REG"\n", | 1635 | printf("dec = "REG" sprg2= "REG"\n", |
1635 | mfspr(SPRN_DEC), mfspr(SPRN_SPRG2)); | 1636 | mfspr(SPRN_DEC), mfspr(SPRN_SPRG2)); |
1636 | printf("sp = "REG" sprg3= "REG"\n", sp, mfspr(SPRN_SPRG3)); | 1637 | printf("sp = "REG" sprg3= "REG"\n", sp, mfspr(SPRN_SPRG3)); |
@@ -1783,7 +1784,7 @@ byterev(unsigned char *val, int size) | |||
1783 | static int brev; | 1784 | static int brev; |
1784 | static int mnoread; | 1785 | static int mnoread; |
1785 | 1786 | ||
1786 | static char *memex_help_string = | 1787 | static char *memex_help_string = |
1787 | "Memory examine command usage:\n" | 1788 | "Memory examine command usage:\n" |
1788 | "m [addr] [flags] examine/change memory\n" | 1789 | "m [addr] [flags] examine/change memory\n" |
1789 | " addr is optional. will start where left off.\n" | 1790 | " addr is optional. will start where left off.\n" |
@@ -1798,7 +1799,7 @@ static char *memex_help_string = | |||
1798 | "NOTE: flags are saved as defaults\n" | 1799 | "NOTE: flags are saved as defaults\n" |
1799 | ""; | 1800 | ""; |
1800 | 1801 | ||
1801 | static char *memex_subcmd_help_string = | 1802 | static char *memex_subcmd_help_string = |
1802 | "Memory examine subcommands:\n" | 1803 | "Memory examine subcommands:\n" |
1803 | " hexval write this val to current location\n" | 1804 | " hexval write this val to current location\n" |
1804 | " 'string' write chars from string to this location\n" | 1805 | " 'string' write chars from string to this location\n" |
@@ -2064,7 +2065,7 @@ prdump(unsigned long adrs, long ndump) | |||
2064 | nr = mread(adrs, temp, r); | 2065 | nr = mread(adrs, temp, r); |
2065 | adrs += nr; | 2066 | adrs += nr; |
2066 | for (m = 0; m < r; ++m) { | 2067 | for (m = 0; m < r; ++m) { |
2067 | if ((m & (sizeof(long) - 1)) == 0 && m > 0) | 2068 | if ((m & (sizeof(long) - 1)) == 0 && m > 0) |
2068 | putchar(' '); | 2069 | putchar(' '); |
2069 | if (m < nr) | 2070 | if (m < nr) |
2070 | printf("%.2x", temp[m]); | 2071 | printf("%.2x", temp[m]); |
@@ -2072,7 +2073,7 @@ prdump(unsigned long adrs, long ndump) | |||
2072 | printf("%s", fault_chars[fault_type]); | 2073 | printf("%s", fault_chars[fault_type]); |
2073 | } | 2074 | } |
2074 | for (; m < 16; ++m) { | 2075 | for (; m < 16; ++m) { |
2075 | if ((m & (sizeof(long) - 1)) == 0) | 2076 | if ((m & (sizeof(long) - 1)) == 0) |
2076 | putchar(' '); | 2077 | putchar(' '); |
2077 | printf(" "); | 2078 | printf(" "); |
2078 | } | 2079 | } |
@@ -2148,45 +2149,28 @@ print_address(unsigned long addr) | |||
2148 | void | 2149 | void |
2149 | dump_log_buf(void) | 2150 | dump_log_buf(void) |
2150 | { | 2151 | { |
2151 | const unsigned long size = 128; | 2152 | struct kmsg_dumper dumper = { .active = 1 }; |
2152 | unsigned long end, addr; | 2153 | unsigned char buf[128]; |
2153 | unsigned char buf[size + 1]; | 2154 | size_t len; |
2154 | 2155 | ||
2155 | addr = 0; | 2156 | if (setjmp(bus_error_jmp) != 0) { |
2156 | buf[size] = '\0'; | 2157 | printf("Error dumping printk buffer!\n"); |
2157 | 2158 | return; | |
2158 | if (setjmp(bus_error_jmp) != 0) { | 2159 | } |
2159 | printf("Unable to lookup symbol __log_buf!\n"); | 2160 | |
2160 | return; | 2161 | catch_memory_errors = 1; |
2161 | } | 2162 | sync(); |
2162 | 2163 | ||
2163 | catch_memory_errors = 1; | 2164 | kmsg_dump_rewind_nolock(&dumper); |
2164 | sync(); | 2165 | while (kmsg_dump_get_line_nolock(&dumper, false, buf, sizeof(buf), &len)) { |
2165 | addr = kallsyms_lookup_name("__log_buf"); | 2166 | buf[len] = '\0'; |
2166 | 2167 | printf("%s", buf); | |
2167 | if (! addr) | 2168 | } |
2168 | printf("Symbol __log_buf not found!\n"); | 2169 | |
2169 | else { | 2170 | sync(); |
2170 | end = addr + (1 << CONFIG_LOG_BUF_SHIFT); | 2171 | /* wait a little while to see if we get a machine check */ |
2171 | while (addr < end) { | 2172 | __delay(200); |
2172 | if (! mread(addr, buf, size)) { | 2173 | catch_memory_errors = 0; |
2173 | printf("Can't read memory at address 0x%lx\n", addr); | ||
2174 | break; | ||
2175 | } | ||
2176 | |||
2177 | printf("%s", buf); | ||
2178 | |||
2179 | if (strlen(buf) < size) | ||
2180 | break; | ||
2181 | |||
2182 | addr += size; | ||
2183 | } | ||
2184 | } | ||
2185 | |||
2186 | sync(); | ||
2187 | /* wait a little while to see if we get a machine check */ | ||
2188 | __delay(200); | ||
2189 | catch_memory_errors = 0; | ||
2190 | } | 2174 | } |
2191 | 2175 | ||
2192 | /* | 2176 | /* |