blob: d940234b09ec9dcacac0ce029c33b85eb568355b [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>
Linus Torvalds1da177e2005-04-16 15:20:36 -070027
28#include <asm/ptrace.h>
29#include <asm/string.h>
30#include <asm/prom.h>
31#include <asm/machdep.h>
Paul Mackerrasf78541dc2005-10-28 22:53:37 +100032#include <asm/xmon.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070033#include <asm/processor.h>
34#include <asm/pgtable.h>
35#include <asm/mmu.h>
36#include <asm/mmu_context.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070037#include <asm/cputable.h>
38#include <asm/rtas.h>
39#include <asm/sstep.h>
Paul Mackerrasf583ffc2006-10-10 11:47:07 +100040#include <asm/irq_regs.h>
Michael Ellermanff8a8f22006-10-24 18:31:27 +020041#include <asm/spu.h>
42#include <asm/spu_priv1.h>
Michael Neulingc3b75bd2008-01-18 15:50:30 +110043#include <asm/setjmp.h>
Anton Vorontsov322b4392008-12-17 10:08:55 +000044#include <asm/reg.h>
David Howellsae3a1972012-03-28 18:30:02 +010045#include <asm/debug.h>
Paul Mackerrasf78541dc2005-10-28 22:53:37 +100046
47#ifdef CONFIG_PPC64
Linus Torvalds1da177e2005-04-16 15:20:36 -070048#include <asm/hvcall.h>
Paul Mackerrasf78541dc2005-10-28 22:53:37 +100049#include <asm/paca.h>
50#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -070051
52#include "nonstdio.h"
Michael Ellermane0426042006-11-23 00:46:45 +010053#include "dis-asm.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070054
Linus Torvalds1da177e2005-04-16 15:20:36 -070055#ifdef CONFIG_SMP
Michael Ellerman1c8950f2008-05-08 14:27:17 +100056static cpumask_t cpus_in_xmon = CPU_MASK_NONE;
Linus Torvalds1da177e2005-04-16 15:20:36 -070057static unsigned long xmon_taken = 1;
58static int xmon_owner;
59static int xmon_gate;
Michael Ellermanddadb6b2012-09-13 23:01:31 +000060#else
61#define xmon_owner 0
Linus Torvalds1da177e2005-04-16 15:20:36 -070062#endif /* CONFIG_SMP */
63
Anton Blanchard5be34922010-01-12 00:50:14 +000064static unsigned long in_xmon __read_mostly = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -070065
66static unsigned long adrs;
67static int size = 1;
68#define MAX_DUMP (128 * 1024)
69static unsigned long ndump = 64;
70static unsigned long nidump = 16;
71static unsigned long ncsum = 4096;
72static int termch;
73static char tmpstr[128];
74
Linus Torvalds1da177e2005-04-16 15:20:36 -070075static long bus_error_jmp[JMP_BUF_LEN];
76static int catch_memory_errors;
77static long *xmon_fault_jmp[NR_CPUS];
Linus Torvalds1da177e2005-04-16 15:20:36 -070078
79/* Breakpoint stuff */
80struct bpt {
81 unsigned long address;
82 unsigned int instr[2];
83 atomic_t ref_count;
84 int enabled;
85 unsigned long pad;
86};
87
88/* Bits in bpt.enabled */
89#define BP_IABR_TE 1 /* IABR translation enabled */
90#define BP_IABR 2
91#define BP_TRAP 8
92#define BP_DABR 0x10
93
94#define NBPTS 256
95static struct bpt bpts[NBPTS];
96static struct bpt dabr;
97static struct bpt *iabr;
98static unsigned bpinstr = 0x7fe00008; /* trap */
99
100#define BP_NUM(bp) ((bp) - bpts + 1)
101
102/* Prototypes */
103static int cmds(struct pt_regs *);
104static int mread(unsigned long, void *, int);
105static int mwrite(unsigned long, void *, int);
106static int handle_fault(struct pt_regs *);
107static void byterev(unsigned char *, int);
108static void memex(void);
109static int bsesc(void);
110static void dump(void);
111static void prdump(unsigned long, long);
112static int ppc_inst_dump(unsigned long, long, int);
Vinay Sridharf312deb2009-05-14 23:13:07 +0000113static void dump_log_buf(void);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700114static void backtrace(struct pt_regs *);
115static void excprint(struct pt_regs *);
116static void prregs(struct pt_regs *);
117static void memops(int);
118static void memlocate(void);
119static void memzcan(void);
120static void memdiffs(unsigned char *, unsigned char *, unsigned, unsigned);
121int skipbl(void);
122int scanhex(unsigned long *valp);
123static void scannl(void);
124static int hexdigit(int);
125void getstring(char *, int);
126static void flush_input(void);
127static int inchar(void);
128static void take_input(char *);
129static unsigned long read_spr(int);
130static void write_spr(int, unsigned long);
131static void super_regs(void);
132static void remove_bpts(void);
133static void insert_bpts(void);
134static void remove_cpu_bpts(void);
135static void insert_cpu_bpts(void);
136static struct bpt *at_breakpoint(unsigned long pc);
137static struct bpt *in_breakpoint_table(unsigned long pc, unsigned long *offp);
138static int do_step(struct pt_regs *);
139static void bpt_cmds(void);
140static void cacheflush(void);
141static int cpu_cmd(void);
142static void csum(void);
143static void bootcmds(void);
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000144static void proccall(void);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700145void dump_segments(void);
146static void symbol_lookup(void);
Olaf Hering26c8af52006-09-08 16:29:21 +0200147static void xmon_show_stack(unsigned long sp, unsigned long lr,
148 unsigned long pc);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700149static void xmon_print_symbol(unsigned long address, const char *mid,
150 const char *after);
151static const char *getvecname(unsigned long vec);
152
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200153static int do_spu_cmd(void);
154
Benjamin Herrenschmidt5a8a1a22007-11-16 18:23:33 +1100155#ifdef CONFIG_44x
156static void dump_tlb_44x(void);
157#endif
Benjamin Herrenschmidt03247152010-07-09 15:34:50 +1000158#ifdef CONFIG_PPC_BOOK3E
159static void dump_tlb_book3e(void);
160#endif
Benjamin Herrenschmidt5a8a1a22007-11-16 18:23:33 +1100161
Michael Ellerman9f1067c22008-05-08 14:27:16 +1000162static int xmon_no_auto_backtrace;
Olaf Hering26c8af52006-09-08 16:29:21 +0200163
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000164extern void xmon_enter(void);
165extern void xmon_leave(void);
166
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000167#ifdef CONFIG_PPC64
168#define REG "%.16lx"
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000169#else
170#define REG "%.8lx"
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000171#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700172
173#define GETWORD(v) (((v)[0] << 24) + ((v)[1] << 16) + ((v)[2] << 8) + (v)[3])
174
175#define isxdigit(c) (('0' <= (c) && (c) <= '9') \
176 || ('a' <= (c) && (c) <= 'f') \
177 || ('A' <= (c) && (c) <= 'F'))
178#define isalnum(c) (('0' <= (c) && (c) <= '9') \
179 || ('a' <= (c) && (c) <= 'z') \
180 || ('A' <= (c) && (c) <= 'Z'))
181#define isspace(c) (c == ' ' || c == '\t' || c == 10 || c == 13 || c == 0)
182
183static char *help_string = "\
184Commands:\n\
185 b show breakpoints\n\
186 bd set data breakpoint\n\
187 bi set instruction breakpoint\n\
188 bc clear breakpoint\n"
189#ifdef CONFIG_SMP
190 "\
191 c print cpus stopped in xmon\n\
192 c# try to switch to cpu number h (in hex)\n"
193#endif
194 "\
195 C checksum\n\
196 d dump bytes\n\
197 di dump instructions\n\
198 df dump float values\n\
199 dd dump double values\n\
Michael Ellermanddadb6b2012-09-13 23:01:31 +0000200 dl dump the kernel log buffer\n"
201#ifdef CONFIG_PPC64
202 "\
203 dp[#] dump paca for current cpu, or cpu #\n\
204 dpa dump paca for all possible cpus\n"
205#endif
206 "\
Olaf Hering7e5b5932006-03-08 20:40:28 +0100207 dr dump stream of raw bytes\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700208 e print exception information\n\
209 f flush cache\n\
210 la lookup symbol+offset of specified address\n\
211 ls lookup address of specified symbol\n\
212 m examine/change memory\n\
213 mm move a block of memory\n\
214 ms set a block of memory\n\
215 md compare two blocks of memory\n\
216 ml locate a block of memory\n\
217 mz zero a block of memory\n\
218 mi show information about memory allocation\n\
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000219 p call a procedure\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700220 r print registers\n\
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200221 s single step\n"
Arnd Bergmanne0555952006-11-27 19:18:55 +0100222#ifdef CONFIG_SPU_BASE
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200223" ss stop execution on all spus\n\
Michael Ellermana8984972006-10-24 18:31:28 +0200224 sr restore execution on stopped spus\n\
Michael Ellerman24a24c82006-11-23 00:46:41 +0100225 sf # dump spu fields for spu # (in hex)\n\
Michael Ellermanc99176a2007-02-26 18:14:06 +0900226 sd # dump spu local store for spu # (in hex)\n\
Michael Ellermanaf89fb82006-11-23 00:46:44 +0100227 sdi # disassemble spu local store for spu # (in hex)\n"
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200228#endif
229" S print special registers\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700230 t print backtrace\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700231 x exit monitor and recover\n\
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000232 X exit monitor and dont recover\n"
Jimi Xenidis79873e82011-09-29 11:25:10 +0000233#if defined(CONFIG_PPC64) && !defined(CONFIG_PPC_BOOK3E)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000234" u dump segment table or SLB\n"
Jimi Xenidis79873e82011-09-29 11:25:10 +0000235#elif defined(CONFIG_PPC_STD_MMU_32)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000236" u dump segment registers\n"
Jimi Xenidis79873e82011-09-29 11:25:10 +0000237#elif defined(CONFIG_44x) || defined(CONFIG_PPC_BOOK3E)
Benjamin Herrenschmidt5a8a1a22007-11-16 18:23:33 +1100238" u dump TLB\n"
239#endif
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000240" ? help\n"
241" zr reboot\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700242 zh halt\n"
243;
244
245static struct pt_regs *xmon_regs;
246
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000247static inline void sync(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700248{
249 asm volatile("sync; isync");
250}
251
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000252static inline void store_inst(void *p)
253{
254 asm volatile ("dcbst 0,%0; sync; icbi 0,%0; isync" : : "r" (p));
255}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700256
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000257static inline void cflush(void *p)
258{
259 asm volatile ("dcbf 0,%0; icbi 0,%0" : : "r" (p));
260}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700261
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000262static inline void cinval(void *p)
263{
264 asm volatile ("dcbi 0,%0; icbi 0,%0" : : "r" (p));
265}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700266
267/*
268 * Disable surveillance (the service processor watchdog function)
269 * while we are in xmon.
270 * XXX we should re-enable it when we leave. :)
271 */
272#define SURVEILLANCE_TOKEN 9000
273
274static inline void disable_surveillance(void)
275{
276#ifdef CONFIG_PPC_PSERIES
277 /* Since this can't be a module, args should end up below 4GB. */
278 static struct rtas_args args;
279
280 /*
281 * At this point we have got all the cpus we can into
282 * xmon, so there is hopefully no other cpu calling RTAS
283 * at the moment, even though we don't take rtas.lock.
284 * If we did try to take rtas.lock there would be a
285 * real possibility of deadlock.
286 */
287 args.token = rtas_token("set-indicator");
288 if (args.token == RTAS_UNKNOWN_SERVICE)
289 return;
290 args.nargs = 3;
291 args.nret = 1;
292 args.rets = &args.args[3];
293 args.args[0] = SURVEILLANCE_TOKEN;
294 args.args[1] = 0;
295 args.args[2] = 0;
296 enter_rtas(__pa(&args));
297#endif /* CONFIG_PPC_PSERIES */
298}
299
300#ifdef CONFIG_SMP
301static int xmon_speaker;
302
303static void get_output_lock(void)
304{
305 int me = smp_processor_id() + 0x100;
306 int last_speaker = 0, prev;
307 long timeout;
308
309 if (xmon_speaker == me)
310 return;
311 for (;;) {
312 if (xmon_speaker == 0) {
313 last_speaker = cmpxchg(&xmon_speaker, 0, me);
314 if (last_speaker == 0)
315 return;
316 }
317 timeout = 10000000;
318 while (xmon_speaker == last_speaker) {
319 if (--timeout > 0)
320 continue;
321 /* hostile takeover */
322 prev = cmpxchg(&xmon_speaker, last_speaker, me);
323 if (prev == last_speaker)
324 return;
325 break;
326 }
327 }
328}
329
330static void release_output_lock(void)
331{
332 xmon_speaker = 0;
333}
Michael Ellerman1c8950f2008-05-08 14:27:17 +1000334
335int cpus_are_in_xmon(void)
336{
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000337 return !cpumask_empty(&cpus_in_xmon);
Michael Ellerman1c8950f2008-05-08 14:27:17 +1000338}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700339#endif
340
Josh Boyerdaf8f402009-09-23 03:51:04 +0000341static inline int unrecoverable_excp(struct pt_regs *regs)
342{
Jimi Xenidis08f6d6a2011-09-29 12:05:28 +0000343#if defined(CONFIG_4xx) || defined(CONFIG_PPC_BOOK3E)
Jimi Xenidis66857b32011-09-23 05:40:46 +0000344 /* We have no MSR_RI bit on 4xx or Book3e, so we simply return false */
Josh Boyerdaf8f402009-09-23 03:51:04 +0000345 return 0;
346#else
347 return ((regs->msr & MSR_RI) == 0);
348#endif
349}
350
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000351static int xmon_core(struct pt_regs *regs, int fromipi)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700352{
353 int cmd = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700354 struct bpt *bp;
355 long recurse_jmp[JMP_BUF_LEN];
356 unsigned long offset;
Anton Blanchardf13659e2007-03-21 01:48:34 +1100357 unsigned long flags;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700358#ifdef CONFIG_SMP
359 int cpu;
360 int secondary;
361 unsigned long timeout;
362#endif
363
Anton Blanchardf13659e2007-03-21 01:48:34 +1100364 local_irq_save(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700365
366 bp = in_breakpoint_table(regs->nip, &offset);
367 if (bp != NULL) {
368 regs->nip = bp->address + offset;
369 atomic_dec(&bp->ref_count);
370 }
371
372 remove_cpu_bpts();
373
374#ifdef CONFIG_SMP
375 cpu = smp_processor_id();
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000376 if (cpumask_test_cpu(cpu, &cpus_in_xmon)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700377 get_output_lock();
378 excprint(regs);
379 printf("cpu 0x%x: Exception %lx %s in xmon, "
380 "returning to main loop\n",
381 cpu, regs->trap, getvecname(TRAP(regs)));
Haren Myneni5cb4cc02005-08-03 15:08:18 +1000382 release_output_lock();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700383 longjmp(xmon_fault_jmp[cpu], 1);
384 }
385
386 if (setjmp(recurse_jmp) != 0) {
387 if (!in_xmon || !xmon_gate) {
Haren Myneni5cb4cc02005-08-03 15:08:18 +1000388 get_output_lock();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700389 printf("xmon: WARNING: bad recursive fault "
390 "on cpu 0x%x\n", cpu);
Haren Myneni5cb4cc02005-08-03 15:08:18 +1000391 release_output_lock();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700392 goto waiting;
393 }
394 secondary = !(xmon_taken && cpu == xmon_owner);
395 goto cmdloop;
396 }
397
398 xmon_fault_jmp[cpu] = recurse_jmp;
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000399 cpumask_set_cpu(cpu, &cpus_in_xmon);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700400
401 bp = NULL;
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000402 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) == (MSR_IR|MSR_64BIT))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700403 bp = at_breakpoint(regs->nip);
Josh Boyerdaf8f402009-09-23 03:51:04 +0000404 if (bp || unrecoverable_excp(regs))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700405 fromipi = 0;
406
407 if (!fromipi) {
408 get_output_lock();
409 excprint(regs);
410 if (bp) {
411 printf("cpu 0x%x stopped at breakpoint 0x%x (",
412 cpu, BP_NUM(bp));
413 xmon_print_symbol(regs->nip, " ", ")\n");
414 }
Josh Boyerdaf8f402009-09-23 03:51:04 +0000415 if (unrecoverable_excp(regs))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700416 printf("WARNING: exception is not recoverable, "
417 "can't continue\n");
418 release_output_lock();
419 }
420
421 waiting:
422 secondary = 1;
423 while (secondary && !xmon_gate) {
424 if (in_xmon == 0) {
425 if (fromipi)
426 goto leave;
427 secondary = test_and_set_bit(0, &in_xmon);
428 }
429 barrier();
430 }
431
432 if (!secondary && !xmon_gate) {
433 /* we are the first cpu to come in */
434 /* interrupt other cpu(s) */
435 int ncpus = num_online_cpus();
436
437 xmon_owner = cpu;
438 mb();
439 if (ncpus > 1) {
Milton Millere0476372011-05-10 19:29:06 +0000440 smp_send_debugger_break();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700441 /* wait for other cpus to come in */
442 for (timeout = 100000000; timeout != 0; --timeout) {
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000443 if (cpumask_weight(&cpus_in_xmon) >= ncpus)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700444 break;
445 barrier();
446 }
447 }
448 remove_bpts();
449 disable_surveillance();
450 /* for breakpoint or single step, print the current instr. */
451 if (bp || TRAP(regs) == 0xd00)
452 ppc_inst_dump(regs->nip, 1, 0);
453 printf("enter ? for help\n");
454 mb();
455 xmon_gate = 1;
456 barrier();
457 }
458
459 cmdloop:
460 while (in_xmon) {
461 if (secondary) {
462 if (cpu == xmon_owner) {
463 if (!test_and_set_bit(0, &xmon_taken)) {
464 secondary = 0;
465 continue;
466 }
467 /* missed it */
468 while (cpu == xmon_owner)
469 barrier();
470 }
471 barrier();
472 } else {
473 cmd = cmds(regs);
474 if (cmd != 0) {
475 /* exiting xmon */
476 insert_bpts();
477 xmon_gate = 0;
478 wmb();
479 in_xmon = 0;
480 break;
481 }
482 /* have switched to some other cpu */
483 secondary = 1;
484 }
485 }
486 leave:
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000487 cpumask_clear_cpu(cpu, &cpus_in_xmon);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700488 xmon_fault_jmp[cpu] = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700489#else
490 /* UP is simple... */
491 if (in_xmon) {
492 printf("Exception %lx %s in xmon, returning to main loop\n",
493 regs->trap, getvecname(TRAP(regs)));
494 longjmp(xmon_fault_jmp[0], 1);
495 }
496 if (setjmp(recurse_jmp) == 0) {
497 xmon_fault_jmp[0] = recurse_jmp;
498 in_xmon = 1;
499
500 excprint(regs);
501 bp = at_breakpoint(regs->nip);
502 if (bp) {
503 printf("Stopped at breakpoint %x (", BP_NUM(bp));
504 xmon_print_symbol(regs->nip, " ", ")\n");
505 }
Josh Boyerdaf8f402009-09-23 03:51:04 +0000506 if (unrecoverable_excp(regs))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700507 printf("WARNING: exception is not recoverable, "
508 "can't continue\n");
509 remove_bpts();
510 disable_surveillance();
511 /* for breakpoint or single step, print the current instr. */
512 if (bp || TRAP(regs) == 0xd00)
513 ppc_inst_dump(regs->nip, 1, 0);
514 printf("enter ? for help\n");
515 }
516
517 cmd = cmds(regs);
518
519 insert_bpts();
520 in_xmon = 0;
521#endif
522
Josh Boyercdd39042009-10-05 04:46:05 +0000523#ifdef CONFIG_BOOKE
524 if (regs->msr & MSR_DE) {
525 bp = at_breakpoint(regs->nip);
526 if (bp != NULL) {
527 regs->nip = (unsigned long) &bp->instr[0];
528 atomic_inc(&bp->ref_count);
529 }
530 }
531#else
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000532 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) == (MSR_IR|MSR_64BIT)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700533 bp = at_breakpoint(regs->nip);
534 if (bp != NULL) {
535 int stepped = emulate_step(regs, bp->instr[0]);
536 if (stepped == 0) {
537 regs->nip = (unsigned long) &bp->instr[0];
538 atomic_inc(&bp->ref_count);
539 } else if (stepped < 0) {
540 printf("Couldn't single-step %s instruction\n",
541 (IS_RFID(bp->instr[0])? "rfid": "mtmsrd"));
542 }
543 }
544 }
Josh Boyercdd39042009-10-05 04:46:05 +0000545#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700546 insert_cpu_bpts();
547
Anton Blanchardf13659e2007-03-21 01:48:34 +1100548 local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700549
Paul Mackerras0a730ae2006-10-03 21:32:49 +1000550 return cmd != 'X' && cmd != EOF;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700551}
552
553int xmon(struct pt_regs *excp)
554{
555 struct pt_regs regs;
556
557 if (excp == NULL) {
Anton Vorontsov322b4392008-12-17 10:08:55 +0000558 ppc_save_regs(&regs);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700559 excp = &regs;
560 }
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200561
Linus Torvalds1da177e2005-04-16 15:20:36 -0700562 return xmon_core(excp, 0);
563}
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000564EXPORT_SYMBOL(xmon);
565
Paul Mackerrasf583ffc2006-10-10 11:47:07 +1000566irqreturn_t xmon_irq(int irq, void *d)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000567{
568 unsigned long flags;
569 local_irq_save(flags);
570 printf("Keyboard interrupt\n");
Paul Mackerrasf583ffc2006-10-10 11:47:07 +1000571 xmon(get_irq_regs());
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000572 local_irq_restore(flags);
573 return IRQ_HANDLED;
574}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700575
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000576static int xmon_bpt(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700577{
578 struct bpt *bp;
579 unsigned long offset;
580
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000581 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) != (MSR_IR|MSR_64BIT))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700582 return 0;
583
584 /* Are we at the trap at bp->instr[1] for some bp? */
585 bp = in_breakpoint_table(regs->nip, &offset);
586 if (bp != NULL && offset == 4) {
587 regs->nip = bp->address + 4;
588 atomic_dec(&bp->ref_count);
589 return 1;
590 }
591
592 /* Are we at a breakpoint? */
593 bp = at_breakpoint(regs->nip);
594 if (!bp)
595 return 0;
596
597 xmon_core(regs, 0);
598
599 return 1;
600}
601
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000602static int xmon_sstep(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700603{
604 if (user_mode(regs))
605 return 0;
606 xmon_core(regs, 0);
607 return 1;
608}
609
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000610static int xmon_dabr_match(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700611{
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000612 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) != (MSR_IR|MSR_64BIT))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700613 return 0;
Anton Blanchardfd9648d2005-09-10 16:01:11 +1000614 if (dabr.enabled == 0)
615 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700616 xmon_core(regs, 0);
617 return 1;
618}
619
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000620static int xmon_iabr_match(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700621{
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000622 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) != (MSR_IR|MSR_64BIT))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700623 return 0;
Michael Ellerman9f1067c22008-05-08 14:27:16 +1000624 if (iabr == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700625 return 0;
626 xmon_core(regs, 0);
627 return 1;
628}
629
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000630static int xmon_ipi(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700631{
632#ifdef CONFIG_SMP
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000633 if (in_xmon && !cpumask_test_cpu(smp_processor_id(), &cpus_in_xmon))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700634 xmon_core(regs, 1);
635#endif
636 return 0;
637}
638
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000639static int xmon_fault_handler(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700640{
641 struct bpt *bp;
642 unsigned long offset;
643
644 if (in_xmon && catch_memory_errors)
645 handle_fault(regs); /* doesn't return */
646
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000647 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) == (MSR_IR|MSR_64BIT)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700648 bp = in_breakpoint_table(regs->nip, &offset);
649 if (bp != NULL) {
650 regs->nip = bp->address + offset;
651 atomic_dec(&bp->ref_count);
652 }
653 }
654
655 return 0;
656}
657
Linus Torvalds1da177e2005-04-16 15:20:36 -0700658static struct bpt *at_breakpoint(unsigned long pc)
659{
660 int i;
661 struct bpt *bp;
662
663 bp = bpts;
664 for (i = 0; i < NBPTS; ++i, ++bp)
665 if (bp->enabled && pc == bp->address)
666 return bp;
667 return NULL;
668}
669
670static struct bpt *in_breakpoint_table(unsigned long nip, unsigned long *offp)
671{
672 unsigned long off;
673
674 off = nip - (unsigned long) bpts;
675 if (off >= sizeof(bpts))
676 return NULL;
677 off %= sizeof(struct bpt);
678 if (off != offsetof(struct bpt, instr[0])
679 && off != offsetof(struct bpt, instr[1]))
680 return NULL;
681 *offp = off - offsetof(struct bpt, instr[0]);
682 return (struct bpt *) (nip - off);
683}
684
685static struct bpt *new_breakpoint(unsigned long a)
686{
687 struct bpt *bp;
688
689 a &= ~3UL;
690 bp = at_breakpoint(a);
691 if (bp)
692 return bp;
693
694 for (bp = bpts; bp < &bpts[NBPTS]; ++bp) {
695 if (!bp->enabled && atomic_read(&bp->ref_count) == 0) {
696 bp->address = a;
697 bp->instr[1] = bpinstr;
698 store_inst(&bp->instr[1]);
699 return bp;
700 }
701 }
702
703 printf("Sorry, no free breakpoints. Please clear one first.\n");
704 return NULL;
705}
706
707static void insert_bpts(void)
708{
709 int i;
710 struct bpt *bp;
711
712 bp = bpts;
713 for (i = 0; i < NBPTS; ++i, ++bp) {
714 if ((bp->enabled & (BP_TRAP|BP_IABR)) == 0)
715 continue;
716 if (mread(bp->address, &bp->instr[0], 4) != 4) {
717 printf("Couldn't read instruction at %lx, "
718 "disabling breakpoint there\n", bp->address);
719 bp->enabled = 0;
720 continue;
721 }
722 if (IS_MTMSRD(bp->instr[0]) || IS_RFID(bp->instr[0])) {
723 printf("Breakpoint at %lx is on an mtmsrd or rfid "
724 "instruction, disabling it\n", bp->address);
725 bp->enabled = 0;
726 continue;
727 }
728 store_inst(&bp->instr[0]);
729 if (bp->enabled & BP_IABR)
730 continue;
731 if (mwrite(bp->address, &bpinstr, 4) != 4) {
732 printf("Couldn't write instruction at %lx, "
733 "disabling breakpoint there\n", bp->address);
734 bp->enabled &= ~BP_TRAP;
735 continue;
736 }
737 store_inst((void *)bp->address);
738 }
739}
740
741static void insert_cpu_bpts(void)
742{
743 if (dabr.enabled)
Michael Neuling4474ef02012-09-06 21:24:56 +0000744 set_dabr(dabr.address | (dabr.enabled & 7), DABRX_ALL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700745 if (iabr && cpu_has_feature(CPU_FTR_IABR))
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000746 mtspr(SPRN_IABR, iabr->address
Linus Torvalds1da177e2005-04-16 15:20:36 -0700747 | (iabr->enabled & (BP_IABR|BP_IABR_TE)));
748}
749
750static void remove_bpts(void)
751{
752 int i;
753 struct bpt *bp;
754 unsigned instr;
755
756 bp = bpts;
757 for (i = 0; i < NBPTS; ++i, ++bp) {
758 if ((bp->enabled & (BP_TRAP|BP_IABR)) != BP_TRAP)
759 continue;
760 if (mread(bp->address, &instr, 4) == 4
761 && instr == bpinstr
762 && mwrite(bp->address, &bp->instr, 4) != 4)
763 printf("Couldn't remove breakpoint at %lx\n",
764 bp->address);
765 else
766 store_inst((void *)bp->address);
767 }
768}
769
770static void remove_cpu_bpts(void)
771{
Michael Neuling4474ef02012-09-06 21:24:56 +0000772 set_dabr(0, 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700773 if (cpu_has_feature(CPU_FTR_IABR))
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000774 mtspr(SPRN_IABR, 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700775}
776
777/* Command interpreting routine */
778static char *last_cmd;
779
780static int
781cmds(struct pt_regs *excp)
782{
783 int cmd = 0;
784
785 last_cmd = NULL;
786 xmon_regs = excp;
Olaf Hering26c8af52006-09-08 16:29:21 +0200787
788 if (!xmon_no_auto_backtrace) {
789 xmon_no_auto_backtrace = 1;
790 xmon_show_stack(excp->gpr[1], excp->link, excp->nip);
791 }
792
Linus Torvalds1da177e2005-04-16 15:20:36 -0700793 for(;;) {
794#ifdef CONFIG_SMP
795 printf("%x:", smp_processor_id());
796#endif /* CONFIG_SMP */
797 printf("mon> ");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700798 flush_input();
799 termch = 0;
800 cmd = skipbl();
801 if( cmd == '\n' ) {
802 if (last_cmd == NULL)
803 continue;
804 take_input(last_cmd);
805 last_cmd = NULL;
806 cmd = inchar();
807 }
808 switch (cmd) {
809 case 'm':
810 cmd = inchar();
811 switch (cmd) {
812 case 'm':
813 case 's':
814 case 'd':
815 memops(cmd);
816 break;
817 case 'l':
818 memlocate();
819 break;
820 case 'z':
821 memzcan();
822 break;
823 case 'i':
David Rientjesb2b755b2011-03-24 15:18:15 -0700824 show_mem(0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700825 break;
826 default:
827 termch = cmd;
828 memex();
829 }
830 break;
831 case 'd':
832 dump();
833 break;
834 case 'l':
835 symbol_lookup();
836 break;
837 case 'r':
838 prregs(excp); /* print regs */
839 break;
840 case 'e':
841 excprint(excp);
842 break;
843 case 'S':
844 super_regs();
845 break;
846 case 't':
847 backtrace(excp);
848 break;
849 case 'f':
850 cacheflush();
851 break;
852 case 's':
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200853 if (do_spu_cmd() == 0)
854 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700855 if (do_step(excp))
856 return cmd;
857 break;
858 case 'x':
859 case 'X':
Benjamin Herrenschmidtbb6b9b22005-11-30 16:54:12 +1100860 return cmd;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700861 case EOF:
Benjamin Herrenschmidtbb6b9b22005-11-30 16:54:12 +1100862 printf(" <no input ...>\n");
863 mdelay(2000);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700864 return cmd;
865 case '?':
Ishizaki Kou4d404ed2007-07-18 19:26:40 +1000866 xmon_puts(help_string);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700867 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700868 case 'b':
869 bpt_cmds();
870 break;
871 case 'C':
872 csum();
873 break;
874 case 'c':
875 if (cpu_cmd())
876 return 0;
877 break;
878 case 'z':
879 bootcmds();
880 break;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000881 case 'p':
882 proccall();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700883 break;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000884#ifdef CONFIG_PPC_STD_MMU
Linus Torvalds1da177e2005-04-16 15:20:36 -0700885 case 'u':
886 dump_segments();
887 break;
Jimi Xenidis79873e82011-09-29 11:25:10 +0000888#elif defined(CONFIG_4xx)
Benjamin Herrenschmidt5a8a1a22007-11-16 18:23:33 +1100889 case 'u':
890 dump_tlb_44x();
891 break;
Jimi Xenidis79873e82011-09-29 11:25:10 +0000892#elif defined(CONFIG_PPC_BOOK3E)
Benjamin Herrenschmidt03247152010-07-09 15:34:50 +1000893 case 'u':
894 dump_tlb_book3e();
895 break;
896#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700897 default:
898 printf("Unrecognized command: ");
Michael Ellermane3bc8042012-08-23 22:09:13 +0000899 do {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700900 if (' ' < cmd && cmd <= '~')
901 putchar(cmd);
902 else
903 printf("\\x%x", cmd);
904 cmd = inchar();
Michael Ellermane3bc8042012-08-23 22:09:13 +0000905 } while (cmd != '\n');
Linus Torvalds1da177e2005-04-16 15:20:36 -0700906 printf(" (type ? for help)\n");
907 break;
908 }
909 }
910}
911
Josh Boyercdd39042009-10-05 04:46:05 +0000912#ifdef CONFIG_BOOKE
913static int do_step(struct pt_regs *regs)
914{
915 regs->msr |= MSR_DE;
916 mtspr(SPRN_DBCR0, mfspr(SPRN_DBCR0) | DBCR0_IC | DBCR0_IDM);
917 return 1;
918}
919#else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700920/*
921 * Step a single instruction.
922 * Some instructions we emulate, others we execute with MSR_SE set.
923 */
924static int do_step(struct pt_regs *regs)
925{
926 unsigned int instr;
927 int stepped;
928
929 /* check we are in 64-bit kernel mode, translation enabled */
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000930 if ((regs->msr & (MSR_64BIT|MSR_PR|MSR_IR)) == (MSR_64BIT|MSR_IR)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700931 if (mread(regs->nip, &instr, 4) == 4) {
932 stepped = emulate_step(regs, instr);
933 if (stepped < 0) {
934 printf("Couldn't single-step %s instruction\n",
935 (IS_RFID(instr)? "rfid": "mtmsrd"));
936 return 0;
937 }
938 if (stepped > 0) {
939 regs->trap = 0xd00 | (regs->trap & 1);
940 printf("stepped to ");
941 xmon_print_symbol(regs->nip, " ", "\n");
942 ppc_inst_dump(regs->nip, 1, 0);
943 return 0;
944 }
945 }
946 }
947 regs->msr |= MSR_SE;
948 return 1;
949}
Josh Boyercdd39042009-10-05 04:46:05 +0000950#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700951
952static void bootcmds(void)
953{
954 int cmd;
955
956 cmd = inchar();
957 if (cmd == 'r')
958 ppc_md.restart(NULL);
959 else if (cmd == 'h')
960 ppc_md.halt();
961 else if (cmd == 'p')
962 ppc_md.power_off();
963}
964
965static int cpu_cmd(void)
966{
967#ifdef CONFIG_SMP
968 unsigned long cpu;
969 int timeout;
970 int count;
971
972 if (!scanhex(&cpu)) {
973 /* print cpus waiting or in xmon */
974 printf("cpus stopped:");
975 count = 0;
Anton Blanchardbc1d7702012-06-28 19:28:57 +0000976 for_each_possible_cpu(cpu) {
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000977 if (cpumask_test_cpu(cpu, &cpus_in_xmon)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700978 if (count == 0)
979 printf(" %x", cpu);
980 ++count;
981 } else {
982 if (count > 1)
983 printf("-%x", cpu - 1);
984 count = 0;
985 }
986 }
987 if (count > 1)
988 printf("-%x", NR_CPUS - 1);
989 printf("\n");
990 return 0;
991 }
992 /* try to switch to cpu specified */
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000993 if (!cpumask_test_cpu(cpu, &cpus_in_xmon)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700994 printf("cpu 0x%x isn't in xmon\n", cpu);
995 return 0;
996 }
997 xmon_taken = 0;
998 mb();
999 xmon_owner = cpu;
1000 timeout = 10000000;
1001 while (!xmon_taken) {
1002 if (--timeout == 0) {
1003 if (test_and_set_bit(0, &xmon_taken))
1004 break;
1005 /* take control back */
1006 mb();
1007 xmon_owner = smp_processor_id();
1008 printf("cpu %u didn't take control\n", cpu);
1009 return 0;
1010 }
1011 barrier();
1012 }
1013 return 1;
1014#else
1015 return 0;
1016#endif /* CONFIG_SMP */
1017}
1018
1019static unsigned short fcstab[256] = {
1020 0x0000, 0x1189, 0x2312, 0x329b, 0x4624, 0x57ad, 0x6536, 0x74bf,
1021 0x8c48, 0x9dc1, 0xaf5a, 0xbed3, 0xca6c, 0xdbe5, 0xe97e, 0xf8f7,
1022 0x1081, 0x0108, 0x3393, 0x221a, 0x56a5, 0x472c, 0x75b7, 0x643e,
1023 0x9cc9, 0x8d40, 0xbfdb, 0xae52, 0xdaed, 0xcb64, 0xf9ff, 0xe876,
1024 0x2102, 0x308b, 0x0210, 0x1399, 0x6726, 0x76af, 0x4434, 0x55bd,
1025 0xad4a, 0xbcc3, 0x8e58, 0x9fd1, 0xeb6e, 0xfae7, 0xc87c, 0xd9f5,
1026 0x3183, 0x200a, 0x1291, 0x0318, 0x77a7, 0x662e, 0x54b5, 0x453c,
1027 0xbdcb, 0xac42, 0x9ed9, 0x8f50, 0xfbef, 0xea66, 0xd8fd, 0xc974,
1028 0x4204, 0x538d, 0x6116, 0x709f, 0x0420, 0x15a9, 0x2732, 0x36bb,
1029 0xce4c, 0xdfc5, 0xed5e, 0xfcd7, 0x8868, 0x99e1, 0xab7a, 0xbaf3,
1030 0x5285, 0x430c, 0x7197, 0x601e, 0x14a1, 0x0528, 0x37b3, 0x263a,
1031 0xdecd, 0xcf44, 0xfddf, 0xec56, 0x98e9, 0x8960, 0xbbfb, 0xaa72,
1032 0x6306, 0x728f, 0x4014, 0x519d, 0x2522, 0x34ab, 0x0630, 0x17b9,
1033 0xef4e, 0xfec7, 0xcc5c, 0xddd5, 0xa96a, 0xb8e3, 0x8a78, 0x9bf1,
1034 0x7387, 0x620e, 0x5095, 0x411c, 0x35a3, 0x242a, 0x16b1, 0x0738,
1035 0xffcf, 0xee46, 0xdcdd, 0xcd54, 0xb9eb, 0xa862, 0x9af9, 0x8b70,
1036 0x8408, 0x9581, 0xa71a, 0xb693, 0xc22c, 0xd3a5, 0xe13e, 0xf0b7,
1037 0x0840, 0x19c9, 0x2b52, 0x3adb, 0x4e64, 0x5fed, 0x6d76, 0x7cff,
1038 0x9489, 0x8500, 0xb79b, 0xa612, 0xd2ad, 0xc324, 0xf1bf, 0xe036,
1039 0x18c1, 0x0948, 0x3bd3, 0x2a5a, 0x5ee5, 0x4f6c, 0x7df7, 0x6c7e,
1040 0xa50a, 0xb483, 0x8618, 0x9791, 0xe32e, 0xf2a7, 0xc03c, 0xd1b5,
1041 0x2942, 0x38cb, 0x0a50, 0x1bd9, 0x6f66, 0x7eef, 0x4c74, 0x5dfd,
1042 0xb58b, 0xa402, 0x9699, 0x8710, 0xf3af, 0xe226, 0xd0bd, 0xc134,
1043 0x39c3, 0x284a, 0x1ad1, 0x0b58, 0x7fe7, 0x6e6e, 0x5cf5, 0x4d7c,
1044 0xc60c, 0xd785, 0xe51e, 0xf497, 0x8028, 0x91a1, 0xa33a, 0xb2b3,
1045 0x4a44, 0x5bcd, 0x6956, 0x78df, 0x0c60, 0x1de9, 0x2f72, 0x3efb,
1046 0xd68d, 0xc704, 0xf59f, 0xe416, 0x90a9, 0x8120, 0xb3bb, 0xa232,
1047 0x5ac5, 0x4b4c, 0x79d7, 0x685e, 0x1ce1, 0x0d68, 0x3ff3, 0x2e7a,
1048 0xe70e, 0xf687, 0xc41c, 0xd595, 0xa12a, 0xb0a3, 0x8238, 0x93b1,
1049 0x6b46, 0x7acf, 0x4854, 0x59dd, 0x2d62, 0x3ceb, 0x0e70, 0x1ff9,
1050 0xf78f, 0xe606, 0xd49d, 0xc514, 0xb1ab, 0xa022, 0x92b9, 0x8330,
1051 0x7bc7, 0x6a4e, 0x58d5, 0x495c, 0x3de3, 0x2c6a, 0x1ef1, 0x0f78
1052};
1053
1054#define FCS(fcs, c) (((fcs) >> 8) ^ fcstab[((fcs) ^ (c)) & 0xff])
1055
1056static void
1057csum(void)
1058{
1059 unsigned int i;
1060 unsigned short fcs;
1061 unsigned char v;
1062
1063 if (!scanhex(&adrs))
1064 return;
1065 if (!scanhex(&ncsum))
1066 return;
1067 fcs = 0xffff;
1068 for (i = 0; i < ncsum; ++i) {
1069 if (mread(adrs+i, &v, 1) == 0) {
1070 printf("csum stopped at %x\n", adrs+i);
1071 break;
1072 }
1073 fcs = FCS(fcs, v);
1074 }
1075 printf("%x\n", fcs);
1076}
1077
1078/*
1079 * Check if this is a suitable place to put a breakpoint.
1080 */
1081static long check_bp_loc(unsigned long addr)
1082{
1083 unsigned int instr;
1084
1085 addr &= ~3;
Michael Ellerman51fae6de2005-12-04 18:39:15 +11001086 if (!is_kernel_addr(addr)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001087 printf("Breakpoints may only be placed at kernel addresses\n");
1088 return 0;
1089 }
1090 if (!mread(addr, &instr, sizeof(instr))) {
1091 printf("Can't read instruction at address %lx\n", addr);
1092 return 0;
1093 }
1094 if (IS_MTMSRD(instr) || IS_RFID(instr)) {
1095 printf("Breakpoints may not be placed on mtmsrd or rfid "
1096 "instructions\n");
1097 return 0;
1098 }
1099 return 1;
1100}
1101
Michael Ellermane3bc8042012-08-23 22:09:13 +00001102static char *breakpoint_help_string =
Linus Torvalds1da177e2005-04-16 15:20:36 -07001103 "Breakpoint command usage:\n"
1104 "b show breakpoints\n"
1105 "b <addr> [cnt] set breakpoint at given instr addr\n"
1106 "bc clear all breakpoints\n"
1107 "bc <n/addr> clear breakpoint number n or at addr\n"
1108 "bi <addr> [cnt] set hardware instr breakpoint (POWER3/RS64 only)\n"
1109 "bd <addr> [cnt] set hardware data breakpoint\n"
1110 "";
1111
1112static void
1113bpt_cmds(void)
1114{
1115 int cmd;
1116 unsigned long a;
1117 int mode, i;
1118 struct bpt *bp;
1119 const char badaddr[] = "Only kernel addresses are permitted "
1120 "for breakpoints\n";
1121
1122 cmd = inchar();
1123 switch (cmd) {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001124#ifndef CONFIG_8xx
Linus Torvalds1da177e2005-04-16 15:20:36 -07001125 case 'd': /* bd - hardware data breakpoint */
1126 mode = 7;
1127 cmd = inchar();
1128 if (cmd == 'r')
1129 mode = 5;
1130 else if (cmd == 'w')
1131 mode = 6;
1132 else
1133 termch = cmd;
1134 dabr.address = 0;
1135 dabr.enabled = 0;
1136 if (scanhex(&dabr.address)) {
Michael Ellerman51fae6de2005-12-04 18:39:15 +11001137 if (!is_kernel_addr(dabr.address)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001138 printf(badaddr);
1139 break;
1140 }
1141 dabr.address &= ~7;
1142 dabr.enabled = mode | BP_DABR;
1143 }
1144 break;
1145
1146 case 'i': /* bi - hardware instr breakpoint */
1147 if (!cpu_has_feature(CPU_FTR_IABR)) {
1148 printf("Hardware instruction breakpoint "
1149 "not supported on this cpu\n");
1150 break;
1151 }
1152 if (iabr) {
1153 iabr->enabled &= ~(BP_IABR | BP_IABR_TE);
1154 iabr = NULL;
1155 }
1156 if (!scanhex(&a))
1157 break;
1158 if (!check_bp_loc(a))
1159 break;
1160 bp = new_breakpoint(a);
1161 if (bp != NULL) {
1162 bp->enabled |= BP_IABR | BP_IABR_TE;
1163 iabr = bp;
1164 }
1165 break;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001166#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001167
1168 case 'c':
1169 if (!scanhex(&a)) {
1170 /* clear all breakpoints */
1171 for (i = 0; i < NBPTS; ++i)
1172 bpts[i].enabled = 0;
1173 iabr = NULL;
1174 dabr.enabled = 0;
1175 printf("All breakpoints cleared\n");
1176 break;
1177 }
1178
1179 if (a <= NBPTS && a >= 1) {
1180 /* assume a breakpoint number */
1181 bp = &bpts[a-1]; /* bp nums are 1 based */
1182 } else {
1183 /* assume a breakpoint address */
1184 bp = at_breakpoint(a);
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001185 if (bp == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001186 printf("No breakpoint at %x\n", a);
1187 break;
1188 }
1189 }
1190
1191 printf("Cleared breakpoint %x (", BP_NUM(bp));
1192 xmon_print_symbol(bp->address, " ", ")\n");
1193 bp->enabled = 0;
1194 break;
1195
1196 default:
1197 termch = cmd;
Michael Ellermane3bc8042012-08-23 22:09:13 +00001198 cmd = skipbl();
Linus Torvalds1da177e2005-04-16 15:20:36 -07001199 if (cmd == '?') {
1200 printf(breakpoint_help_string);
1201 break;
1202 }
1203 termch = cmd;
1204 if (!scanhex(&a)) {
1205 /* print all breakpoints */
1206 printf(" type address\n");
1207 if (dabr.enabled) {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001208 printf(" data "REG" [", dabr.address);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001209 if (dabr.enabled & 1)
1210 printf("r");
1211 if (dabr.enabled & 2)
1212 printf("w");
1213 printf("]\n");
1214 }
1215 for (bp = bpts; bp < &bpts[NBPTS]; ++bp) {
1216 if (!bp->enabled)
1217 continue;
1218 printf("%2x %s ", BP_NUM(bp),
1219 (bp->enabled & BP_IABR)? "inst": "trap");
1220 xmon_print_symbol(bp->address, " ", "\n");
1221 }
1222 break;
1223 }
1224
1225 if (!check_bp_loc(a))
1226 break;
1227 bp = new_breakpoint(a);
1228 if (bp != NULL)
1229 bp->enabled |= BP_TRAP;
1230 break;
1231 }
1232}
1233
1234/* Very cheap human name for vector lookup. */
1235static
1236const char *getvecname(unsigned long vec)
1237{
1238 char *ret;
1239
1240 switch (vec) {
1241 case 0x100: ret = "(System Reset)"; break;
1242 case 0x200: ret = "(Machine Check)"; break;
1243 case 0x300: ret = "(Data Access)"; break;
1244 case 0x380: ret = "(Data SLB Access)"; break;
1245 case 0x400: ret = "(Instruction Access)"; break;
1246 case 0x480: ret = "(Instruction SLB Access)"; break;
1247 case 0x500: ret = "(Hardware Interrupt)"; break;
1248 case 0x600: ret = "(Alignment)"; break;
1249 case 0x700: ret = "(Program Check)"; break;
1250 case 0x800: ret = "(FPU Unavailable)"; break;
1251 case 0x900: ret = "(Decrementer)"; break;
1252 case 0xc00: ret = "(System Call)"; break;
1253 case 0xd00: ret = "(Single Step)"; break;
1254 case 0xf00: ret = "(Performance Monitor)"; break;
1255 case 0xf20: ret = "(Altivec Unavailable)"; break;
1256 case 0x1300: ret = "(Instruction Breakpoint)"; break;
1257 default: ret = "";
1258 }
1259 return ret;
1260}
1261
1262static void get_function_bounds(unsigned long pc, unsigned long *startp,
1263 unsigned long *endp)
1264{
1265 unsigned long size, offset;
1266 const char *name;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001267
1268 *startp = *endp = 0;
1269 if (pc == 0)
1270 return;
1271 if (setjmp(bus_error_jmp) == 0) {
1272 catch_memory_errors = 1;
1273 sync();
Alexey Dobriyanffb45122007-05-08 00:28:41 -07001274 name = kallsyms_lookup(pc, &size, &offset, NULL, tmpstr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001275 if (name != NULL) {
1276 *startp = pc - offset;
1277 *endp = pc - offset + size;
1278 }
1279 sync();
1280 }
1281 catch_memory_errors = 0;
1282}
1283
1284static int xmon_depth_to_print = 64;
1285
Benjamin Herrenschmidtec2b36b2008-04-17 14:34:59 +10001286#define LRSAVE_OFFSET (STACK_FRAME_LR_SAVE * sizeof(unsigned long))
1287#define MARKER_OFFSET (STACK_FRAME_MARKER * sizeof(unsigned long))
1288
Linus Torvalds1da177e2005-04-16 15:20:36 -07001289static void xmon_show_stack(unsigned long sp, unsigned long lr,
1290 unsigned long pc)
1291{
1292 unsigned long ip;
1293 unsigned long newsp;
1294 unsigned long marker;
1295 int count = 0;
1296 struct pt_regs regs;
1297
1298 do {
1299 if (sp < PAGE_OFFSET) {
1300 if (sp != 0)
1301 printf("SP (%lx) is in userspace\n", sp);
1302 break;
1303 }
1304
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001305 if (!mread(sp + LRSAVE_OFFSET, &ip, sizeof(unsigned long))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001306 || !mread(sp, &newsp, sizeof(unsigned long))) {
1307 printf("Couldn't read stack frame at %lx\n", sp);
1308 break;
1309 }
1310
1311 /*
1312 * For the first stack frame, try to work out if
1313 * LR and/or the saved LR value in the bottommost
1314 * stack frame are valid.
1315 */
1316 if ((pc | lr) != 0) {
1317 unsigned long fnstart, fnend;
1318 unsigned long nextip;
1319 int printip = 1;
1320
1321 get_function_bounds(pc, &fnstart, &fnend);
1322 nextip = 0;
1323 if (newsp > sp)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001324 mread(newsp + LRSAVE_OFFSET, &nextip,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001325 sizeof(unsigned long));
1326 if (lr == ip) {
1327 if (lr < PAGE_OFFSET
1328 || (fnstart <= lr && lr < fnend))
1329 printip = 0;
1330 } else if (lr == nextip) {
1331 printip = 0;
1332 } else if (lr >= PAGE_OFFSET
1333 && !(fnstart <= lr && lr < fnend)) {
1334 printf("[link register ] ");
1335 xmon_print_symbol(lr, " ", "\n");
1336 }
1337 if (printip) {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001338 printf("["REG"] ", sp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001339 xmon_print_symbol(ip, " ", " (unreliable)\n");
1340 }
1341 pc = lr = 0;
1342
1343 } else {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001344 printf("["REG"] ", sp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001345 xmon_print_symbol(ip, " ", "\n");
1346 }
1347
1348 /* Look for "regshere" marker to see if this is
1349 an exception frame. */
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001350 if (mread(sp + MARKER_OFFSET, &marker, sizeof(unsigned long))
Benjamin Herrenschmidtec2b36b2008-04-17 14:34:59 +10001351 && marker == STACK_FRAME_REGS_MARKER) {
Michael Ellermanc4de3802012-10-09 04:20:35 +00001352 if (mread(sp + STACK_FRAME_OVERHEAD, &regs, sizeof(regs))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001353 != sizeof(regs)) {
1354 printf("Couldn't read registers at %lx\n",
Michael Ellermanc4de3802012-10-09 04:20:35 +00001355 sp + STACK_FRAME_OVERHEAD);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001356 break;
1357 }
Michael Ellermane3bc8042012-08-23 22:09:13 +00001358 printf("--- Exception: %lx %s at ", regs.trap,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001359 getvecname(TRAP(&regs)));
1360 pc = regs.nip;
1361 lr = regs.link;
1362 xmon_print_symbol(pc, " ", "\n");
1363 }
1364
1365 if (newsp == 0)
1366 break;
1367
1368 sp = newsp;
1369 } while (count++ < xmon_depth_to_print);
1370}
1371
1372static void backtrace(struct pt_regs *excp)
1373{
1374 unsigned long sp;
1375
1376 if (scanhex(&sp))
1377 xmon_show_stack(sp, 0, 0);
1378 else
1379 xmon_show_stack(excp->gpr[1], excp->link, excp->nip);
1380 scannl();
1381}
1382
1383static void print_bug_trap(struct pt_regs *regs)
1384{
Paul Mackerrasebdba9a2008-10-31 21:34:09 +11001385#ifdef CONFIG_BUG
Jeremy Fitzhardinge73c9cea2006-12-08 03:30:41 -08001386 const struct bug_entry *bug;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001387 unsigned long addr;
1388
1389 if (regs->msr & MSR_PR)
1390 return; /* not in kernel */
1391 addr = regs->nip; /* address of trap instruction */
1392 if (addr < PAGE_OFFSET)
1393 return;
1394 bug = find_bug(regs->nip);
1395 if (bug == NULL)
1396 return;
Jeremy Fitzhardinge73c9cea2006-12-08 03:30:41 -08001397 if (is_warning_bug(bug))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001398 return;
1399
Stephen Rothwell0a7c7ef2007-03-04 17:05:34 +11001400#ifdef CONFIG_DEBUG_BUGVERBOSE
Jeremy Fitzhardinge73c9cea2006-12-08 03:30:41 -08001401 printf("kernel BUG at %s:%u!\n",
1402 bug->file, bug->line);
Stephen Rothwell0a7c7ef2007-03-04 17:05:34 +11001403#else
1404 printf("kernel BUG at %p!\n", (void *)bug->bug_addr);
1405#endif
Paul Mackerrasebdba9a2008-10-31 21:34:09 +11001406#endif /* CONFIG_BUG */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001407}
1408
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001409static void excprint(struct pt_regs *fp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001410{
1411 unsigned long trap;
1412
1413#ifdef CONFIG_SMP
1414 printf("cpu 0x%x: ", smp_processor_id());
1415#endif /* CONFIG_SMP */
1416
1417 trap = TRAP(fp);
1418 printf("Vector: %lx %s at [%lx]\n", fp->trap, getvecname(trap), fp);
1419 printf(" pc: ");
1420 xmon_print_symbol(fp->nip, ": ", "\n");
1421
1422 printf(" lr: ", fp->link);
1423 xmon_print_symbol(fp->link, ": ", "\n");
1424
1425 printf(" sp: %lx\n", fp->gpr[1]);
1426 printf(" msr: %lx\n", fp->msr);
1427
1428 if (trap == 0x300 || trap == 0x380 || trap == 0x600) {
1429 printf(" dar: %lx\n", fp->dar);
1430 if (trap != 0x380)
1431 printf(" dsisr: %lx\n", fp->dsisr);
1432 }
1433
1434 printf(" current = 0x%lx\n", current);
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001435#ifdef CONFIG_PPC64
Benjamin Herrenschmidt7230c562012-03-06 18:27:59 +11001436 printf(" paca = 0x%lx\t softe: %d\t irq_happened: 0x%02x\n",
1437 local_paca, local_paca->soft_enabled, local_paca->irq_happened);
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001438#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001439 if (current) {
1440 printf(" pid = %ld, comm = %s\n",
1441 current->pid, current->comm);
1442 }
1443
1444 if (trap == 0x700)
1445 print_bug_trap(fp);
1446}
1447
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001448static void prregs(struct pt_regs *fp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001449{
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001450 int n, trap;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001451 unsigned long base;
1452 struct pt_regs regs;
1453
1454 if (scanhex(&base)) {
1455 if (setjmp(bus_error_jmp) == 0) {
1456 catch_memory_errors = 1;
1457 sync();
1458 regs = *(struct pt_regs *)base;
1459 sync();
1460 __delay(200);
1461 } else {
1462 catch_memory_errors = 0;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001463 printf("*** Error reading registers from "REG"\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001464 base);
1465 return;
1466 }
1467 catch_memory_errors = 0;
1468 fp = &regs;
1469 }
1470
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001471#ifdef CONFIG_PPC64
Linus Torvalds1da177e2005-04-16 15:20:36 -07001472 if (FULL_REGS(fp)) {
1473 for (n = 0; n < 16; ++n)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001474 printf("R%.2ld = "REG" R%.2ld = "REG"\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001475 n, fp->gpr[n], n+16, fp->gpr[n+16]);
1476 } else {
1477 for (n = 0; n < 7; ++n)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001478 printf("R%.2ld = "REG" R%.2ld = "REG"\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001479 n, fp->gpr[n], n+7, fp->gpr[n+7]);
1480 }
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001481#else
1482 for (n = 0; n < 32; ++n) {
1483 printf("R%.2d = %.8x%s", n, fp->gpr[n],
1484 (n & 3) == 3? "\n": " ");
1485 if (n == 12 && !FULL_REGS(fp)) {
1486 printf("\n");
1487 break;
1488 }
1489 }
1490#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001491 printf("pc = ");
1492 xmon_print_symbol(fp->nip, " ", "\n");
Paul Mackerras48404f22011-05-01 19:48:20 +00001493 if (TRAP(fp) != 0xc00 && cpu_has_feature(CPU_FTR_CFAR)) {
1494 printf("cfar= ");
1495 xmon_print_symbol(fp->orig_gpr3, " ", "\n");
1496 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001497 printf("lr = ");
1498 xmon_print_symbol(fp->link, " ", "\n");
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001499 printf("msr = "REG" cr = %.8lx\n", fp->msr, fp->ccr);
1500 printf("ctr = "REG" xer = "REG" trap = %4lx\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001501 fp->ctr, fp->xer, fp->trap);
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001502 trap = TRAP(fp);
1503 if (trap == 0x300 || trap == 0x380 || trap == 0x600)
1504 printf("dar = "REG" dsisr = %.8lx\n", fp->dar, fp->dsisr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001505}
1506
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001507static void cacheflush(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001508{
1509 int cmd;
1510 unsigned long nflush;
1511
1512 cmd = inchar();
1513 if (cmd != 'i')
1514 termch = cmd;
1515 scanhex((void *)&adrs);
1516 if (termch != '\n')
1517 termch = 0;
1518 nflush = 1;
1519 scanhex(&nflush);
1520 nflush = (nflush + L1_CACHE_BYTES - 1) / L1_CACHE_BYTES;
1521 if (setjmp(bus_error_jmp) == 0) {
1522 catch_memory_errors = 1;
1523 sync();
1524
1525 if (cmd != 'i') {
1526 for (; nflush > 0; --nflush, adrs += L1_CACHE_BYTES)
1527 cflush((void *) adrs);
1528 } else {
1529 for (; nflush > 0; --nflush, adrs += L1_CACHE_BYTES)
1530 cinval((void *) adrs);
1531 }
1532 sync();
1533 /* wait a little while to see if we get a machine check */
1534 __delay(200);
1535 }
1536 catch_memory_errors = 0;
1537}
1538
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001539static unsigned long
Linus Torvalds1da177e2005-04-16 15:20:36 -07001540read_spr(int n)
1541{
1542 unsigned int instrs[2];
1543 unsigned long (*code)(void);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001544 unsigned long ret = -1UL;
Paul Mackerras548cceb2005-11-11 22:36:34 +11001545#ifdef CONFIG_PPC64
1546 unsigned long opd[3];
Linus Torvalds1da177e2005-04-16 15:20:36 -07001547
Linus Torvalds1da177e2005-04-16 15:20:36 -07001548 opd[0] = (unsigned long)instrs;
1549 opd[1] = 0;
1550 opd[2] = 0;
Paul Mackerras548cceb2005-11-11 22:36:34 +11001551 code = (unsigned long (*)(void)) opd;
1552#else
1553 code = (unsigned long (*)(void)) instrs;
1554#endif
1555
1556 /* mfspr r3,n; blr */
1557 instrs[0] = 0x7c6002a6 + ((n & 0x1F) << 16) + ((n & 0x3e0) << 6);
1558 instrs[1] = 0x4e800020;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001559 store_inst(instrs);
1560 store_inst(instrs+1);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001561
1562 if (setjmp(bus_error_jmp) == 0) {
1563 catch_memory_errors = 1;
1564 sync();
1565
1566 ret = code();
1567
1568 sync();
1569 /* wait a little while to see if we get a machine check */
1570 __delay(200);
1571 n = size;
1572 }
1573
1574 return ret;
1575}
1576
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001577static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07001578write_spr(int n, unsigned long val)
1579{
1580 unsigned int instrs[2];
1581 unsigned long (*code)(unsigned long);
Paul Mackerras548cceb2005-11-11 22:36:34 +11001582#ifdef CONFIG_PPC64
Linus Torvalds1da177e2005-04-16 15:20:36 -07001583 unsigned long opd[3];
1584
Linus Torvalds1da177e2005-04-16 15:20:36 -07001585 opd[0] = (unsigned long)instrs;
1586 opd[1] = 0;
1587 opd[2] = 0;
Paul Mackerras548cceb2005-11-11 22:36:34 +11001588 code = (unsigned long (*)(unsigned long)) opd;
1589#else
1590 code = (unsigned long (*)(unsigned long)) instrs;
1591#endif
1592
1593 instrs[0] = 0x7c6003a6 + ((n & 0x1F) << 16) + ((n & 0x3e0) << 6);
1594 instrs[1] = 0x4e800020;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001595 store_inst(instrs);
1596 store_inst(instrs+1);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001597
1598 if (setjmp(bus_error_jmp) == 0) {
1599 catch_memory_errors = 1;
1600 sync();
1601
1602 code(val);
1603
1604 sync();
1605 /* wait a little while to see if we get a machine check */
1606 __delay(200);
1607 n = size;
1608 }
1609}
1610
1611static unsigned long regno;
1612extern char exc_prolog;
1613extern char dec_exc;
1614
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001615static void super_regs(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001616{
1617 int cmd;
1618 unsigned long val;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001619
1620 cmd = skipbl();
1621 if (cmd == '\n') {
Michael Ellermane3bc8042012-08-23 22:09:13 +00001622 unsigned long sp, toc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001623 asm("mr %0,1" : "=r" (sp) :);
1624 asm("mr %0,2" : "=r" (toc) :);
1625
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001626 printf("msr = "REG" sprg0= "REG"\n",
1627 mfmsr(), mfspr(SPRN_SPRG0));
1628 printf("pvr = "REG" sprg1= "REG"\n",
Michael Ellermane3bc8042012-08-23 22:09:13 +00001629 mfspr(SPRN_PVR), mfspr(SPRN_SPRG1));
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001630 printf("dec = "REG" sprg2= "REG"\n",
1631 mfspr(SPRN_DEC), mfspr(SPRN_SPRG2));
1632 printf("sp = "REG" sprg3= "REG"\n", sp, mfspr(SPRN_SPRG3));
1633 printf("toc = "REG" dar = "REG"\n", toc, mfspr(SPRN_DAR));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001634
1635 return;
1636 }
1637
1638 scanhex(&regno);
1639 switch (cmd) {
1640 case 'w':
1641 val = read_spr(regno);
1642 scanhex(&val);
1643 write_spr(regno, val);
1644 /* fall through */
1645 case 'r':
1646 printf("spr %lx = %lx\n", regno, read_spr(regno));
1647 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001648 }
1649 scannl();
1650}
1651
1652/*
1653 * Stuff for reading and writing memory safely
1654 */
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001655static int
Linus Torvalds1da177e2005-04-16 15:20:36 -07001656mread(unsigned long adrs, void *buf, int size)
1657{
1658 volatile int n;
1659 char *p, *q;
1660
1661 n = 0;
1662 if (setjmp(bus_error_jmp) == 0) {
1663 catch_memory_errors = 1;
1664 sync();
1665 p = (char *)adrs;
1666 q = (char *)buf;
1667 switch (size) {
1668 case 2:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001669 *(u16 *)q = *(u16 *)p;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001670 break;
1671 case 4:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001672 *(u32 *)q = *(u32 *)p;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001673 break;
1674 case 8:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001675 *(u64 *)q = *(u64 *)p;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001676 break;
1677 default:
1678 for( ; n < size; ++n) {
1679 *q++ = *p++;
1680 sync();
1681 }
1682 }
1683 sync();
1684 /* wait a little while to see if we get a machine check */
1685 __delay(200);
1686 n = size;
1687 }
1688 catch_memory_errors = 0;
1689 return n;
1690}
1691
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001692static int
Linus Torvalds1da177e2005-04-16 15:20:36 -07001693mwrite(unsigned long adrs, void *buf, int size)
1694{
1695 volatile int n;
1696 char *p, *q;
1697
1698 n = 0;
1699 if (setjmp(bus_error_jmp) == 0) {
1700 catch_memory_errors = 1;
1701 sync();
1702 p = (char *) adrs;
1703 q = (char *) buf;
1704 switch (size) {
1705 case 2:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001706 *(u16 *)p = *(u16 *)q;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001707 break;
1708 case 4:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001709 *(u32 *)p = *(u32 *)q;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001710 break;
1711 case 8:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001712 *(u64 *)p = *(u64 *)q;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001713 break;
1714 default:
1715 for ( ; n < size; ++n) {
1716 *p++ = *q++;
1717 sync();
1718 }
1719 }
1720 sync();
1721 /* wait a little while to see if we get a machine check */
1722 __delay(200);
1723 n = size;
1724 } else {
1725 printf("*** Error writing address %x\n", adrs + n);
1726 }
1727 catch_memory_errors = 0;
1728 return n;
1729}
1730
1731static int fault_type;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001732static int fault_except;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001733static char *fault_chars[] = { "--", "**", "##" };
1734
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001735static int handle_fault(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001736{
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001737 fault_except = TRAP(regs);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001738 switch (TRAP(regs)) {
1739 case 0x200:
1740 fault_type = 0;
1741 break;
1742 case 0x300:
1743 case 0x380:
1744 fault_type = 1;
1745 break;
1746 default:
1747 fault_type = 2;
1748 }
1749
1750 longjmp(bus_error_jmp, 1);
1751
1752 return 0;
1753}
1754
1755#define SWAP(a, b, t) ((t) = (a), (a) = (b), (b) = (t))
1756
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001757static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07001758byterev(unsigned char *val, int size)
1759{
1760 int t;
1761
1762 switch (size) {
1763 case 2:
1764 SWAP(val[0], val[1], t);
1765 break;
1766 case 4:
1767 SWAP(val[0], val[3], t);
1768 SWAP(val[1], val[2], t);
1769 break;
1770 case 8: /* is there really any use for this? */
1771 SWAP(val[0], val[7], t);
1772 SWAP(val[1], val[6], t);
1773 SWAP(val[2], val[5], t);
1774 SWAP(val[3], val[4], t);
1775 break;
1776 }
1777}
1778
1779static int brev;
1780static int mnoread;
1781
Michael Ellermane3bc8042012-08-23 22:09:13 +00001782static char *memex_help_string =
Linus Torvalds1da177e2005-04-16 15:20:36 -07001783 "Memory examine command usage:\n"
1784 "m [addr] [flags] examine/change memory\n"
1785 " addr is optional. will start where left off.\n"
1786 " flags may include chars from this set:\n"
1787 " b modify by bytes (default)\n"
1788 " w modify by words (2 byte)\n"
1789 " l modify by longs (4 byte)\n"
1790 " d modify by doubleword (8 byte)\n"
1791 " r toggle reverse byte order mode\n"
1792 " n do not read memory (for i/o spaces)\n"
1793 " . ok to read (default)\n"
1794 "NOTE: flags are saved as defaults\n"
1795 "";
1796
Michael Ellermane3bc8042012-08-23 22:09:13 +00001797static char *memex_subcmd_help_string =
Linus Torvalds1da177e2005-04-16 15:20:36 -07001798 "Memory examine subcommands:\n"
1799 " hexval write this val to current location\n"
1800 " 'string' write chars from string to this location\n"
1801 " ' increment address\n"
1802 " ^ decrement address\n"
1803 " / increment addr by 0x10. //=0x100, ///=0x1000, etc\n"
1804 " \\ decrement addr by 0x10. \\\\=0x100, \\\\\\=0x1000, etc\n"
1805 " ` clear no-read flag\n"
1806 " ; stay at this addr\n"
1807 " v change to byte mode\n"
1808 " w change to word (2 byte) mode\n"
1809 " l change to long (4 byte) mode\n"
1810 " u change to doubleword (8 byte) mode\n"
1811 " m addr change current addr\n"
1812 " n toggle no-read flag\n"
1813 " r toggle byte reverse flag\n"
1814 " < count back up count bytes\n"
1815 " > count skip forward count bytes\n"
1816 " x exit this mode\n"
1817 "";
1818
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001819static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07001820memex(void)
1821{
1822 int cmd, inc, i, nslash;
1823 unsigned long n;
1824 unsigned char val[16];
1825
1826 scanhex((void *)&adrs);
1827 cmd = skipbl();
1828 if (cmd == '?') {
1829 printf(memex_help_string);
1830 return;
1831 } else {
1832 termch = cmd;
1833 }
1834 last_cmd = "m\n";
1835 while ((cmd = skipbl()) != '\n') {
1836 switch( cmd ){
1837 case 'b': size = 1; break;
1838 case 'w': size = 2; break;
1839 case 'l': size = 4; break;
1840 case 'd': size = 8; break;
1841 case 'r': brev = !brev; break;
1842 case 'n': mnoread = 1; break;
1843 case '.': mnoread = 0; break;
1844 }
1845 }
1846 if( size <= 0 )
1847 size = 1;
1848 else if( size > 8 )
1849 size = 8;
1850 for(;;){
1851 if (!mnoread)
1852 n = mread(adrs, val, size);
Paul Mackerrase1449ed2005-11-10 14:30:20 +11001853 printf(REG"%c", adrs, brev? 'r': ' ');
Linus Torvalds1da177e2005-04-16 15:20:36 -07001854 if (!mnoread) {
1855 if (brev)
1856 byterev(val, size);
1857 putchar(' ');
1858 for (i = 0; i < n; ++i)
1859 printf("%.2x", val[i]);
1860 for (; i < size; ++i)
1861 printf("%s", fault_chars[fault_type]);
1862 }
1863 putchar(' ');
1864 inc = size;
1865 nslash = 0;
1866 for(;;){
1867 if( scanhex(&n) ){
1868 for (i = 0; i < size; ++i)
1869 val[i] = n >> (i * 8);
1870 if (!brev)
1871 byterev(val, size);
1872 mwrite(adrs, val, size);
1873 inc = size;
1874 }
1875 cmd = skipbl();
1876 if (cmd == '\n')
1877 break;
1878 inc = 0;
1879 switch (cmd) {
1880 case '\'':
1881 for(;;){
1882 n = inchar();
1883 if( n == '\\' )
1884 n = bsesc();
1885 else if( n == '\'' )
1886 break;
1887 for (i = 0; i < size; ++i)
1888 val[i] = n >> (i * 8);
1889 if (!brev)
1890 byterev(val, size);
1891 mwrite(adrs, val, size);
1892 adrs += size;
1893 }
1894 adrs -= size;
1895 inc = size;
1896 break;
1897 case ',':
1898 adrs += size;
1899 break;
1900 case '.':
1901 mnoread = 0;
1902 break;
1903 case ';':
1904 break;
1905 case 'x':
1906 case EOF:
1907 scannl();
1908 return;
1909 case 'b':
1910 case 'v':
1911 size = 1;
1912 break;
1913 case 'w':
1914 size = 2;
1915 break;
1916 case 'l':
1917 size = 4;
1918 break;
1919 case 'u':
1920 size = 8;
1921 break;
1922 case '^':
1923 adrs -= size;
1924 break;
1925 break;
1926 case '/':
1927 if (nslash > 0)
1928 adrs -= 1 << nslash;
1929 else
1930 nslash = 0;
1931 nslash += 4;
1932 adrs += 1 << nslash;
1933 break;
1934 case '\\':
1935 if (nslash < 0)
1936 adrs += 1 << -nslash;
1937 else
1938 nslash = 0;
1939 nslash -= 4;
1940 adrs -= 1 << -nslash;
1941 break;
1942 case 'm':
1943 scanhex((void *)&adrs);
1944 break;
1945 case 'n':
1946 mnoread = 1;
1947 break;
1948 case 'r':
1949 brev = !brev;
1950 break;
1951 case '<':
1952 n = size;
1953 scanhex(&n);
1954 adrs -= n;
1955 break;
1956 case '>':
1957 n = size;
1958 scanhex(&n);
1959 adrs += n;
1960 break;
1961 case '?':
1962 printf(memex_subcmd_help_string);
1963 break;
1964 }
1965 }
1966 adrs += inc;
1967 }
1968}
1969
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001970static int
Linus Torvalds1da177e2005-04-16 15:20:36 -07001971bsesc(void)
1972{
1973 int c;
1974
1975 c = inchar();
1976 switch( c ){
1977 case 'n': c = '\n'; break;
1978 case 'r': c = '\r'; break;
1979 case 'b': c = '\b'; break;
1980 case 't': c = '\t'; break;
1981 }
1982 return c;
1983}
1984
Olaf Hering7e5b5932006-03-08 20:40:28 +01001985static void xmon_rawdump (unsigned long adrs, long ndump)
1986{
1987 long n, m, r, nr;
1988 unsigned char temp[16];
1989
1990 for (n = ndump; n > 0;) {
1991 r = n < 16? n: 16;
1992 nr = mread(adrs, temp, r);
1993 adrs += nr;
1994 for (m = 0; m < r; ++m) {
1995 if (m < nr)
1996 printf("%.2x", temp[m]);
1997 else
1998 printf("%s", fault_chars[fault_type]);
1999 }
2000 n -= r;
2001 if (nr < r)
2002 break;
2003 }
2004 printf("\n");
2005}
2006
Michael Ellermanddadb6b2012-09-13 23:01:31 +00002007#ifdef CONFIG_PPC64
2008static void dump_one_paca(int cpu)
2009{
2010 struct paca_struct *p;
2011
2012 if (setjmp(bus_error_jmp) != 0) {
2013 printf("*** Error dumping paca for cpu 0x%x!\n", cpu);
2014 return;
2015 }
2016
2017 catch_memory_errors = 1;
2018 sync();
2019
2020 p = &paca[cpu];
2021
2022 printf("paca for cpu 0x%x @ %p:\n", cpu, p);
2023
2024 printf(" %-*s = %s\n", 16, "possible", cpu_possible(cpu) ? "yes" : "no");
2025 printf(" %-*s = %s\n", 16, "present", cpu_present(cpu) ? "yes" : "no");
2026 printf(" %-*s = %s\n", 16, "online", cpu_online(cpu) ? "yes" : "no");
2027
2028#define DUMP(paca, name, format) \
2029 printf(" %-*s = %#-*"format"\t(0x%lx)\n", 16, #name, 18, paca->name, \
2030 offsetof(struct paca_struct, name));
2031
2032 DUMP(p, lock_token, "x");
2033 DUMP(p, paca_index, "x");
2034 DUMP(p, kernel_toc, "lx");
2035 DUMP(p, kernelbase, "lx");
2036 DUMP(p, kernel_msr, "lx");
2037#ifdef CONFIG_PPC_STD_MMU_64
2038 DUMP(p, stab_real, "lx");
2039 DUMP(p, stab_addr, "lx");
2040#endif
2041 DUMP(p, emergency_sp, "p");
2042 DUMP(p, data_offset, "lx");
2043 DUMP(p, hw_cpu_id, "x");
2044 DUMP(p, cpu_start, "x");
2045 DUMP(p, kexec_state, "x");
2046 DUMP(p, __current, "p");
2047 DUMP(p, kstack, "lx");
2048 DUMP(p, stab_rr, "lx");
2049 DUMP(p, saved_r1, "lx");
2050 DUMP(p, trap_save, "x");
2051 DUMP(p, soft_enabled, "x");
2052 DUMP(p, irq_happened, "x");
2053 DUMP(p, io_sync, "x");
2054 DUMP(p, irq_work_pending, "x");
2055 DUMP(p, nap_state_lost, "x");
2056
2057#undef DUMP
2058
2059 catch_memory_errors = 0;
2060 sync();
2061}
2062
2063static void dump_all_pacas(void)
2064{
2065 int cpu;
2066
2067 if (num_possible_cpus() == 0) {
2068 printf("No possible cpus, use 'dp #' to dump individual cpus\n");
2069 return;
2070 }
2071
2072 for_each_possible_cpu(cpu)
2073 dump_one_paca(cpu);
2074}
2075
2076static void dump_pacas(void)
2077{
2078 unsigned long num;
2079 int c;
2080
2081 c = inchar();
2082 if (c == 'a') {
2083 dump_all_pacas();
2084 return;
2085 }
2086
2087 termch = c; /* Put c back, it wasn't 'a' */
2088
2089 if (scanhex(&num))
2090 dump_one_paca(num);
2091 else
2092 dump_one_paca(xmon_owner);
2093}
2094#endif
2095
Linus Torvalds1da177e2005-04-16 15:20:36 -07002096#define isxdigit(c) (('0' <= (c) && (c) <= '9') \
2097 || ('a' <= (c) && (c) <= 'f') \
2098 || ('A' <= (c) && (c) <= 'F'))
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002099static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002100dump(void)
2101{
2102 int c;
2103
2104 c = inchar();
Michael Ellermanddadb6b2012-09-13 23:01:31 +00002105
2106#ifdef CONFIG_PPC64
2107 if (c == 'p') {
2108 dump_pacas();
2109 return;
2110 }
2111#endif
2112
Linus Torvalds1da177e2005-04-16 15:20:36 -07002113 if ((isxdigit(c) && c != 'f' && c != 'd') || c == '\n')
2114 termch = c;
2115 scanhex((void *)&adrs);
2116 if (termch != '\n')
2117 termch = 0;
2118 if (c == 'i') {
2119 scanhex(&nidump);
2120 if (nidump == 0)
2121 nidump = 16;
2122 else if (nidump > MAX_DUMP)
2123 nidump = MAX_DUMP;
2124 adrs += ppc_inst_dump(adrs, nidump, 1);
2125 last_cmd = "di\n";
Vinay Sridharf312deb2009-05-14 23:13:07 +00002126 } else if (c == 'l') {
2127 dump_log_buf();
Olaf Hering7e5b5932006-03-08 20:40:28 +01002128 } else if (c == 'r') {
2129 scanhex(&ndump);
2130 if (ndump == 0)
2131 ndump = 64;
2132 xmon_rawdump(adrs, ndump);
2133 adrs += ndump;
2134 last_cmd = "dr\n";
Linus Torvalds1da177e2005-04-16 15:20:36 -07002135 } else {
2136 scanhex(&ndump);
2137 if (ndump == 0)
2138 ndump = 64;
2139 else if (ndump > MAX_DUMP)
2140 ndump = MAX_DUMP;
2141 prdump(adrs, ndump);
2142 adrs += ndump;
2143 last_cmd = "d\n";
2144 }
2145}
2146
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002147static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002148prdump(unsigned long adrs, long ndump)
2149{
2150 long n, m, c, r, nr;
2151 unsigned char temp[16];
2152
2153 for (n = ndump; n > 0;) {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002154 printf(REG, adrs);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002155 putchar(' ');
2156 r = n < 16? n: 16;
2157 nr = mread(adrs, temp, r);
2158 adrs += nr;
2159 for (m = 0; m < r; ++m) {
Michael Ellermane3bc8042012-08-23 22:09:13 +00002160 if ((m & (sizeof(long) - 1)) == 0 && m > 0)
Paul Mackerrase1449ed2005-11-10 14:30:20 +11002161 putchar(' ');
Linus Torvalds1da177e2005-04-16 15:20:36 -07002162 if (m < nr)
2163 printf("%.2x", temp[m]);
2164 else
2165 printf("%s", fault_chars[fault_type]);
2166 }
Paul Mackerrase1449ed2005-11-10 14:30:20 +11002167 for (; m < 16; ++m) {
Michael Ellermane3bc8042012-08-23 22:09:13 +00002168 if ((m & (sizeof(long) - 1)) == 0)
Paul Mackerrase1449ed2005-11-10 14:30:20 +11002169 putchar(' ');
Linus Torvalds1da177e2005-04-16 15:20:36 -07002170 printf(" ");
Paul Mackerrase1449ed2005-11-10 14:30:20 +11002171 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002172 printf(" |");
2173 for (m = 0; m < r; ++m) {
2174 if (m < nr) {
2175 c = temp[m];
2176 putchar(' ' <= c && c <= '~'? c: '.');
2177 } else
2178 putchar(' ');
2179 }
2180 n -= r;
2181 for (; m < 16; ++m)
2182 putchar(' ');
2183 printf("|\n");
2184 if (nr < r)
2185 break;
2186 }
2187}
2188
Michael Ellerman4c4c8722006-11-23 00:46:42 +01002189typedef int (*instruction_dump_func)(unsigned long inst, unsigned long addr);
2190
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002191static int
Michael Ellerman4c4c8722006-11-23 00:46:42 +01002192generic_inst_dump(unsigned long adr, long count, int praddr,
2193 instruction_dump_func dump_func)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002194{
2195 int nr, dotted;
2196 unsigned long first_adr;
2197 unsigned long inst, last_inst = 0;
2198 unsigned char val[4];
2199
2200 dotted = 0;
2201 for (first_adr = adr; count > 0; --count, adr += 4) {
2202 nr = mread(adr, val, 4);
2203 if (nr == 0) {
2204 if (praddr) {
2205 const char *x = fault_chars[fault_type];
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002206 printf(REG" %s%s%s%s\n", adr, x, x, x, x);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002207 }
2208 break;
2209 }
2210 inst = GETWORD(val);
2211 if (adr > first_adr && inst == last_inst) {
2212 if (!dotted) {
2213 printf(" ...\n");
2214 dotted = 1;
2215 }
2216 continue;
2217 }
2218 dotted = 0;
2219 last_inst = inst;
2220 if (praddr)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002221 printf(REG" %.8x", adr, inst);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002222 printf("\t");
Michael Ellerman4c4c8722006-11-23 00:46:42 +01002223 dump_func(inst, adr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002224 printf("\n");
2225 }
2226 return adr - first_adr;
2227}
2228
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002229static int
Michael Ellerman4c4c8722006-11-23 00:46:42 +01002230ppc_inst_dump(unsigned long adr, long count, int praddr)
2231{
2232 return generic_inst_dump(adr, count, praddr, print_insn_powerpc);
2233}
2234
Linus Torvalds1da177e2005-04-16 15:20:36 -07002235void
2236print_address(unsigned long addr)
2237{
2238 xmon_print_symbol(addr, "\t# ", "");
2239}
2240
Vinay Sridharf312deb2009-05-14 23:13:07 +00002241void
2242dump_log_buf(void)
2243{
Michael Ellermanca5dd392012-08-23 22:09:12 +00002244 struct kmsg_dumper dumper = { .active = 1 };
2245 unsigned char buf[128];
2246 size_t len;
Vinay Sridharf312deb2009-05-14 23:13:07 +00002247
Michael Ellermane3bc8042012-08-23 22:09:13 +00002248 if (setjmp(bus_error_jmp) != 0) {
Michael Ellermanca5dd392012-08-23 22:09:12 +00002249 printf("Error dumping printk buffer!\n");
Michael Ellermane3bc8042012-08-23 22:09:13 +00002250 return;
2251 }
Vinay Sridharf312deb2009-05-14 23:13:07 +00002252
Michael Ellermane3bc8042012-08-23 22:09:13 +00002253 catch_memory_errors = 1;
2254 sync();
Vinay Sridharf312deb2009-05-14 23:13:07 +00002255
Michael Ellermanca5dd392012-08-23 22:09:12 +00002256 kmsg_dump_rewind_nolock(&dumper);
2257 while (kmsg_dump_get_line_nolock(&dumper, false, buf, sizeof(buf), &len)) {
2258 buf[len] = '\0';
2259 printf("%s", buf);
2260 }
Vinay Sridharf312deb2009-05-14 23:13:07 +00002261
Michael Ellermane3bc8042012-08-23 22:09:13 +00002262 sync();
2263 /* wait a little while to see if we get a machine check */
2264 __delay(200);
2265 catch_memory_errors = 0;
Vinay Sridharf312deb2009-05-14 23:13:07 +00002266}
Linus Torvalds1da177e2005-04-16 15:20:36 -07002267
2268/*
2269 * Memory operations - move, set, print differences
2270 */
2271static unsigned long mdest; /* destination address */
2272static unsigned long msrc; /* source address */
2273static unsigned long mval; /* byte value to set memory to */
2274static unsigned long mcount; /* # bytes to affect */
2275static unsigned long mdiffs; /* max # differences to print */
2276
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002277static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002278memops(int cmd)
2279{
2280 scanhex((void *)&mdest);
2281 if( termch != '\n' )
2282 termch = 0;
2283 scanhex((void *)(cmd == 's'? &mval: &msrc));
2284 if( termch != '\n' )
2285 termch = 0;
2286 scanhex((void *)&mcount);
2287 switch( cmd ){
2288 case 'm':
2289 memmove((void *)mdest, (void *)msrc, mcount);
2290 break;
2291 case 's':
2292 memset((void *)mdest, mval, mcount);
2293 break;
2294 case 'd':
2295 if( termch != '\n' )
2296 termch = 0;
2297 scanhex((void *)&mdiffs);
2298 memdiffs((unsigned char *)mdest, (unsigned char *)msrc, mcount, mdiffs);
2299 break;
2300 }
2301}
2302
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002303static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002304memdiffs(unsigned char *p1, unsigned char *p2, unsigned nb, unsigned maxpr)
2305{
2306 unsigned n, prt;
2307
2308 prt = 0;
2309 for( n = nb; n > 0; --n )
2310 if( *p1++ != *p2++ )
2311 if( ++prt <= maxpr )
2312 printf("%.16x %.2x # %.16x %.2x\n", p1 - 1,
2313 p1[-1], p2 - 1, p2[-1]);
2314 if( prt > maxpr )
2315 printf("Total of %d differences\n", prt);
2316}
2317
2318static unsigned mend;
2319static unsigned mask;
2320
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002321static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002322memlocate(void)
2323{
2324 unsigned a, n;
2325 unsigned char val[4];
2326
2327 last_cmd = "ml";
2328 scanhex((void *)&mdest);
2329 if (termch != '\n') {
2330 termch = 0;
2331 scanhex((void *)&mend);
2332 if (termch != '\n') {
2333 termch = 0;
2334 scanhex((void *)&mval);
2335 mask = ~0;
2336 if (termch != '\n') termch = 0;
2337 scanhex((void *)&mask);
2338 }
2339 }
2340 n = 0;
2341 for (a = mdest; a < mend; a += 4) {
2342 if (mread(a, val, 4) == 4
2343 && ((GETWORD(val) ^ mval) & mask) == 0) {
2344 printf("%.16x: %.16x\n", a, GETWORD(val));
2345 if (++n >= 10)
2346 break;
2347 }
2348 }
2349}
2350
2351static unsigned long mskip = 0x1000;
2352static unsigned long mlim = 0xffffffff;
2353
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002354static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002355memzcan(void)
2356{
2357 unsigned char v;
2358 unsigned a;
2359 int ok, ook;
2360
2361 scanhex(&mdest);
2362 if (termch != '\n') termch = 0;
2363 scanhex(&mskip);
2364 if (termch != '\n') termch = 0;
2365 scanhex(&mlim);
2366 ook = 0;
2367 for (a = mdest; a < mlim; a += mskip) {
2368 ok = mread(a, &v, 1);
2369 if (ok && !ook) {
2370 printf("%.8x .. ", a);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002371 } else if (!ok && ook)
2372 printf("%.8x\n", a - mskip);
2373 ook = ok;
2374 if (a + mskip < a)
2375 break;
2376 }
2377 if (ook)
2378 printf("%.8x\n", a - mskip);
2379}
2380
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002381static void proccall(void)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002382{
2383 unsigned long args[8];
2384 unsigned long ret;
2385 int i;
2386 typedef unsigned long (*callfunc_t)(unsigned long, unsigned long,
2387 unsigned long, unsigned long, unsigned long,
2388 unsigned long, unsigned long, unsigned long);
2389 callfunc_t func;
2390
2391 if (!scanhex(&adrs))
2392 return;
2393 if (termch != '\n')
2394 termch = 0;
2395 for (i = 0; i < 8; ++i)
2396 args[i] = 0;
2397 for (i = 0; i < 8; ++i) {
2398 if (!scanhex(&args[i]) || termch == '\n')
2399 break;
2400 termch = 0;
2401 }
2402 func = (callfunc_t) adrs;
2403 ret = 0;
2404 if (setjmp(bus_error_jmp) == 0) {
2405 catch_memory_errors = 1;
2406 sync();
2407 ret = func(args[0], args[1], args[2], args[3],
2408 args[4], args[5], args[6], args[7]);
2409 sync();
2410 printf("return value is %x\n", ret);
2411 } else {
2412 printf("*** %x exception occurred\n", fault_except);
2413 }
2414 catch_memory_errors = 0;
2415}
2416
Linus Torvalds1da177e2005-04-16 15:20:36 -07002417/* Input scanning routines */
2418int
2419skipbl(void)
2420{
2421 int c;
2422
2423 if( termch != 0 ){
2424 c = termch;
2425 termch = 0;
2426 } else
2427 c = inchar();
2428 while( c == ' ' || c == '\t' )
2429 c = inchar();
2430 return c;
2431}
2432
2433#define N_PTREGS 44
2434static char *regnames[N_PTREGS] = {
2435 "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7",
2436 "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15",
2437 "r16", "r17", "r18", "r19", "r20", "r21", "r22", "r23",
2438 "r24", "r25", "r26", "r27", "r28", "r29", "r30", "r31",
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002439 "pc", "msr", "or3", "ctr", "lr", "xer", "ccr",
2440#ifdef CONFIG_PPC64
2441 "softe",
2442#else
2443 "mq",
2444#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07002445 "trap", "dar", "dsisr", "res"
2446};
2447
2448int
2449scanhex(unsigned long *vp)
2450{
2451 int c, d;
2452 unsigned long v;
2453
2454 c = skipbl();
2455 if (c == '%') {
2456 /* parse register name */
2457 char regname[8];
2458 int i;
2459
2460 for (i = 0; i < sizeof(regname) - 1; ++i) {
2461 c = inchar();
2462 if (!isalnum(c)) {
2463 termch = c;
2464 break;
2465 }
2466 regname[i] = c;
2467 }
2468 regname[i] = 0;
2469 for (i = 0; i < N_PTREGS; ++i) {
2470 if (strcmp(regnames[i], regname) == 0) {
2471 if (xmon_regs == NULL) {
2472 printf("regs not available\n");
2473 return 0;
2474 }
2475 *vp = ((unsigned long *)xmon_regs)[i];
2476 return 1;
2477 }
2478 }
2479 printf("invalid register name '%%%s'\n", regname);
2480 return 0;
2481 }
2482
2483 /* skip leading "0x" if any */
2484
2485 if (c == '0') {
2486 c = inchar();
2487 if (c == 'x') {
2488 c = inchar();
2489 } else {
2490 d = hexdigit(c);
2491 if (d == EOF) {
2492 termch = c;
2493 *vp = 0;
2494 return 1;
2495 }
2496 }
2497 } else if (c == '$') {
2498 int i;
2499 for (i=0; i<63; i++) {
2500 c = inchar();
2501 if (isspace(c)) {
2502 termch = c;
2503 break;
2504 }
2505 tmpstr[i] = c;
2506 }
2507 tmpstr[i++] = 0;
Benjamin Herrenschmidt6879dc12005-06-21 17:15:30 -07002508 *vp = 0;
2509 if (setjmp(bus_error_jmp) == 0) {
2510 catch_memory_errors = 1;
2511 sync();
2512 *vp = kallsyms_lookup_name(tmpstr);
2513 sync();
2514 }
2515 catch_memory_errors = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002516 if (!(*vp)) {
2517 printf("unknown symbol '%s'\n", tmpstr);
2518 return 0;
2519 }
2520 return 1;
2521 }
2522
2523 d = hexdigit(c);
2524 if (d == EOF) {
2525 termch = c;
2526 return 0;
2527 }
2528 v = 0;
2529 do {
2530 v = (v << 4) + d;
2531 c = inchar();
2532 d = hexdigit(c);
2533 } while (d != EOF);
2534 termch = c;
2535 *vp = v;
2536 return 1;
2537}
2538
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002539static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002540scannl(void)
2541{
2542 int c;
2543
2544 c = termch;
2545 termch = 0;
2546 while( c != '\n' )
2547 c = inchar();
2548}
2549
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002550static int hexdigit(int c)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002551{
2552 if( '0' <= c && c <= '9' )
2553 return c - '0';
2554 if( 'A' <= c && c <= 'F' )
2555 return c - ('A' - 10);
2556 if( 'a' <= c && c <= 'f' )
2557 return c - ('a' - 10);
2558 return EOF;
2559}
2560
2561void
2562getstring(char *s, int size)
2563{
2564 int c;
2565
2566 c = skipbl();
2567 do {
2568 if( size > 1 ){
2569 *s++ = c;
2570 --size;
2571 }
2572 c = inchar();
2573 } while( c != ' ' && c != '\t' && c != '\n' );
2574 termch = c;
2575 *s = 0;
2576}
2577
2578static char line[256];
2579static char *lineptr;
2580
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002581static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002582flush_input(void)
2583{
2584 lineptr = NULL;
2585}
2586
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002587static int
Linus Torvalds1da177e2005-04-16 15:20:36 -07002588inchar(void)
2589{
2590 if (lineptr == NULL || *lineptr == 0) {
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11002591 if (xmon_gets(line, sizeof(line)) == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002592 lineptr = NULL;
2593 return EOF;
2594 }
2595 lineptr = line;
2596 }
2597 return *lineptr++;
2598}
2599
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002600static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002601take_input(char *str)
2602{
2603 lineptr = str;
2604}
2605
2606
2607static void
2608symbol_lookup(void)
2609{
2610 int type = inchar();
2611 unsigned long addr;
2612 static char tmp[64];
2613
2614 switch (type) {
2615 case 'a':
2616 if (scanhex(&addr))
2617 xmon_print_symbol(addr, ": ", "\n");
2618 termch = 0;
2619 break;
2620 case 's':
2621 getstring(tmp, 64);
2622 if (setjmp(bus_error_jmp) == 0) {
2623 catch_memory_errors = 1;
2624 sync();
2625 addr = kallsyms_lookup_name(tmp);
2626 if (addr)
2627 printf("%s: %lx\n", tmp, addr);
2628 else
2629 printf("Symbol '%s' not found.\n", tmp);
2630 sync();
2631 }
2632 catch_memory_errors = 0;
2633 termch = 0;
2634 break;
2635 }
2636}
2637
2638
2639/* Print an address in numeric and symbolic form (if possible) */
2640static void xmon_print_symbol(unsigned long address, const char *mid,
2641 const char *after)
2642{
2643 char *modname;
2644 const char *name = NULL;
2645 unsigned long offset, size;
2646
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002647 printf(REG, address);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002648 if (setjmp(bus_error_jmp) == 0) {
2649 catch_memory_errors = 1;
2650 sync();
2651 name = kallsyms_lookup(address, &size, &offset, &modname,
2652 tmpstr);
2653 sync();
2654 /* wait a little while to see if we get a machine check */
2655 __delay(200);
2656 }
2657
2658 catch_memory_errors = 0;
2659
2660 if (name) {
2661 printf("%s%s+%#lx/%#lx", mid, name, offset, size);
2662 if (modname)
2663 printf(" [%s]", modname);
2664 }
2665 printf("%s", after);
2666}
2667
Benjamin Herrenschmidt2d27cfd2009-07-23 23:15:59 +00002668#ifdef CONFIG_PPC_BOOK3S_64
Linus Torvalds1da177e2005-04-16 15:20:36 -07002669static void dump_slb(void)
2670{
2671 int i;
will schmidtb3b95952007-12-07 08:22:23 +11002672 unsigned long esid,vsid,valid;
2673 unsigned long llp;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002674
2675 printf("SLB contents of cpu %x\n", smp_processor_id());
2676
Michael Neuling584f8b72007-12-06 17:24:48 +11002677 for (i = 0; i < mmu_slb_size; i++) {
will schmidtb3b95952007-12-07 08:22:23 +11002678 asm volatile("slbmfee %0,%1" : "=r" (esid) : "r" (i));
2679 asm volatile("slbmfev %0,%1" : "=r" (vsid) : "r" (i));
2680 valid = (esid & SLB_ESID_V);
2681 if (valid | esid | vsid) {
2682 printf("%02d %016lx %016lx", i, esid, vsid);
2683 if (valid) {
2684 llp = vsid & SLB_VSID_LLP;
2685 if (vsid & SLB_VSID_B_1T) {
2686 printf(" 1T ESID=%9lx VSID=%13lx LLP:%3lx \n",
2687 GET_ESID_1T(esid),
2688 (vsid & ~SLB_VSID_B) >> SLB_VSID_SHIFT_1T,
2689 llp);
2690 } else {
2691 printf(" 256M ESID=%9lx VSID=%13lx LLP:%3lx \n",
2692 GET_ESID(esid),
2693 (vsid & ~SLB_VSID_B) >> SLB_VSID_SHIFT,
2694 llp);
2695 }
2696 } else
2697 printf("\n");
2698 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002699 }
2700}
2701
2702static void dump_stab(void)
2703{
2704 int i;
Benjamin Herrenschmidt7ac21cd2012-03-02 10:10:09 +11002705 unsigned long *tmp = (unsigned long *)local_paca->stab_addr;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002706
2707 printf("Segment table contents of cpu %x\n", smp_processor_id());
2708
2709 for (i = 0; i < PAGE_SIZE/16; i++) {
2710 unsigned long a, b;
2711
2712 a = *tmp++;
2713 b = *tmp++;
2714
2715 if (a || b) {
2716 printf("%03d %016lx ", i, a);
2717 printf("%016lx\n", b);
2718 }
2719 }
2720}
2721
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002722void dump_segments(void)
2723{
Matt Evans44ae3ab2011-04-06 19:48:50 +00002724 if (mmu_has_feature(MMU_FTR_SLB))
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002725 dump_slb();
2726 else
2727 dump_stab();
2728}
2729#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;
2922 __debugger_dabr_match = xmon_dabr_match;
2923 __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;
2930 __debugger_dabr_match = NULL;
2931 __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,
2945 .help_msg = "Xmon",
2946 .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