blob: ef18021d52e1e58cf52d0bd425a97709bd8d077f [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * Routines providing a simple monitor for use on the PowerMac.
3 *
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11004 * Copyright (C) 1996-2005 Paul Mackerras.
Michael Ellerman476792832006-10-03 14:12:08 +10005 * Copyright (C) 2001 PPC64 Team, IBM Corp
6 * Copyrignt (C) 2006 Michael Ellerman, IBM Corp
Linus Torvalds1da177e2005-04-16 15:20:36 -07007 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version
11 * 2 of the License, or (at your option) any later version.
12 */
Linus Torvalds1da177e2005-04-16 15:20:36 -070013#include <linux/errno.h>
14#include <linux/sched.h>
15#include <linux/smp.h>
16#include <linux/mm.h>
17#include <linux/reboot.h>
18#include <linux/delay.h>
19#include <linux/kallsyms.h>
Michael Ellermanca5dd392012-08-23 22:09:12 +000020#include <linux/kmsg_dump.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070021#include <linux/cpumask.h>
Paul Gortmaker4b16f8e2011-07-22 18:24:23 -040022#include <linux/export.h>
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +110023#include <linux/sysrq.h>
Andrew Morton4694ca02005-11-13 16:06:50 -080024#include <linux/interrupt.h>
David Howells7d12e782006-10-05 14:55:46 +010025#include <linux/irq.h>
Jeremy Fitzhardinge73c9cea2006-12-08 03:30:41 -080026#include <linux/bug.h>
Anton Blancharda71d64b2014-08-05 14:55:00 +100027#include <linux/nmi.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070028
29#include <asm/ptrace.h>
30#include <asm/string.h>
31#include <asm/prom.h>
32#include <asm/machdep.h>
Paul Mackerrasf78541dc2005-10-28 22:53:37 +100033#include <asm/xmon.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070034#include <asm/processor.h>
35#include <asm/pgtable.h>
36#include <asm/mmu.h>
37#include <asm/mmu_context.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070038#include <asm/cputable.h>
39#include <asm/rtas.h>
40#include <asm/sstep.h>
Paul Mackerrasf583ffc2006-10-10 11:47:07 +100041#include <asm/irq_regs.h>
Michael Ellermanff8a8f22006-10-24 18:31:27 +020042#include <asm/spu.h>
43#include <asm/spu_priv1.h>
Michael Neulingc3b75bd2008-01-18 15:50:30 +110044#include <asm/setjmp.h>
Anton Vorontsov322b4392008-12-17 10:08:55 +000045#include <asm/reg.h>
David Howellsae3a1972012-03-28 18:30:02 +010046#include <asm/debug.h>
Michael Neuling9422de32012-12-20 14:06:44 +000047#include <asm/hw_breakpoint.h>
Paul Mackerrasf78541dc2005-10-28 22:53:37 +100048
49#ifdef CONFIG_PPC64
Linus Torvalds1da177e2005-04-16 15:20:36 -070050#include <asm/hvcall.h>
Paul Mackerrasf78541dc2005-10-28 22:53:37 +100051#include <asm/paca.h>
52#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -070053
54#include "nonstdio.h"
Michael Ellermane0426042006-11-23 00:46:45 +010055#include "dis-asm.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070056
Linus Torvalds1da177e2005-04-16 15:20:36 -070057#ifdef CONFIG_SMP
Michael Ellerman1c8950f2008-05-08 14:27:17 +100058static cpumask_t cpus_in_xmon = CPU_MASK_NONE;
Linus Torvalds1da177e2005-04-16 15:20:36 -070059static unsigned long xmon_taken = 1;
60static int xmon_owner;
61static int xmon_gate;
Michael Ellermanddadb6b2012-09-13 23:01:31 +000062#else
63#define xmon_owner 0
Linus Torvalds1da177e2005-04-16 15:20:36 -070064#endif /* CONFIG_SMP */
65
Anton Blanchard5be34922010-01-12 00:50:14 +000066static unsigned long in_xmon __read_mostly = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -070067
68static unsigned long adrs;
69static int size = 1;
70#define MAX_DUMP (128 * 1024)
71static unsigned long ndump = 64;
72static unsigned long nidump = 16;
73static unsigned long ncsum = 4096;
74static int termch;
75static char tmpstr[128];
76
Linus Torvalds1da177e2005-04-16 15:20:36 -070077static long bus_error_jmp[JMP_BUF_LEN];
78static int catch_memory_errors;
79static long *xmon_fault_jmp[NR_CPUS];
Linus Torvalds1da177e2005-04-16 15:20:36 -070080
81/* Breakpoint stuff */
82struct bpt {
83 unsigned long address;
84 unsigned int instr[2];
85 atomic_t ref_count;
86 int enabled;
87 unsigned long pad;
88};
89
90/* Bits in bpt.enabled */
91#define BP_IABR_TE 1 /* IABR translation enabled */
92#define BP_IABR 2
93#define BP_TRAP 8
94#define BP_DABR 0x10
95
96#define NBPTS 256
97static struct bpt bpts[NBPTS];
98static struct bpt dabr;
99static struct bpt *iabr;
100static unsigned bpinstr = 0x7fe00008; /* trap */
101
102#define BP_NUM(bp) ((bp) - bpts + 1)
103
104/* Prototypes */
105static int cmds(struct pt_regs *);
106static int mread(unsigned long, void *, int);
107static int mwrite(unsigned long, void *, int);
108static int handle_fault(struct pt_regs *);
109static void byterev(unsigned char *, int);
110static void memex(void);
111static int bsesc(void);
112static void dump(void);
113static void prdump(unsigned long, long);
114static int ppc_inst_dump(unsigned long, long, int);
Vinay Sridharf312deb2009-05-14 23:13:07 +0000115static void dump_log_buf(void);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700116static void backtrace(struct pt_regs *);
117static void excprint(struct pt_regs *);
118static void prregs(struct pt_regs *);
119static void memops(int);
120static void memlocate(void);
121static void memzcan(void);
122static void memdiffs(unsigned char *, unsigned char *, unsigned, unsigned);
123int skipbl(void);
124int scanhex(unsigned long *valp);
125static void scannl(void);
126static int hexdigit(int);
127void getstring(char *, int);
128static void flush_input(void);
129static int inchar(void);
130static void take_input(char *);
131static unsigned long read_spr(int);
132static void write_spr(int, unsigned long);
133static void super_regs(void);
134static void remove_bpts(void);
135static void insert_bpts(void);
136static void remove_cpu_bpts(void);
137static void insert_cpu_bpts(void);
138static struct bpt *at_breakpoint(unsigned long pc);
139static struct bpt *in_breakpoint_table(unsigned long pc, unsigned long *offp);
140static int do_step(struct pt_regs *);
141static void bpt_cmds(void);
142static void cacheflush(void);
143static int cpu_cmd(void);
144static void csum(void);
145static void bootcmds(void);
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000146static void proccall(void);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700147void dump_segments(void);
148static void symbol_lookup(void);
Olaf Hering26c8af52006-09-08 16:29:21 +0200149static void xmon_show_stack(unsigned long sp, unsigned long lr,
150 unsigned long pc);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700151static void xmon_print_symbol(unsigned long address, const char *mid,
152 const char *after);
153static const char *getvecname(unsigned long vec);
154
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200155static int do_spu_cmd(void);
156
Benjamin Herrenschmidt5a8a1a22007-11-16 18:23:33 +1100157#ifdef CONFIG_44x
158static void dump_tlb_44x(void);
159#endif
Benjamin Herrenschmidt03247152010-07-09 15:34:50 +1000160#ifdef CONFIG_PPC_BOOK3E
161static void dump_tlb_book3e(void);
162#endif
Benjamin Herrenschmidt5a8a1a22007-11-16 18:23:33 +1100163
Michael Ellerman9f1067c22008-05-08 14:27:16 +1000164static int xmon_no_auto_backtrace;
Olaf Hering26c8af52006-09-08 16:29:21 +0200165
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000166extern void xmon_enter(void);
167extern void xmon_leave(void);
168
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000169#ifdef CONFIG_PPC64
170#define REG "%.16lx"
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000171#else
172#define REG "%.8lx"
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000173#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700174
Philippe Bergheaud72eceef2013-12-02 10:10:12 +0100175#ifdef __LITTLE_ENDIAN__
176#define GETWORD(v) (((v)[3] << 24) + ((v)[2] << 16) + ((v)[1] << 8) + (v)[0])
177#else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700178#define GETWORD(v) (((v)[0] << 24) + ((v)[1] << 16) + ((v)[2] << 8) + (v)[3])
Philippe Bergheaud72eceef2013-12-02 10:10:12 +0100179#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700180
181#define isxdigit(c) (('0' <= (c) && (c) <= '9') \
182 || ('a' <= (c) && (c) <= 'f') \
183 || ('A' <= (c) && (c) <= 'F'))
184#define isalnum(c) (('0' <= (c) && (c) <= '9') \
185 || ('a' <= (c) && (c) <= 'z') \
186 || ('A' <= (c) && (c) <= 'Z'))
187#define isspace(c) (c == ' ' || c == '\t' || c == 10 || c == 13 || c == 0)
188
189static char *help_string = "\
190Commands:\n\
191 b show breakpoints\n\
192 bd set data breakpoint\n\
193 bi set instruction breakpoint\n\
194 bc clear breakpoint\n"
195#ifdef CONFIG_SMP
196 "\
197 c print cpus stopped in xmon\n\
198 c# try to switch to cpu number h (in hex)\n"
199#endif
200 "\
201 C checksum\n\
202 d dump bytes\n\
203 di dump instructions\n\
204 df dump float values\n\
205 dd dump double values\n\
Michael Ellermanddadb6b2012-09-13 23:01:31 +0000206 dl dump the kernel log buffer\n"
207#ifdef CONFIG_PPC64
208 "\
209 dp[#] dump paca for current cpu, or cpu #\n\
210 dpa dump paca for all possible cpus\n"
211#endif
212 "\
Olaf Hering7e5b5932006-03-08 20:40:28 +0100213 dr dump stream of raw bytes\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700214 e print exception information\n\
215 f flush cache\n\
216 la lookup symbol+offset of specified address\n\
217 ls lookup address of specified symbol\n\
218 m examine/change memory\n\
219 mm move a block of memory\n\
220 ms set a block of memory\n\
221 md compare two blocks of memory\n\
222 ml locate a block of memory\n\
223 mz zero a block of memory\n\
224 mi show information about memory allocation\n\
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000225 p call a procedure\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700226 r print registers\n\
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200227 s single step\n"
Arnd Bergmanne0555952006-11-27 19:18:55 +0100228#ifdef CONFIG_SPU_BASE
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200229" ss stop execution on all spus\n\
Michael Ellermana8984972006-10-24 18:31:28 +0200230 sr restore execution on stopped spus\n\
Michael Ellerman24a24c82006-11-23 00:46:41 +0100231 sf # dump spu fields for spu # (in hex)\n\
Michael Ellermanc99176a2007-02-26 18:14:06 +0900232 sd # dump spu local store for spu # (in hex)\n\
Michael Ellermanaf89fb82006-11-23 00:46:44 +0100233 sdi # disassemble spu local store for spu # (in hex)\n"
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200234#endif
235" S print special registers\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700236 t print backtrace\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700237 x exit monitor and recover\n\
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000238 X exit monitor and dont recover\n"
Jimi Xenidis79873e82011-09-29 11:25:10 +0000239#if defined(CONFIG_PPC64) && !defined(CONFIG_PPC_BOOK3E)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000240" u dump segment table or SLB\n"
Jimi Xenidis79873e82011-09-29 11:25:10 +0000241#elif defined(CONFIG_PPC_STD_MMU_32)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000242" u dump segment registers\n"
Jimi Xenidis79873e82011-09-29 11:25:10 +0000243#elif defined(CONFIG_44x) || defined(CONFIG_PPC_BOOK3E)
Benjamin Herrenschmidt5a8a1a22007-11-16 18:23:33 +1100244" u dump TLB\n"
245#endif
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000246" ? help\n"
247" zr reboot\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700248 zh halt\n"
249;
250
251static struct pt_regs *xmon_regs;
252
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000253static inline void sync(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700254{
255 asm volatile("sync; isync");
256}
257
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000258static inline void store_inst(void *p)
259{
260 asm volatile ("dcbst 0,%0; sync; icbi 0,%0; isync" : : "r" (p));
261}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700262
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000263static inline void cflush(void *p)
264{
265 asm volatile ("dcbf 0,%0; icbi 0,%0" : : "r" (p));
266}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700267
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000268static inline void cinval(void *p)
269{
270 asm volatile ("dcbi 0,%0; icbi 0,%0" : : "r" (p));
271}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700272
273/*
274 * Disable surveillance (the service processor watchdog function)
275 * while we are in xmon.
276 * XXX we should re-enable it when we leave. :)
277 */
278#define SURVEILLANCE_TOKEN 9000
279
280static inline void disable_surveillance(void)
281{
282#ifdef CONFIG_PPC_PSERIES
283 /* Since this can't be a module, args should end up below 4GB. */
284 static struct rtas_args args;
285
286 /*
287 * At this point we have got all the cpus we can into
288 * xmon, so there is hopefully no other cpu calling RTAS
289 * at the moment, even though we don't take rtas.lock.
290 * If we did try to take rtas.lock there would be a
291 * real possibility of deadlock.
292 */
293 args.token = rtas_token("set-indicator");
294 if (args.token == RTAS_UNKNOWN_SERVICE)
295 return;
296 args.nargs = 3;
297 args.nret = 1;
298 args.rets = &args.args[3];
299 args.args[0] = SURVEILLANCE_TOKEN;
300 args.args[1] = 0;
301 args.args[2] = 0;
302 enter_rtas(__pa(&args));
303#endif /* CONFIG_PPC_PSERIES */
304}
305
306#ifdef CONFIG_SMP
307static int xmon_speaker;
308
309static void get_output_lock(void)
310{
311 int me = smp_processor_id() + 0x100;
312 int last_speaker = 0, prev;
313 long timeout;
314
315 if (xmon_speaker == me)
316 return;
Michael Ellerman730efb62013-12-23 23:46:04 +1100317
Linus Torvalds1da177e2005-04-16 15:20:36 -0700318 for (;;) {
Michael Ellerman730efb62013-12-23 23:46:04 +1100319 last_speaker = cmpxchg(&xmon_speaker, 0, me);
320 if (last_speaker == 0)
321 return;
322
Michael Ellerman15075892013-12-23 23:46:05 +1100323 /*
324 * Wait a full second for the lock, we might be on a slow
325 * console, but check every 100us.
326 */
327 timeout = 10000;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700328 while (xmon_speaker == last_speaker) {
Michael Ellerman15075892013-12-23 23:46:05 +1100329 if (--timeout > 0) {
330 udelay(100);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700331 continue;
Michael Ellerman15075892013-12-23 23:46:05 +1100332 }
333
Linus Torvalds1da177e2005-04-16 15:20:36 -0700334 /* hostile takeover */
335 prev = cmpxchg(&xmon_speaker, last_speaker, me);
336 if (prev == last_speaker)
337 return;
338 break;
339 }
340 }
341}
342
343static void release_output_lock(void)
344{
345 xmon_speaker = 0;
346}
Michael Ellerman1c8950f2008-05-08 14:27:17 +1000347
348int cpus_are_in_xmon(void)
349{
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000350 return !cpumask_empty(&cpus_in_xmon);
Michael Ellerman1c8950f2008-05-08 14:27:17 +1000351}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700352#endif
353
Josh Boyerdaf8f402009-09-23 03:51:04 +0000354static inline int unrecoverable_excp(struct pt_regs *regs)
355{
Jimi Xenidis08f6d6a2011-09-29 12:05:28 +0000356#if defined(CONFIG_4xx) || defined(CONFIG_PPC_BOOK3E)
Jimi Xenidis66857b32011-09-23 05:40:46 +0000357 /* We have no MSR_RI bit on 4xx or Book3e, so we simply return false */
Josh Boyerdaf8f402009-09-23 03:51:04 +0000358 return 0;
359#else
360 return ((regs->msr & MSR_RI) == 0);
361#endif
362}
363
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000364static int xmon_core(struct pt_regs *regs, int fromipi)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700365{
366 int cmd = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700367 struct bpt *bp;
368 long recurse_jmp[JMP_BUF_LEN];
369 unsigned long offset;
Anton Blanchardf13659e2007-03-21 01:48:34 +1100370 unsigned long flags;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700371#ifdef CONFIG_SMP
372 int cpu;
373 int secondary;
374 unsigned long timeout;
375#endif
376
Anton Blanchardf13659e2007-03-21 01:48:34 +1100377 local_irq_save(flags);
Anton Blancharda71d64b2014-08-05 14:55:00 +1000378 hard_irq_disable();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700379
380 bp = in_breakpoint_table(regs->nip, &offset);
381 if (bp != NULL) {
382 regs->nip = bp->address + offset;
383 atomic_dec(&bp->ref_count);
384 }
385
386 remove_cpu_bpts();
387
388#ifdef CONFIG_SMP
389 cpu = smp_processor_id();
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000390 if (cpumask_test_cpu(cpu, &cpus_in_xmon)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700391 get_output_lock();
392 excprint(regs);
393 printf("cpu 0x%x: Exception %lx %s in xmon, "
394 "returning to main loop\n",
395 cpu, regs->trap, getvecname(TRAP(regs)));
Haren Myneni5cb4cc02005-08-03 15:08:18 +1000396 release_output_lock();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700397 longjmp(xmon_fault_jmp[cpu], 1);
398 }
399
400 if (setjmp(recurse_jmp) != 0) {
401 if (!in_xmon || !xmon_gate) {
Haren Myneni5cb4cc02005-08-03 15:08:18 +1000402 get_output_lock();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700403 printf("xmon: WARNING: bad recursive fault "
404 "on cpu 0x%x\n", cpu);
Haren Myneni5cb4cc02005-08-03 15:08:18 +1000405 release_output_lock();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700406 goto waiting;
407 }
408 secondary = !(xmon_taken && cpu == xmon_owner);
409 goto cmdloop;
410 }
411
412 xmon_fault_jmp[cpu] = recurse_jmp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700413
414 bp = NULL;
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000415 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) == (MSR_IR|MSR_64BIT))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700416 bp = at_breakpoint(regs->nip);
Josh Boyerdaf8f402009-09-23 03:51:04 +0000417 if (bp || unrecoverable_excp(regs))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700418 fromipi = 0;
419
420 if (!fromipi) {
421 get_output_lock();
422 excprint(regs);
423 if (bp) {
Michael Ellerman736256e2014-05-26 21:02:14 +1000424 printf("cpu 0x%x stopped at breakpoint 0x%lx (",
Linus Torvalds1da177e2005-04-16 15:20:36 -0700425 cpu, BP_NUM(bp));
426 xmon_print_symbol(regs->nip, " ", ")\n");
427 }
Josh Boyerdaf8f402009-09-23 03:51:04 +0000428 if (unrecoverable_excp(regs))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700429 printf("WARNING: exception is not recoverable, "
430 "can't continue\n");
431 release_output_lock();
432 }
433
Michael Ellermand2b496e2013-12-23 23:46:06 +1100434 cpumask_set_cpu(cpu, &cpus_in_xmon);
435
Linus Torvalds1da177e2005-04-16 15:20:36 -0700436 waiting:
437 secondary = 1;
438 while (secondary && !xmon_gate) {
439 if (in_xmon == 0) {
440 if (fromipi)
441 goto leave;
442 secondary = test_and_set_bit(0, &in_xmon);
443 }
444 barrier();
445 }
446
447 if (!secondary && !xmon_gate) {
448 /* we are the first cpu to come in */
449 /* interrupt other cpu(s) */
450 int ncpus = num_online_cpus();
451
452 xmon_owner = cpu;
453 mb();
454 if (ncpus > 1) {
Milton Millere0476372011-05-10 19:29:06 +0000455 smp_send_debugger_break();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700456 /* wait for other cpus to come in */
457 for (timeout = 100000000; timeout != 0; --timeout) {
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000458 if (cpumask_weight(&cpus_in_xmon) >= ncpus)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700459 break;
460 barrier();
461 }
462 }
463 remove_bpts();
464 disable_surveillance();
465 /* for breakpoint or single step, print the current instr. */
466 if (bp || TRAP(regs) == 0xd00)
467 ppc_inst_dump(regs->nip, 1, 0);
468 printf("enter ? for help\n");
469 mb();
470 xmon_gate = 1;
471 barrier();
472 }
473
474 cmdloop:
475 while (in_xmon) {
476 if (secondary) {
477 if (cpu == xmon_owner) {
478 if (!test_and_set_bit(0, &xmon_taken)) {
479 secondary = 0;
480 continue;
481 }
482 /* missed it */
483 while (cpu == xmon_owner)
484 barrier();
485 }
486 barrier();
487 } else {
488 cmd = cmds(regs);
489 if (cmd != 0) {
490 /* exiting xmon */
491 insert_bpts();
492 xmon_gate = 0;
493 wmb();
494 in_xmon = 0;
495 break;
496 }
497 /* have switched to some other cpu */
498 secondary = 1;
499 }
500 }
501 leave:
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000502 cpumask_clear_cpu(cpu, &cpus_in_xmon);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700503 xmon_fault_jmp[cpu] = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700504#else
505 /* UP is simple... */
506 if (in_xmon) {
507 printf("Exception %lx %s in xmon, returning to main loop\n",
508 regs->trap, getvecname(TRAP(regs)));
509 longjmp(xmon_fault_jmp[0], 1);
510 }
511 if (setjmp(recurse_jmp) == 0) {
512 xmon_fault_jmp[0] = recurse_jmp;
513 in_xmon = 1;
514
515 excprint(regs);
516 bp = at_breakpoint(regs->nip);
517 if (bp) {
Michael Ellerman736256e2014-05-26 21:02:14 +1000518 printf("Stopped at breakpoint %lx (", BP_NUM(bp));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700519 xmon_print_symbol(regs->nip, " ", ")\n");
520 }
Josh Boyerdaf8f402009-09-23 03:51:04 +0000521 if (unrecoverable_excp(regs))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700522 printf("WARNING: exception is not recoverable, "
523 "can't continue\n");
524 remove_bpts();
525 disable_surveillance();
526 /* for breakpoint or single step, print the current instr. */
527 if (bp || TRAP(regs) == 0xd00)
528 ppc_inst_dump(regs->nip, 1, 0);
529 printf("enter ? for help\n");
530 }
531
532 cmd = cmds(regs);
533
534 insert_bpts();
535 in_xmon = 0;
536#endif
537
Josh Boyercdd39042009-10-05 04:46:05 +0000538#ifdef CONFIG_BOOKE
539 if (regs->msr & MSR_DE) {
540 bp = at_breakpoint(regs->nip);
541 if (bp != NULL) {
542 regs->nip = (unsigned long) &bp->instr[0];
543 atomic_inc(&bp->ref_count);
544 }
545 }
546#else
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000547 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) == (MSR_IR|MSR_64BIT)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700548 bp = at_breakpoint(regs->nip);
549 if (bp != NULL) {
550 int stepped = emulate_step(regs, bp->instr[0]);
551 if (stepped == 0) {
552 regs->nip = (unsigned long) &bp->instr[0];
553 atomic_inc(&bp->ref_count);
554 } else if (stepped < 0) {
555 printf("Couldn't single-step %s instruction\n",
556 (IS_RFID(bp->instr[0])? "rfid": "mtmsrd"));
557 }
558 }
559 }
Josh Boyercdd39042009-10-05 04:46:05 +0000560#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700561 insert_cpu_bpts();
562
Anton Blancharda71d64b2014-08-05 14:55:00 +1000563 touch_nmi_watchdog();
Anton Blanchardf13659e2007-03-21 01:48:34 +1100564 local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700565
Paul Mackerras0a730ae2006-10-03 21:32:49 +1000566 return cmd != 'X' && cmd != EOF;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700567}
568
569int xmon(struct pt_regs *excp)
570{
571 struct pt_regs regs;
572
573 if (excp == NULL) {
Anton Vorontsov322b4392008-12-17 10:08:55 +0000574 ppc_save_regs(&regs);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700575 excp = &regs;
576 }
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200577
Linus Torvalds1da177e2005-04-16 15:20:36 -0700578 return xmon_core(excp, 0);
579}
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000580EXPORT_SYMBOL(xmon);
581
Paul Mackerrasf583ffc2006-10-10 11:47:07 +1000582irqreturn_t xmon_irq(int irq, void *d)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000583{
584 unsigned long flags;
585 local_irq_save(flags);
586 printf("Keyboard interrupt\n");
Paul Mackerrasf583ffc2006-10-10 11:47:07 +1000587 xmon(get_irq_regs());
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000588 local_irq_restore(flags);
589 return IRQ_HANDLED;
590}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700591
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000592static int xmon_bpt(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700593{
594 struct bpt *bp;
595 unsigned long offset;
596
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000597 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) != (MSR_IR|MSR_64BIT))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700598 return 0;
599
600 /* Are we at the trap at bp->instr[1] for some bp? */
601 bp = in_breakpoint_table(regs->nip, &offset);
602 if (bp != NULL && offset == 4) {
603 regs->nip = bp->address + 4;
604 atomic_dec(&bp->ref_count);
605 return 1;
606 }
607
608 /* Are we at a breakpoint? */
609 bp = at_breakpoint(regs->nip);
610 if (!bp)
611 return 0;
612
613 xmon_core(regs, 0);
614
615 return 1;
616}
617
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000618static int xmon_sstep(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700619{
620 if (user_mode(regs))
621 return 0;
622 xmon_core(regs, 0);
623 return 1;
624}
625
Michael Neuling9422de32012-12-20 14:06:44 +0000626static int xmon_break_match(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700627{
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000628 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) != (MSR_IR|MSR_64BIT))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700629 return 0;
Anton Blanchardfd9648d2005-09-10 16:01:11 +1000630 if (dabr.enabled == 0)
631 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700632 xmon_core(regs, 0);
633 return 1;
634}
635
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000636static int xmon_iabr_match(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700637{
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000638 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) != (MSR_IR|MSR_64BIT))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700639 return 0;
Michael Ellerman9f1067c22008-05-08 14:27:16 +1000640 if (iabr == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700641 return 0;
642 xmon_core(regs, 0);
643 return 1;
644}
645
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000646static int xmon_ipi(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700647{
648#ifdef CONFIG_SMP
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000649 if (in_xmon && !cpumask_test_cpu(smp_processor_id(), &cpus_in_xmon))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700650 xmon_core(regs, 1);
651#endif
652 return 0;
653}
654
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000655static int xmon_fault_handler(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700656{
657 struct bpt *bp;
658 unsigned long offset;
659
660 if (in_xmon && catch_memory_errors)
661 handle_fault(regs); /* doesn't return */
662
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000663 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) == (MSR_IR|MSR_64BIT)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700664 bp = in_breakpoint_table(regs->nip, &offset);
665 if (bp != NULL) {
666 regs->nip = bp->address + offset;
667 atomic_dec(&bp->ref_count);
668 }
669 }
670
671 return 0;
672}
673
Linus Torvalds1da177e2005-04-16 15:20:36 -0700674static struct bpt *at_breakpoint(unsigned long pc)
675{
676 int i;
677 struct bpt *bp;
678
679 bp = bpts;
680 for (i = 0; i < NBPTS; ++i, ++bp)
681 if (bp->enabled && pc == bp->address)
682 return bp;
683 return NULL;
684}
685
686static struct bpt *in_breakpoint_table(unsigned long nip, unsigned long *offp)
687{
688 unsigned long off;
689
690 off = nip - (unsigned long) bpts;
691 if (off >= sizeof(bpts))
692 return NULL;
693 off %= sizeof(struct bpt);
694 if (off != offsetof(struct bpt, instr[0])
695 && off != offsetof(struct bpt, instr[1]))
696 return NULL;
697 *offp = off - offsetof(struct bpt, instr[0]);
698 return (struct bpt *) (nip - off);
699}
700
701static struct bpt *new_breakpoint(unsigned long a)
702{
703 struct bpt *bp;
704
705 a &= ~3UL;
706 bp = at_breakpoint(a);
707 if (bp)
708 return bp;
709
710 for (bp = bpts; bp < &bpts[NBPTS]; ++bp) {
711 if (!bp->enabled && atomic_read(&bp->ref_count) == 0) {
712 bp->address = a;
713 bp->instr[1] = bpinstr;
714 store_inst(&bp->instr[1]);
715 return bp;
716 }
717 }
718
719 printf("Sorry, no free breakpoints. Please clear one first.\n");
720 return NULL;
721}
722
723static void insert_bpts(void)
724{
725 int i;
726 struct bpt *bp;
727
728 bp = bpts;
729 for (i = 0; i < NBPTS; ++i, ++bp) {
730 if ((bp->enabled & (BP_TRAP|BP_IABR)) == 0)
731 continue;
732 if (mread(bp->address, &bp->instr[0], 4) != 4) {
733 printf("Couldn't read instruction at %lx, "
734 "disabling breakpoint there\n", bp->address);
735 bp->enabled = 0;
736 continue;
737 }
738 if (IS_MTMSRD(bp->instr[0]) || IS_RFID(bp->instr[0])) {
739 printf("Breakpoint at %lx is on an mtmsrd or rfid "
740 "instruction, disabling it\n", bp->address);
741 bp->enabled = 0;
742 continue;
743 }
744 store_inst(&bp->instr[0]);
745 if (bp->enabled & BP_IABR)
746 continue;
747 if (mwrite(bp->address, &bpinstr, 4) != 4) {
748 printf("Couldn't write instruction at %lx, "
749 "disabling breakpoint there\n", bp->address);
750 bp->enabled &= ~BP_TRAP;
751 continue;
752 }
753 store_inst((void *)bp->address);
754 }
755}
756
757static void insert_cpu_bpts(void)
758{
Michael Neuling9422de32012-12-20 14:06:44 +0000759 struct arch_hw_breakpoint brk;
760
761 if (dabr.enabled) {
762 brk.address = dabr.address;
763 brk.type = (dabr.enabled & HW_BRK_TYPE_DABR) | HW_BRK_TYPE_PRIV_ALL;
764 brk.len = 8;
Paul Gortmaker21f58502014-04-29 15:25:17 -0400765 __set_breakpoint(&brk);
Michael Neuling9422de32012-12-20 14:06:44 +0000766 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700767 if (iabr && cpu_has_feature(CPU_FTR_IABR))
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000768 mtspr(SPRN_IABR, iabr->address
Linus Torvalds1da177e2005-04-16 15:20:36 -0700769 | (iabr->enabled & (BP_IABR|BP_IABR_TE)));
770}
771
772static void remove_bpts(void)
773{
774 int i;
775 struct bpt *bp;
776 unsigned instr;
777
778 bp = bpts;
779 for (i = 0; i < NBPTS; ++i, ++bp) {
780 if ((bp->enabled & (BP_TRAP|BP_IABR)) != BP_TRAP)
781 continue;
782 if (mread(bp->address, &instr, 4) == 4
783 && instr == bpinstr
784 && mwrite(bp->address, &bp->instr, 4) != 4)
785 printf("Couldn't remove breakpoint at %lx\n",
786 bp->address);
787 else
788 store_inst((void *)bp->address);
789 }
790}
791
792static void remove_cpu_bpts(void)
793{
Michael Neuling9422de32012-12-20 14:06:44 +0000794 hw_breakpoint_disable();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700795 if (cpu_has_feature(CPU_FTR_IABR))
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000796 mtspr(SPRN_IABR, 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700797}
798
799/* Command interpreting routine */
800static char *last_cmd;
801
802static int
803cmds(struct pt_regs *excp)
804{
805 int cmd = 0;
806
807 last_cmd = NULL;
808 xmon_regs = excp;
Olaf Hering26c8af52006-09-08 16:29:21 +0200809
810 if (!xmon_no_auto_backtrace) {
811 xmon_no_auto_backtrace = 1;
812 xmon_show_stack(excp->gpr[1], excp->link, excp->nip);
813 }
814
Linus Torvalds1da177e2005-04-16 15:20:36 -0700815 for(;;) {
816#ifdef CONFIG_SMP
817 printf("%x:", smp_processor_id());
818#endif /* CONFIG_SMP */
819 printf("mon> ");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700820 flush_input();
821 termch = 0;
822 cmd = skipbl();
823 if( cmd == '\n' ) {
824 if (last_cmd == NULL)
825 continue;
826 take_input(last_cmd);
827 last_cmd = NULL;
828 cmd = inchar();
829 }
830 switch (cmd) {
831 case 'm':
832 cmd = inchar();
833 switch (cmd) {
834 case 'm':
835 case 's':
836 case 'd':
837 memops(cmd);
838 break;
839 case 'l':
840 memlocate();
841 break;
842 case 'z':
843 memzcan();
844 break;
845 case 'i':
David Rientjesb2b755b2011-03-24 15:18:15 -0700846 show_mem(0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700847 break;
848 default:
849 termch = cmd;
850 memex();
851 }
852 break;
853 case 'd':
854 dump();
855 break;
856 case 'l':
857 symbol_lookup();
858 break;
859 case 'r':
860 prregs(excp); /* print regs */
861 break;
862 case 'e':
863 excprint(excp);
864 break;
865 case 'S':
866 super_regs();
867 break;
868 case 't':
869 backtrace(excp);
870 break;
871 case 'f':
872 cacheflush();
873 break;
874 case 's':
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200875 if (do_spu_cmd() == 0)
876 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700877 if (do_step(excp))
878 return cmd;
879 break;
880 case 'x':
881 case 'X':
Benjamin Herrenschmidtbb6b9b22005-11-30 16:54:12 +1100882 return cmd;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700883 case EOF:
Benjamin Herrenschmidtbb6b9b22005-11-30 16:54:12 +1100884 printf(" <no input ...>\n");
885 mdelay(2000);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700886 return cmd;
887 case '?':
Ishizaki Kou4d404ed2007-07-18 19:26:40 +1000888 xmon_puts(help_string);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700889 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700890 case 'b':
891 bpt_cmds();
892 break;
893 case 'C':
894 csum();
895 break;
896 case 'c':
897 if (cpu_cmd())
898 return 0;
899 break;
900 case 'z':
901 bootcmds();
902 break;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000903 case 'p':
904 proccall();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700905 break;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000906#ifdef CONFIG_PPC_STD_MMU
Linus Torvalds1da177e2005-04-16 15:20:36 -0700907 case 'u':
908 dump_segments();
909 break;
Michael Ellermand8ee6f32014-11-12 16:54:54 +1100910#elif defined(CONFIG_44x)
Benjamin Herrenschmidt5a8a1a22007-11-16 18:23:33 +1100911 case 'u':
912 dump_tlb_44x();
913 break;
Jimi Xenidis79873e82011-09-29 11:25:10 +0000914#elif defined(CONFIG_PPC_BOOK3E)
Benjamin Herrenschmidt03247152010-07-09 15:34:50 +1000915 case 'u':
916 dump_tlb_book3e();
917 break;
918#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700919 default:
920 printf("Unrecognized command: ");
Michael Ellermane3bc8042012-08-23 22:09:13 +0000921 do {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700922 if (' ' < cmd && cmd <= '~')
923 putchar(cmd);
924 else
925 printf("\\x%x", cmd);
926 cmd = inchar();
Michael Ellermane3bc8042012-08-23 22:09:13 +0000927 } while (cmd != '\n');
Linus Torvalds1da177e2005-04-16 15:20:36 -0700928 printf(" (type ? for help)\n");
929 break;
930 }
931 }
932}
933
Josh Boyercdd39042009-10-05 04:46:05 +0000934#ifdef CONFIG_BOOKE
935static int do_step(struct pt_regs *regs)
936{
937 regs->msr |= MSR_DE;
938 mtspr(SPRN_DBCR0, mfspr(SPRN_DBCR0) | DBCR0_IC | DBCR0_IDM);
939 return 1;
940}
941#else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700942/*
943 * Step a single instruction.
944 * Some instructions we emulate, others we execute with MSR_SE set.
945 */
946static int do_step(struct pt_regs *regs)
947{
948 unsigned int instr;
949 int stepped;
950
951 /* check we are in 64-bit kernel mode, translation enabled */
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000952 if ((regs->msr & (MSR_64BIT|MSR_PR|MSR_IR)) == (MSR_64BIT|MSR_IR)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700953 if (mread(regs->nip, &instr, 4) == 4) {
954 stepped = emulate_step(regs, instr);
955 if (stepped < 0) {
956 printf("Couldn't single-step %s instruction\n",
957 (IS_RFID(instr)? "rfid": "mtmsrd"));
958 return 0;
959 }
960 if (stepped > 0) {
961 regs->trap = 0xd00 | (regs->trap & 1);
962 printf("stepped to ");
963 xmon_print_symbol(regs->nip, " ", "\n");
964 ppc_inst_dump(regs->nip, 1, 0);
965 return 0;
966 }
967 }
968 }
969 regs->msr |= MSR_SE;
970 return 1;
971}
Josh Boyercdd39042009-10-05 04:46:05 +0000972#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700973
974static void bootcmds(void)
975{
976 int cmd;
977
978 cmd = inchar();
979 if (cmd == 'r')
980 ppc_md.restart(NULL);
981 else if (cmd == 'h')
982 ppc_md.halt();
983 else if (cmd == 'p')
Alexander Graf9178ba22014-10-13 16:01:09 +0200984 if (pm_power_off)
985 pm_power_off();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700986}
987
988static int cpu_cmd(void)
989{
990#ifdef CONFIG_SMP
Paul Mackerrasfd3bb912013-09-03 20:16:23 +1000991 unsigned long cpu, first_cpu, last_cpu;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700992 int timeout;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700993
994 if (!scanhex(&cpu)) {
995 /* print cpus waiting or in xmon */
996 printf("cpus stopped:");
Paul Mackerrasfd3bb912013-09-03 20:16:23 +1000997 last_cpu = first_cpu = NR_CPUS;
Anton Blanchardbc1d7702012-06-28 19:28:57 +0000998 for_each_possible_cpu(cpu) {
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000999 if (cpumask_test_cpu(cpu, &cpus_in_xmon)) {
Paul Mackerrasfd3bb912013-09-03 20:16:23 +10001000 if (cpu == last_cpu + 1) {
1001 last_cpu = cpu;
1002 } else {
1003 if (last_cpu != first_cpu)
Michael Ellerman736256e2014-05-26 21:02:14 +10001004 printf("-0x%lx", last_cpu);
Paul Mackerrasfd3bb912013-09-03 20:16:23 +10001005 last_cpu = first_cpu = cpu;
Michael Ellerman736256e2014-05-26 21:02:14 +10001006 printf(" 0x%lx", cpu);
Paul Mackerrasfd3bb912013-09-03 20:16:23 +10001007 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001008 }
1009 }
Paul Mackerrasfd3bb912013-09-03 20:16:23 +10001010 if (last_cpu != first_cpu)
Michael Ellerman736256e2014-05-26 21:02:14 +10001011 printf("-0x%lx", last_cpu);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001012 printf("\n");
1013 return 0;
1014 }
1015 /* try to switch to cpu specified */
KOSAKI Motohiro104699c2011-04-28 05:07:23 +00001016 if (!cpumask_test_cpu(cpu, &cpus_in_xmon)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001017 printf("cpu 0x%x isn't in xmon\n", cpu);
1018 return 0;
1019 }
1020 xmon_taken = 0;
1021 mb();
1022 xmon_owner = cpu;
1023 timeout = 10000000;
1024 while (!xmon_taken) {
1025 if (--timeout == 0) {
1026 if (test_and_set_bit(0, &xmon_taken))
1027 break;
1028 /* take control back */
1029 mb();
1030 xmon_owner = smp_processor_id();
Michael Ellerman736256e2014-05-26 21:02:14 +10001031 printf("cpu 0x%x didn't take control\n", cpu);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001032 return 0;
1033 }
1034 barrier();
1035 }
1036 return 1;
1037#else
1038 return 0;
1039#endif /* CONFIG_SMP */
1040}
1041
1042static unsigned short fcstab[256] = {
1043 0x0000, 0x1189, 0x2312, 0x329b, 0x4624, 0x57ad, 0x6536, 0x74bf,
1044 0x8c48, 0x9dc1, 0xaf5a, 0xbed3, 0xca6c, 0xdbe5, 0xe97e, 0xf8f7,
1045 0x1081, 0x0108, 0x3393, 0x221a, 0x56a5, 0x472c, 0x75b7, 0x643e,
1046 0x9cc9, 0x8d40, 0xbfdb, 0xae52, 0xdaed, 0xcb64, 0xf9ff, 0xe876,
1047 0x2102, 0x308b, 0x0210, 0x1399, 0x6726, 0x76af, 0x4434, 0x55bd,
1048 0xad4a, 0xbcc3, 0x8e58, 0x9fd1, 0xeb6e, 0xfae7, 0xc87c, 0xd9f5,
1049 0x3183, 0x200a, 0x1291, 0x0318, 0x77a7, 0x662e, 0x54b5, 0x453c,
1050 0xbdcb, 0xac42, 0x9ed9, 0x8f50, 0xfbef, 0xea66, 0xd8fd, 0xc974,
1051 0x4204, 0x538d, 0x6116, 0x709f, 0x0420, 0x15a9, 0x2732, 0x36bb,
1052 0xce4c, 0xdfc5, 0xed5e, 0xfcd7, 0x8868, 0x99e1, 0xab7a, 0xbaf3,
1053 0x5285, 0x430c, 0x7197, 0x601e, 0x14a1, 0x0528, 0x37b3, 0x263a,
1054 0xdecd, 0xcf44, 0xfddf, 0xec56, 0x98e9, 0x8960, 0xbbfb, 0xaa72,
1055 0x6306, 0x728f, 0x4014, 0x519d, 0x2522, 0x34ab, 0x0630, 0x17b9,
1056 0xef4e, 0xfec7, 0xcc5c, 0xddd5, 0xa96a, 0xb8e3, 0x8a78, 0x9bf1,
1057 0x7387, 0x620e, 0x5095, 0x411c, 0x35a3, 0x242a, 0x16b1, 0x0738,
1058 0xffcf, 0xee46, 0xdcdd, 0xcd54, 0xb9eb, 0xa862, 0x9af9, 0x8b70,
1059 0x8408, 0x9581, 0xa71a, 0xb693, 0xc22c, 0xd3a5, 0xe13e, 0xf0b7,
1060 0x0840, 0x19c9, 0x2b52, 0x3adb, 0x4e64, 0x5fed, 0x6d76, 0x7cff,
1061 0x9489, 0x8500, 0xb79b, 0xa612, 0xd2ad, 0xc324, 0xf1bf, 0xe036,
1062 0x18c1, 0x0948, 0x3bd3, 0x2a5a, 0x5ee5, 0x4f6c, 0x7df7, 0x6c7e,
1063 0xa50a, 0xb483, 0x8618, 0x9791, 0xe32e, 0xf2a7, 0xc03c, 0xd1b5,
1064 0x2942, 0x38cb, 0x0a50, 0x1bd9, 0x6f66, 0x7eef, 0x4c74, 0x5dfd,
1065 0xb58b, 0xa402, 0x9699, 0x8710, 0xf3af, 0xe226, 0xd0bd, 0xc134,
1066 0x39c3, 0x284a, 0x1ad1, 0x0b58, 0x7fe7, 0x6e6e, 0x5cf5, 0x4d7c,
1067 0xc60c, 0xd785, 0xe51e, 0xf497, 0x8028, 0x91a1, 0xa33a, 0xb2b3,
1068 0x4a44, 0x5bcd, 0x6956, 0x78df, 0x0c60, 0x1de9, 0x2f72, 0x3efb,
1069 0xd68d, 0xc704, 0xf59f, 0xe416, 0x90a9, 0x8120, 0xb3bb, 0xa232,
1070 0x5ac5, 0x4b4c, 0x79d7, 0x685e, 0x1ce1, 0x0d68, 0x3ff3, 0x2e7a,
1071 0xe70e, 0xf687, 0xc41c, 0xd595, 0xa12a, 0xb0a3, 0x8238, 0x93b1,
1072 0x6b46, 0x7acf, 0x4854, 0x59dd, 0x2d62, 0x3ceb, 0x0e70, 0x1ff9,
1073 0xf78f, 0xe606, 0xd49d, 0xc514, 0xb1ab, 0xa022, 0x92b9, 0x8330,
1074 0x7bc7, 0x6a4e, 0x58d5, 0x495c, 0x3de3, 0x2c6a, 0x1ef1, 0x0f78
1075};
1076
1077#define FCS(fcs, c) (((fcs) >> 8) ^ fcstab[((fcs) ^ (c)) & 0xff])
1078
1079static void
1080csum(void)
1081{
1082 unsigned int i;
1083 unsigned short fcs;
1084 unsigned char v;
1085
1086 if (!scanhex(&adrs))
1087 return;
1088 if (!scanhex(&ncsum))
1089 return;
1090 fcs = 0xffff;
1091 for (i = 0; i < ncsum; ++i) {
1092 if (mread(adrs+i, &v, 1) == 0) {
Michael Ellerman736256e2014-05-26 21:02:14 +10001093 printf("csum stopped at "REG"\n", adrs+i);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001094 break;
1095 }
1096 fcs = FCS(fcs, v);
1097 }
1098 printf("%x\n", fcs);
1099}
1100
1101/*
1102 * Check if this is a suitable place to put a breakpoint.
1103 */
1104static long check_bp_loc(unsigned long addr)
1105{
1106 unsigned int instr;
1107
1108 addr &= ~3;
Michael Ellerman51fae6de2005-12-04 18:39:15 +11001109 if (!is_kernel_addr(addr)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001110 printf("Breakpoints may only be placed at kernel addresses\n");
1111 return 0;
1112 }
1113 if (!mread(addr, &instr, sizeof(instr))) {
1114 printf("Can't read instruction at address %lx\n", addr);
1115 return 0;
1116 }
1117 if (IS_MTMSRD(instr) || IS_RFID(instr)) {
1118 printf("Breakpoints may not be placed on mtmsrd or rfid "
1119 "instructions\n");
1120 return 0;
1121 }
1122 return 1;
1123}
1124
Michael Ellermane3bc8042012-08-23 22:09:13 +00001125static char *breakpoint_help_string =
Linus Torvalds1da177e2005-04-16 15:20:36 -07001126 "Breakpoint command usage:\n"
1127 "b show breakpoints\n"
1128 "b <addr> [cnt] set breakpoint at given instr addr\n"
1129 "bc clear all breakpoints\n"
1130 "bc <n/addr> clear breakpoint number n or at addr\n"
1131 "bi <addr> [cnt] set hardware instr breakpoint (POWER3/RS64 only)\n"
1132 "bd <addr> [cnt] set hardware data breakpoint\n"
1133 "";
1134
1135static void
1136bpt_cmds(void)
1137{
1138 int cmd;
1139 unsigned long a;
1140 int mode, i;
1141 struct bpt *bp;
1142 const char badaddr[] = "Only kernel addresses are permitted "
1143 "for breakpoints\n";
1144
1145 cmd = inchar();
1146 switch (cmd) {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001147#ifndef CONFIG_8xx
Linus Torvalds1da177e2005-04-16 15:20:36 -07001148 case 'd': /* bd - hardware data breakpoint */
1149 mode = 7;
1150 cmd = inchar();
1151 if (cmd == 'r')
1152 mode = 5;
1153 else if (cmd == 'w')
1154 mode = 6;
1155 else
1156 termch = cmd;
1157 dabr.address = 0;
1158 dabr.enabled = 0;
1159 if (scanhex(&dabr.address)) {
Michael Ellerman51fae6de2005-12-04 18:39:15 +11001160 if (!is_kernel_addr(dabr.address)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001161 printf(badaddr);
1162 break;
1163 }
Michael Neuling9422de32012-12-20 14:06:44 +00001164 dabr.address &= ~HW_BRK_TYPE_DABR;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001165 dabr.enabled = mode | BP_DABR;
1166 }
1167 break;
1168
1169 case 'i': /* bi - hardware instr breakpoint */
1170 if (!cpu_has_feature(CPU_FTR_IABR)) {
1171 printf("Hardware instruction breakpoint "
1172 "not supported on this cpu\n");
1173 break;
1174 }
1175 if (iabr) {
1176 iabr->enabled &= ~(BP_IABR | BP_IABR_TE);
1177 iabr = NULL;
1178 }
1179 if (!scanhex(&a))
1180 break;
1181 if (!check_bp_loc(a))
1182 break;
1183 bp = new_breakpoint(a);
1184 if (bp != NULL) {
1185 bp->enabled |= BP_IABR | BP_IABR_TE;
1186 iabr = bp;
1187 }
1188 break;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001189#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001190
1191 case 'c':
1192 if (!scanhex(&a)) {
1193 /* clear all breakpoints */
1194 for (i = 0; i < NBPTS; ++i)
1195 bpts[i].enabled = 0;
1196 iabr = NULL;
1197 dabr.enabled = 0;
1198 printf("All breakpoints cleared\n");
1199 break;
1200 }
1201
1202 if (a <= NBPTS && a >= 1) {
1203 /* assume a breakpoint number */
1204 bp = &bpts[a-1]; /* bp nums are 1 based */
1205 } else {
1206 /* assume a breakpoint address */
1207 bp = at_breakpoint(a);
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001208 if (bp == NULL) {
Michael Ellerman736256e2014-05-26 21:02:14 +10001209 printf("No breakpoint at %lx\n", a);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001210 break;
1211 }
1212 }
1213
Michael Ellerman736256e2014-05-26 21:02:14 +10001214 printf("Cleared breakpoint %lx (", BP_NUM(bp));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001215 xmon_print_symbol(bp->address, " ", ")\n");
1216 bp->enabled = 0;
1217 break;
1218
1219 default:
1220 termch = cmd;
Michael Ellermane3bc8042012-08-23 22:09:13 +00001221 cmd = skipbl();
Linus Torvalds1da177e2005-04-16 15:20:36 -07001222 if (cmd == '?') {
1223 printf(breakpoint_help_string);
1224 break;
1225 }
1226 termch = cmd;
1227 if (!scanhex(&a)) {
1228 /* print all breakpoints */
1229 printf(" type address\n");
1230 if (dabr.enabled) {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001231 printf(" data "REG" [", dabr.address);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001232 if (dabr.enabled & 1)
1233 printf("r");
1234 if (dabr.enabled & 2)
1235 printf("w");
1236 printf("]\n");
1237 }
1238 for (bp = bpts; bp < &bpts[NBPTS]; ++bp) {
1239 if (!bp->enabled)
1240 continue;
1241 printf("%2x %s ", BP_NUM(bp),
1242 (bp->enabled & BP_IABR)? "inst": "trap");
1243 xmon_print_symbol(bp->address, " ", "\n");
1244 }
1245 break;
1246 }
1247
1248 if (!check_bp_loc(a))
1249 break;
1250 bp = new_breakpoint(a);
1251 if (bp != NULL)
1252 bp->enabled |= BP_TRAP;
1253 break;
1254 }
1255}
1256
1257/* Very cheap human name for vector lookup. */
1258static
1259const char *getvecname(unsigned long vec)
1260{
1261 char *ret;
1262
1263 switch (vec) {
1264 case 0x100: ret = "(System Reset)"; break;
1265 case 0x200: ret = "(Machine Check)"; break;
1266 case 0x300: ret = "(Data Access)"; break;
1267 case 0x380: ret = "(Data SLB Access)"; break;
1268 case 0x400: ret = "(Instruction Access)"; break;
1269 case 0x480: ret = "(Instruction SLB Access)"; break;
1270 case 0x500: ret = "(Hardware Interrupt)"; break;
1271 case 0x600: ret = "(Alignment)"; break;
1272 case 0x700: ret = "(Program Check)"; break;
1273 case 0x800: ret = "(FPU Unavailable)"; break;
1274 case 0x900: ret = "(Decrementer)"; break;
Michael Ellerman660e0342013-08-15 15:22:16 +10001275 case 0x980: ret = "(Hypervisor Decrementer)"; break;
1276 case 0xa00: ret = "(Doorbell)"; break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001277 case 0xc00: ret = "(System Call)"; break;
1278 case 0xd00: ret = "(Single Step)"; break;
Michael Ellerman660e0342013-08-15 15:22:16 +10001279 case 0xe40: ret = "(Emulation Assist)"; break;
1280 case 0xe60: ret = "(HMI)"; break;
1281 case 0xe80: ret = "(Hypervisor Doorbell)"; break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001282 case 0xf00: ret = "(Performance Monitor)"; break;
1283 case 0xf20: ret = "(Altivec Unavailable)"; break;
1284 case 0x1300: ret = "(Instruction Breakpoint)"; break;
Michael Ellerman660e0342013-08-15 15:22:16 +10001285 case 0x1500: ret = "(Denormalisation)"; break;
1286 case 0x1700: ret = "(Altivec Assist)"; break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001287 default: ret = "";
1288 }
1289 return ret;
1290}
1291
1292static void get_function_bounds(unsigned long pc, unsigned long *startp,
1293 unsigned long *endp)
1294{
1295 unsigned long size, offset;
1296 const char *name;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001297
1298 *startp = *endp = 0;
1299 if (pc == 0)
1300 return;
1301 if (setjmp(bus_error_jmp) == 0) {
1302 catch_memory_errors = 1;
1303 sync();
Alexey Dobriyanffb45122007-05-08 00:28:41 -07001304 name = kallsyms_lookup(pc, &size, &offset, NULL, tmpstr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001305 if (name != NULL) {
1306 *startp = pc - offset;
1307 *endp = pc - offset + size;
1308 }
1309 sync();
1310 }
1311 catch_memory_errors = 0;
1312}
1313
Benjamin Herrenschmidtec2b36b2008-04-17 14:34:59 +10001314#define LRSAVE_OFFSET (STACK_FRAME_LR_SAVE * sizeof(unsigned long))
1315#define MARKER_OFFSET (STACK_FRAME_MARKER * sizeof(unsigned long))
1316
Linus Torvalds1da177e2005-04-16 15:20:36 -07001317static void xmon_show_stack(unsigned long sp, unsigned long lr,
1318 unsigned long pc)
1319{
Michael Ellerman0104cd62012-10-09 04:20:36 +00001320 int max_to_print = 64;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001321 unsigned long ip;
1322 unsigned long newsp;
1323 unsigned long marker;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001324 struct pt_regs regs;
1325
Michael Ellerman0104cd62012-10-09 04:20:36 +00001326 while (max_to_print--) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001327 if (sp < PAGE_OFFSET) {
1328 if (sp != 0)
1329 printf("SP (%lx) is in userspace\n", sp);
1330 break;
1331 }
1332
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001333 if (!mread(sp + LRSAVE_OFFSET, &ip, sizeof(unsigned long))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001334 || !mread(sp, &newsp, sizeof(unsigned long))) {
1335 printf("Couldn't read stack frame at %lx\n", sp);
1336 break;
1337 }
1338
1339 /*
1340 * For the first stack frame, try to work out if
1341 * LR and/or the saved LR value in the bottommost
1342 * stack frame are valid.
1343 */
1344 if ((pc | lr) != 0) {
1345 unsigned long fnstart, fnend;
1346 unsigned long nextip;
1347 int printip = 1;
1348
1349 get_function_bounds(pc, &fnstart, &fnend);
1350 nextip = 0;
1351 if (newsp > sp)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001352 mread(newsp + LRSAVE_OFFSET, &nextip,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001353 sizeof(unsigned long));
1354 if (lr == ip) {
1355 if (lr < PAGE_OFFSET
1356 || (fnstart <= lr && lr < fnend))
1357 printip = 0;
1358 } else if (lr == nextip) {
1359 printip = 0;
1360 } else if (lr >= PAGE_OFFSET
1361 && !(fnstart <= lr && lr < fnend)) {
1362 printf("[link register ] ");
1363 xmon_print_symbol(lr, " ", "\n");
1364 }
1365 if (printip) {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001366 printf("["REG"] ", sp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001367 xmon_print_symbol(ip, " ", " (unreliable)\n");
1368 }
1369 pc = lr = 0;
1370
1371 } else {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001372 printf("["REG"] ", sp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001373 xmon_print_symbol(ip, " ", "\n");
1374 }
1375
1376 /* Look for "regshere" marker to see if this is
1377 an exception frame. */
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001378 if (mread(sp + MARKER_OFFSET, &marker, sizeof(unsigned long))
Benjamin Herrenschmidtec2b36b2008-04-17 14:34:59 +10001379 && marker == STACK_FRAME_REGS_MARKER) {
Michael Ellermanc4de3802012-10-09 04:20:35 +00001380 if (mread(sp + STACK_FRAME_OVERHEAD, &regs, sizeof(regs))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001381 != sizeof(regs)) {
1382 printf("Couldn't read registers at %lx\n",
Michael Ellermanc4de3802012-10-09 04:20:35 +00001383 sp + STACK_FRAME_OVERHEAD);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001384 break;
1385 }
Michael Ellermane3bc8042012-08-23 22:09:13 +00001386 printf("--- Exception: %lx %s at ", regs.trap,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001387 getvecname(TRAP(&regs)));
1388 pc = regs.nip;
1389 lr = regs.link;
1390 xmon_print_symbol(pc, " ", "\n");
1391 }
1392
1393 if (newsp == 0)
1394 break;
1395
1396 sp = newsp;
Michael Ellerman0104cd62012-10-09 04:20:36 +00001397 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001398}
1399
1400static void backtrace(struct pt_regs *excp)
1401{
1402 unsigned long sp;
1403
1404 if (scanhex(&sp))
1405 xmon_show_stack(sp, 0, 0);
1406 else
1407 xmon_show_stack(excp->gpr[1], excp->link, excp->nip);
1408 scannl();
1409}
1410
1411static void print_bug_trap(struct pt_regs *regs)
1412{
Paul Mackerrasebdba9a2008-10-31 21:34:09 +11001413#ifdef CONFIG_BUG
Jeremy Fitzhardinge73c9cea2006-12-08 03:30:41 -08001414 const struct bug_entry *bug;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001415 unsigned long addr;
1416
1417 if (regs->msr & MSR_PR)
1418 return; /* not in kernel */
1419 addr = regs->nip; /* address of trap instruction */
1420 if (addr < PAGE_OFFSET)
1421 return;
1422 bug = find_bug(regs->nip);
1423 if (bug == NULL)
1424 return;
Jeremy Fitzhardinge73c9cea2006-12-08 03:30:41 -08001425 if (is_warning_bug(bug))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001426 return;
1427
Stephen Rothwell0a7c7ef2007-03-04 17:05:34 +11001428#ifdef CONFIG_DEBUG_BUGVERBOSE
Jeremy Fitzhardinge73c9cea2006-12-08 03:30:41 -08001429 printf("kernel BUG at %s:%u!\n",
1430 bug->file, bug->line);
Stephen Rothwell0a7c7ef2007-03-04 17:05:34 +11001431#else
1432 printf("kernel BUG at %p!\n", (void *)bug->bug_addr);
1433#endif
Paul Mackerrasebdba9a2008-10-31 21:34:09 +11001434#endif /* CONFIG_BUG */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001435}
1436
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001437static void excprint(struct pt_regs *fp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001438{
1439 unsigned long trap;
1440
1441#ifdef CONFIG_SMP
1442 printf("cpu 0x%x: ", smp_processor_id());
1443#endif /* CONFIG_SMP */
1444
1445 trap = TRAP(fp);
1446 printf("Vector: %lx %s at [%lx]\n", fp->trap, getvecname(trap), fp);
1447 printf(" pc: ");
1448 xmon_print_symbol(fp->nip, ": ", "\n");
1449
1450 printf(" lr: ", fp->link);
1451 xmon_print_symbol(fp->link, ": ", "\n");
1452
1453 printf(" sp: %lx\n", fp->gpr[1]);
1454 printf(" msr: %lx\n", fp->msr);
1455
Aneesh Kumar K.Vce541522013-04-28 09:37:26 +00001456 if (trap == 0x300 || trap == 0x380 || trap == 0x600 || trap == 0x200) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001457 printf(" dar: %lx\n", fp->dar);
1458 if (trap != 0x380)
1459 printf(" dsisr: %lx\n", fp->dsisr);
1460 }
1461
1462 printf(" current = 0x%lx\n", current);
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001463#ifdef CONFIG_PPC64
Benjamin Herrenschmidt7230c562012-03-06 18:27:59 +11001464 printf(" paca = 0x%lx\t softe: %d\t irq_happened: 0x%02x\n",
1465 local_paca, local_paca->soft_enabled, local_paca->irq_happened);
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001466#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001467 if (current) {
1468 printf(" pid = %ld, comm = %s\n",
1469 current->pid, current->comm);
1470 }
1471
1472 if (trap == 0x700)
1473 print_bug_trap(fp);
1474}
1475
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001476static void prregs(struct pt_regs *fp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001477{
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001478 int n, trap;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001479 unsigned long base;
1480 struct pt_regs regs;
1481
1482 if (scanhex(&base)) {
1483 if (setjmp(bus_error_jmp) == 0) {
1484 catch_memory_errors = 1;
1485 sync();
1486 regs = *(struct pt_regs *)base;
1487 sync();
1488 __delay(200);
1489 } else {
1490 catch_memory_errors = 0;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001491 printf("*** Error reading registers from "REG"\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001492 base);
1493 return;
1494 }
1495 catch_memory_errors = 0;
1496 fp = &regs;
1497 }
1498
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001499#ifdef CONFIG_PPC64
Linus Torvalds1da177e2005-04-16 15:20:36 -07001500 if (FULL_REGS(fp)) {
1501 for (n = 0; n < 16; ++n)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001502 printf("R%.2ld = "REG" R%.2ld = "REG"\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001503 n, fp->gpr[n], n+16, fp->gpr[n+16]);
1504 } else {
1505 for (n = 0; n < 7; ++n)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001506 printf("R%.2ld = "REG" R%.2ld = "REG"\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001507 n, fp->gpr[n], n+7, fp->gpr[n+7]);
1508 }
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001509#else
1510 for (n = 0; n < 32; ++n) {
1511 printf("R%.2d = %.8x%s", n, fp->gpr[n],
1512 (n & 3) == 3? "\n": " ");
1513 if (n == 12 && !FULL_REGS(fp)) {
1514 printf("\n");
1515 break;
1516 }
1517 }
1518#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001519 printf("pc = ");
1520 xmon_print_symbol(fp->nip, " ", "\n");
Paul Mackerras48404f22011-05-01 19:48:20 +00001521 if (TRAP(fp) != 0xc00 && cpu_has_feature(CPU_FTR_CFAR)) {
1522 printf("cfar= ");
1523 xmon_print_symbol(fp->orig_gpr3, " ", "\n");
1524 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001525 printf("lr = ");
1526 xmon_print_symbol(fp->link, " ", "\n");
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001527 printf("msr = "REG" cr = %.8lx\n", fp->msr, fp->ccr);
1528 printf("ctr = "REG" xer = "REG" trap = %4lx\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001529 fp->ctr, fp->xer, fp->trap);
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001530 trap = TRAP(fp);
1531 if (trap == 0x300 || trap == 0x380 || trap == 0x600)
1532 printf("dar = "REG" dsisr = %.8lx\n", fp->dar, fp->dsisr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001533}
1534
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001535static void cacheflush(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001536{
1537 int cmd;
1538 unsigned long nflush;
1539
1540 cmd = inchar();
1541 if (cmd != 'i')
1542 termch = cmd;
1543 scanhex((void *)&adrs);
1544 if (termch != '\n')
1545 termch = 0;
1546 nflush = 1;
1547 scanhex(&nflush);
1548 nflush = (nflush + L1_CACHE_BYTES - 1) / L1_CACHE_BYTES;
1549 if (setjmp(bus_error_jmp) == 0) {
1550 catch_memory_errors = 1;
1551 sync();
1552
1553 if (cmd != 'i') {
1554 for (; nflush > 0; --nflush, adrs += L1_CACHE_BYTES)
1555 cflush((void *) adrs);
1556 } else {
1557 for (; nflush > 0; --nflush, adrs += L1_CACHE_BYTES)
1558 cinval((void *) adrs);
1559 }
1560 sync();
1561 /* wait a little while to see if we get a machine check */
1562 __delay(200);
1563 }
1564 catch_memory_errors = 0;
1565}
1566
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001567static unsigned long
Linus Torvalds1da177e2005-04-16 15:20:36 -07001568read_spr(int n)
1569{
1570 unsigned int instrs[2];
1571 unsigned long (*code)(void);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001572 unsigned long ret = -1UL;
Paul Mackerras548cceb2005-11-11 22:36:34 +11001573#ifdef CONFIG_PPC64
1574 unsigned long opd[3];
Linus Torvalds1da177e2005-04-16 15:20:36 -07001575
Linus Torvalds1da177e2005-04-16 15:20:36 -07001576 opd[0] = (unsigned long)instrs;
1577 opd[1] = 0;
1578 opd[2] = 0;
Paul Mackerras548cceb2005-11-11 22:36:34 +11001579 code = (unsigned long (*)(void)) opd;
1580#else
1581 code = (unsigned long (*)(void)) instrs;
1582#endif
1583
1584 /* mfspr r3,n; blr */
1585 instrs[0] = 0x7c6002a6 + ((n & 0x1F) << 16) + ((n & 0x3e0) << 6);
1586 instrs[1] = 0x4e800020;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001587 store_inst(instrs);
1588 store_inst(instrs+1);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001589
1590 if (setjmp(bus_error_jmp) == 0) {
1591 catch_memory_errors = 1;
1592 sync();
1593
1594 ret = code();
1595
1596 sync();
1597 /* wait a little while to see if we get a machine check */
1598 __delay(200);
1599 n = size;
1600 }
1601
1602 return ret;
1603}
1604
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001605static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07001606write_spr(int n, unsigned long val)
1607{
1608 unsigned int instrs[2];
1609 unsigned long (*code)(unsigned long);
Paul Mackerras548cceb2005-11-11 22:36:34 +11001610#ifdef CONFIG_PPC64
Linus Torvalds1da177e2005-04-16 15:20:36 -07001611 unsigned long opd[3];
1612
Linus Torvalds1da177e2005-04-16 15:20:36 -07001613 opd[0] = (unsigned long)instrs;
1614 opd[1] = 0;
1615 opd[2] = 0;
Paul Mackerras548cceb2005-11-11 22:36:34 +11001616 code = (unsigned long (*)(unsigned long)) opd;
1617#else
1618 code = (unsigned long (*)(unsigned long)) instrs;
1619#endif
1620
1621 instrs[0] = 0x7c6003a6 + ((n & 0x1F) << 16) + ((n & 0x3e0) << 6);
1622 instrs[1] = 0x4e800020;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001623 store_inst(instrs);
1624 store_inst(instrs+1);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001625
1626 if (setjmp(bus_error_jmp) == 0) {
1627 catch_memory_errors = 1;
1628 sync();
1629
1630 code(val);
1631
1632 sync();
1633 /* wait a little while to see if we get a machine check */
1634 __delay(200);
1635 n = size;
1636 }
1637}
1638
1639static unsigned long regno;
1640extern char exc_prolog;
1641extern char dec_exc;
1642
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001643static void super_regs(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001644{
1645 int cmd;
1646 unsigned long val;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001647
1648 cmd = skipbl();
1649 if (cmd == '\n') {
Michael Ellermane3bc8042012-08-23 22:09:13 +00001650 unsigned long sp, toc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001651 asm("mr %0,1" : "=r" (sp) :);
1652 asm("mr %0,2" : "=r" (toc) :);
1653
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001654 printf("msr = "REG" sprg0= "REG"\n",
1655 mfmsr(), mfspr(SPRN_SPRG0));
1656 printf("pvr = "REG" sprg1= "REG"\n",
Michael Ellermane3bc8042012-08-23 22:09:13 +00001657 mfspr(SPRN_PVR), mfspr(SPRN_SPRG1));
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001658 printf("dec = "REG" sprg2= "REG"\n",
1659 mfspr(SPRN_DEC), mfspr(SPRN_SPRG2));
1660 printf("sp = "REG" sprg3= "REG"\n", sp, mfspr(SPRN_SPRG3));
1661 printf("toc = "REG" dar = "REG"\n", toc, mfspr(SPRN_DAR));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001662
1663 return;
1664 }
1665
1666 scanhex(&regno);
1667 switch (cmd) {
1668 case 'w':
1669 val = read_spr(regno);
1670 scanhex(&val);
1671 write_spr(regno, val);
1672 /* fall through */
1673 case 'r':
1674 printf("spr %lx = %lx\n", regno, read_spr(regno));
1675 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001676 }
1677 scannl();
1678}
1679
1680/*
1681 * Stuff for reading and writing memory safely
1682 */
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001683static int
Linus Torvalds1da177e2005-04-16 15:20:36 -07001684mread(unsigned long adrs, void *buf, int size)
1685{
1686 volatile int n;
1687 char *p, *q;
1688
1689 n = 0;
1690 if (setjmp(bus_error_jmp) == 0) {
1691 catch_memory_errors = 1;
1692 sync();
1693 p = (char *)adrs;
1694 q = (char *)buf;
1695 switch (size) {
1696 case 2:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001697 *(u16 *)q = *(u16 *)p;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001698 break;
1699 case 4:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001700 *(u32 *)q = *(u32 *)p;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001701 break;
1702 case 8:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001703 *(u64 *)q = *(u64 *)p;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001704 break;
1705 default:
1706 for( ; n < size; ++n) {
1707 *q++ = *p++;
1708 sync();
1709 }
1710 }
1711 sync();
1712 /* wait a little while to see if we get a machine check */
1713 __delay(200);
1714 n = size;
1715 }
1716 catch_memory_errors = 0;
1717 return n;
1718}
1719
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001720static int
Linus Torvalds1da177e2005-04-16 15:20:36 -07001721mwrite(unsigned long adrs, void *buf, int size)
1722{
1723 volatile int n;
1724 char *p, *q;
1725
1726 n = 0;
1727 if (setjmp(bus_error_jmp) == 0) {
1728 catch_memory_errors = 1;
1729 sync();
1730 p = (char *) adrs;
1731 q = (char *) buf;
1732 switch (size) {
1733 case 2:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001734 *(u16 *)p = *(u16 *)q;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001735 break;
1736 case 4:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001737 *(u32 *)p = *(u32 *)q;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001738 break;
1739 case 8:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001740 *(u64 *)p = *(u64 *)q;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001741 break;
1742 default:
1743 for ( ; n < size; ++n) {
1744 *p++ = *q++;
1745 sync();
1746 }
1747 }
1748 sync();
1749 /* wait a little while to see if we get a machine check */
1750 __delay(200);
1751 n = size;
1752 } else {
Michael Ellerman736256e2014-05-26 21:02:14 +10001753 printf("*** Error writing address "REG"\n", adrs + n);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001754 }
1755 catch_memory_errors = 0;
1756 return n;
1757}
1758
1759static int fault_type;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001760static int fault_except;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001761static char *fault_chars[] = { "--", "**", "##" };
1762
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001763static int handle_fault(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001764{
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001765 fault_except = TRAP(regs);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001766 switch (TRAP(regs)) {
1767 case 0x200:
1768 fault_type = 0;
1769 break;
1770 case 0x300:
1771 case 0x380:
1772 fault_type = 1;
1773 break;
1774 default:
1775 fault_type = 2;
1776 }
1777
1778 longjmp(bus_error_jmp, 1);
1779
1780 return 0;
1781}
1782
1783#define SWAP(a, b, t) ((t) = (a), (a) = (b), (b) = (t))
1784
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001785static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07001786byterev(unsigned char *val, int size)
1787{
1788 int t;
1789
1790 switch (size) {
1791 case 2:
1792 SWAP(val[0], val[1], t);
1793 break;
1794 case 4:
1795 SWAP(val[0], val[3], t);
1796 SWAP(val[1], val[2], t);
1797 break;
1798 case 8: /* is there really any use for this? */
1799 SWAP(val[0], val[7], t);
1800 SWAP(val[1], val[6], t);
1801 SWAP(val[2], val[5], t);
1802 SWAP(val[3], val[4], t);
1803 break;
1804 }
1805}
1806
1807static int brev;
1808static int mnoread;
1809
Michael Ellermane3bc8042012-08-23 22:09:13 +00001810static char *memex_help_string =
Linus Torvalds1da177e2005-04-16 15:20:36 -07001811 "Memory examine command usage:\n"
1812 "m [addr] [flags] examine/change memory\n"
1813 " addr is optional. will start where left off.\n"
1814 " flags may include chars from this set:\n"
1815 " b modify by bytes (default)\n"
1816 " w modify by words (2 byte)\n"
1817 " l modify by longs (4 byte)\n"
1818 " d modify by doubleword (8 byte)\n"
1819 " r toggle reverse byte order mode\n"
1820 " n do not read memory (for i/o spaces)\n"
1821 " . ok to read (default)\n"
1822 "NOTE: flags are saved as defaults\n"
1823 "";
1824
Michael Ellermane3bc8042012-08-23 22:09:13 +00001825static char *memex_subcmd_help_string =
Linus Torvalds1da177e2005-04-16 15:20:36 -07001826 "Memory examine subcommands:\n"
1827 " hexval write this val to current location\n"
1828 " 'string' write chars from string to this location\n"
1829 " ' increment address\n"
1830 " ^ decrement address\n"
1831 " / increment addr by 0x10. //=0x100, ///=0x1000, etc\n"
1832 " \\ decrement addr by 0x10. \\\\=0x100, \\\\\\=0x1000, etc\n"
1833 " ` clear no-read flag\n"
1834 " ; stay at this addr\n"
1835 " v change to byte mode\n"
1836 " w change to word (2 byte) mode\n"
1837 " l change to long (4 byte) mode\n"
1838 " u change to doubleword (8 byte) mode\n"
1839 " m addr change current addr\n"
1840 " n toggle no-read flag\n"
1841 " r toggle byte reverse flag\n"
1842 " < count back up count bytes\n"
1843 " > count skip forward count bytes\n"
1844 " x exit this mode\n"
1845 "";
1846
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001847static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07001848memex(void)
1849{
1850 int cmd, inc, i, nslash;
1851 unsigned long n;
1852 unsigned char val[16];
1853
1854 scanhex((void *)&adrs);
1855 cmd = skipbl();
1856 if (cmd == '?') {
1857 printf(memex_help_string);
1858 return;
1859 } else {
1860 termch = cmd;
1861 }
1862 last_cmd = "m\n";
1863 while ((cmd = skipbl()) != '\n') {
1864 switch( cmd ){
1865 case 'b': size = 1; break;
1866 case 'w': size = 2; break;
1867 case 'l': size = 4; break;
1868 case 'd': size = 8; break;
1869 case 'r': brev = !brev; break;
1870 case 'n': mnoread = 1; break;
1871 case '.': mnoread = 0; break;
1872 }
1873 }
1874 if( size <= 0 )
1875 size = 1;
1876 else if( size > 8 )
1877 size = 8;
1878 for(;;){
1879 if (!mnoread)
1880 n = mread(adrs, val, size);
Paul Mackerrase1449ed2005-11-10 14:30:20 +11001881 printf(REG"%c", adrs, brev? 'r': ' ');
Linus Torvalds1da177e2005-04-16 15:20:36 -07001882 if (!mnoread) {
1883 if (brev)
1884 byterev(val, size);
1885 putchar(' ');
1886 for (i = 0; i < n; ++i)
1887 printf("%.2x", val[i]);
1888 for (; i < size; ++i)
1889 printf("%s", fault_chars[fault_type]);
1890 }
1891 putchar(' ');
1892 inc = size;
1893 nslash = 0;
1894 for(;;){
1895 if( scanhex(&n) ){
1896 for (i = 0; i < size; ++i)
1897 val[i] = n >> (i * 8);
1898 if (!brev)
1899 byterev(val, size);
1900 mwrite(adrs, val, size);
1901 inc = size;
1902 }
1903 cmd = skipbl();
1904 if (cmd == '\n')
1905 break;
1906 inc = 0;
1907 switch (cmd) {
1908 case '\'':
1909 for(;;){
1910 n = inchar();
1911 if( n == '\\' )
1912 n = bsesc();
1913 else if( n == '\'' )
1914 break;
1915 for (i = 0; i < size; ++i)
1916 val[i] = n >> (i * 8);
1917 if (!brev)
1918 byterev(val, size);
1919 mwrite(adrs, val, size);
1920 adrs += size;
1921 }
1922 adrs -= size;
1923 inc = size;
1924 break;
1925 case ',':
1926 adrs += size;
1927 break;
1928 case '.':
1929 mnoread = 0;
1930 break;
1931 case ';':
1932 break;
1933 case 'x':
1934 case EOF:
1935 scannl();
1936 return;
1937 case 'b':
1938 case 'v':
1939 size = 1;
1940 break;
1941 case 'w':
1942 size = 2;
1943 break;
1944 case 'l':
1945 size = 4;
1946 break;
1947 case 'u':
1948 size = 8;
1949 break;
1950 case '^':
1951 adrs -= size;
1952 break;
1953 break;
1954 case '/':
1955 if (nslash > 0)
1956 adrs -= 1 << nslash;
1957 else
1958 nslash = 0;
1959 nslash += 4;
1960 adrs += 1 << nslash;
1961 break;
1962 case '\\':
1963 if (nslash < 0)
1964 adrs += 1 << -nslash;
1965 else
1966 nslash = 0;
1967 nslash -= 4;
1968 adrs -= 1 << -nslash;
1969 break;
1970 case 'm':
1971 scanhex((void *)&adrs);
1972 break;
1973 case 'n':
1974 mnoread = 1;
1975 break;
1976 case 'r':
1977 brev = !brev;
1978 break;
1979 case '<':
1980 n = size;
1981 scanhex(&n);
1982 adrs -= n;
1983 break;
1984 case '>':
1985 n = size;
1986 scanhex(&n);
1987 adrs += n;
1988 break;
1989 case '?':
1990 printf(memex_subcmd_help_string);
1991 break;
1992 }
1993 }
1994 adrs += inc;
1995 }
1996}
1997
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001998static int
Linus Torvalds1da177e2005-04-16 15:20:36 -07001999bsesc(void)
2000{
2001 int c;
2002
2003 c = inchar();
2004 switch( c ){
2005 case 'n': c = '\n'; break;
2006 case 'r': c = '\r'; break;
2007 case 'b': c = '\b'; break;
2008 case 't': c = '\t'; break;
2009 }
2010 return c;
2011}
2012
Olaf Hering7e5b5932006-03-08 20:40:28 +01002013static void xmon_rawdump (unsigned long adrs, long ndump)
2014{
2015 long n, m, r, nr;
2016 unsigned char temp[16];
2017
2018 for (n = ndump; n > 0;) {
2019 r = n < 16? n: 16;
2020 nr = mread(adrs, temp, r);
2021 adrs += nr;
2022 for (m = 0; m < r; ++m) {
2023 if (m < nr)
2024 printf("%.2x", temp[m]);
2025 else
2026 printf("%s", fault_chars[fault_type]);
2027 }
2028 n -= r;
2029 if (nr < r)
2030 break;
2031 }
2032 printf("\n");
2033}
2034
Michael Ellermanddadb6b2012-09-13 23:01:31 +00002035#ifdef CONFIG_PPC64
2036static void dump_one_paca(int cpu)
2037{
2038 struct paca_struct *p;
2039
2040 if (setjmp(bus_error_jmp) != 0) {
2041 printf("*** Error dumping paca for cpu 0x%x!\n", cpu);
2042 return;
2043 }
2044
2045 catch_memory_errors = 1;
2046 sync();
2047
2048 p = &paca[cpu];
2049
2050 printf("paca for cpu 0x%x @ %p:\n", cpu, p);
2051
2052 printf(" %-*s = %s\n", 16, "possible", cpu_possible(cpu) ? "yes" : "no");
2053 printf(" %-*s = %s\n", 16, "present", cpu_present(cpu) ? "yes" : "no");
2054 printf(" %-*s = %s\n", 16, "online", cpu_online(cpu) ? "yes" : "no");
2055
2056#define DUMP(paca, name, format) \
2057 printf(" %-*s = %#-*"format"\t(0x%lx)\n", 16, #name, 18, paca->name, \
2058 offsetof(struct paca_struct, name));
2059
2060 DUMP(p, lock_token, "x");
2061 DUMP(p, paca_index, "x");
2062 DUMP(p, kernel_toc, "lx");
2063 DUMP(p, kernelbase, "lx");
2064 DUMP(p, kernel_msr, "lx");
Michael Ellermanddadb6b2012-09-13 23:01:31 +00002065 DUMP(p, emergency_sp, "p");
Mahesh Salgaonkar729b0f72013-10-30 20:04:00 +05302066#ifdef CONFIG_PPC_BOOK3S_64
2067 DUMP(p, mc_emergency_sp, "p");
2068 DUMP(p, in_mce, "x");
2069#endif
Michael Ellermanddadb6b2012-09-13 23:01:31 +00002070 DUMP(p, data_offset, "lx");
2071 DUMP(p, hw_cpu_id, "x");
2072 DUMP(p, cpu_start, "x");
2073 DUMP(p, kexec_state, "x");
2074 DUMP(p, __current, "p");
2075 DUMP(p, kstack, "lx");
2076 DUMP(p, stab_rr, "lx");
2077 DUMP(p, saved_r1, "lx");
2078 DUMP(p, trap_save, "x");
2079 DUMP(p, soft_enabled, "x");
2080 DUMP(p, irq_happened, "x");
2081 DUMP(p, io_sync, "x");
2082 DUMP(p, irq_work_pending, "x");
2083 DUMP(p, nap_state_lost, "x");
2084
2085#undef DUMP
2086
2087 catch_memory_errors = 0;
2088 sync();
2089}
2090
2091static void dump_all_pacas(void)
2092{
2093 int cpu;
2094
2095 if (num_possible_cpus() == 0) {
2096 printf("No possible cpus, use 'dp #' to dump individual cpus\n");
2097 return;
2098 }
2099
2100 for_each_possible_cpu(cpu)
2101 dump_one_paca(cpu);
2102}
2103
2104static void dump_pacas(void)
2105{
2106 unsigned long num;
2107 int c;
2108
2109 c = inchar();
2110 if (c == 'a') {
2111 dump_all_pacas();
2112 return;
2113 }
2114
2115 termch = c; /* Put c back, it wasn't 'a' */
2116
2117 if (scanhex(&num))
2118 dump_one_paca(num);
2119 else
2120 dump_one_paca(xmon_owner);
2121}
2122#endif
2123
Linus Torvalds1da177e2005-04-16 15:20:36 -07002124#define isxdigit(c) (('0' <= (c) && (c) <= '9') \
2125 || ('a' <= (c) && (c) <= 'f') \
2126 || ('A' <= (c) && (c) <= 'F'))
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002127static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002128dump(void)
2129{
2130 int c;
2131
2132 c = inchar();
Michael Ellermanddadb6b2012-09-13 23:01:31 +00002133
2134#ifdef CONFIG_PPC64
2135 if (c == 'p') {
2136 dump_pacas();
2137 return;
2138 }
2139#endif
2140
Linus Torvalds1da177e2005-04-16 15:20:36 -07002141 if ((isxdigit(c) && c != 'f' && c != 'd') || c == '\n')
2142 termch = c;
2143 scanhex((void *)&adrs);
2144 if (termch != '\n')
2145 termch = 0;
2146 if (c == 'i') {
2147 scanhex(&nidump);
2148 if (nidump == 0)
2149 nidump = 16;
2150 else if (nidump > MAX_DUMP)
2151 nidump = MAX_DUMP;
2152 adrs += ppc_inst_dump(adrs, nidump, 1);
2153 last_cmd = "di\n";
Vinay Sridharf312deb2009-05-14 23:13:07 +00002154 } else if (c == 'l') {
2155 dump_log_buf();
Olaf Hering7e5b5932006-03-08 20:40:28 +01002156 } else if (c == 'r') {
2157 scanhex(&ndump);
2158 if (ndump == 0)
2159 ndump = 64;
2160 xmon_rawdump(adrs, ndump);
2161 adrs += ndump;
2162 last_cmd = "dr\n";
Linus Torvalds1da177e2005-04-16 15:20:36 -07002163 } else {
2164 scanhex(&ndump);
2165 if (ndump == 0)
2166 ndump = 64;
2167 else if (ndump > MAX_DUMP)
2168 ndump = MAX_DUMP;
2169 prdump(adrs, ndump);
2170 adrs += ndump;
2171 last_cmd = "d\n";
2172 }
2173}
2174
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002175static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002176prdump(unsigned long adrs, long ndump)
2177{
2178 long n, m, c, r, nr;
2179 unsigned char temp[16];
2180
2181 for (n = ndump; n > 0;) {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002182 printf(REG, adrs);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002183 putchar(' ');
2184 r = n < 16? n: 16;
2185 nr = mread(adrs, temp, r);
2186 adrs += nr;
2187 for (m = 0; m < r; ++m) {
Michael Ellermane3bc8042012-08-23 22:09:13 +00002188 if ((m & (sizeof(long) - 1)) == 0 && m > 0)
Paul Mackerrase1449ed2005-11-10 14:30:20 +11002189 putchar(' ');
Linus Torvalds1da177e2005-04-16 15:20:36 -07002190 if (m < nr)
2191 printf("%.2x", temp[m]);
2192 else
2193 printf("%s", fault_chars[fault_type]);
2194 }
Paul Mackerrase1449ed2005-11-10 14:30:20 +11002195 for (; m < 16; ++m) {
Michael Ellermane3bc8042012-08-23 22:09:13 +00002196 if ((m & (sizeof(long) - 1)) == 0)
Paul Mackerrase1449ed2005-11-10 14:30:20 +11002197 putchar(' ');
Linus Torvalds1da177e2005-04-16 15:20:36 -07002198 printf(" ");
Paul Mackerrase1449ed2005-11-10 14:30:20 +11002199 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002200 printf(" |");
2201 for (m = 0; m < r; ++m) {
2202 if (m < nr) {
2203 c = temp[m];
2204 putchar(' ' <= c && c <= '~'? c: '.');
2205 } else
2206 putchar(' ');
2207 }
2208 n -= r;
2209 for (; m < 16; ++m)
2210 putchar(' ');
2211 printf("|\n");
2212 if (nr < r)
2213 break;
2214 }
2215}
2216
Michael Ellerman4c4c8722006-11-23 00:46:42 +01002217typedef int (*instruction_dump_func)(unsigned long inst, unsigned long addr);
2218
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002219static int
Michael Ellerman4c4c8722006-11-23 00:46:42 +01002220generic_inst_dump(unsigned long adr, long count, int praddr,
2221 instruction_dump_func dump_func)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002222{
2223 int nr, dotted;
2224 unsigned long first_adr;
2225 unsigned long inst, last_inst = 0;
2226 unsigned char val[4];
2227
2228 dotted = 0;
2229 for (first_adr = adr; count > 0; --count, adr += 4) {
2230 nr = mread(adr, val, 4);
2231 if (nr == 0) {
2232 if (praddr) {
2233 const char *x = fault_chars[fault_type];
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002234 printf(REG" %s%s%s%s\n", adr, x, x, x, x);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002235 }
2236 break;
2237 }
2238 inst = GETWORD(val);
2239 if (adr > first_adr && inst == last_inst) {
2240 if (!dotted) {
2241 printf(" ...\n");
2242 dotted = 1;
2243 }
2244 continue;
2245 }
2246 dotted = 0;
2247 last_inst = inst;
2248 if (praddr)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002249 printf(REG" %.8x", adr, inst);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002250 printf("\t");
Michael Ellerman4c4c8722006-11-23 00:46:42 +01002251 dump_func(inst, adr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002252 printf("\n");
2253 }
2254 return adr - first_adr;
2255}
2256
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002257static int
Michael Ellerman4c4c8722006-11-23 00:46:42 +01002258ppc_inst_dump(unsigned long adr, long count, int praddr)
2259{
2260 return generic_inst_dump(adr, count, praddr, print_insn_powerpc);
2261}
2262
Linus Torvalds1da177e2005-04-16 15:20:36 -07002263void
2264print_address(unsigned long addr)
2265{
2266 xmon_print_symbol(addr, "\t# ", "");
2267}
2268
Vinay Sridharf312deb2009-05-14 23:13:07 +00002269void
2270dump_log_buf(void)
2271{
Michael Ellermanca5dd392012-08-23 22:09:12 +00002272 struct kmsg_dumper dumper = { .active = 1 };
2273 unsigned char buf[128];
2274 size_t len;
Vinay Sridharf312deb2009-05-14 23:13:07 +00002275
Michael Ellermane3bc8042012-08-23 22:09:13 +00002276 if (setjmp(bus_error_jmp) != 0) {
Michael Ellermanca5dd392012-08-23 22:09:12 +00002277 printf("Error dumping printk buffer!\n");
Michael Ellermane3bc8042012-08-23 22:09:13 +00002278 return;
2279 }
Vinay Sridharf312deb2009-05-14 23:13:07 +00002280
Michael Ellermane3bc8042012-08-23 22:09:13 +00002281 catch_memory_errors = 1;
2282 sync();
Vinay Sridharf312deb2009-05-14 23:13:07 +00002283
Michael Ellermanca5dd392012-08-23 22:09:12 +00002284 kmsg_dump_rewind_nolock(&dumper);
2285 while (kmsg_dump_get_line_nolock(&dumper, false, buf, sizeof(buf), &len)) {
2286 buf[len] = '\0';
2287 printf("%s", buf);
2288 }
Vinay Sridharf312deb2009-05-14 23:13:07 +00002289
Michael Ellermane3bc8042012-08-23 22:09:13 +00002290 sync();
2291 /* wait a little while to see if we get a machine check */
2292 __delay(200);
2293 catch_memory_errors = 0;
Vinay Sridharf312deb2009-05-14 23:13:07 +00002294}
Linus Torvalds1da177e2005-04-16 15:20:36 -07002295
2296/*
2297 * Memory operations - move, set, print differences
2298 */
2299static unsigned long mdest; /* destination address */
2300static unsigned long msrc; /* source address */
2301static unsigned long mval; /* byte value to set memory to */
2302static unsigned long mcount; /* # bytes to affect */
2303static unsigned long mdiffs; /* max # differences to print */
2304
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002305static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002306memops(int cmd)
2307{
2308 scanhex((void *)&mdest);
2309 if( termch != '\n' )
2310 termch = 0;
2311 scanhex((void *)(cmd == 's'? &mval: &msrc));
2312 if( termch != '\n' )
2313 termch = 0;
2314 scanhex((void *)&mcount);
2315 switch( cmd ){
2316 case 'm':
2317 memmove((void *)mdest, (void *)msrc, mcount);
2318 break;
2319 case 's':
2320 memset((void *)mdest, mval, mcount);
2321 break;
2322 case 'd':
2323 if( termch != '\n' )
2324 termch = 0;
2325 scanhex((void *)&mdiffs);
2326 memdiffs((unsigned char *)mdest, (unsigned char *)msrc, mcount, mdiffs);
2327 break;
2328 }
2329}
2330
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002331static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002332memdiffs(unsigned char *p1, unsigned char *p2, unsigned nb, unsigned maxpr)
2333{
2334 unsigned n, prt;
2335
2336 prt = 0;
2337 for( n = nb; n > 0; --n )
2338 if( *p1++ != *p2++ )
2339 if( ++prt <= maxpr )
2340 printf("%.16x %.2x # %.16x %.2x\n", p1 - 1,
2341 p1[-1], p2 - 1, p2[-1]);
2342 if( prt > maxpr )
2343 printf("Total of %d differences\n", prt);
2344}
2345
2346static unsigned mend;
2347static unsigned mask;
2348
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002349static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002350memlocate(void)
2351{
2352 unsigned a, n;
2353 unsigned char val[4];
2354
2355 last_cmd = "ml";
2356 scanhex((void *)&mdest);
2357 if (termch != '\n') {
2358 termch = 0;
2359 scanhex((void *)&mend);
2360 if (termch != '\n') {
2361 termch = 0;
2362 scanhex((void *)&mval);
2363 mask = ~0;
2364 if (termch != '\n') termch = 0;
2365 scanhex((void *)&mask);
2366 }
2367 }
2368 n = 0;
2369 for (a = mdest; a < mend; a += 4) {
2370 if (mread(a, val, 4) == 4
2371 && ((GETWORD(val) ^ mval) & mask) == 0) {
2372 printf("%.16x: %.16x\n", a, GETWORD(val));
2373 if (++n >= 10)
2374 break;
2375 }
2376 }
2377}
2378
2379static unsigned long mskip = 0x1000;
2380static unsigned long mlim = 0xffffffff;
2381
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002382static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002383memzcan(void)
2384{
2385 unsigned char v;
2386 unsigned a;
2387 int ok, ook;
2388
2389 scanhex(&mdest);
2390 if (termch != '\n') termch = 0;
2391 scanhex(&mskip);
2392 if (termch != '\n') termch = 0;
2393 scanhex(&mlim);
2394 ook = 0;
2395 for (a = mdest; a < mlim; a += mskip) {
2396 ok = mread(a, &v, 1);
2397 if (ok && !ook) {
2398 printf("%.8x .. ", a);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002399 } else if (!ok && ook)
2400 printf("%.8x\n", a - mskip);
2401 ook = ok;
2402 if (a + mskip < a)
2403 break;
2404 }
2405 if (ook)
2406 printf("%.8x\n", a - mskip);
2407}
2408
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002409static void proccall(void)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002410{
2411 unsigned long args[8];
2412 unsigned long ret;
2413 int i;
2414 typedef unsigned long (*callfunc_t)(unsigned long, unsigned long,
2415 unsigned long, unsigned long, unsigned long,
2416 unsigned long, unsigned long, unsigned long);
2417 callfunc_t func;
2418
2419 if (!scanhex(&adrs))
2420 return;
2421 if (termch != '\n')
2422 termch = 0;
2423 for (i = 0; i < 8; ++i)
2424 args[i] = 0;
2425 for (i = 0; i < 8; ++i) {
2426 if (!scanhex(&args[i]) || termch == '\n')
2427 break;
2428 termch = 0;
2429 }
2430 func = (callfunc_t) adrs;
2431 ret = 0;
2432 if (setjmp(bus_error_jmp) == 0) {
2433 catch_memory_errors = 1;
2434 sync();
2435 ret = func(args[0], args[1], args[2], args[3],
2436 args[4], args[5], args[6], args[7]);
2437 sync();
Michael Ellerman736256e2014-05-26 21:02:14 +10002438 printf("return value is 0x%lx\n", ret);
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002439 } else {
2440 printf("*** %x exception occurred\n", fault_except);
2441 }
2442 catch_memory_errors = 0;
2443}
2444
Linus Torvalds1da177e2005-04-16 15:20:36 -07002445/* Input scanning routines */
2446int
2447skipbl(void)
2448{
2449 int c;
2450
2451 if( termch != 0 ){
2452 c = termch;
2453 termch = 0;
2454 } else
2455 c = inchar();
2456 while( c == ' ' || c == '\t' )
2457 c = inchar();
2458 return c;
2459}
2460
2461#define N_PTREGS 44
2462static char *regnames[N_PTREGS] = {
2463 "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7",
2464 "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15",
2465 "r16", "r17", "r18", "r19", "r20", "r21", "r22", "r23",
2466 "r24", "r25", "r26", "r27", "r28", "r29", "r30", "r31",
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002467 "pc", "msr", "or3", "ctr", "lr", "xer", "ccr",
2468#ifdef CONFIG_PPC64
2469 "softe",
2470#else
2471 "mq",
2472#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07002473 "trap", "dar", "dsisr", "res"
2474};
2475
2476int
2477scanhex(unsigned long *vp)
2478{
2479 int c, d;
2480 unsigned long v;
2481
2482 c = skipbl();
2483 if (c == '%') {
2484 /* parse register name */
2485 char regname[8];
2486 int i;
2487
2488 for (i = 0; i < sizeof(regname) - 1; ++i) {
2489 c = inchar();
2490 if (!isalnum(c)) {
2491 termch = c;
2492 break;
2493 }
2494 regname[i] = c;
2495 }
2496 regname[i] = 0;
2497 for (i = 0; i < N_PTREGS; ++i) {
2498 if (strcmp(regnames[i], regname) == 0) {
2499 if (xmon_regs == NULL) {
2500 printf("regs not available\n");
2501 return 0;
2502 }
2503 *vp = ((unsigned long *)xmon_regs)[i];
2504 return 1;
2505 }
2506 }
2507 printf("invalid register name '%%%s'\n", regname);
2508 return 0;
2509 }
2510
2511 /* skip leading "0x" if any */
2512
2513 if (c == '0') {
2514 c = inchar();
2515 if (c == 'x') {
2516 c = inchar();
2517 } else {
2518 d = hexdigit(c);
2519 if (d == EOF) {
2520 termch = c;
2521 *vp = 0;
2522 return 1;
2523 }
2524 }
2525 } else if (c == '$') {
2526 int i;
2527 for (i=0; i<63; i++) {
2528 c = inchar();
2529 if (isspace(c)) {
2530 termch = c;
2531 break;
2532 }
2533 tmpstr[i] = c;
2534 }
2535 tmpstr[i++] = 0;
Benjamin Herrenschmidt6879dc12005-06-21 17:15:30 -07002536 *vp = 0;
2537 if (setjmp(bus_error_jmp) == 0) {
2538 catch_memory_errors = 1;
2539 sync();
2540 *vp = kallsyms_lookup_name(tmpstr);
2541 sync();
2542 }
2543 catch_memory_errors = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002544 if (!(*vp)) {
2545 printf("unknown symbol '%s'\n", tmpstr);
2546 return 0;
2547 }
2548 return 1;
2549 }
2550
2551 d = hexdigit(c);
2552 if (d == EOF) {
2553 termch = c;
2554 return 0;
2555 }
2556 v = 0;
2557 do {
2558 v = (v << 4) + d;
2559 c = inchar();
2560 d = hexdigit(c);
2561 } while (d != EOF);
2562 termch = c;
2563 *vp = v;
2564 return 1;
2565}
2566
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002567static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002568scannl(void)
2569{
2570 int c;
2571
2572 c = termch;
2573 termch = 0;
2574 while( c != '\n' )
2575 c = inchar();
2576}
2577
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002578static int hexdigit(int c)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002579{
2580 if( '0' <= c && c <= '9' )
2581 return c - '0';
2582 if( 'A' <= c && c <= 'F' )
2583 return c - ('A' - 10);
2584 if( 'a' <= c && c <= 'f' )
2585 return c - ('a' - 10);
2586 return EOF;
2587}
2588
2589void
2590getstring(char *s, int size)
2591{
2592 int c;
2593
2594 c = skipbl();
2595 do {
2596 if( size > 1 ){
2597 *s++ = c;
2598 --size;
2599 }
2600 c = inchar();
2601 } while( c != ' ' && c != '\t' && c != '\n' );
2602 termch = c;
2603 *s = 0;
2604}
2605
2606static char line[256];
2607static char *lineptr;
2608
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002609static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002610flush_input(void)
2611{
2612 lineptr = NULL;
2613}
2614
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002615static int
Linus Torvalds1da177e2005-04-16 15:20:36 -07002616inchar(void)
2617{
2618 if (lineptr == NULL || *lineptr == 0) {
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11002619 if (xmon_gets(line, sizeof(line)) == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002620 lineptr = NULL;
2621 return EOF;
2622 }
2623 lineptr = line;
2624 }
2625 return *lineptr++;
2626}
2627
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002628static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002629take_input(char *str)
2630{
2631 lineptr = str;
2632}
2633
2634
2635static void
2636symbol_lookup(void)
2637{
2638 int type = inchar();
2639 unsigned long addr;
2640 static char tmp[64];
2641
2642 switch (type) {
2643 case 'a':
2644 if (scanhex(&addr))
2645 xmon_print_symbol(addr, ": ", "\n");
2646 termch = 0;
2647 break;
2648 case 's':
2649 getstring(tmp, 64);
2650 if (setjmp(bus_error_jmp) == 0) {
2651 catch_memory_errors = 1;
2652 sync();
2653 addr = kallsyms_lookup_name(tmp);
2654 if (addr)
2655 printf("%s: %lx\n", tmp, addr);
2656 else
2657 printf("Symbol '%s' not found.\n", tmp);
2658 sync();
2659 }
2660 catch_memory_errors = 0;
2661 termch = 0;
2662 break;
2663 }
2664}
2665
2666
2667/* Print an address in numeric and symbolic form (if possible) */
2668static void xmon_print_symbol(unsigned long address, const char *mid,
2669 const char *after)
2670{
2671 char *modname;
2672 const char *name = NULL;
2673 unsigned long offset, size;
2674
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002675 printf(REG, address);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002676 if (setjmp(bus_error_jmp) == 0) {
2677 catch_memory_errors = 1;
2678 sync();
2679 name = kallsyms_lookup(address, &size, &offset, &modname,
2680 tmpstr);
2681 sync();
2682 /* wait a little while to see if we get a machine check */
2683 __delay(200);
2684 }
2685
2686 catch_memory_errors = 0;
2687
2688 if (name) {
2689 printf("%s%s+%#lx/%#lx", mid, name, offset, size);
2690 if (modname)
2691 printf(" [%s]", modname);
2692 }
2693 printf("%s", after);
2694}
2695
Benjamin Herrenschmidt2d27cfd2009-07-23 23:15:59 +00002696#ifdef CONFIG_PPC_BOOK3S_64
Michael Ellerman13b3d132014-07-10 12:29:20 +10002697void dump_segments(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002698{
2699 int i;
will schmidtb3b95952007-12-07 08:22:23 +11002700 unsigned long esid,vsid,valid;
2701 unsigned long llp;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002702
Michael Ellerman736256e2014-05-26 21:02:14 +10002703 printf("SLB contents of cpu 0x%x\n", smp_processor_id());
Linus Torvalds1da177e2005-04-16 15:20:36 -07002704
Michael Neuling584f8b72007-12-06 17:24:48 +11002705 for (i = 0; i < mmu_slb_size; i++) {
will schmidtb3b95952007-12-07 08:22:23 +11002706 asm volatile("slbmfee %0,%1" : "=r" (esid) : "r" (i));
2707 asm volatile("slbmfev %0,%1" : "=r" (vsid) : "r" (i));
2708 valid = (esid & SLB_ESID_V);
2709 if (valid | esid | vsid) {
2710 printf("%02d %016lx %016lx", i, esid, vsid);
2711 if (valid) {
2712 llp = vsid & SLB_VSID_LLP;
2713 if (vsid & SLB_VSID_B_1T) {
2714 printf(" 1T ESID=%9lx VSID=%13lx LLP:%3lx \n",
2715 GET_ESID_1T(esid),
2716 (vsid & ~SLB_VSID_B) >> SLB_VSID_SHIFT_1T,
2717 llp);
2718 } else {
2719 printf(" 256M ESID=%9lx VSID=%13lx LLP:%3lx \n",
2720 GET_ESID(esid),
2721 (vsid & ~SLB_VSID_B) >> SLB_VSID_SHIFT,
2722 llp);
2723 }
2724 } else
2725 printf("\n");
2726 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002727 }
2728}
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002729#endif
2730
2731#ifdef CONFIG_PPC_STD_MMU_32
2732void dump_segments(void)
2733{
2734 int i;
2735
2736 printf("sr0-15 =");
2737 for (i = 0; i < 16; ++i)
2738 printf(" %x", mfsrin(i));
2739 printf("\n");
2740}
2741#endif
2742
Benjamin Herrenschmidt5a8a1a22007-11-16 18:23:33 +11002743#ifdef CONFIG_44x
2744static void dump_tlb_44x(void)
2745{
2746 int i;
2747
2748 for (i = 0; i < PPC44x_TLB_SIZE; i++) {
2749 unsigned long w0,w1,w2;
2750 asm volatile("tlbre %0,%1,0" : "=r" (w0) : "r" (i));
2751 asm volatile("tlbre %0,%1,1" : "=r" (w1) : "r" (i));
2752 asm volatile("tlbre %0,%1,2" : "=r" (w2) : "r" (i));
2753 printf("[%02x] %08x %08x %08x ", i, w0, w1, w2);
2754 if (w0 & PPC44x_TLB_VALID) {
2755 printf("V %08x -> %01x%08x %c%c%c%c%c",
2756 w0 & PPC44x_TLB_EPN_MASK,
2757 w1 & PPC44x_TLB_ERPN_MASK,
2758 w1 & PPC44x_TLB_RPN_MASK,
2759 (w2 & PPC44x_TLB_W) ? 'W' : 'w',
2760 (w2 & PPC44x_TLB_I) ? 'I' : 'i',
2761 (w2 & PPC44x_TLB_M) ? 'M' : 'm',
2762 (w2 & PPC44x_TLB_G) ? 'G' : 'g',
2763 (w2 & PPC44x_TLB_E) ? 'E' : 'e');
2764 }
2765 printf("\n");
2766 }
2767}
2768#endif /* CONFIG_44x */
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002769
Benjamin Herrenschmidt03247152010-07-09 15:34:50 +10002770#ifdef CONFIG_PPC_BOOK3E
2771static void dump_tlb_book3e(void)
2772{
2773 u32 mmucfg, pidmask, lpidmask;
2774 u64 ramask;
2775 int i, tlb, ntlbs, pidsz, lpidsz, rasz, lrat = 0;
2776 int mmu_version;
2777 static const char *pgsz_names[] = {
2778 " 1K",
2779 " 2K",
2780 " 4K",
2781 " 8K",
2782 " 16K",
2783 " 32K",
2784 " 64K",
2785 "128K",
2786 "256K",
2787 "512K",
2788 " 1M",
2789 " 2M",
2790 " 4M",
2791 " 8M",
2792 " 16M",
2793 " 32M",
2794 " 64M",
2795 "128M",
2796 "256M",
2797 "512M",
2798 " 1G",
2799 " 2G",
2800 " 4G",
2801 " 8G",
2802 " 16G",
2803 " 32G",
2804 " 64G",
2805 "128G",
2806 "256G",
2807 "512G",
2808 " 1T",
2809 " 2T",
2810 };
2811
2812 /* Gather some infos about the MMU */
2813 mmucfg = mfspr(SPRN_MMUCFG);
2814 mmu_version = (mmucfg & 3) + 1;
2815 ntlbs = ((mmucfg >> 2) & 3) + 1;
2816 pidsz = ((mmucfg >> 6) & 0x1f) + 1;
2817 lpidsz = (mmucfg >> 24) & 0xf;
2818 rasz = (mmucfg >> 16) & 0x7f;
2819 if ((mmu_version > 1) && (mmucfg & 0x10000))
2820 lrat = 1;
2821 printf("Book3E MMU MAV=%d.0,%d TLBs,%d-bit PID,%d-bit LPID,%d-bit RA\n",
2822 mmu_version, ntlbs, pidsz, lpidsz, rasz);
2823 pidmask = (1ul << pidsz) - 1;
2824 lpidmask = (1ul << lpidsz) - 1;
2825 ramask = (1ull << rasz) - 1;
2826
2827 for (tlb = 0; tlb < ntlbs; tlb++) {
2828 u32 tlbcfg;
2829 int nent, assoc, new_cc = 1;
2830 printf("TLB %d:\n------\n", tlb);
2831 switch(tlb) {
2832 case 0:
2833 tlbcfg = mfspr(SPRN_TLB0CFG);
2834 break;
2835 case 1:
2836 tlbcfg = mfspr(SPRN_TLB1CFG);
2837 break;
2838 case 2:
2839 tlbcfg = mfspr(SPRN_TLB2CFG);
2840 break;
2841 case 3:
2842 tlbcfg = mfspr(SPRN_TLB3CFG);
2843 break;
2844 default:
2845 printf("Unsupported TLB number !\n");
2846 continue;
2847 }
2848 nent = tlbcfg & 0xfff;
2849 assoc = (tlbcfg >> 24) & 0xff;
2850 for (i = 0; i < nent; i++) {
2851 u32 mas0 = MAS0_TLBSEL(tlb);
2852 u32 mas1 = MAS1_TSIZE(BOOK3E_PAGESZ_4K);
2853 u64 mas2 = 0;
2854 u64 mas7_mas3;
2855 int esel = i, cc = i;
2856
2857 if (assoc != 0) {
2858 cc = i / assoc;
2859 esel = i % assoc;
2860 mas2 = cc * 0x1000;
2861 }
2862
2863 mas0 |= MAS0_ESEL(esel);
2864 mtspr(SPRN_MAS0, mas0);
2865 mtspr(SPRN_MAS1, mas1);
2866 mtspr(SPRN_MAS2, mas2);
2867 asm volatile("tlbre 0,0,0" : : : "memory");
2868 mas1 = mfspr(SPRN_MAS1);
2869 mas2 = mfspr(SPRN_MAS2);
2870 mas7_mas3 = mfspr(SPRN_MAS7_MAS3);
2871 if (assoc && (i % assoc) == 0)
2872 new_cc = 1;
2873 if (!(mas1 & MAS1_VALID))
2874 continue;
2875 if (assoc == 0)
2876 printf("%04x- ", i);
2877 else if (new_cc)
2878 printf("%04x-%c", cc, 'A' + esel);
2879 else
2880 printf(" |%c", 'A' + esel);
2881 new_cc = 0;
2882 printf(" %016llx %04x %s %c%c AS%c",
2883 mas2 & ~0x3ffull,
2884 (mas1 >> 16) & 0x3fff,
2885 pgsz_names[(mas1 >> 7) & 0x1f],
2886 mas1 & MAS1_IND ? 'I' : ' ',
2887 mas1 & MAS1_IPROT ? 'P' : ' ',
2888 mas1 & MAS1_TS ? '1' : '0');
2889 printf(" %c%c%c%c%c%c%c",
2890 mas2 & MAS2_X0 ? 'a' : ' ',
2891 mas2 & MAS2_X1 ? 'v' : ' ',
2892 mas2 & MAS2_W ? 'w' : ' ',
2893 mas2 & MAS2_I ? 'i' : ' ',
2894 mas2 & MAS2_M ? 'm' : ' ',
2895 mas2 & MAS2_G ? 'g' : ' ',
2896 mas2 & MAS2_E ? 'e' : ' ');
2897 printf(" %016llx", mas7_mas3 & ramask & ~0x7ffull);
2898 if (mas1 & MAS1_IND)
2899 printf(" %s\n",
2900 pgsz_names[(mas7_mas3 >> 1) & 0x1f]);
2901 else
2902 printf(" U%c%c%c S%c%c%c\n",
2903 mas7_mas3 & MAS3_UX ? 'x' : ' ',
2904 mas7_mas3 & MAS3_UW ? 'w' : ' ',
2905 mas7_mas3 & MAS3_UR ? 'r' : ' ',
2906 mas7_mas3 & MAS3_SX ? 'x' : ' ',
2907 mas7_mas3 & MAS3_SW ? 'w' : ' ',
2908 mas7_mas3 & MAS3_SR ? 'r' : ' ');
2909 }
2910 }
2911}
2912#endif /* CONFIG_PPC_BOOK3E */
2913
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002914static void xmon_init(int enable)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002915{
Olaf Heringb13cfd172005-08-04 19:26:42 +02002916 if (enable) {
2917 __debugger = xmon;
2918 __debugger_ipi = xmon_ipi;
2919 __debugger_bpt = xmon_bpt;
2920 __debugger_sstep = xmon_sstep;
2921 __debugger_iabr_match = xmon_iabr_match;
Michael Neuling9422de32012-12-20 14:06:44 +00002922 __debugger_break_match = xmon_break_match;
Olaf Heringb13cfd172005-08-04 19:26:42 +02002923 __debugger_fault_handler = xmon_fault_handler;
2924 } else {
2925 __debugger = NULL;
2926 __debugger_ipi = NULL;
2927 __debugger_bpt = NULL;
2928 __debugger_sstep = NULL;
2929 __debugger_iabr_match = NULL;
Michael Neuling9422de32012-12-20 14:06:44 +00002930 __debugger_break_match = NULL;
Olaf Heringb13cfd172005-08-04 19:26:42 +02002931 __debugger_fault_handler = NULL;
2932 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002933}
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11002934
2935#ifdef CONFIG_MAGIC_SYSRQ
Dmitry Torokhov1495cc92010-08-17 21:15:46 -07002936static void sysrq_handle_xmon(int key)
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11002937{
2938 /* ensure xmon is enabled */
2939 xmon_init(1);
David Howells7d12e782006-10-05 14:55:46 +01002940 debugger(get_irq_regs());
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11002941}
2942
Dmitry Torokhov1495cc92010-08-17 21:15:46 -07002943static struct sysrq_key_op sysrq_xmon_op = {
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11002944 .handler = sysrq_handle_xmon,
zhangwei(Jovi)90a102e2013-04-30 15:28:54 -07002945 .help_msg = "xmon(x)",
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11002946 .action_msg = "Entering xmon",
2947};
2948
2949static int __init setup_xmon_sysrq(void)
2950{
2951 register_sysrq_key('x', &sysrq_xmon_op);
2952 return 0;
2953}
2954__initcall(setup_xmon_sysrq);
2955#endif /* CONFIG_MAGIC_SYSRQ */
Michael Ellerman476792832006-10-03 14:12:08 +10002956
Olaf Heringf5e6a282007-06-24 16:57:08 +10002957static int __initdata xmon_early, xmon_off;
Michael Ellerman476792832006-10-03 14:12:08 +10002958
2959static int __init early_parse_xmon(char *p)
2960{
2961 if (!p || strncmp(p, "early", 5) == 0) {
2962 /* just "xmon" is equivalent to "xmon=early" */
2963 xmon_init(1);
2964 xmon_early = 1;
2965 } else if (strncmp(p, "on", 2) == 0)
2966 xmon_init(1);
2967 else if (strncmp(p, "off", 3) == 0)
2968 xmon_off = 1;
2969 else if (strncmp(p, "nobt", 4) == 0)
2970 xmon_no_auto_backtrace = 1;
2971 else
2972 return 1;
2973
2974 return 0;
2975}
2976early_param("xmon", early_parse_xmon);
2977
2978void __init xmon_setup(void)
2979{
2980#ifdef CONFIG_XMON_DEFAULT
2981 if (!xmon_off)
2982 xmon_init(1);
2983#endif
2984 if (xmon_early)
2985 debugger(NULL);
2986}
Michael Ellermanff8a8f22006-10-24 18:31:27 +02002987
Arnd Bergmanne0555952006-11-27 19:18:55 +01002988#ifdef CONFIG_SPU_BASE
Michael Ellermanff8a8f22006-10-24 18:31:27 +02002989
2990struct spu_info {
2991 struct spu *spu;
2992 u64 saved_mfc_sr1_RW;
2993 u32 saved_spu_runcntl_RW;
Michael Ellerman24a24c82006-11-23 00:46:41 +01002994 unsigned long dump_addr;
Michael Ellermanff8a8f22006-10-24 18:31:27 +02002995 u8 stopped_ok;
2996};
2997
2998#define XMON_NUM_SPUS 16 /* Enough for current hardware */
2999
3000static struct spu_info spu_info[XMON_NUM_SPUS];
3001
3002void xmon_register_spus(struct list_head *list)
3003{
3004 struct spu *spu;
3005
3006 list_for_each_entry(spu, list, full_list) {
3007 if (spu->number >= XMON_NUM_SPUS) {
3008 WARN_ON(1);
3009 continue;
3010 }
3011
3012 spu_info[spu->number].spu = spu;
3013 spu_info[spu->number].stopped_ok = 0;
Michael Ellerman24a24c82006-11-23 00:46:41 +01003014 spu_info[spu->number].dump_addr = (unsigned long)
3015 spu_info[spu->number].spu->local_store;
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003016 }
3017}
3018
3019static void stop_spus(void)
3020{
3021 struct spu *spu;
3022 int i;
3023 u64 tmp;
3024
3025 for (i = 0; i < XMON_NUM_SPUS; i++) {
3026 if (!spu_info[i].spu)
3027 continue;
3028
3029 if (setjmp(bus_error_jmp) == 0) {
3030 catch_memory_errors = 1;
3031 sync();
3032
3033 spu = spu_info[i].spu;
3034
3035 spu_info[i].saved_spu_runcntl_RW =
3036 in_be32(&spu->problem->spu_runcntl_RW);
3037
3038 tmp = spu_mfc_sr1_get(spu);
3039 spu_info[i].saved_mfc_sr1_RW = tmp;
3040
3041 tmp &= ~MFC_STATE1_MASTER_RUN_CONTROL_MASK;
3042 spu_mfc_sr1_set(spu, tmp);
3043
3044 sync();
3045 __delay(200);
3046
3047 spu_info[i].stopped_ok = 1;
Michael Ellerman2a144422006-11-23 00:46:40 +01003048
3049 printf("Stopped spu %.2d (was %s)\n", i,
3050 spu_info[i].saved_spu_runcntl_RW ?
3051 "running" : "stopped");
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003052 } else {
3053 catch_memory_errors = 0;
3054 printf("*** Error stopping spu %.2d\n", i);
3055 }
3056 catch_memory_errors = 0;
3057 }
3058}
3059
3060static void restart_spus(void)
3061{
3062 struct spu *spu;
3063 int i;
3064
3065 for (i = 0; i < XMON_NUM_SPUS; i++) {
3066 if (!spu_info[i].spu)
3067 continue;
3068
3069 if (!spu_info[i].stopped_ok) {
3070 printf("*** Error, spu %d was not successfully stopped"
3071 ", not restarting\n", i);
3072 continue;
3073 }
3074
3075 if (setjmp(bus_error_jmp) == 0) {
3076 catch_memory_errors = 1;
3077 sync();
3078
3079 spu = spu_info[i].spu;
3080 spu_mfc_sr1_set(spu, spu_info[i].saved_mfc_sr1_RW);
3081 out_be32(&spu->problem->spu_runcntl_RW,
3082 spu_info[i].saved_spu_runcntl_RW);
3083
3084 sync();
3085 __delay(200);
3086
3087 printf("Restarted spu %.2d\n", i);
3088 } else {
3089 catch_memory_errors = 0;
3090 printf("*** Error restarting spu %.2d\n", i);
3091 }
3092 catch_memory_errors = 0;
3093 }
3094}
3095
Michael Ellermana8984972006-10-24 18:31:28 +02003096#define DUMP_WIDTH 23
Michael Ellerman437a0702006-11-23 00:46:39 +01003097#define DUMP_VALUE(format, field, value) \
Michael Ellermana8984972006-10-24 18:31:28 +02003098do { \
3099 if (setjmp(bus_error_jmp) == 0) { \
3100 catch_memory_errors = 1; \
3101 sync(); \
3102 printf(" %-*s = "format"\n", DUMP_WIDTH, \
Michael Ellerman437a0702006-11-23 00:46:39 +01003103 #field, value); \
Michael Ellermana8984972006-10-24 18:31:28 +02003104 sync(); \
3105 __delay(200); \
3106 } else { \
3107 catch_memory_errors = 0; \
3108 printf(" %-*s = *** Error reading field.\n", \
3109 DUMP_WIDTH, #field); \
3110 } \
3111 catch_memory_errors = 0; \
3112} while (0)
3113
Michael Ellerman437a0702006-11-23 00:46:39 +01003114#define DUMP_FIELD(obj, format, field) \
3115 DUMP_VALUE(format, field, obj->field)
3116
Michael Ellermana8984972006-10-24 18:31:28 +02003117static void dump_spu_fields(struct spu *spu)
3118{
3119 printf("Dumping spu fields at address %p:\n", spu);
3120
3121 DUMP_FIELD(spu, "0x%x", number);
3122 DUMP_FIELD(spu, "%s", name);
Michael Ellermana8984972006-10-24 18:31:28 +02003123 DUMP_FIELD(spu, "0x%lx", local_store_phys);
3124 DUMP_FIELD(spu, "0x%p", local_store);
3125 DUMP_FIELD(spu, "0x%lx", ls_size);
3126 DUMP_FIELD(spu, "0x%x", node);
3127 DUMP_FIELD(spu, "0x%lx", flags);
Michael Ellermana8984972006-10-24 18:31:28 +02003128 DUMP_FIELD(spu, "%d", class_0_pending);
Luke Browningf3d69e02008-04-27 18:41:55 +00003129 DUMP_FIELD(spu, "0x%lx", class_0_dar);
Luke Browningf3d69e02008-04-27 18:41:55 +00003130 DUMP_FIELD(spu, "0x%lx", class_1_dar);
3131 DUMP_FIELD(spu, "0x%lx", class_1_dsisr);
Michael Ellermana8984972006-10-24 18:31:28 +02003132 DUMP_FIELD(spu, "0x%lx", irqs[0]);
3133 DUMP_FIELD(spu, "0x%lx", irqs[1]);
3134 DUMP_FIELD(spu, "0x%lx", irqs[2]);
3135 DUMP_FIELD(spu, "0x%x", slb_replace);
3136 DUMP_FIELD(spu, "%d", pid);
Michael Ellermana8984972006-10-24 18:31:28 +02003137 DUMP_FIELD(spu, "0x%p", mm);
3138 DUMP_FIELD(spu, "0x%p", ctx);
3139 DUMP_FIELD(spu, "0x%p", rq);
3140 DUMP_FIELD(spu, "0x%p", timestamp);
3141 DUMP_FIELD(spu, "0x%lx", problem_phys);
3142 DUMP_FIELD(spu, "0x%p", problem);
Michael Ellerman437a0702006-11-23 00:46:39 +01003143 DUMP_VALUE("0x%x", problem->spu_runcntl_RW,
3144 in_be32(&spu->problem->spu_runcntl_RW));
3145 DUMP_VALUE("0x%x", problem->spu_status_R,
3146 in_be32(&spu->problem->spu_status_R));
3147 DUMP_VALUE("0x%x", problem->spu_npc_RW,
3148 in_be32(&spu->problem->spu_npc_RW));
Michael Ellermana8984972006-10-24 18:31:28 +02003149 DUMP_FIELD(spu, "0x%p", priv2);
Michael Ellermana9852392006-11-23 00:46:50 +01003150 DUMP_FIELD(spu, "0x%p", pdata);
Michael Ellermana8984972006-10-24 18:31:28 +02003151}
3152
Michael Ellermanaf89fb82006-11-23 00:46:44 +01003153int
3154spu_inst_dump(unsigned long adr, long count, int praddr)
3155{
3156 return generic_inst_dump(adr, count, praddr, print_insn_spu);
3157}
3158
3159static void dump_spu_ls(unsigned long num, int subcmd)
Michael Ellerman24a24c82006-11-23 00:46:41 +01003160{
3161 unsigned long offset, addr, ls_addr;
3162
3163 if (setjmp(bus_error_jmp) == 0) {
3164 catch_memory_errors = 1;
3165 sync();
3166 ls_addr = (unsigned long)spu_info[num].spu->local_store;
3167 sync();
3168 __delay(200);
3169 } else {
3170 catch_memory_errors = 0;
3171 printf("*** Error: accessing spu info for spu %d\n", num);
3172 return;
3173 }
3174 catch_memory_errors = 0;
3175
3176 if (scanhex(&offset))
3177 addr = ls_addr + offset;
3178 else
3179 addr = spu_info[num].dump_addr;
3180
3181 if (addr >= ls_addr + LS_SIZE) {
3182 printf("*** Error: address outside of local store\n");
3183 return;
3184 }
3185
Michael Ellermanaf89fb82006-11-23 00:46:44 +01003186 switch (subcmd) {
3187 case 'i':
3188 addr += spu_inst_dump(addr, 16, 1);
3189 last_cmd = "sdi\n";
3190 break;
3191 default:
3192 prdump(addr, 64);
3193 addr += 64;
3194 last_cmd = "sd\n";
3195 break;
3196 }
Michael Ellerman24a24c82006-11-23 00:46:41 +01003197
3198 spu_info[num].dump_addr = addr;
3199}
3200
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003201static int do_spu_cmd(void)
3202{
Michael Ellerman24a24c82006-11-23 00:46:41 +01003203 static unsigned long num = 0;
Michael Ellermanaf89fb82006-11-23 00:46:44 +01003204 int cmd, subcmd = 0;
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003205
3206 cmd = inchar();
3207 switch (cmd) {
3208 case 's':
3209 stop_spus();
3210 break;
3211 case 'r':
3212 restart_spus();
3213 break;
Michael Ellerman24a24c82006-11-23 00:46:41 +01003214 case 'd':
Michael Ellermanaf89fb82006-11-23 00:46:44 +01003215 subcmd = inchar();
3216 if (isxdigit(subcmd) || subcmd == '\n')
3217 termch = subcmd;
3218 case 'f':
Michael Ellerman24a24c82006-11-23 00:46:41 +01003219 scanhex(&num);
3220 if (num >= XMON_NUM_SPUS || !spu_info[num].spu) {
Michael Ellermana8984972006-10-24 18:31:28 +02003221 printf("*** Error: invalid spu number\n");
Michael Ellerman24a24c82006-11-23 00:46:41 +01003222 return 0;
3223 }
3224
3225 switch (cmd) {
3226 case 'f':
3227 dump_spu_fields(spu_info[num].spu);
3228 break;
3229 default:
Michael Ellermanaf89fb82006-11-23 00:46:44 +01003230 dump_spu_ls(num, subcmd);
Michael Ellerman24a24c82006-11-23 00:46:41 +01003231 break;
3232 }
3233
Michael Ellermana8984972006-10-24 18:31:28 +02003234 break;
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003235 default:
3236 return -1;
3237 }
3238
3239 return 0;
3240}
Arnd Bergmanne0555952006-11-27 19:18:55 +01003241#else /* ! CONFIG_SPU_BASE */
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003242static int do_spu_cmd(void)
3243{
3244 return -1;
3245}
3246#endif