blob: e66ace703a69454a611d54e3a9b832449b51cf3a [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * Routines providing a simple monitor for use on the PowerMac.
3 *
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11004 * Copyright (C) 1996-2005 Paul Mackerras.
Michael Ellerman476792832006-10-03 14:12:08 +10005 * Copyright (C) 2001 PPC64 Team, IBM Corp
6 * Copyrignt (C) 2006 Michael Ellerman, IBM Corp
Linus Torvalds1da177e2005-04-16 15:20:36 -07007 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version
11 * 2 of the License, or (at your option) any later version.
12 */
Linus Torvalds1da177e2005-04-16 15:20:36 -070013#include <linux/errno.h>
14#include <linux/sched.h>
15#include <linux/smp.h>
16#include <linux/mm.h>
17#include <linux/reboot.h>
18#include <linux/delay.h>
19#include <linux/kallsyms.h>
Michael Ellermanca5dd392012-08-23 22:09:12 +000020#include <linux/kmsg_dump.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070021#include <linux/cpumask.h>
Paul Gortmaker4b16f8e2011-07-22 18:24:23 -040022#include <linux/export.h>
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +110023#include <linux/sysrq.h>
Andrew Morton4694ca02005-11-13 16:06:50 -080024#include <linux/interrupt.h>
David Howells7d12e782006-10-05 14:55:46 +010025#include <linux/irq.h>
Jeremy Fitzhardinge73c9cea2006-12-08 03:30:41 -080026#include <linux/bug.h>
Anton Blancharda71d64b2014-08-05 14:55:00 +100027#include <linux/nmi.h>
Vincent Bernat05b981f2014-07-15 13:43:47 +020028#include <linux/ctype.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070029
30#include <asm/ptrace.h>
31#include <asm/string.h>
32#include <asm/prom.h>
33#include <asm/machdep.h>
Paul Mackerrasf78541dc2005-10-28 22:53:37 +100034#include <asm/xmon.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070035#include <asm/processor.h>
36#include <asm/pgtable.h>
37#include <asm/mmu.h>
38#include <asm/mmu_context.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070039#include <asm/cputable.h>
40#include <asm/rtas.h>
41#include <asm/sstep.h>
Paul Mackerrasf583ffc2006-10-10 11:47:07 +100042#include <asm/irq_regs.h>
Michael Ellermanff8a8f22006-10-24 18:31:27 +020043#include <asm/spu.h>
44#include <asm/spu_priv1.h>
Michael Neulingc3b75bd2008-01-18 15:50:30 +110045#include <asm/setjmp.h>
Anton Vorontsov322b4392008-12-17 10:08:55 +000046#include <asm/reg.h>
David Howellsae3a1972012-03-28 18:30:02 +010047#include <asm/debug.h>
Michael Neuling9422de32012-12-20 14:06:44 +000048#include <asm/hw_breakpoint.h>
Paul Mackerrasf78541dc2005-10-28 22:53:37 +100049
50#ifdef CONFIG_PPC64
Linus Torvalds1da177e2005-04-16 15:20:36 -070051#include <asm/hvcall.h>
Paul Mackerrasf78541dc2005-10-28 22:53:37 +100052#include <asm/paca.h>
53#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -070054
Anshuman Khandual1ad7d702014-11-28 10:06:42 +053055#if defined(CONFIG_PPC_SPLPAR)
56#include <asm/plpar_wrappers.h>
57#else
58static inline long plapr_set_ciabr(unsigned long ciabr) {return 0; };
59#endif
60
Linus Torvalds1da177e2005-04-16 15:20:36 -070061#include "nonstdio.h"
Michael Ellermane0426042006-11-23 00:46:45 +010062#include "dis-asm.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070063
Linus Torvalds1da177e2005-04-16 15:20:36 -070064#ifdef CONFIG_SMP
Michael Ellerman1c8950f2008-05-08 14:27:17 +100065static cpumask_t cpus_in_xmon = CPU_MASK_NONE;
Linus Torvalds1da177e2005-04-16 15:20:36 -070066static unsigned long xmon_taken = 1;
67static int xmon_owner;
68static int xmon_gate;
Michael Ellermanddadb6b2012-09-13 23:01:31 +000069#else
70#define xmon_owner 0
Linus Torvalds1da177e2005-04-16 15:20:36 -070071#endif /* CONFIG_SMP */
72
Anton Blanchard5be34922010-01-12 00:50:14 +000073static unsigned long in_xmon __read_mostly = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -070074
75static unsigned long adrs;
76static int size = 1;
77#define MAX_DUMP (128 * 1024)
78static unsigned long ndump = 64;
79static unsigned long nidump = 16;
80static unsigned long ncsum = 4096;
81static int termch;
82static char tmpstr[128];
83
Linus Torvalds1da177e2005-04-16 15:20:36 -070084static long bus_error_jmp[JMP_BUF_LEN];
85static int catch_memory_errors;
86static long *xmon_fault_jmp[NR_CPUS];
Linus Torvalds1da177e2005-04-16 15:20:36 -070087
88/* Breakpoint stuff */
89struct bpt {
90 unsigned long address;
91 unsigned int instr[2];
92 atomic_t ref_count;
93 int enabled;
94 unsigned long pad;
95};
96
97/* Bits in bpt.enabled */
Michael Ellermanabb90ee2014-12-01 16:54:13 +110098#define BP_CIABR 1
99#define BP_TRAP 2
100#define BP_DABR 4
Linus Torvalds1da177e2005-04-16 15:20:36 -0700101
102#define NBPTS 256
103static struct bpt bpts[NBPTS];
104static struct bpt dabr;
105static struct bpt *iabr;
106static unsigned bpinstr = 0x7fe00008; /* trap */
107
108#define BP_NUM(bp) ((bp) - bpts + 1)
109
110/* Prototypes */
111static int cmds(struct pt_regs *);
112static int mread(unsigned long, void *, int);
113static int mwrite(unsigned long, void *, int);
114static int handle_fault(struct pt_regs *);
115static void byterev(unsigned char *, int);
116static void memex(void);
117static int bsesc(void);
118static void dump(void);
119static void prdump(unsigned long, long);
120static int ppc_inst_dump(unsigned long, long, int);
Vinay Sridharf312deb2009-05-14 23:13:07 +0000121static void dump_log_buf(void);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700122static void backtrace(struct pt_regs *);
123static void excprint(struct pt_regs *);
124static void prregs(struct pt_regs *);
125static void memops(int);
126static void memlocate(void);
127static void memzcan(void);
128static void memdiffs(unsigned char *, unsigned char *, unsigned, unsigned);
129int skipbl(void);
130int scanhex(unsigned long *valp);
131static void scannl(void);
132static int hexdigit(int);
133void getstring(char *, int);
134static void flush_input(void);
135static int inchar(void);
136static void take_input(char *);
137static unsigned long read_spr(int);
138static void write_spr(int, unsigned long);
139static void super_regs(void);
140static void remove_bpts(void);
141static void insert_bpts(void);
142static void remove_cpu_bpts(void);
143static void insert_cpu_bpts(void);
144static struct bpt *at_breakpoint(unsigned long pc);
145static struct bpt *in_breakpoint_table(unsigned long pc, unsigned long *offp);
146static int do_step(struct pt_regs *);
147static void bpt_cmds(void);
148static void cacheflush(void);
149static int cpu_cmd(void);
150static void csum(void);
151static void bootcmds(void);
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000152static void proccall(void);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700153void dump_segments(void);
154static void symbol_lookup(void);
Olaf Hering26c8af52006-09-08 16:29:21 +0200155static void xmon_show_stack(unsigned long sp, unsigned long lr,
156 unsigned long pc);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700157static void xmon_print_symbol(unsigned long address, const char *mid,
158 const char *after);
159static const char *getvecname(unsigned long vec);
160
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200161static int do_spu_cmd(void);
162
Benjamin Herrenschmidt5a8a1a22007-11-16 18:23:33 +1100163#ifdef CONFIG_44x
164static void dump_tlb_44x(void);
165#endif
Benjamin Herrenschmidt03247152010-07-09 15:34:50 +1000166#ifdef CONFIG_PPC_BOOK3E
167static void dump_tlb_book3e(void);
168#endif
Benjamin Herrenschmidt5a8a1a22007-11-16 18:23:33 +1100169
Michael Ellerman9f1067c22008-05-08 14:27:16 +1000170static int xmon_no_auto_backtrace;
Olaf Hering26c8af52006-09-08 16:29:21 +0200171
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000172extern void xmon_enter(void);
173extern void xmon_leave(void);
174
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000175#ifdef CONFIG_PPC64
176#define REG "%.16lx"
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000177#else
178#define REG "%.8lx"
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000179#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700180
Philippe Bergheaud72eceef2013-12-02 10:10:12 +0100181#ifdef __LITTLE_ENDIAN__
182#define GETWORD(v) (((v)[3] << 24) + ((v)[2] << 16) + ((v)[1] << 8) + (v)[0])
183#else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700184#define GETWORD(v) (((v)[0] << 24) + ((v)[1] << 16) + ((v)[2] << 8) + (v)[3])
Philippe Bergheaud72eceef2013-12-02 10:10:12 +0100185#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700186
Linus Torvalds1da177e2005-04-16 15:20:36 -0700187static char *help_string = "\
188Commands:\n\
189 b show breakpoints\n\
190 bd set data breakpoint\n\
191 bi set instruction breakpoint\n\
192 bc clear breakpoint\n"
193#ifdef CONFIG_SMP
194 "\
195 c print cpus stopped in xmon\n\
196 c# try to switch to cpu number h (in hex)\n"
197#endif
198 "\
199 C checksum\n\
200 d dump bytes\n\
201 di dump instructions\n\
202 df dump float values\n\
203 dd dump double values\n\
Michael Ellermanddadb6b2012-09-13 23:01:31 +0000204 dl dump the kernel log buffer\n"
205#ifdef CONFIG_PPC64
206 "\
207 dp[#] dump paca for current cpu, or cpu #\n\
208 dpa dump paca for all possible cpus\n"
209#endif
210 "\
Olaf Hering7e5b5932006-03-08 20:40:28 +0100211 dr dump stream of raw bytes\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700212 e print exception information\n\
213 f flush cache\n\
214 la lookup symbol+offset of specified address\n\
215 ls lookup address of specified symbol\n\
216 m examine/change memory\n\
217 mm move a block of memory\n\
218 ms set a block of memory\n\
219 md compare two blocks of memory\n\
220 ml locate a block of memory\n\
221 mz zero a block of memory\n\
222 mi show information about memory allocation\n\
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000223 p call a procedure\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700224 r print registers\n\
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200225 s single step\n"
Arnd Bergmanne0555952006-11-27 19:18:55 +0100226#ifdef CONFIG_SPU_BASE
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200227" ss stop execution on all spus\n\
Michael Ellermana8984972006-10-24 18:31:28 +0200228 sr restore execution on stopped spus\n\
Michael Ellerman24a24c82006-11-23 00:46:41 +0100229 sf # dump spu fields for spu # (in hex)\n\
Michael Ellermanc99176a2007-02-26 18:14:06 +0900230 sd # dump spu local store for spu # (in hex)\n\
Michael Ellermanaf89fb82006-11-23 00:46:44 +0100231 sdi # disassemble spu local store for spu # (in hex)\n"
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200232#endif
233" S print special registers\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700234 t print backtrace\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700235 x exit monitor and recover\n\
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000236 X exit monitor and dont recover\n"
Jimi Xenidis79873e82011-09-29 11:25:10 +0000237#if defined(CONFIG_PPC64) && !defined(CONFIG_PPC_BOOK3E)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000238" u dump segment table or SLB\n"
Jimi Xenidis79873e82011-09-29 11:25:10 +0000239#elif defined(CONFIG_PPC_STD_MMU_32)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000240" u dump segment registers\n"
Jimi Xenidis79873e82011-09-29 11:25:10 +0000241#elif defined(CONFIG_44x) || defined(CONFIG_PPC_BOOK3E)
Benjamin Herrenschmidt5a8a1a22007-11-16 18:23:33 +1100242" u dump TLB\n"
243#endif
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000244" ? help\n"
245" zr reboot\n\
Linus Torvalds1da177e2005-04-16 15:20:36 -0700246 zh halt\n"
247;
248
249static struct pt_regs *xmon_regs;
250
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000251static inline void sync(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700252{
253 asm volatile("sync; isync");
254}
255
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000256static inline void store_inst(void *p)
257{
258 asm volatile ("dcbst 0,%0; sync; icbi 0,%0; isync" : : "r" (p));
259}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700260
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000261static inline void cflush(void *p)
262{
263 asm volatile ("dcbf 0,%0; icbi 0,%0" : : "r" (p));
264}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700265
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000266static inline void cinval(void *p)
267{
268 asm volatile ("dcbi 0,%0; icbi 0,%0" : : "r" (p));
269}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700270
Anshuman Khandual1ad7d702014-11-28 10:06:42 +0530271/**
272 * write_ciabr() - write the CIABR SPR
273 * @ciabr: The value to write.
274 *
275 * This function writes a value to the CIARB register either directly
276 * through mtspr instruction if the kernel is in HV privilege mode or
277 * call a hypervisor function to achieve the same in case the kernel
278 * is in supervisor privilege mode.
279 */
280static void write_ciabr(unsigned long ciabr)
281{
282 if (!cpu_has_feature(CPU_FTR_ARCH_207S))
283 return;
284
285 if (cpu_has_feature(CPU_FTR_HVMODE)) {
286 mtspr(SPRN_CIABR, ciabr);
287 return;
288 }
289 plapr_set_ciabr(ciabr);
290}
291
292/**
293 * set_ciabr() - set the CIABR
294 * @addr: The value to set.
295 *
296 * This function sets the correct privilege value into the the HW
297 * breakpoint address before writing it up in the CIABR register.
298 */
299static void set_ciabr(unsigned long addr)
300{
301 addr &= ~CIABR_PRIV;
302
303 if (cpu_has_feature(CPU_FTR_HVMODE))
304 addr |= CIABR_PRIV_HYPER;
305 else
306 addr |= CIABR_PRIV_SUPER;
307 write_ciabr(addr);
308}
309
Linus Torvalds1da177e2005-04-16 15:20:36 -0700310/*
311 * Disable surveillance (the service processor watchdog function)
312 * while we are in xmon.
313 * XXX we should re-enable it when we leave. :)
314 */
315#define SURVEILLANCE_TOKEN 9000
316
317static inline void disable_surveillance(void)
318{
319#ifdef CONFIG_PPC_PSERIES
320 /* Since this can't be a module, args should end up below 4GB. */
321 static struct rtas_args args;
322
323 /*
324 * At this point we have got all the cpus we can into
325 * xmon, so there is hopefully no other cpu calling RTAS
326 * at the moment, even though we don't take rtas.lock.
327 * If we did try to take rtas.lock there would be a
328 * real possibility of deadlock.
329 */
330 args.token = rtas_token("set-indicator");
331 if (args.token == RTAS_UNKNOWN_SERVICE)
332 return;
Laurent Dufour3b8a3c02014-11-24 15:07:53 +0100333 args.nargs = cpu_to_be32(3);
334 args.nret = cpu_to_be32(1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700335 args.rets = &args.args[3];
Laurent Dufour3b8a3c02014-11-24 15:07:53 +0100336 args.args[0] = cpu_to_be32(SURVEILLANCE_TOKEN);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700337 args.args[1] = 0;
338 args.args[2] = 0;
339 enter_rtas(__pa(&args));
340#endif /* CONFIG_PPC_PSERIES */
341}
342
343#ifdef CONFIG_SMP
344static int xmon_speaker;
345
346static void get_output_lock(void)
347{
348 int me = smp_processor_id() + 0x100;
349 int last_speaker = 0, prev;
350 long timeout;
351
352 if (xmon_speaker == me)
353 return;
Michael Ellerman730efb62013-12-23 23:46:04 +1100354
Linus Torvalds1da177e2005-04-16 15:20:36 -0700355 for (;;) {
Michael Ellerman730efb62013-12-23 23:46:04 +1100356 last_speaker = cmpxchg(&xmon_speaker, 0, me);
357 if (last_speaker == 0)
358 return;
359
Michael Ellerman15075892013-12-23 23:46:05 +1100360 /*
361 * Wait a full second for the lock, we might be on a slow
362 * console, but check every 100us.
363 */
364 timeout = 10000;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700365 while (xmon_speaker == last_speaker) {
Michael Ellerman15075892013-12-23 23:46:05 +1100366 if (--timeout > 0) {
367 udelay(100);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700368 continue;
Michael Ellerman15075892013-12-23 23:46:05 +1100369 }
370
Linus Torvalds1da177e2005-04-16 15:20:36 -0700371 /* hostile takeover */
372 prev = cmpxchg(&xmon_speaker, last_speaker, me);
373 if (prev == last_speaker)
374 return;
375 break;
376 }
377 }
378}
379
380static void release_output_lock(void)
381{
382 xmon_speaker = 0;
383}
Michael Ellerman1c8950f2008-05-08 14:27:17 +1000384
385int cpus_are_in_xmon(void)
386{
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000387 return !cpumask_empty(&cpus_in_xmon);
Michael Ellerman1c8950f2008-05-08 14:27:17 +1000388}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700389#endif
390
Josh Boyerdaf8f402009-09-23 03:51:04 +0000391static inline int unrecoverable_excp(struct pt_regs *regs)
392{
Jimi Xenidis08f6d6a2011-09-29 12:05:28 +0000393#if defined(CONFIG_4xx) || defined(CONFIG_PPC_BOOK3E)
Jimi Xenidis66857b32011-09-23 05:40:46 +0000394 /* We have no MSR_RI bit on 4xx or Book3e, so we simply return false */
Josh Boyerdaf8f402009-09-23 03:51:04 +0000395 return 0;
396#else
397 return ((regs->msr & MSR_RI) == 0);
398#endif
399}
400
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000401static int xmon_core(struct pt_regs *regs, int fromipi)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700402{
403 int cmd = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700404 struct bpt *bp;
405 long recurse_jmp[JMP_BUF_LEN];
406 unsigned long offset;
Anton Blanchardf13659e2007-03-21 01:48:34 +1100407 unsigned long flags;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700408#ifdef CONFIG_SMP
409 int cpu;
410 int secondary;
411 unsigned long timeout;
412#endif
413
Anton Blanchardf13659e2007-03-21 01:48:34 +1100414 local_irq_save(flags);
Anton Blancharda71d64b2014-08-05 14:55:00 +1000415 hard_irq_disable();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700416
417 bp = in_breakpoint_table(regs->nip, &offset);
418 if (bp != NULL) {
419 regs->nip = bp->address + offset;
420 atomic_dec(&bp->ref_count);
421 }
422
423 remove_cpu_bpts();
424
425#ifdef CONFIG_SMP
426 cpu = smp_processor_id();
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000427 if (cpumask_test_cpu(cpu, &cpus_in_xmon)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700428 get_output_lock();
429 excprint(regs);
430 printf("cpu 0x%x: Exception %lx %s in xmon, "
431 "returning to main loop\n",
432 cpu, regs->trap, getvecname(TRAP(regs)));
Haren Myneni5cb4cc02005-08-03 15:08:18 +1000433 release_output_lock();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700434 longjmp(xmon_fault_jmp[cpu], 1);
435 }
436
437 if (setjmp(recurse_jmp) != 0) {
438 if (!in_xmon || !xmon_gate) {
Haren Myneni5cb4cc02005-08-03 15:08:18 +1000439 get_output_lock();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700440 printf("xmon: WARNING: bad recursive fault "
441 "on cpu 0x%x\n", cpu);
Haren Myneni5cb4cc02005-08-03 15:08:18 +1000442 release_output_lock();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700443 goto waiting;
444 }
445 secondary = !(xmon_taken && cpu == xmon_owner);
446 goto cmdloop;
447 }
448
449 xmon_fault_jmp[cpu] = recurse_jmp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700450
451 bp = NULL;
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000452 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) == (MSR_IR|MSR_64BIT))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700453 bp = at_breakpoint(regs->nip);
Josh Boyerdaf8f402009-09-23 03:51:04 +0000454 if (bp || unrecoverable_excp(regs))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700455 fromipi = 0;
456
457 if (!fromipi) {
458 get_output_lock();
459 excprint(regs);
460 if (bp) {
Michael Ellerman736256e2014-05-26 21:02:14 +1000461 printf("cpu 0x%x stopped at breakpoint 0x%lx (",
Linus Torvalds1da177e2005-04-16 15:20:36 -0700462 cpu, BP_NUM(bp));
463 xmon_print_symbol(regs->nip, " ", ")\n");
464 }
Josh Boyerdaf8f402009-09-23 03:51:04 +0000465 if (unrecoverable_excp(regs))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700466 printf("WARNING: exception is not recoverable, "
467 "can't continue\n");
468 release_output_lock();
469 }
470
Michael Ellermand2b496e2013-12-23 23:46:06 +1100471 cpumask_set_cpu(cpu, &cpus_in_xmon);
472
Linus Torvalds1da177e2005-04-16 15:20:36 -0700473 waiting:
474 secondary = 1;
475 while (secondary && !xmon_gate) {
476 if (in_xmon == 0) {
477 if (fromipi)
478 goto leave;
479 secondary = test_and_set_bit(0, &in_xmon);
480 }
481 barrier();
482 }
483
484 if (!secondary && !xmon_gate) {
485 /* we are the first cpu to come in */
486 /* interrupt other cpu(s) */
487 int ncpus = num_online_cpus();
488
489 xmon_owner = cpu;
490 mb();
491 if (ncpus > 1) {
Milton Millere0476372011-05-10 19:29:06 +0000492 smp_send_debugger_break();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700493 /* wait for other cpus to come in */
494 for (timeout = 100000000; timeout != 0; --timeout) {
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000495 if (cpumask_weight(&cpus_in_xmon) >= ncpus)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700496 break;
497 barrier();
498 }
499 }
500 remove_bpts();
501 disable_surveillance();
502 /* for breakpoint or single step, print the current instr. */
503 if (bp || TRAP(regs) == 0xd00)
504 ppc_inst_dump(regs->nip, 1, 0);
505 printf("enter ? for help\n");
506 mb();
507 xmon_gate = 1;
508 barrier();
509 }
510
511 cmdloop:
512 while (in_xmon) {
513 if (secondary) {
514 if (cpu == xmon_owner) {
515 if (!test_and_set_bit(0, &xmon_taken)) {
516 secondary = 0;
517 continue;
518 }
519 /* missed it */
520 while (cpu == xmon_owner)
521 barrier();
522 }
523 barrier();
524 } else {
525 cmd = cmds(regs);
526 if (cmd != 0) {
527 /* exiting xmon */
528 insert_bpts();
529 xmon_gate = 0;
530 wmb();
531 in_xmon = 0;
532 break;
533 }
534 /* have switched to some other cpu */
535 secondary = 1;
536 }
537 }
538 leave:
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000539 cpumask_clear_cpu(cpu, &cpus_in_xmon);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700540 xmon_fault_jmp[cpu] = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700541#else
542 /* UP is simple... */
543 if (in_xmon) {
544 printf("Exception %lx %s in xmon, returning to main loop\n",
545 regs->trap, getvecname(TRAP(regs)));
546 longjmp(xmon_fault_jmp[0], 1);
547 }
548 if (setjmp(recurse_jmp) == 0) {
549 xmon_fault_jmp[0] = recurse_jmp;
550 in_xmon = 1;
551
552 excprint(regs);
553 bp = at_breakpoint(regs->nip);
554 if (bp) {
Michael Ellerman736256e2014-05-26 21:02:14 +1000555 printf("Stopped at breakpoint %lx (", BP_NUM(bp));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700556 xmon_print_symbol(regs->nip, " ", ")\n");
557 }
Josh Boyerdaf8f402009-09-23 03:51:04 +0000558 if (unrecoverable_excp(regs))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700559 printf("WARNING: exception is not recoverable, "
560 "can't continue\n");
561 remove_bpts();
562 disable_surveillance();
563 /* for breakpoint or single step, print the current instr. */
564 if (bp || TRAP(regs) == 0xd00)
565 ppc_inst_dump(regs->nip, 1, 0);
566 printf("enter ? for help\n");
567 }
568
569 cmd = cmds(regs);
570
571 insert_bpts();
572 in_xmon = 0;
573#endif
574
Josh Boyercdd39042009-10-05 04:46:05 +0000575#ifdef CONFIG_BOOKE
576 if (regs->msr & MSR_DE) {
577 bp = at_breakpoint(regs->nip);
578 if (bp != NULL) {
579 regs->nip = (unsigned long) &bp->instr[0];
580 atomic_inc(&bp->ref_count);
581 }
582 }
583#else
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000584 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) == (MSR_IR|MSR_64BIT)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700585 bp = at_breakpoint(regs->nip);
586 if (bp != NULL) {
587 int stepped = emulate_step(regs, bp->instr[0]);
588 if (stepped == 0) {
589 regs->nip = (unsigned long) &bp->instr[0];
590 atomic_inc(&bp->ref_count);
591 } else if (stepped < 0) {
592 printf("Couldn't single-step %s instruction\n",
593 (IS_RFID(bp->instr[0])? "rfid": "mtmsrd"));
594 }
595 }
596 }
Josh Boyercdd39042009-10-05 04:46:05 +0000597#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700598 insert_cpu_bpts();
599
Anton Blancharda71d64b2014-08-05 14:55:00 +1000600 touch_nmi_watchdog();
Anton Blanchardf13659e2007-03-21 01:48:34 +1100601 local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700602
Paul Mackerras0a730ae2006-10-03 21:32:49 +1000603 return cmd != 'X' && cmd != EOF;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700604}
605
606int xmon(struct pt_regs *excp)
607{
608 struct pt_regs regs;
609
610 if (excp == NULL) {
Anton Vorontsov322b4392008-12-17 10:08:55 +0000611 ppc_save_regs(&regs);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700612 excp = &regs;
613 }
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200614
Linus Torvalds1da177e2005-04-16 15:20:36 -0700615 return xmon_core(excp, 0);
616}
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000617EXPORT_SYMBOL(xmon);
618
Paul Mackerrasf583ffc2006-10-10 11:47:07 +1000619irqreturn_t xmon_irq(int irq, void *d)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000620{
621 unsigned long flags;
622 local_irq_save(flags);
623 printf("Keyboard interrupt\n");
Paul Mackerrasf583ffc2006-10-10 11:47:07 +1000624 xmon(get_irq_regs());
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000625 local_irq_restore(flags);
626 return IRQ_HANDLED;
627}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700628
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000629static int xmon_bpt(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700630{
631 struct bpt *bp;
632 unsigned long offset;
633
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000634 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) != (MSR_IR|MSR_64BIT))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700635 return 0;
636
637 /* Are we at the trap at bp->instr[1] for some bp? */
638 bp = in_breakpoint_table(regs->nip, &offset);
639 if (bp != NULL && offset == 4) {
640 regs->nip = bp->address + 4;
641 atomic_dec(&bp->ref_count);
642 return 1;
643 }
644
645 /* Are we at a breakpoint? */
646 bp = at_breakpoint(regs->nip);
647 if (!bp)
648 return 0;
649
650 xmon_core(regs, 0);
651
652 return 1;
653}
654
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000655static int xmon_sstep(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700656{
657 if (user_mode(regs))
658 return 0;
659 xmon_core(regs, 0);
660 return 1;
661}
662
Michael Neuling9422de32012-12-20 14:06:44 +0000663static int xmon_break_match(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700664{
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000665 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) != (MSR_IR|MSR_64BIT))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700666 return 0;
Anton Blanchardfd9648d2005-09-10 16:01:11 +1000667 if (dabr.enabled == 0)
668 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700669 xmon_core(regs, 0);
670 return 1;
671}
672
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000673static int xmon_iabr_match(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700674{
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000675 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) != (MSR_IR|MSR_64BIT))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700676 return 0;
Michael Ellerman9f1067c22008-05-08 14:27:16 +1000677 if (iabr == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700678 return 0;
679 xmon_core(regs, 0);
680 return 1;
681}
682
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000683static int xmon_ipi(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700684{
685#ifdef CONFIG_SMP
KOSAKI Motohiro104699c2011-04-28 05:07:23 +0000686 if (in_xmon && !cpumask_test_cpu(smp_processor_id(), &cpus_in_xmon))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700687 xmon_core(regs, 1);
688#endif
689 return 0;
690}
691
Arnd Bergmannb0da9852006-01-11 00:00:05 +0000692static int xmon_fault_handler(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700693{
694 struct bpt *bp;
695 unsigned long offset;
696
697 if (in_xmon && catch_memory_errors)
698 handle_fault(regs); /* doesn't return */
699
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000700 if ((regs->msr & (MSR_IR|MSR_PR|MSR_64BIT)) == (MSR_IR|MSR_64BIT)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700701 bp = in_breakpoint_table(regs->nip, &offset);
702 if (bp != NULL) {
703 regs->nip = bp->address + offset;
704 atomic_dec(&bp->ref_count);
705 }
706 }
707
708 return 0;
709}
710
Linus Torvalds1da177e2005-04-16 15:20:36 -0700711static struct bpt *at_breakpoint(unsigned long pc)
712{
713 int i;
714 struct bpt *bp;
715
716 bp = bpts;
717 for (i = 0; i < NBPTS; ++i, ++bp)
718 if (bp->enabled && pc == bp->address)
719 return bp;
720 return NULL;
721}
722
723static struct bpt *in_breakpoint_table(unsigned long nip, unsigned long *offp)
724{
725 unsigned long off;
726
727 off = nip - (unsigned long) bpts;
728 if (off >= sizeof(bpts))
729 return NULL;
730 off %= sizeof(struct bpt);
731 if (off != offsetof(struct bpt, instr[0])
732 && off != offsetof(struct bpt, instr[1]))
733 return NULL;
734 *offp = off - offsetof(struct bpt, instr[0]);
735 return (struct bpt *) (nip - off);
736}
737
738static struct bpt *new_breakpoint(unsigned long a)
739{
740 struct bpt *bp;
741
742 a &= ~3UL;
743 bp = at_breakpoint(a);
744 if (bp)
745 return bp;
746
747 for (bp = bpts; bp < &bpts[NBPTS]; ++bp) {
748 if (!bp->enabled && atomic_read(&bp->ref_count) == 0) {
749 bp->address = a;
750 bp->instr[1] = bpinstr;
751 store_inst(&bp->instr[1]);
752 return bp;
753 }
754 }
755
756 printf("Sorry, no free breakpoints. Please clear one first.\n");
757 return NULL;
758}
759
760static void insert_bpts(void)
761{
762 int i;
763 struct bpt *bp;
764
765 bp = bpts;
766 for (i = 0; i < NBPTS; ++i, ++bp) {
Michael Ellermanabb90ee2014-12-01 16:54:13 +1100767 if ((bp->enabled & (BP_TRAP|BP_CIABR)) == 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700768 continue;
769 if (mread(bp->address, &bp->instr[0], 4) != 4) {
770 printf("Couldn't read instruction at %lx, "
771 "disabling breakpoint there\n", bp->address);
772 bp->enabled = 0;
773 continue;
774 }
775 if (IS_MTMSRD(bp->instr[0]) || IS_RFID(bp->instr[0])) {
776 printf("Breakpoint at %lx is on an mtmsrd or rfid "
777 "instruction, disabling it\n", bp->address);
778 bp->enabled = 0;
779 continue;
780 }
781 store_inst(&bp->instr[0]);
Michael Ellermanabb90ee2014-12-01 16:54:13 +1100782 if (bp->enabled & BP_CIABR)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700783 continue;
784 if (mwrite(bp->address, &bpinstr, 4) != 4) {
785 printf("Couldn't write instruction at %lx, "
786 "disabling breakpoint there\n", bp->address);
787 bp->enabled &= ~BP_TRAP;
788 continue;
789 }
790 store_inst((void *)bp->address);
791 }
792}
793
794static void insert_cpu_bpts(void)
795{
Michael Neuling9422de32012-12-20 14:06:44 +0000796 struct arch_hw_breakpoint brk;
797
798 if (dabr.enabled) {
799 brk.address = dabr.address;
800 brk.type = (dabr.enabled & HW_BRK_TYPE_DABR) | HW_BRK_TYPE_PRIV_ALL;
801 brk.len = 8;
Paul Gortmaker21f58502014-04-29 15:25:17 -0400802 __set_breakpoint(&brk);
Michael Neuling9422de32012-12-20 14:06:44 +0000803 }
Anshuman Khandual1ad7d702014-11-28 10:06:42 +0530804
805 if (iabr)
806 set_ciabr(iabr->address);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700807}
808
809static void remove_bpts(void)
810{
811 int i;
812 struct bpt *bp;
813 unsigned instr;
814
815 bp = bpts;
816 for (i = 0; i < NBPTS; ++i, ++bp) {
Michael Ellermanabb90ee2014-12-01 16:54:13 +1100817 if ((bp->enabled & (BP_TRAP|BP_CIABR)) != BP_TRAP)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700818 continue;
819 if (mread(bp->address, &instr, 4) == 4
820 && instr == bpinstr
821 && mwrite(bp->address, &bp->instr, 4) != 4)
822 printf("Couldn't remove breakpoint at %lx\n",
823 bp->address);
824 else
825 store_inst((void *)bp->address);
826 }
827}
828
829static void remove_cpu_bpts(void)
830{
Michael Neuling9422de32012-12-20 14:06:44 +0000831 hw_breakpoint_disable();
Anshuman Khandual1ad7d702014-11-28 10:06:42 +0530832 write_ciabr(0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700833}
834
835/* Command interpreting routine */
836static char *last_cmd;
837
838static int
839cmds(struct pt_regs *excp)
840{
841 int cmd = 0;
842
843 last_cmd = NULL;
844 xmon_regs = excp;
Olaf Hering26c8af52006-09-08 16:29:21 +0200845
846 if (!xmon_no_auto_backtrace) {
847 xmon_no_auto_backtrace = 1;
848 xmon_show_stack(excp->gpr[1], excp->link, excp->nip);
849 }
850
Linus Torvalds1da177e2005-04-16 15:20:36 -0700851 for(;;) {
852#ifdef CONFIG_SMP
853 printf("%x:", smp_processor_id());
854#endif /* CONFIG_SMP */
855 printf("mon> ");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700856 flush_input();
857 termch = 0;
858 cmd = skipbl();
859 if( cmd == '\n' ) {
860 if (last_cmd == NULL)
861 continue;
862 take_input(last_cmd);
863 last_cmd = NULL;
864 cmd = inchar();
865 }
866 switch (cmd) {
867 case 'm':
868 cmd = inchar();
869 switch (cmd) {
870 case 'm':
871 case 's':
872 case 'd':
873 memops(cmd);
874 break;
875 case 'l':
876 memlocate();
877 break;
878 case 'z':
879 memzcan();
880 break;
881 case 'i':
David Rientjesb2b755b2011-03-24 15:18:15 -0700882 show_mem(0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700883 break;
884 default:
885 termch = cmd;
886 memex();
887 }
888 break;
889 case 'd':
890 dump();
891 break;
892 case 'l':
893 symbol_lookup();
894 break;
895 case 'r':
896 prregs(excp); /* print regs */
897 break;
898 case 'e':
899 excprint(excp);
900 break;
901 case 'S':
902 super_regs();
903 break;
904 case 't':
905 backtrace(excp);
906 break;
907 case 'f':
908 cacheflush();
909 break;
910 case 's':
Michael Ellermanff8a8f22006-10-24 18:31:27 +0200911 if (do_spu_cmd() == 0)
912 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700913 if (do_step(excp))
914 return cmd;
915 break;
916 case 'x':
917 case 'X':
Benjamin Herrenschmidtbb6b9b22005-11-30 16:54:12 +1100918 return cmd;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700919 case EOF:
Benjamin Herrenschmidtbb6b9b22005-11-30 16:54:12 +1100920 printf(" <no input ...>\n");
921 mdelay(2000);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700922 return cmd;
923 case '?':
Ishizaki Kou4d404ed2007-07-18 19:26:40 +1000924 xmon_puts(help_string);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700925 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700926 case 'b':
927 bpt_cmds();
928 break;
929 case 'C':
930 csum();
931 break;
932 case 'c':
933 if (cpu_cmd())
934 return 0;
935 break;
936 case 'z':
937 bootcmds();
938 break;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000939 case 'p':
940 proccall();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700941 break;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +1000942#ifdef CONFIG_PPC_STD_MMU
Linus Torvalds1da177e2005-04-16 15:20:36 -0700943 case 'u':
944 dump_segments();
945 break;
Michael Ellermand8ee6f32014-11-12 16:54:54 +1100946#elif defined(CONFIG_44x)
Benjamin Herrenschmidt5a8a1a22007-11-16 18:23:33 +1100947 case 'u':
948 dump_tlb_44x();
949 break;
Jimi Xenidis79873e82011-09-29 11:25:10 +0000950#elif defined(CONFIG_PPC_BOOK3E)
Benjamin Herrenschmidt03247152010-07-09 15:34:50 +1000951 case 'u':
952 dump_tlb_book3e();
953 break;
954#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700955 default:
956 printf("Unrecognized command: ");
Michael Ellermane3bc8042012-08-23 22:09:13 +0000957 do {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700958 if (' ' < cmd && cmd <= '~')
959 putchar(cmd);
960 else
961 printf("\\x%x", cmd);
962 cmd = inchar();
Michael Ellermane3bc8042012-08-23 22:09:13 +0000963 } while (cmd != '\n');
Linus Torvalds1da177e2005-04-16 15:20:36 -0700964 printf(" (type ? for help)\n");
965 break;
966 }
967 }
968}
969
Josh Boyercdd39042009-10-05 04:46:05 +0000970#ifdef CONFIG_BOOKE
971static int do_step(struct pt_regs *regs)
972{
973 regs->msr |= MSR_DE;
974 mtspr(SPRN_DBCR0, mfspr(SPRN_DBCR0) | DBCR0_IC | DBCR0_IDM);
975 return 1;
976}
977#else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700978/*
979 * Step a single instruction.
980 * Some instructions we emulate, others we execute with MSR_SE set.
981 */
982static int do_step(struct pt_regs *regs)
983{
984 unsigned int instr;
985 int stepped;
986
987 /* check we are in 64-bit kernel mode, translation enabled */
Michael Ellerman9f0b0792011-04-07 21:56:03 +0000988 if ((regs->msr & (MSR_64BIT|MSR_PR|MSR_IR)) == (MSR_64BIT|MSR_IR)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700989 if (mread(regs->nip, &instr, 4) == 4) {
990 stepped = emulate_step(regs, instr);
991 if (stepped < 0) {
992 printf("Couldn't single-step %s instruction\n",
993 (IS_RFID(instr)? "rfid": "mtmsrd"));
994 return 0;
995 }
996 if (stepped > 0) {
997 regs->trap = 0xd00 | (regs->trap & 1);
998 printf("stepped to ");
999 xmon_print_symbol(regs->nip, " ", "\n");
1000 ppc_inst_dump(regs->nip, 1, 0);
1001 return 0;
1002 }
1003 }
1004 }
1005 regs->msr |= MSR_SE;
1006 return 1;
1007}
Josh Boyercdd39042009-10-05 04:46:05 +00001008#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001009
1010static void bootcmds(void)
1011{
1012 int cmd;
1013
1014 cmd = inchar();
1015 if (cmd == 'r')
1016 ppc_md.restart(NULL);
1017 else if (cmd == 'h')
1018 ppc_md.halt();
1019 else if (cmd == 'p')
Alexander Graf9178ba22014-10-13 16:01:09 +02001020 if (pm_power_off)
1021 pm_power_off();
Linus Torvalds1da177e2005-04-16 15:20:36 -07001022}
1023
1024static int cpu_cmd(void)
1025{
1026#ifdef CONFIG_SMP
Paul Mackerrasfd3bb912013-09-03 20:16:23 +10001027 unsigned long cpu, first_cpu, last_cpu;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001028 int timeout;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001029
1030 if (!scanhex(&cpu)) {
1031 /* print cpus waiting or in xmon */
1032 printf("cpus stopped:");
Paul Mackerrasfd3bb912013-09-03 20:16:23 +10001033 last_cpu = first_cpu = NR_CPUS;
Anton Blanchardbc1d7702012-06-28 19:28:57 +00001034 for_each_possible_cpu(cpu) {
KOSAKI Motohiro104699c2011-04-28 05:07:23 +00001035 if (cpumask_test_cpu(cpu, &cpus_in_xmon)) {
Paul Mackerrasfd3bb912013-09-03 20:16:23 +10001036 if (cpu == last_cpu + 1) {
1037 last_cpu = cpu;
1038 } else {
1039 if (last_cpu != first_cpu)
Michael Ellerman736256e2014-05-26 21:02:14 +10001040 printf("-0x%lx", last_cpu);
Paul Mackerrasfd3bb912013-09-03 20:16:23 +10001041 last_cpu = first_cpu = cpu;
Michael Ellerman736256e2014-05-26 21:02:14 +10001042 printf(" 0x%lx", cpu);
Paul Mackerrasfd3bb912013-09-03 20:16:23 +10001043 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001044 }
1045 }
Paul Mackerrasfd3bb912013-09-03 20:16:23 +10001046 if (last_cpu != first_cpu)
Michael Ellerman736256e2014-05-26 21:02:14 +10001047 printf("-0x%lx", last_cpu);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001048 printf("\n");
1049 return 0;
1050 }
1051 /* try to switch to cpu specified */
KOSAKI Motohiro104699c2011-04-28 05:07:23 +00001052 if (!cpumask_test_cpu(cpu, &cpus_in_xmon)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001053 printf("cpu 0x%x isn't in xmon\n", cpu);
1054 return 0;
1055 }
1056 xmon_taken = 0;
1057 mb();
1058 xmon_owner = cpu;
1059 timeout = 10000000;
1060 while (!xmon_taken) {
1061 if (--timeout == 0) {
1062 if (test_and_set_bit(0, &xmon_taken))
1063 break;
1064 /* take control back */
1065 mb();
1066 xmon_owner = smp_processor_id();
Michael Ellerman736256e2014-05-26 21:02:14 +10001067 printf("cpu 0x%x didn't take control\n", cpu);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001068 return 0;
1069 }
1070 barrier();
1071 }
1072 return 1;
1073#else
1074 return 0;
1075#endif /* CONFIG_SMP */
1076}
1077
1078static unsigned short fcstab[256] = {
1079 0x0000, 0x1189, 0x2312, 0x329b, 0x4624, 0x57ad, 0x6536, 0x74bf,
1080 0x8c48, 0x9dc1, 0xaf5a, 0xbed3, 0xca6c, 0xdbe5, 0xe97e, 0xf8f7,
1081 0x1081, 0x0108, 0x3393, 0x221a, 0x56a5, 0x472c, 0x75b7, 0x643e,
1082 0x9cc9, 0x8d40, 0xbfdb, 0xae52, 0xdaed, 0xcb64, 0xf9ff, 0xe876,
1083 0x2102, 0x308b, 0x0210, 0x1399, 0x6726, 0x76af, 0x4434, 0x55bd,
1084 0xad4a, 0xbcc3, 0x8e58, 0x9fd1, 0xeb6e, 0xfae7, 0xc87c, 0xd9f5,
1085 0x3183, 0x200a, 0x1291, 0x0318, 0x77a7, 0x662e, 0x54b5, 0x453c,
1086 0xbdcb, 0xac42, 0x9ed9, 0x8f50, 0xfbef, 0xea66, 0xd8fd, 0xc974,
1087 0x4204, 0x538d, 0x6116, 0x709f, 0x0420, 0x15a9, 0x2732, 0x36bb,
1088 0xce4c, 0xdfc5, 0xed5e, 0xfcd7, 0x8868, 0x99e1, 0xab7a, 0xbaf3,
1089 0x5285, 0x430c, 0x7197, 0x601e, 0x14a1, 0x0528, 0x37b3, 0x263a,
1090 0xdecd, 0xcf44, 0xfddf, 0xec56, 0x98e9, 0x8960, 0xbbfb, 0xaa72,
1091 0x6306, 0x728f, 0x4014, 0x519d, 0x2522, 0x34ab, 0x0630, 0x17b9,
1092 0xef4e, 0xfec7, 0xcc5c, 0xddd5, 0xa96a, 0xb8e3, 0x8a78, 0x9bf1,
1093 0x7387, 0x620e, 0x5095, 0x411c, 0x35a3, 0x242a, 0x16b1, 0x0738,
1094 0xffcf, 0xee46, 0xdcdd, 0xcd54, 0xb9eb, 0xa862, 0x9af9, 0x8b70,
1095 0x8408, 0x9581, 0xa71a, 0xb693, 0xc22c, 0xd3a5, 0xe13e, 0xf0b7,
1096 0x0840, 0x19c9, 0x2b52, 0x3adb, 0x4e64, 0x5fed, 0x6d76, 0x7cff,
1097 0x9489, 0x8500, 0xb79b, 0xa612, 0xd2ad, 0xc324, 0xf1bf, 0xe036,
1098 0x18c1, 0x0948, 0x3bd3, 0x2a5a, 0x5ee5, 0x4f6c, 0x7df7, 0x6c7e,
1099 0xa50a, 0xb483, 0x8618, 0x9791, 0xe32e, 0xf2a7, 0xc03c, 0xd1b5,
1100 0x2942, 0x38cb, 0x0a50, 0x1bd9, 0x6f66, 0x7eef, 0x4c74, 0x5dfd,
1101 0xb58b, 0xa402, 0x9699, 0x8710, 0xf3af, 0xe226, 0xd0bd, 0xc134,
1102 0x39c3, 0x284a, 0x1ad1, 0x0b58, 0x7fe7, 0x6e6e, 0x5cf5, 0x4d7c,
1103 0xc60c, 0xd785, 0xe51e, 0xf497, 0x8028, 0x91a1, 0xa33a, 0xb2b3,
1104 0x4a44, 0x5bcd, 0x6956, 0x78df, 0x0c60, 0x1de9, 0x2f72, 0x3efb,
1105 0xd68d, 0xc704, 0xf59f, 0xe416, 0x90a9, 0x8120, 0xb3bb, 0xa232,
1106 0x5ac5, 0x4b4c, 0x79d7, 0x685e, 0x1ce1, 0x0d68, 0x3ff3, 0x2e7a,
1107 0xe70e, 0xf687, 0xc41c, 0xd595, 0xa12a, 0xb0a3, 0x8238, 0x93b1,
1108 0x6b46, 0x7acf, 0x4854, 0x59dd, 0x2d62, 0x3ceb, 0x0e70, 0x1ff9,
1109 0xf78f, 0xe606, 0xd49d, 0xc514, 0xb1ab, 0xa022, 0x92b9, 0x8330,
1110 0x7bc7, 0x6a4e, 0x58d5, 0x495c, 0x3de3, 0x2c6a, 0x1ef1, 0x0f78
1111};
1112
1113#define FCS(fcs, c) (((fcs) >> 8) ^ fcstab[((fcs) ^ (c)) & 0xff])
1114
1115static void
1116csum(void)
1117{
1118 unsigned int i;
1119 unsigned short fcs;
1120 unsigned char v;
1121
1122 if (!scanhex(&adrs))
1123 return;
1124 if (!scanhex(&ncsum))
1125 return;
1126 fcs = 0xffff;
1127 for (i = 0; i < ncsum; ++i) {
1128 if (mread(adrs+i, &v, 1) == 0) {
Michael Ellerman736256e2014-05-26 21:02:14 +10001129 printf("csum stopped at "REG"\n", adrs+i);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001130 break;
1131 }
1132 fcs = FCS(fcs, v);
1133 }
1134 printf("%x\n", fcs);
1135}
1136
1137/*
1138 * Check if this is a suitable place to put a breakpoint.
1139 */
1140static long check_bp_loc(unsigned long addr)
1141{
1142 unsigned int instr;
1143
1144 addr &= ~3;
Michael Ellerman51fae6de2005-12-04 18:39:15 +11001145 if (!is_kernel_addr(addr)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001146 printf("Breakpoints may only be placed at kernel addresses\n");
1147 return 0;
1148 }
1149 if (!mread(addr, &instr, sizeof(instr))) {
1150 printf("Can't read instruction at address %lx\n", addr);
1151 return 0;
1152 }
1153 if (IS_MTMSRD(instr) || IS_RFID(instr)) {
1154 printf("Breakpoints may not be placed on mtmsrd or rfid "
1155 "instructions\n");
1156 return 0;
1157 }
1158 return 1;
1159}
1160
Michael Ellermane3bc8042012-08-23 22:09:13 +00001161static char *breakpoint_help_string =
Linus Torvalds1da177e2005-04-16 15:20:36 -07001162 "Breakpoint command usage:\n"
1163 "b show breakpoints\n"
1164 "b <addr> [cnt] set breakpoint at given instr addr\n"
1165 "bc clear all breakpoints\n"
1166 "bc <n/addr> clear breakpoint number n or at addr\n"
Anshuman Khandual1ad7d702014-11-28 10:06:42 +05301167 "bi <addr> [cnt] set hardware instr breakpoint (POWER8 only)\n"
Linus Torvalds1da177e2005-04-16 15:20:36 -07001168 "bd <addr> [cnt] set hardware data breakpoint\n"
1169 "";
1170
1171static void
1172bpt_cmds(void)
1173{
1174 int cmd;
1175 unsigned long a;
1176 int mode, i;
1177 struct bpt *bp;
1178 const char badaddr[] = "Only kernel addresses are permitted "
1179 "for breakpoints\n";
1180
1181 cmd = inchar();
1182 switch (cmd) {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001183#ifndef CONFIG_8xx
Linus Torvalds1da177e2005-04-16 15:20:36 -07001184 case 'd': /* bd - hardware data breakpoint */
1185 mode = 7;
1186 cmd = inchar();
1187 if (cmd == 'r')
1188 mode = 5;
1189 else if (cmd == 'w')
1190 mode = 6;
1191 else
1192 termch = cmd;
1193 dabr.address = 0;
1194 dabr.enabled = 0;
1195 if (scanhex(&dabr.address)) {
Michael Ellerman51fae6de2005-12-04 18:39:15 +11001196 if (!is_kernel_addr(dabr.address)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001197 printf(badaddr);
1198 break;
1199 }
Michael Neuling9422de32012-12-20 14:06:44 +00001200 dabr.address &= ~HW_BRK_TYPE_DABR;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001201 dabr.enabled = mode | BP_DABR;
1202 }
1203 break;
1204
1205 case 'i': /* bi - hardware instr breakpoint */
Anshuman Khandual1ad7d702014-11-28 10:06:42 +05301206 if (!cpu_has_feature(CPU_FTR_ARCH_207S)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001207 printf("Hardware instruction breakpoint "
1208 "not supported on this cpu\n");
1209 break;
1210 }
1211 if (iabr) {
Michael Ellermanabb90ee2014-12-01 16:54:13 +11001212 iabr->enabled &= ~BP_CIABR;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001213 iabr = NULL;
1214 }
1215 if (!scanhex(&a))
1216 break;
1217 if (!check_bp_loc(a))
1218 break;
1219 bp = new_breakpoint(a);
1220 if (bp != NULL) {
Michael Ellermanabb90ee2014-12-01 16:54:13 +11001221 bp->enabled |= BP_CIABR;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001222 iabr = bp;
1223 }
1224 break;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001225#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001226
1227 case 'c':
1228 if (!scanhex(&a)) {
1229 /* clear all breakpoints */
1230 for (i = 0; i < NBPTS; ++i)
1231 bpts[i].enabled = 0;
1232 iabr = NULL;
1233 dabr.enabled = 0;
1234 printf("All breakpoints cleared\n");
1235 break;
1236 }
1237
1238 if (a <= NBPTS && a >= 1) {
1239 /* assume a breakpoint number */
1240 bp = &bpts[a-1]; /* bp nums are 1 based */
1241 } else {
1242 /* assume a breakpoint address */
1243 bp = at_breakpoint(a);
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001244 if (bp == NULL) {
Michael Ellerman736256e2014-05-26 21:02:14 +10001245 printf("No breakpoint at %lx\n", a);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001246 break;
1247 }
1248 }
1249
Michael Ellerman736256e2014-05-26 21:02:14 +10001250 printf("Cleared breakpoint %lx (", BP_NUM(bp));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001251 xmon_print_symbol(bp->address, " ", ")\n");
1252 bp->enabled = 0;
1253 break;
1254
1255 default:
1256 termch = cmd;
Michael Ellermane3bc8042012-08-23 22:09:13 +00001257 cmd = skipbl();
Linus Torvalds1da177e2005-04-16 15:20:36 -07001258 if (cmd == '?') {
1259 printf(breakpoint_help_string);
1260 break;
1261 }
1262 termch = cmd;
1263 if (!scanhex(&a)) {
1264 /* print all breakpoints */
1265 printf(" type address\n");
1266 if (dabr.enabled) {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001267 printf(" data "REG" [", dabr.address);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001268 if (dabr.enabled & 1)
1269 printf("r");
1270 if (dabr.enabled & 2)
1271 printf("w");
1272 printf("]\n");
1273 }
1274 for (bp = bpts; bp < &bpts[NBPTS]; ++bp) {
1275 if (!bp->enabled)
1276 continue;
1277 printf("%2x %s ", BP_NUM(bp),
Michael Ellermanabb90ee2014-12-01 16:54:13 +11001278 (bp->enabled & BP_CIABR) ? "inst": "trap");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001279 xmon_print_symbol(bp->address, " ", "\n");
1280 }
1281 break;
1282 }
1283
1284 if (!check_bp_loc(a))
1285 break;
1286 bp = new_breakpoint(a);
1287 if (bp != NULL)
1288 bp->enabled |= BP_TRAP;
1289 break;
1290 }
1291}
1292
1293/* Very cheap human name for vector lookup. */
1294static
1295const char *getvecname(unsigned long vec)
1296{
1297 char *ret;
1298
1299 switch (vec) {
1300 case 0x100: ret = "(System Reset)"; break;
1301 case 0x200: ret = "(Machine Check)"; break;
1302 case 0x300: ret = "(Data Access)"; break;
1303 case 0x380: ret = "(Data SLB Access)"; break;
1304 case 0x400: ret = "(Instruction Access)"; break;
1305 case 0x480: ret = "(Instruction SLB Access)"; break;
1306 case 0x500: ret = "(Hardware Interrupt)"; break;
1307 case 0x600: ret = "(Alignment)"; break;
1308 case 0x700: ret = "(Program Check)"; break;
1309 case 0x800: ret = "(FPU Unavailable)"; break;
1310 case 0x900: ret = "(Decrementer)"; break;
Michael Ellerman660e0342013-08-15 15:22:16 +10001311 case 0x980: ret = "(Hypervisor Decrementer)"; break;
1312 case 0xa00: ret = "(Doorbell)"; break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001313 case 0xc00: ret = "(System Call)"; break;
1314 case 0xd00: ret = "(Single Step)"; break;
Michael Ellerman660e0342013-08-15 15:22:16 +10001315 case 0xe40: ret = "(Emulation Assist)"; break;
1316 case 0xe60: ret = "(HMI)"; break;
1317 case 0xe80: ret = "(Hypervisor Doorbell)"; break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001318 case 0xf00: ret = "(Performance Monitor)"; break;
1319 case 0xf20: ret = "(Altivec Unavailable)"; break;
1320 case 0x1300: ret = "(Instruction Breakpoint)"; break;
Michael Ellerman660e0342013-08-15 15:22:16 +10001321 case 0x1500: ret = "(Denormalisation)"; break;
1322 case 0x1700: ret = "(Altivec Assist)"; break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001323 default: ret = "";
1324 }
1325 return ret;
1326}
1327
1328static void get_function_bounds(unsigned long pc, unsigned long *startp,
1329 unsigned long *endp)
1330{
1331 unsigned long size, offset;
1332 const char *name;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001333
1334 *startp = *endp = 0;
1335 if (pc == 0)
1336 return;
1337 if (setjmp(bus_error_jmp) == 0) {
1338 catch_memory_errors = 1;
1339 sync();
Alexey Dobriyanffb45122007-05-08 00:28:41 -07001340 name = kallsyms_lookup(pc, &size, &offset, NULL, tmpstr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001341 if (name != NULL) {
1342 *startp = pc - offset;
1343 *endp = pc - offset + size;
1344 }
1345 sync();
1346 }
1347 catch_memory_errors = 0;
1348}
1349
Benjamin Herrenschmidtec2b36b2008-04-17 14:34:59 +10001350#define LRSAVE_OFFSET (STACK_FRAME_LR_SAVE * sizeof(unsigned long))
1351#define MARKER_OFFSET (STACK_FRAME_MARKER * sizeof(unsigned long))
1352
Linus Torvalds1da177e2005-04-16 15:20:36 -07001353static void xmon_show_stack(unsigned long sp, unsigned long lr,
1354 unsigned long pc)
1355{
Michael Ellerman0104cd62012-10-09 04:20:36 +00001356 int max_to_print = 64;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001357 unsigned long ip;
1358 unsigned long newsp;
1359 unsigned long marker;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001360 struct pt_regs regs;
1361
Michael Ellerman0104cd62012-10-09 04:20:36 +00001362 while (max_to_print--) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001363 if (sp < PAGE_OFFSET) {
1364 if (sp != 0)
1365 printf("SP (%lx) is in userspace\n", sp);
1366 break;
1367 }
1368
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001369 if (!mread(sp + LRSAVE_OFFSET, &ip, sizeof(unsigned long))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001370 || !mread(sp, &newsp, sizeof(unsigned long))) {
1371 printf("Couldn't read stack frame at %lx\n", sp);
1372 break;
1373 }
1374
1375 /*
1376 * For the first stack frame, try to work out if
1377 * LR and/or the saved LR value in the bottommost
1378 * stack frame are valid.
1379 */
1380 if ((pc | lr) != 0) {
1381 unsigned long fnstart, fnend;
1382 unsigned long nextip;
1383 int printip = 1;
1384
1385 get_function_bounds(pc, &fnstart, &fnend);
1386 nextip = 0;
1387 if (newsp > sp)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001388 mread(newsp + LRSAVE_OFFSET, &nextip,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001389 sizeof(unsigned long));
1390 if (lr == ip) {
1391 if (lr < PAGE_OFFSET
1392 || (fnstart <= lr && lr < fnend))
1393 printip = 0;
1394 } else if (lr == nextip) {
1395 printip = 0;
1396 } else if (lr >= PAGE_OFFSET
1397 && !(fnstart <= lr && lr < fnend)) {
1398 printf("[link register ] ");
1399 xmon_print_symbol(lr, " ", "\n");
1400 }
1401 if (printip) {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001402 printf("["REG"] ", sp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001403 xmon_print_symbol(ip, " ", " (unreliable)\n");
1404 }
1405 pc = lr = 0;
1406
1407 } else {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001408 printf("["REG"] ", sp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001409 xmon_print_symbol(ip, " ", "\n");
1410 }
1411
1412 /* Look for "regshere" marker to see if this is
1413 an exception frame. */
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001414 if (mread(sp + MARKER_OFFSET, &marker, sizeof(unsigned long))
Benjamin Herrenschmidtec2b36b2008-04-17 14:34:59 +10001415 && marker == STACK_FRAME_REGS_MARKER) {
Michael Ellermanc4de3802012-10-09 04:20:35 +00001416 if (mread(sp + STACK_FRAME_OVERHEAD, &regs, sizeof(regs))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001417 != sizeof(regs)) {
1418 printf("Couldn't read registers at %lx\n",
Michael Ellermanc4de3802012-10-09 04:20:35 +00001419 sp + STACK_FRAME_OVERHEAD);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001420 break;
1421 }
Michael Ellermane3bc8042012-08-23 22:09:13 +00001422 printf("--- Exception: %lx %s at ", regs.trap,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001423 getvecname(TRAP(&regs)));
1424 pc = regs.nip;
1425 lr = regs.link;
1426 xmon_print_symbol(pc, " ", "\n");
1427 }
1428
1429 if (newsp == 0)
1430 break;
1431
1432 sp = newsp;
Michael Ellerman0104cd62012-10-09 04:20:36 +00001433 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001434}
1435
1436static void backtrace(struct pt_regs *excp)
1437{
1438 unsigned long sp;
1439
1440 if (scanhex(&sp))
1441 xmon_show_stack(sp, 0, 0);
1442 else
1443 xmon_show_stack(excp->gpr[1], excp->link, excp->nip);
1444 scannl();
1445}
1446
1447static void print_bug_trap(struct pt_regs *regs)
1448{
Paul Mackerrasebdba9a2008-10-31 21:34:09 +11001449#ifdef CONFIG_BUG
Jeremy Fitzhardinge73c9cea2006-12-08 03:30:41 -08001450 const struct bug_entry *bug;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001451 unsigned long addr;
1452
1453 if (regs->msr & MSR_PR)
1454 return; /* not in kernel */
1455 addr = regs->nip; /* address of trap instruction */
1456 if (addr < PAGE_OFFSET)
1457 return;
1458 bug = find_bug(regs->nip);
1459 if (bug == NULL)
1460 return;
Jeremy Fitzhardinge73c9cea2006-12-08 03:30:41 -08001461 if (is_warning_bug(bug))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001462 return;
1463
Stephen Rothwell0a7c7ef2007-03-04 17:05:34 +11001464#ifdef CONFIG_DEBUG_BUGVERBOSE
Jeremy Fitzhardinge73c9cea2006-12-08 03:30:41 -08001465 printf("kernel BUG at %s:%u!\n",
1466 bug->file, bug->line);
Stephen Rothwell0a7c7ef2007-03-04 17:05:34 +11001467#else
1468 printf("kernel BUG at %p!\n", (void *)bug->bug_addr);
1469#endif
Paul Mackerrasebdba9a2008-10-31 21:34:09 +11001470#endif /* CONFIG_BUG */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001471}
1472
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001473static void excprint(struct pt_regs *fp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001474{
1475 unsigned long trap;
1476
1477#ifdef CONFIG_SMP
1478 printf("cpu 0x%x: ", smp_processor_id());
1479#endif /* CONFIG_SMP */
1480
1481 trap = TRAP(fp);
1482 printf("Vector: %lx %s at [%lx]\n", fp->trap, getvecname(trap), fp);
1483 printf(" pc: ");
1484 xmon_print_symbol(fp->nip, ": ", "\n");
1485
1486 printf(" lr: ", fp->link);
1487 xmon_print_symbol(fp->link, ": ", "\n");
1488
1489 printf(" sp: %lx\n", fp->gpr[1]);
1490 printf(" msr: %lx\n", fp->msr);
1491
Aneesh Kumar K.Vce541522013-04-28 09:37:26 +00001492 if (trap == 0x300 || trap == 0x380 || trap == 0x600 || trap == 0x200) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001493 printf(" dar: %lx\n", fp->dar);
1494 if (trap != 0x380)
1495 printf(" dsisr: %lx\n", fp->dsisr);
1496 }
1497
1498 printf(" current = 0x%lx\n", current);
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001499#ifdef CONFIG_PPC64
Benjamin Herrenschmidt7230c562012-03-06 18:27:59 +11001500 printf(" paca = 0x%lx\t softe: %d\t irq_happened: 0x%02x\n",
1501 local_paca, local_paca->soft_enabled, local_paca->irq_happened);
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001502#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001503 if (current) {
1504 printf(" pid = %ld, comm = %s\n",
1505 current->pid, current->comm);
1506 }
1507
1508 if (trap == 0x700)
1509 print_bug_trap(fp);
1510}
1511
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001512static void prregs(struct pt_regs *fp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001513{
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001514 int n, trap;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001515 unsigned long base;
1516 struct pt_regs regs;
1517
1518 if (scanhex(&base)) {
1519 if (setjmp(bus_error_jmp) == 0) {
1520 catch_memory_errors = 1;
1521 sync();
1522 regs = *(struct pt_regs *)base;
1523 sync();
1524 __delay(200);
1525 } else {
1526 catch_memory_errors = 0;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001527 printf("*** Error reading registers from "REG"\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001528 base);
1529 return;
1530 }
1531 catch_memory_errors = 0;
1532 fp = &regs;
1533 }
1534
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001535#ifdef CONFIG_PPC64
Linus Torvalds1da177e2005-04-16 15:20:36 -07001536 if (FULL_REGS(fp)) {
1537 for (n = 0; n < 16; ++n)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001538 printf("R%.2ld = "REG" R%.2ld = "REG"\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001539 n, fp->gpr[n], n+16, fp->gpr[n+16]);
1540 } else {
1541 for (n = 0; n < 7; ++n)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001542 printf("R%.2ld = "REG" R%.2ld = "REG"\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001543 n, fp->gpr[n], n+7, fp->gpr[n+7]);
1544 }
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001545#else
1546 for (n = 0; n < 32; ++n) {
1547 printf("R%.2d = %.8x%s", n, fp->gpr[n],
1548 (n & 3) == 3? "\n": " ");
1549 if (n == 12 && !FULL_REGS(fp)) {
1550 printf("\n");
1551 break;
1552 }
1553 }
1554#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001555 printf("pc = ");
1556 xmon_print_symbol(fp->nip, " ", "\n");
Paul Mackerras48404f22011-05-01 19:48:20 +00001557 if (TRAP(fp) != 0xc00 && cpu_has_feature(CPU_FTR_CFAR)) {
1558 printf("cfar= ");
1559 xmon_print_symbol(fp->orig_gpr3, " ", "\n");
1560 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001561 printf("lr = ");
1562 xmon_print_symbol(fp->link, " ", "\n");
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001563 printf("msr = "REG" cr = %.8lx\n", fp->msr, fp->ccr);
1564 printf("ctr = "REG" xer = "REG" trap = %4lx\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001565 fp->ctr, fp->xer, fp->trap);
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001566 trap = TRAP(fp);
1567 if (trap == 0x300 || trap == 0x380 || trap == 0x600)
1568 printf("dar = "REG" dsisr = %.8lx\n", fp->dar, fp->dsisr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001569}
1570
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001571static void cacheflush(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001572{
1573 int cmd;
1574 unsigned long nflush;
1575
1576 cmd = inchar();
1577 if (cmd != 'i')
1578 termch = cmd;
1579 scanhex((void *)&adrs);
1580 if (termch != '\n')
1581 termch = 0;
1582 nflush = 1;
1583 scanhex(&nflush);
1584 nflush = (nflush + L1_CACHE_BYTES - 1) / L1_CACHE_BYTES;
1585 if (setjmp(bus_error_jmp) == 0) {
1586 catch_memory_errors = 1;
1587 sync();
1588
1589 if (cmd != 'i') {
1590 for (; nflush > 0; --nflush, adrs += L1_CACHE_BYTES)
1591 cflush((void *) adrs);
1592 } else {
1593 for (; nflush > 0; --nflush, adrs += L1_CACHE_BYTES)
1594 cinval((void *) adrs);
1595 }
1596 sync();
1597 /* wait a little while to see if we get a machine check */
1598 __delay(200);
1599 }
1600 catch_memory_errors = 0;
1601}
1602
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001603static unsigned long
Linus Torvalds1da177e2005-04-16 15:20:36 -07001604read_spr(int n)
1605{
1606 unsigned int instrs[2];
1607 unsigned long (*code)(void);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001608 unsigned long ret = -1UL;
Paul Mackerras548cceb2005-11-11 22:36:34 +11001609#ifdef CONFIG_PPC64
1610 unsigned long opd[3];
Linus Torvalds1da177e2005-04-16 15:20:36 -07001611
Linus Torvalds1da177e2005-04-16 15:20:36 -07001612 opd[0] = (unsigned long)instrs;
1613 opd[1] = 0;
1614 opd[2] = 0;
Paul Mackerras548cceb2005-11-11 22:36:34 +11001615 code = (unsigned long (*)(void)) opd;
1616#else
1617 code = (unsigned long (*)(void)) instrs;
1618#endif
1619
1620 /* mfspr r3,n; blr */
1621 instrs[0] = 0x7c6002a6 + ((n & 0x1F) << 16) + ((n & 0x3e0) << 6);
1622 instrs[1] = 0x4e800020;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001623 store_inst(instrs);
1624 store_inst(instrs+1);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001625
1626 if (setjmp(bus_error_jmp) == 0) {
1627 catch_memory_errors = 1;
1628 sync();
1629
1630 ret = code();
1631
1632 sync();
1633 /* wait a little while to see if we get a machine check */
1634 __delay(200);
1635 n = size;
1636 }
1637
1638 return ret;
1639}
1640
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001641static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07001642write_spr(int n, unsigned long val)
1643{
1644 unsigned int instrs[2];
1645 unsigned long (*code)(unsigned long);
Paul Mackerras548cceb2005-11-11 22:36:34 +11001646#ifdef CONFIG_PPC64
Linus Torvalds1da177e2005-04-16 15:20:36 -07001647 unsigned long opd[3];
1648
Linus Torvalds1da177e2005-04-16 15:20:36 -07001649 opd[0] = (unsigned long)instrs;
1650 opd[1] = 0;
1651 opd[2] = 0;
Paul Mackerras548cceb2005-11-11 22:36:34 +11001652 code = (unsigned long (*)(unsigned long)) opd;
1653#else
1654 code = (unsigned long (*)(unsigned long)) instrs;
1655#endif
1656
1657 instrs[0] = 0x7c6003a6 + ((n & 0x1F) << 16) + ((n & 0x3e0) << 6);
1658 instrs[1] = 0x4e800020;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001659 store_inst(instrs);
1660 store_inst(instrs+1);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001661
1662 if (setjmp(bus_error_jmp) == 0) {
1663 catch_memory_errors = 1;
1664 sync();
1665
1666 code(val);
1667
1668 sync();
1669 /* wait a little while to see if we get a machine check */
1670 __delay(200);
1671 n = size;
1672 }
1673}
1674
1675static unsigned long regno;
1676extern char exc_prolog;
1677extern char dec_exc;
1678
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001679static void super_regs(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001680{
1681 int cmd;
1682 unsigned long val;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001683
1684 cmd = skipbl();
1685 if (cmd == '\n') {
Michael Ellermane3bc8042012-08-23 22:09:13 +00001686 unsigned long sp, toc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001687 asm("mr %0,1" : "=r" (sp) :);
1688 asm("mr %0,2" : "=r" (toc) :);
1689
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001690 printf("msr = "REG" sprg0= "REG"\n",
1691 mfmsr(), mfspr(SPRN_SPRG0));
1692 printf("pvr = "REG" sprg1= "REG"\n",
Michael Ellermane3bc8042012-08-23 22:09:13 +00001693 mfspr(SPRN_PVR), mfspr(SPRN_SPRG1));
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001694 printf("dec = "REG" sprg2= "REG"\n",
1695 mfspr(SPRN_DEC), mfspr(SPRN_SPRG2));
1696 printf("sp = "REG" sprg3= "REG"\n", sp, mfspr(SPRN_SPRG3));
1697 printf("toc = "REG" dar = "REG"\n", toc, mfspr(SPRN_DAR));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001698
1699 return;
1700 }
1701
1702 scanhex(&regno);
1703 switch (cmd) {
1704 case 'w':
1705 val = read_spr(regno);
1706 scanhex(&val);
1707 write_spr(regno, val);
1708 /* fall through */
1709 case 'r':
1710 printf("spr %lx = %lx\n", regno, read_spr(regno));
1711 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001712 }
1713 scannl();
1714}
1715
1716/*
1717 * Stuff for reading and writing memory safely
1718 */
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001719static int
Linus Torvalds1da177e2005-04-16 15:20:36 -07001720mread(unsigned long adrs, void *buf, int size)
1721{
1722 volatile int n;
1723 char *p, *q;
1724
1725 n = 0;
1726 if (setjmp(bus_error_jmp) == 0) {
1727 catch_memory_errors = 1;
1728 sync();
1729 p = (char *)adrs;
1730 q = (char *)buf;
1731 switch (size) {
1732 case 2:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001733 *(u16 *)q = *(u16 *)p;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001734 break;
1735 case 4:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001736 *(u32 *)q = *(u32 *)p;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001737 break;
1738 case 8:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001739 *(u64 *)q = *(u64 *)p;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001740 break;
1741 default:
1742 for( ; n < size; ++n) {
1743 *q++ = *p++;
1744 sync();
1745 }
1746 }
1747 sync();
1748 /* wait a little while to see if we get a machine check */
1749 __delay(200);
1750 n = size;
1751 }
1752 catch_memory_errors = 0;
1753 return n;
1754}
1755
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001756static int
Linus Torvalds1da177e2005-04-16 15:20:36 -07001757mwrite(unsigned long adrs, void *buf, int size)
1758{
1759 volatile int n;
1760 char *p, *q;
1761
1762 n = 0;
1763 if (setjmp(bus_error_jmp) == 0) {
1764 catch_memory_errors = 1;
1765 sync();
1766 p = (char *) adrs;
1767 q = (char *) buf;
1768 switch (size) {
1769 case 2:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001770 *(u16 *)p = *(u16 *)q;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001771 break;
1772 case 4:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001773 *(u32 *)p = *(u32 *)q;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001774 break;
1775 case 8:
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001776 *(u64 *)p = *(u64 *)q;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001777 break;
1778 default:
1779 for ( ; n < size; ++n) {
1780 *p++ = *q++;
1781 sync();
1782 }
1783 }
1784 sync();
1785 /* wait a little while to see if we get a machine check */
1786 __delay(200);
1787 n = size;
1788 } else {
Michael Ellerman736256e2014-05-26 21:02:14 +10001789 printf("*** Error writing address "REG"\n", adrs + n);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001790 }
1791 catch_memory_errors = 0;
1792 return n;
1793}
1794
1795static int fault_type;
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001796static int fault_except;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001797static char *fault_chars[] = { "--", "**", "##" };
1798
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001799static int handle_fault(struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001800{
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10001801 fault_except = TRAP(regs);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001802 switch (TRAP(regs)) {
1803 case 0x200:
1804 fault_type = 0;
1805 break;
1806 case 0x300:
1807 case 0x380:
1808 fault_type = 1;
1809 break;
1810 default:
1811 fault_type = 2;
1812 }
1813
1814 longjmp(bus_error_jmp, 1);
1815
1816 return 0;
1817}
1818
1819#define SWAP(a, b, t) ((t) = (a), (a) = (b), (b) = (t))
1820
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001821static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07001822byterev(unsigned char *val, int size)
1823{
1824 int t;
1825
1826 switch (size) {
1827 case 2:
1828 SWAP(val[0], val[1], t);
1829 break;
1830 case 4:
1831 SWAP(val[0], val[3], t);
1832 SWAP(val[1], val[2], t);
1833 break;
1834 case 8: /* is there really any use for this? */
1835 SWAP(val[0], val[7], t);
1836 SWAP(val[1], val[6], t);
1837 SWAP(val[2], val[5], t);
1838 SWAP(val[3], val[4], t);
1839 break;
1840 }
1841}
1842
1843static int brev;
1844static int mnoread;
1845
Michael Ellermane3bc8042012-08-23 22:09:13 +00001846static char *memex_help_string =
Linus Torvalds1da177e2005-04-16 15:20:36 -07001847 "Memory examine command usage:\n"
1848 "m [addr] [flags] examine/change memory\n"
1849 " addr is optional. will start where left off.\n"
1850 " flags may include chars from this set:\n"
1851 " b modify by bytes (default)\n"
1852 " w modify by words (2 byte)\n"
1853 " l modify by longs (4 byte)\n"
1854 " d modify by doubleword (8 byte)\n"
1855 " r toggle reverse byte order mode\n"
1856 " n do not read memory (for i/o spaces)\n"
1857 " . ok to read (default)\n"
1858 "NOTE: flags are saved as defaults\n"
1859 "";
1860
Michael Ellermane3bc8042012-08-23 22:09:13 +00001861static char *memex_subcmd_help_string =
Linus Torvalds1da177e2005-04-16 15:20:36 -07001862 "Memory examine subcommands:\n"
1863 " hexval write this val to current location\n"
1864 " 'string' write chars from string to this location\n"
1865 " ' increment address\n"
1866 " ^ decrement address\n"
1867 " / increment addr by 0x10. //=0x100, ///=0x1000, etc\n"
1868 " \\ decrement addr by 0x10. \\\\=0x100, \\\\\\=0x1000, etc\n"
1869 " ` clear no-read flag\n"
1870 " ; stay at this addr\n"
1871 " v change to byte mode\n"
1872 " w change to word (2 byte) mode\n"
1873 " l change to long (4 byte) mode\n"
1874 " u change to doubleword (8 byte) mode\n"
1875 " m addr change current addr\n"
1876 " n toggle no-read flag\n"
1877 " r toggle byte reverse flag\n"
1878 " < count back up count bytes\n"
1879 " > count skip forward count bytes\n"
1880 " x exit this mode\n"
1881 "";
1882
Michael Ellerman9f1067c22008-05-08 14:27:16 +10001883static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07001884memex(void)
1885{
1886 int cmd, inc, i, nslash;
1887 unsigned long n;
1888 unsigned char val[16];
1889
1890 scanhex((void *)&adrs);
1891 cmd = skipbl();
1892 if (cmd == '?') {
1893 printf(memex_help_string);
1894 return;
1895 } else {
1896 termch = cmd;
1897 }
1898 last_cmd = "m\n";
1899 while ((cmd = skipbl()) != '\n') {
1900 switch( cmd ){
1901 case 'b': size = 1; break;
1902 case 'w': size = 2; break;
1903 case 'l': size = 4; break;
1904 case 'd': size = 8; break;
1905 case 'r': brev = !brev; break;
1906 case 'n': mnoread = 1; break;
1907 case '.': mnoread = 0; break;
1908 }
1909 }
1910 if( size <= 0 )
1911 size = 1;
1912 else if( size > 8 )
1913 size = 8;
1914 for(;;){
1915 if (!mnoread)
1916 n = mread(adrs, val, size);
Paul Mackerrase1449ed2005-11-10 14:30:20 +11001917 printf(REG"%c", adrs, brev? 'r': ' ');
Linus Torvalds1da177e2005-04-16 15:20:36 -07001918 if (!mnoread) {
1919 if (brev)
1920 byterev(val, size);
1921 putchar(' ');
1922 for (i = 0; i < n; ++i)
1923 printf("%.2x", val[i]);
1924 for (; i < size; ++i)
1925 printf("%s", fault_chars[fault_type]);
1926 }
1927 putchar(' ');
1928 inc = size;
1929 nslash = 0;
1930 for(;;){
1931 if( scanhex(&n) ){
1932 for (i = 0; i < size; ++i)
1933 val[i] = n >> (i * 8);
1934 if (!brev)
1935 byterev(val, size);
1936 mwrite(adrs, val, size);
1937 inc = size;
1938 }
1939 cmd = skipbl();
1940 if (cmd == '\n')
1941 break;
1942 inc = 0;
1943 switch (cmd) {
1944 case '\'':
1945 for(;;){
1946 n = inchar();
1947 if( n == '\\' )
1948 n = bsesc();
1949 else if( n == '\'' )
1950 break;
1951 for (i = 0; i < size; ++i)
1952 val[i] = n >> (i * 8);
1953 if (!brev)
1954 byterev(val, size);
1955 mwrite(adrs, val, size);
1956 adrs += size;
1957 }
1958 adrs -= size;
1959 inc = size;
1960 break;
1961 case ',':
1962 adrs += size;
1963 break;
1964 case '.':
1965 mnoread = 0;
1966 break;
1967 case ';':
1968 break;
1969 case 'x':
1970 case EOF:
1971 scannl();
1972 return;
1973 case 'b':
1974 case 'v':
1975 size = 1;
1976 break;
1977 case 'w':
1978 size = 2;
1979 break;
1980 case 'l':
1981 size = 4;
1982 break;
1983 case 'u':
1984 size = 8;
1985 break;
1986 case '^':
1987 adrs -= size;
1988 break;
1989 break;
1990 case '/':
1991 if (nslash > 0)
1992 adrs -= 1 << nslash;
1993 else
1994 nslash = 0;
1995 nslash += 4;
1996 adrs += 1 << nslash;
1997 break;
1998 case '\\':
1999 if (nslash < 0)
2000 adrs += 1 << -nslash;
2001 else
2002 nslash = 0;
2003 nslash -= 4;
2004 adrs -= 1 << -nslash;
2005 break;
2006 case 'm':
2007 scanhex((void *)&adrs);
2008 break;
2009 case 'n':
2010 mnoread = 1;
2011 break;
2012 case 'r':
2013 brev = !brev;
2014 break;
2015 case '<':
2016 n = size;
2017 scanhex(&n);
2018 adrs -= n;
2019 break;
2020 case '>':
2021 n = size;
2022 scanhex(&n);
2023 adrs += n;
2024 break;
2025 case '?':
2026 printf(memex_subcmd_help_string);
2027 break;
2028 }
2029 }
2030 adrs += inc;
2031 }
2032}
2033
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002034static int
Linus Torvalds1da177e2005-04-16 15:20:36 -07002035bsesc(void)
2036{
2037 int c;
2038
2039 c = inchar();
2040 switch( c ){
2041 case 'n': c = '\n'; break;
2042 case 'r': c = '\r'; break;
2043 case 'b': c = '\b'; break;
2044 case 't': c = '\t'; break;
2045 }
2046 return c;
2047}
2048
Olaf Hering7e5b5932006-03-08 20:40:28 +01002049static void xmon_rawdump (unsigned long adrs, long ndump)
2050{
2051 long n, m, r, nr;
2052 unsigned char temp[16];
2053
2054 for (n = ndump; n > 0;) {
2055 r = n < 16? n: 16;
2056 nr = mread(adrs, temp, r);
2057 adrs += nr;
2058 for (m = 0; m < r; ++m) {
2059 if (m < nr)
2060 printf("%.2x", temp[m]);
2061 else
2062 printf("%s", fault_chars[fault_type]);
2063 }
2064 n -= r;
2065 if (nr < r)
2066 break;
2067 }
2068 printf("\n");
2069}
2070
Michael Ellermanddadb6b2012-09-13 23:01:31 +00002071#ifdef CONFIG_PPC64
2072static void dump_one_paca(int cpu)
2073{
2074 struct paca_struct *p;
2075
2076 if (setjmp(bus_error_jmp) != 0) {
2077 printf("*** Error dumping paca for cpu 0x%x!\n", cpu);
2078 return;
2079 }
2080
2081 catch_memory_errors = 1;
2082 sync();
2083
2084 p = &paca[cpu];
2085
2086 printf("paca for cpu 0x%x @ %p:\n", cpu, p);
2087
2088 printf(" %-*s = %s\n", 16, "possible", cpu_possible(cpu) ? "yes" : "no");
2089 printf(" %-*s = %s\n", 16, "present", cpu_present(cpu) ? "yes" : "no");
2090 printf(" %-*s = %s\n", 16, "online", cpu_online(cpu) ? "yes" : "no");
2091
2092#define DUMP(paca, name, format) \
2093 printf(" %-*s = %#-*"format"\t(0x%lx)\n", 16, #name, 18, paca->name, \
2094 offsetof(struct paca_struct, name));
2095
2096 DUMP(p, lock_token, "x");
2097 DUMP(p, paca_index, "x");
2098 DUMP(p, kernel_toc, "lx");
2099 DUMP(p, kernelbase, "lx");
2100 DUMP(p, kernel_msr, "lx");
Michael Ellermanddadb6b2012-09-13 23:01:31 +00002101 DUMP(p, emergency_sp, "p");
Mahesh Salgaonkar729b0f72013-10-30 20:04:00 +05302102#ifdef CONFIG_PPC_BOOK3S_64
2103 DUMP(p, mc_emergency_sp, "p");
2104 DUMP(p, in_mce, "x");
2105#endif
Michael Ellermanddadb6b2012-09-13 23:01:31 +00002106 DUMP(p, data_offset, "lx");
2107 DUMP(p, hw_cpu_id, "x");
2108 DUMP(p, cpu_start, "x");
2109 DUMP(p, kexec_state, "x");
2110 DUMP(p, __current, "p");
2111 DUMP(p, kstack, "lx");
2112 DUMP(p, stab_rr, "lx");
2113 DUMP(p, saved_r1, "lx");
2114 DUMP(p, trap_save, "x");
2115 DUMP(p, soft_enabled, "x");
2116 DUMP(p, irq_happened, "x");
2117 DUMP(p, io_sync, "x");
2118 DUMP(p, irq_work_pending, "x");
2119 DUMP(p, nap_state_lost, "x");
2120
2121#undef DUMP
2122
2123 catch_memory_errors = 0;
2124 sync();
2125}
2126
2127static void dump_all_pacas(void)
2128{
2129 int cpu;
2130
2131 if (num_possible_cpus() == 0) {
2132 printf("No possible cpus, use 'dp #' to dump individual cpus\n");
2133 return;
2134 }
2135
2136 for_each_possible_cpu(cpu)
2137 dump_one_paca(cpu);
2138}
2139
2140static void dump_pacas(void)
2141{
2142 unsigned long num;
2143 int c;
2144
2145 c = inchar();
2146 if (c == 'a') {
2147 dump_all_pacas();
2148 return;
2149 }
2150
2151 termch = c; /* Put c back, it wasn't 'a' */
2152
2153 if (scanhex(&num))
2154 dump_one_paca(num);
2155 else
2156 dump_one_paca(xmon_owner);
2157}
2158#endif
2159
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002160static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002161dump(void)
2162{
2163 int c;
2164
2165 c = inchar();
Michael Ellermanddadb6b2012-09-13 23:01:31 +00002166
2167#ifdef CONFIG_PPC64
2168 if (c == 'p') {
2169 dump_pacas();
2170 return;
2171 }
2172#endif
2173
Linus Torvalds1da177e2005-04-16 15:20:36 -07002174 if ((isxdigit(c) && c != 'f' && c != 'd') || c == '\n')
2175 termch = c;
2176 scanhex((void *)&adrs);
2177 if (termch != '\n')
2178 termch = 0;
2179 if (c == 'i') {
2180 scanhex(&nidump);
2181 if (nidump == 0)
2182 nidump = 16;
2183 else if (nidump > MAX_DUMP)
2184 nidump = MAX_DUMP;
2185 adrs += ppc_inst_dump(adrs, nidump, 1);
2186 last_cmd = "di\n";
Vinay Sridharf312deb2009-05-14 23:13:07 +00002187 } else if (c == 'l') {
2188 dump_log_buf();
Olaf Hering7e5b5932006-03-08 20:40:28 +01002189 } else if (c == 'r') {
2190 scanhex(&ndump);
2191 if (ndump == 0)
2192 ndump = 64;
2193 xmon_rawdump(adrs, ndump);
2194 adrs += ndump;
2195 last_cmd = "dr\n";
Linus Torvalds1da177e2005-04-16 15:20:36 -07002196 } else {
2197 scanhex(&ndump);
2198 if (ndump == 0)
2199 ndump = 64;
2200 else if (ndump > MAX_DUMP)
2201 ndump = MAX_DUMP;
2202 prdump(adrs, ndump);
2203 adrs += ndump;
2204 last_cmd = "d\n";
2205 }
2206}
2207
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002208static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002209prdump(unsigned long adrs, long ndump)
2210{
2211 long n, m, c, r, nr;
2212 unsigned char temp[16];
2213
2214 for (n = ndump; n > 0;) {
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002215 printf(REG, adrs);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002216 putchar(' ');
2217 r = n < 16? n: 16;
2218 nr = mread(adrs, temp, r);
2219 adrs += nr;
2220 for (m = 0; m < r; ++m) {
Michael Ellermane3bc8042012-08-23 22:09:13 +00002221 if ((m & (sizeof(long) - 1)) == 0 && m > 0)
Paul Mackerrase1449ed2005-11-10 14:30:20 +11002222 putchar(' ');
Linus Torvalds1da177e2005-04-16 15:20:36 -07002223 if (m < nr)
2224 printf("%.2x", temp[m]);
2225 else
2226 printf("%s", fault_chars[fault_type]);
2227 }
Paul Mackerrase1449ed2005-11-10 14:30:20 +11002228 for (; m < 16; ++m) {
Michael Ellermane3bc8042012-08-23 22:09:13 +00002229 if ((m & (sizeof(long) - 1)) == 0)
Paul Mackerrase1449ed2005-11-10 14:30:20 +11002230 putchar(' ');
Linus Torvalds1da177e2005-04-16 15:20:36 -07002231 printf(" ");
Paul Mackerrase1449ed2005-11-10 14:30:20 +11002232 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002233 printf(" |");
2234 for (m = 0; m < r; ++m) {
2235 if (m < nr) {
2236 c = temp[m];
2237 putchar(' ' <= c && c <= '~'? c: '.');
2238 } else
2239 putchar(' ');
2240 }
2241 n -= r;
2242 for (; m < 16; ++m)
2243 putchar(' ');
2244 printf("|\n");
2245 if (nr < r)
2246 break;
2247 }
2248}
2249
Michael Ellerman4c4c8722006-11-23 00:46:42 +01002250typedef int (*instruction_dump_func)(unsigned long inst, unsigned long addr);
2251
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002252static int
Michael Ellerman4c4c8722006-11-23 00:46:42 +01002253generic_inst_dump(unsigned long adr, long count, int praddr,
2254 instruction_dump_func dump_func)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002255{
2256 int nr, dotted;
2257 unsigned long first_adr;
2258 unsigned long inst, last_inst = 0;
2259 unsigned char val[4];
2260
2261 dotted = 0;
2262 for (first_adr = adr; count > 0; --count, adr += 4) {
2263 nr = mread(adr, val, 4);
2264 if (nr == 0) {
2265 if (praddr) {
2266 const char *x = fault_chars[fault_type];
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002267 printf(REG" %s%s%s%s\n", adr, x, x, x, x);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002268 }
2269 break;
2270 }
2271 inst = GETWORD(val);
2272 if (adr > first_adr && inst == last_inst) {
2273 if (!dotted) {
2274 printf(" ...\n");
2275 dotted = 1;
2276 }
2277 continue;
2278 }
2279 dotted = 0;
2280 last_inst = inst;
2281 if (praddr)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002282 printf(REG" %.8x", adr, inst);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002283 printf("\t");
Michael Ellerman4c4c8722006-11-23 00:46:42 +01002284 dump_func(inst, adr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002285 printf("\n");
2286 }
2287 return adr - first_adr;
2288}
2289
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002290static int
Michael Ellerman4c4c8722006-11-23 00:46:42 +01002291ppc_inst_dump(unsigned long adr, long count, int praddr)
2292{
2293 return generic_inst_dump(adr, count, praddr, print_insn_powerpc);
2294}
2295
Linus Torvalds1da177e2005-04-16 15:20:36 -07002296void
2297print_address(unsigned long addr)
2298{
2299 xmon_print_symbol(addr, "\t# ", "");
2300}
2301
Vinay Sridharf312deb2009-05-14 23:13:07 +00002302void
2303dump_log_buf(void)
2304{
Michael Ellermanca5dd392012-08-23 22:09:12 +00002305 struct kmsg_dumper dumper = { .active = 1 };
2306 unsigned char buf[128];
2307 size_t len;
Vinay Sridharf312deb2009-05-14 23:13:07 +00002308
Michael Ellermane3bc8042012-08-23 22:09:13 +00002309 if (setjmp(bus_error_jmp) != 0) {
Michael Ellermanca5dd392012-08-23 22:09:12 +00002310 printf("Error dumping printk buffer!\n");
Michael Ellermane3bc8042012-08-23 22:09:13 +00002311 return;
2312 }
Vinay Sridharf312deb2009-05-14 23:13:07 +00002313
Michael Ellermane3bc8042012-08-23 22:09:13 +00002314 catch_memory_errors = 1;
2315 sync();
Vinay Sridharf312deb2009-05-14 23:13:07 +00002316
Michael Ellermanca5dd392012-08-23 22:09:12 +00002317 kmsg_dump_rewind_nolock(&dumper);
2318 while (kmsg_dump_get_line_nolock(&dumper, false, buf, sizeof(buf), &len)) {
2319 buf[len] = '\0';
2320 printf("%s", buf);
2321 }
Vinay Sridharf312deb2009-05-14 23:13:07 +00002322
Michael Ellermane3bc8042012-08-23 22:09:13 +00002323 sync();
2324 /* wait a little while to see if we get a machine check */
2325 __delay(200);
2326 catch_memory_errors = 0;
Vinay Sridharf312deb2009-05-14 23:13:07 +00002327}
Linus Torvalds1da177e2005-04-16 15:20:36 -07002328
2329/*
2330 * Memory operations - move, set, print differences
2331 */
2332static unsigned long mdest; /* destination address */
2333static unsigned long msrc; /* source address */
2334static unsigned long mval; /* byte value to set memory to */
2335static unsigned long mcount; /* # bytes to affect */
2336static unsigned long mdiffs; /* max # differences to print */
2337
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002338static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002339memops(int cmd)
2340{
2341 scanhex((void *)&mdest);
2342 if( termch != '\n' )
2343 termch = 0;
2344 scanhex((void *)(cmd == 's'? &mval: &msrc));
2345 if( termch != '\n' )
2346 termch = 0;
2347 scanhex((void *)&mcount);
2348 switch( cmd ){
2349 case 'm':
2350 memmove((void *)mdest, (void *)msrc, mcount);
2351 break;
2352 case 's':
2353 memset((void *)mdest, mval, mcount);
2354 break;
2355 case 'd':
2356 if( termch != '\n' )
2357 termch = 0;
2358 scanhex((void *)&mdiffs);
2359 memdiffs((unsigned char *)mdest, (unsigned char *)msrc, mcount, mdiffs);
2360 break;
2361 }
2362}
2363
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002364static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002365memdiffs(unsigned char *p1, unsigned char *p2, unsigned nb, unsigned maxpr)
2366{
2367 unsigned n, prt;
2368
2369 prt = 0;
2370 for( n = nb; n > 0; --n )
2371 if( *p1++ != *p2++ )
2372 if( ++prt <= maxpr )
2373 printf("%.16x %.2x # %.16x %.2x\n", p1 - 1,
2374 p1[-1], p2 - 1, p2[-1]);
2375 if( prt > maxpr )
2376 printf("Total of %d differences\n", prt);
2377}
2378
2379static unsigned mend;
2380static unsigned mask;
2381
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002382static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002383memlocate(void)
2384{
2385 unsigned a, n;
2386 unsigned char val[4];
2387
2388 last_cmd = "ml";
2389 scanhex((void *)&mdest);
2390 if (termch != '\n') {
2391 termch = 0;
2392 scanhex((void *)&mend);
2393 if (termch != '\n') {
2394 termch = 0;
2395 scanhex((void *)&mval);
2396 mask = ~0;
2397 if (termch != '\n') termch = 0;
2398 scanhex((void *)&mask);
2399 }
2400 }
2401 n = 0;
2402 for (a = mdest; a < mend; a += 4) {
2403 if (mread(a, val, 4) == 4
2404 && ((GETWORD(val) ^ mval) & mask) == 0) {
2405 printf("%.16x: %.16x\n", a, GETWORD(val));
2406 if (++n >= 10)
2407 break;
2408 }
2409 }
2410}
2411
2412static unsigned long mskip = 0x1000;
2413static unsigned long mlim = 0xffffffff;
2414
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002415static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002416memzcan(void)
2417{
2418 unsigned char v;
2419 unsigned a;
2420 int ok, ook;
2421
2422 scanhex(&mdest);
2423 if (termch != '\n') termch = 0;
2424 scanhex(&mskip);
2425 if (termch != '\n') termch = 0;
2426 scanhex(&mlim);
2427 ook = 0;
2428 for (a = mdest; a < mlim; a += mskip) {
2429 ok = mread(a, &v, 1);
2430 if (ok && !ook) {
2431 printf("%.8x .. ", a);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002432 } else if (!ok && ook)
2433 printf("%.8x\n", a - mskip);
2434 ook = ok;
2435 if (a + mskip < a)
2436 break;
2437 }
2438 if (ook)
2439 printf("%.8x\n", a - mskip);
2440}
2441
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002442static void proccall(void)
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002443{
2444 unsigned long args[8];
2445 unsigned long ret;
2446 int i;
2447 typedef unsigned long (*callfunc_t)(unsigned long, unsigned long,
2448 unsigned long, unsigned long, unsigned long,
2449 unsigned long, unsigned long, unsigned long);
2450 callfunc_t func;
2451
2452 if (!scanhex(&adrs))
2453 return;
2454 if (termch != '\n')
2455 termch = 0;
2456 for (i = 0; i < 8; ++i)
2457 args[i] = 0;
2458 for (i = 0; i < 8; ++i) {
2459 if (!scanhex(&args[i]) || termch == '\n')
2460 break;
2461 termch = 0;
2462 }
2463 func = (callfunc_t) adrs;
2464 ret = 0;
2465 if (setjmp(bus_error_jmp) == 0) {
2466 catch_memory_errors = 1;
2467 sync();
2468 ret = func(args[0], args[1], args[2], args[3],
2469 args[4], args[5], args[6], args[7]);
2470 sync();
Michael Ellerman736256e2014-05-26 21:02:14 +10002471 printf("return value is 0x%lx\n", ret);
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002472 } else {
2473 printf("*** %x exception occurred\n", fault_except);
2474 }
2475 catch_memory_errors = 0;
2476}
2477
Linus Torvalds1da177e2005-04-16 15:20:36 -07002478/* Input scanning routines */
2479int
2480skipbl(void)
2481{
2482 int c;
2483
2484 if( termch != 0 ){
2485 c = termch;
2486 termch = 0;
2487 } else
2488 c = inchar();
2489 while( c == ' ' || c == '\t' )
2490 c = inchar();
2491 return c;
2492}
2493
2494#define N_PTREGS 44
2495static char *regnames[N_PTREGS] = {
2496 "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7",
2497 "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15",
2498 "r16", "r17", "r18", "r19", "r20", "r21", "r22", "r23",
2499 "r24", "r25", "r26", "r27", "r28", "r29", "r30", "r31",
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002500 "pc", "msr", "or3", "ctr", "lr", "xer", "ccr",
2501#ifdef CONFIG_PPC64
2502 "softe",
2503#else
2504 "mq",
2505#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07002506 "trap", "dar", "dsisr", "res"
2507};
2508
2509int
2510scanhex(unsigned long *vp)
2511{
2512 int c, d;
2513 unsigned long v;
2514
2515 c = skipbl();
2516 if (c == '%') {
2517 /* parse register name */
2518 char regname[8];
2519 int i;
2520
2521 for (i = 0; i < sizeof(regname) - 1; ++i) {
2522 c = inchar();
2523 if (!isalnum(c)) {
2524 termch = c;
2525 break;
2526 }
2527 regname[i] = c;
2528 }
2529 regname[i] = 0;
2530 for (i = 0; i < N_PTREGS; ++i) {
2531 if (strcmp(regnames[i], regname) == 0) {
2532 if (xmon_regs == NULL) {
2533 printf("regs not available\n");
2534 return 0;
2535 }
2536 *vp = ((unsigned long *)xmon_regs)[i];
2537 return 1;
2538 }
2539 }
2540 printf("invalid register name '%%%s'\n", regname);
2541 return 0;
2542 }
2543
2544 /* skip leading "0x" if any */
2545
2546 if (c == '0') {
2547 c = inchar();
2548 if (c == 'x') {
2549 c = inchar();
2550 } else {
2551 d = hexdigit(c);
2552 if (d == EOF) {
2553 termch = c;
2554 *vp = 0;
2555 return 1;
2556 }
2557 }
2558 } else if (c == '$') {
2559 int i;
2560 for (i=0; i<63; i++) {
2561 c = inchar();
Vincent Bernat05b981f2014-07-15 13:43:47 +02002562 if (isspace(c) || c == '\0') {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002563 termch = c;
2564 break;
2565 }
2566 tmpstr[i] = c;
2567 }
2568 tmpstr[i++] = 0;
Benjamin Herrenschmidt6879dc12005-06-21 17:15:30 -07002569 *vp = 0;
2570 if (setjmp(bus_error_jmp) == 0) {
2571 catch_memory_errors = 1;
2572 sync();
2573 *vp = kallsyms_lookup_name(tmpstr);
2574 sync();
2575 }
2576 catch_memory_errors = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002577 if (!(*vp)) {
2578 printf("unknown symbol '%s'\n", tmpstr);
2579 return 0;
2580 }
2581 return 1;
2582 }
2583
2584 d = hexdigit(c);
2585 if (d == EOF) {
2586 termch = c;
2587 return 0;
2588 }
2589 v = 0;
2590 do {
2591 v = (v << 4) + d;
2592 c = inchar();
2593 d = hexdigit(c);
2594 } while (d != EOF);
2595 termch = c;
2596 *vp = v;
2597 return 1;
2598}
2599
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002600static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002601scannl(void)
2602{
2603 int c;
2604
2605 c = termch;
2606 termch = 0;
2607 while( c != '\n' )
2608 c = inchar();
2609}
2610
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002611static int hexdigit(int c)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002612{
2613 if( '0' <= c && c <= '9' )
2614 return c - '0';
2615 if( 'A' <= c && c <= 'F' )
2616 return c - ('A' - 10);
2617 if( 'a' <= c && c <= 'f' )
2618 return c - ('a' - 10);
2619 return EOF;
2620}
2621
2622void
2623getstring(char *s, int size)
2624{
2625 int c;
2626
2627 c = skipbl();
2628 do {
2629 if( size > 1 ){
2630 *s++ = c;
2631 --size;
2632 }
2633 c = inchar();
2634 } while( c != ' ' && c != '\t' && c != '\n' );
2635 termch = c;
2636 *s = 0;
2637}
2638
2639static char line[256];
2640static char *lineptr;
2641
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002642static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002643flush_input(void)
2644{
2645 lineptr = NULL;
2646}
2647
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002648static int
Linus Torvalds1da177e2005-04-16 15:20:36 -07002649inchar(void)
2650{
2651 if (lineptr == NULL || *lineptr == 0) {
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11002652 if (xmon_gets(line, sizeof(line)) == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002653 lineptr = NULL;
2654 return EOF;
2655 }
2656 lineptr = line;
2657 }
2658 return *lineptr++;
2659}
2660
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002661static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07002662take_input(char *str)
2663{
2664 lineptr = str;
2665}
2666
2667
2668static void
2669symbol_lookup(void)
2670{
2671 int type = inchar();
2672 unsigned long addr;
2673 static char tmp[64];
2674
2675 switch (type) {
2676 case 'a':
2677 if (scanhex(&addr))
2678 xmon_print_symbol(addr, ": ", "\n");
2679 termch = 0;
2680 break;
2681 case 's':
2682 getstring(tmp, 64);
2683 if (setjmp(bus_error_jmp) == 0) {
2684 catch_memory_errors = 1;
2685 sync();
2686 addr = kallsyms_lookup_name(tmp);
2687 if (addr)
2688 printf("%s: %lx\n", tmp, addr);
2689 else
2690 printf("Symbol '%s' not found.\n", tmp);
2691 sync();
2692 }
2693 catch_memory_errors = 0;
2694 termch = 0;
2695 break;
2696 }
2697}
2698
2699
2700/* Print an address in numeric and symbolic form (if possible) */
2701static void xmon_print_symbol(unsigned long address, const char *mid,
2702 const char *after)
2703{
2704 char *modname;
2705 const char *name = NULL;
2706 unsigned long offset, size;
2707
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002708 printf(REG, address);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002709 if (setjmp(bus_error_jmp) == 0) {
2710 catch_memory_errors = 1;
2711 sync();
2712 name = kallsyms_lookup(address, &size, &offset, &modname,
2713 tmpstr);
2714 sync();
2715 /* wait a little while to see if we get a machine check */
2716 __delay(200);
2717 }
2718
2719 catch_memory_errors = 0;
2720
2721 if (name) {
2722 printf("%s%s+%#lx/%#lx", mid, name, offset, size);
2723 if (modname)
2724 printf(" [%s]", modname);
2725 }
2726 printf("%s", after);
2727}
2728
Benjamin Herrenschmidt2d27cfd2009-07-23 23:15:59 +00002729#ifdef CONFIG_PPC_BOOK3S_64
Michael Ellerman13b3d132014-07-10 12:29:20 +10002730void dump_segments(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002731{
2732 int i;
will schmidtb3b95952007-12-07 08:22:23 +11002733 unsigned long esid,vsid,valid;
2734 unsigned long llp;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002735
Michael Ellerman736256e2014-05-26 21:02:14 +10002736 printf("SLB contents of cpu 0x%x\n", smp_processor_id());
Linus Torvalds1da177e2005-04-16 15:20:36 -07002737
Michael Neuling584f8b72007-12-06 17:24:48 +11002738 for (i = 0; i < mmu_slb_size; i++) {
will schmidtb3b95952007-12-07 08:22:23 +11002739 asm volatile("slbmfee %0,%1" : "=r" (esid) : "r" (i));
2740 asm volatile("slbmfev %0,%1" : "=r" (vsid) : "r" (i));
2741 valid = (esid & SLB_ESID_V);
2742 if (valid | esid | vsid) {
2743 printf("%02d %016lx %016lx", i, esid, vsid);
2744 if (valid) {
2745 llp = vsid & SLB_VSID_LLP;
2746 if (vsid & SLB_VSID_B_1T) {
2747 printf(" 1T ESID=%9lx VSID=%13lx LLP:%3lx \n",
2748 GET_ESID_1T(esid),
2749 (vsid & ~SLB_VSID_B) >> SLB_VSID_SHIFT_1T,
2750 llp);
2751 } else {
2752 printf(" 256M ESID=%9lx VSID=%13lx LLP:%3lx \n",
2753 GET_ESID(esid),
2754 (vsid & ~SLB_VSID_B) >> SLB_VSID_SHIFT,
2755 llp);
2756 }
2757 } else
2758 printf("\n");
2759 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002760 }
2761}
Paul Mackerrasf78541dc2005-10-28 22:53:37 +10002762#endif
2763
2764#ifdef CONFIG_PPC_STD_MMU_32
2765void dump_segments(void)
2766{
2767 int i;
2768
2769 printf("sr0-15 =");
2770 for (i = 0; i < 16; ++i)
2771 printf(" %x", mfsrin(i));
2772 printf("\n");
2773}
2774#endif
2775
Benjamin Herrenschmidt5a8a1a22007-11-16 18:23:33 +11002776#ifdef CONFIG_44x
2777static void dump_tlb_44x(void)
2778{
2779 int i;
2780
2781 for (i = 0; i < PPC44x_TLB_SIZE; i++) {
2782 unsigned long w0,w1,w2;
2783 asm volatile("tlbre %0,%1,0" : "=r" (w0) : "r" (i));
2784 asm volatile("tlbre %0,%1,1" : "=r" (w1) : "r" (i));
2785 asm volatile("tlbre %0,%1,2" : "=r" (w2) : "r" (i));
2786 printf("[%02x] %08x %08x %08x ", i, w0, w1, w2);
2787 if (w0 & PPC44x_TLB_VALID) {
2788 printf("V %08x -> %01x%08x %c%c%c%c%c",
2789 w0 & PPC44x_TLB_EPN_MASK,
2790 w1 & PPC44x_TLB_ERPN_MASK,
2791 w1 & PPC44x_TLB_RPN_MASK,
2792 (w2 & PPC44x_TLB_W) ? 'W' : 'w',
2793 (w2 & PPC44x_TLB_I) ? 'I' : 'i',
2794 (w2 & PPC44x_TLB_M) ? 'M' : 'm',
2795 (w2 & PPC44x_TLB_G) ? 'G' : 'g',
2796 (w2 & PPC44x_TLB_E) ? 'E' : 'e');
2797 }
2798 printf("\n");
2799 }
2800}
2801#endif /* CONFIG_44x */
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002802
Benjamin Herrenschmidt03247152010-07-09 15:34:50 +10002803#ifdef CONFIG_PPC_BOOK3E
2804static void dump_tlb_book3e(void)
2805{
2806 u32 mmucfg, pidmask, lpidmask;
2807 u64 ramask;
2808 int i, tlb, ntlbs, pidsz, lpidsz, rasz, lrat = 0;
2809 int mmu_version;
2810 static const char *pgsz_names[] = {
2811 " 1K",
2812 " 2K",
2813 " 4K",
2814 " 8K",
2815 " 16K",
2816 " 32K",
2817 " 64K",
2818 "128K",
2819 "256K",
2820 "512K",
2821 " 1M",
2822 " 2M",
2823 " 4M",
2824 " 8M",
2825 " 16M",
2826 " 32M",
2827 " 64M",
2828 "128M",
2829 "256M",
2830 "512M",
2831 " 1G",
2832 " 2G",
2833 " 4G",
2834 " 8G",
2835 " 16G",
2836 " 32G",
2837 " 64G",
2838 "128G",
2839 "256G",
2840 "512G",
2841 " 1T",
2842 " 2T",
2843 };
2844
2845 /* Gather some infos about the MMU */
2846 mmucfg = mfspr(SPRN_MMUCFG);
2847 mmu_version = (mmucfg & 3) + 1;
2848 ntlbs = ((mmucfg >> 2) & 3) + 1;
2849 pidsz = ((mmucfg >> 6) & 0x1f) + 1;
2850 lpidsz = (mmucfg >> 24) & 0xf;
2851 rasz = (mmucfg >> 16) & 0x7f;
2852 if ((mmu_version > 1) && (mmucfg & 0x10000))
2853 lrat = 1;
2854 printf("Book3E MMU MAV=%d.0,%d TLBs,%d-bit PID,%d-bit LPID,%d-bit RA\n",
2855 mmu_version, ntlbs, pidsz, lpidsz, rasz);
2856 pidmask = (1ul << pidsz) - 1;
2857 lpidmask = (1ul << lpidsz) - 1;
2858 ramask = (1ull << rasz) - 1;
2859
2860 for (tlb = 0; tlb < ntlbs; tlb++) {
2861 u32 tlbcfg;
2862 int nent, assoc, new_cc = 1;
2863 printf("TLB %d:\n------\n", tlb);
2864 switch(tlb) {
2865 case 0:
2866 tlbcfg = mfspr(SPRN_TLB0CFG);
2867 break;
2868 case 1:
2869 tlbcfg = mfspr(SPRN_TLB1CFG);
2870 break;
2871 case 2:
2872 tlbcfg = mfspr(SPRN_TLB2CFG);
2873 break;
2874 case 3:
2875 tlbcfg = mfspr(SPRN_TLB3CFG);
2876 break;
2877 default:
2878 printf("Unsupported TLB number !\n");
2879 continue;
2880 }
2881 nent = tlbcfg & 0xfff;
2882 assoc = (tlbcfg >> 24) & 0xff;
2883 for (i = 0; i < nent; i++) {
2884 u32 mas0 = MAS0_TLBSEL(tlb);
2885 u32 mas1 = MAS1_TSIZE(BOOK3E_PAGESZ_4K);
2886 u64 mas2 = 0;
2887 u64 mas7_mas3;
2888 int esel = i, cc = i;
2889
2890 if (assoc != 0) {
2891 cc = i / assoc;
2892 esel = i % assoc;
2893 mas2 = cc * 0x1000;
2894 }
2895
2896 mas0 |= MAS0_ESEL(esel);
2897 mtspr(SPRN_MAS0, mas0);
2898 mtspr(SPRN_MAS1, mas1);
2899 mtspr(SPRN_MAS2, mas2);
2900 asm volatile("tlbre 0,0,0" : : : "memory");
2901 mas1 = mfspr(SPRN_MAS1);
2902 mas2 = mfspr(SPRN_MAS2);
2903 mas7_mas3 = mfspr(SPRN_MAS7_MAS3);
2904 if (assoc && (i % assoc) == 0)
2905 new_cc = 1;
2906 if (!(mas1 & MAS1_VALID))
2907 continue;
2908 if (assoc == 0)
2909 printf("%04x- ", i);
2910 else if (new_cc)
2911 printf("%04x-%c", cc, 'A' + esel);
2912 else
2913 printf(" |%c", 'A' + esel);
2914 new_cc = 0;
2915 printf(" %016llx %04x %s %c%c AS%c",
2916 mas2 & ~0x3ffull,
2917 (mas1 >> 16) & 0x3fff,
2918 pgsz_names[(mas1 >> 7) & 0x1f],
2919 mas1 & MAS1_IND ? 'I' : ' ',
2920 mas1 & MAS1_IPROT ? 'P' : ' ',
2921 mas1 & MAS1_TS ? '1' : '0');
2922 printf(" %c%c%c%c%c%c%c",
2923 mas2 & MAS2_X0 ? 'a' : ' ',
2924 mas2 & MAS2_X1 ? 'v' : ' ',
2925 mas2 & MAS2_W ? 'w' : ' ',
2926 mas2 & MAS2_I ? 'i' : ' ',
2927 mas2 & MAS2_M ? 'm' : ' ',
2928 mas2 & MAS2_G ? 'g' : ' ',
2929 mas2 & MAS2_E ? 'e' : ' ');
2930 printf(" %016llx", mas7_mas3 & ramask & ~0x7ffull);
2931 if (mas1 & MAS1_IND)
2932 printf(" %s\n",
2933 pgsz_names[(mas7_mas3 >> 1) & 0x1f]);
2934 else
2935 printf(" U%c%c%c S%c%c%c\n",
2936 mas7_mas3 & MAS3_UX ? 'x' : ' ',
2937 mas7_mas3 & MAS3_UW ? 'w' : ' ',
2938 mas7_mas3 & MAS3_UR ? 'r' : ' ',
2939 mas7_mas3 & MAS3_SX ? 'x' : ' ',
2940 mas7_mas3 & MAS3_SW ? 'w' : ' ',
2941 mas7_mas3 & MAS3_SR ? 'r' : ' ');
2942 }
2943 }
2944}
2945#endif /* CONFIG_PPC_BOOK3E */
2946
Michael Ellerman9f1067c22008-05-08 14:27:16 +10002947static void xmon_init(int enable)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002948{
Olaf Heringb13cfd172005-08-04 19:26:42 +02002949 if (enable) {
2950 __debugger = xmon;
2951 __debugger_ipi = xmon_ipi;
2952 __debugger_bpt = xmon_bpt;
2953 __debugger_sstep = xmon_sstep;
2954 __debugger_iabr_match = xmon_iabr_match;
Michael Neuling9422de32012-12-20 14:06:44 +00002955 __debugger_break_match = xmon_break_match;
Olaf Heringb13cfd172005-08-04 19:26:42 +02002956 __debugger_fault_handler = xmon_fault_handler;
2957 } else {
2958 __debugger = NULL;
2959 __debugger_ipi = NULL;
2960 __debugger_bpt = NULL;
2961 __debugger_sstep = NULL;
2962 __debugger_iabr_match = NULL;
Michael Neuling9422de32012-12-20 14:06:44 +00002963 __debugger_break_match = NULL;
Olaf Heringb13cfd172005-08-04 19:26:42 +02002964 __debugger_fault_handler = NULL;
2965 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002966}
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11002967
2968#ifdef CONFIG_MAGIC_SYSRQ
Dmitry Torokhov1495cc92010-08-17 21:15:46 -07002969static void sysrq_handle_xmon(int key)
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11002970{
2971 /* ensure xmon is enabled */
2972 xmon_init(1);
David Howells7d12e782006-10-05 14:55:46 +01002973 debugger(get_irq_regs());
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11002974}
2975
Dmitry Torokhov1495cc92010-08-17 21:15:46 -07002976static struct sysrq_key_op sysrq_xmon_op = {
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11002977 .handler = sysrq_handle_xmon,
zhangwei(Jovi)90a102e2013-04-30 15:28:54 -07002978 .help_msg = "xmon(x)",
Paul Mackerrasfca5dcd2005-11-08 22:55:08 +11002979 .action_msg = "Entering xmon",
2980};
2981
2982static int __init setup_xmon_sysrq(void)
2983{
2984 register_sysrq_key('x', &sysrq_xmon_op);
2985 return 0;
2986}
2987__initcall(setup_xmon_sysrq);
2988#endif /* CONFIG_MAGIC_SYSRQ */
Michael Ellerman476792832006-10-03 14:12:08 +10002989
Olaf Heringf5e6a282007-06-24 16:57:08 +10002990static int __initdata xmon_early, xmon_off;
Michael Ellerman476792832006-10-03 14:12:08 +10002991
2992static int __init early_parse_xmon(char *p)
2993{
2994 if (!p || strncmp(p, "early", 5) == 0) {
2995 /* just "xmon" is equivalent to "xmon=early" */
2996 xmon_init(1);
2997 xmon_early = 1;
2998 } else if (strncmp(p, "on", 2) == 0)
2999 xmon_init(1);
3000 else if (strncmp(p, "off", 3) == 0)
3001 xmon_off = 1;
3002 else if (strncmp(p, "nobt", 4) == 0)
3003 xmon_no_auto_backtrace = 1;
3004 else
3005 return 1;
3006
3007 return 0;
3008}
3009early_param("xmon", early_parse_xmon);
3010
3011void __init xmon_setup(void)
3012{
3013#ifdef CONFIG_XMON_DEFAULT
3014 if (!xmon_off)
3015 xmon_init(1);
3016#endif
3017 if (xmon_early)
3018 debugger(NULL);
3019}
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003020
Arnd Bergmanne0555952006-11-27 19:18:55 +01003021#ifdef CONFIG_SPU_BASE
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003022
3023struct spu_info {
3024 struct spu *spu;
3025 u64 saved_mfc_sr1_RW;
3026 u32 saved_spu_runcntl_RW;
Michael Ellerman24a24c82006-11-23 00:46:41 +01003027 unsigned long dump_addr;
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003028 u8 stopped_ok;
3029};
3030
3031#define XMON_NUM_SPUS 16 /* Enough for current hardware */
3032
3033static struct spu_info spu_info[XMON_NUM_SPUS];
3034
3035void xmon_register_spus(struct list_head *list)
3036{
3037 struct spu *spu;
3038
3039 list_for_each_entry(spu, list, full_list) {
3040 if (spu->number >= XMON_NUM_SPUS) {
3041 WARN_ON(1);
3042 continue;
3043 }
3044
3045 spu_info[spu->number].spu = spu;
3046 spu_info[spu->number].stopped_ok = 0;
Michael Ellerman24a24c82006-11-23 00:46:41 +01003047 spu_info[spu->number].dump_addr = (unsigned long)
3048 spu_info[spu->number].spu->local_store;
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003049 }
3050}
3051
3052static void stop_spus(void)
3053{
3054 struct spu *spu;
3055 int i;
3056 u64 tmp;
3057
3058 for (i = 0; i < XMON_NUM_SPUS; i++) {
3059 if (!spu_info[i].spu)
3060 continue;
3061
3062 if (setjmp(bus_error_jmp) == 0) {
3063 catch_memory_errors = 1;
3064 sync();
3065
3066 spu = spu_info[i].spu;
3067
3068 spu_info[i].saved_spu_runcntl_RW =
3069 in_be32(&spu->problem->spu_runcntl_RW);
3070
3071 tmp = spu_mfc_sr1_get(spu);
3072 spu_info[i].saved_mfc_sr1_RW = tmp;
3073
3074 tmp &= ~MFC_STATE1_MASTER_RUN_CONTROL_MASK;
3075 spu_mfc_sr1_set(spu, tmp);
3076
3077 sync();
3078 __delay(200);
3079
3080 spu_info[i].stopped_ok = 1;
Michael Ellerman2a144422006-11-23 00:46:40 +01003081
3082 printf("Stopped spu %.2d (was %s)\n", i,
3083 spu_info[i].saved_spu_runcntl_RW ?
3084 "running" : "stopped");
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003085 } else {
3086 catch_memory_errors = 0;
3087 printf("*** Error stopping spu %.2d\n", i);
3088 }
3089 catch_memory_errors = 0;
3090 }
3091}
3092
3093static void restart_spus(void)
3094{
3095 struct spu *spu;
3096 int i;
3097
3098 for (i = 0; i < XMON_NUM_SPUS; i++) {
3099 if (!spu_info[i].spu)
3100 continue;
3101
3102 if (!spu_info[i].stopped_ok) {
3103 printf("*** Error, spu %d was not successfully stopped"
3104 ", not restarting\n", i);
3105 continue;
3106 }
3107
3108 if (setjmp(bus_error_jmp) == 0) {
3109 catch_memory_errors = 1;
3110 sync();
3111
3112 spu = spu_info[i].spu;
3113 spu_mfc_sr1_set(spu, spu_info[i].saved_mfc_sr1_RW);
3114 out_be32(&spu->problem->spu_runcntl_RW,
3115 spu_info[i].saved_spu_runcntl_RW);
3116
3117 sync();
3118 __delay(200);
3119
3120 printf("Restarted spu %.2d\n", i);
3121 } else {
3122 catch_memory_errors = 0;
3123 printf("*** Error restarting spu %.2d\n", i);
3124 }
3125 catch_memory_errors = 0;
3126 }
3127}
3128
Michael Ellermana8984972006-10-24 18:31:28 +02003129#define DUMP_WIDTH 23
Michael Ellerman437a0702006-11-23 00:46:39 +01003130#define DUMP_VALUE(format, field, value) \
Michael Ellermana8984972006-10-24 18:31:28 +02003131do { \
3132 if (setjmp(bus_error_jmp) == 0) { \
3133 catch_memory_errors = 1; \
3134 sync(); \
3135 printf(" %-*s = "format"\n", DUMP_WIDTH, \
Michael Ellerman437a0702006-11-23 00:46:39 +01003136 #field, value); \
Michael Ellermana8984972006-10-24 18:31:28 +02003137 sync(); \
3138 __delay(200); \
3139 } else { \
3140 catch_memory_errors = 0; \
3141 printf(" %-*s = *** Error reading field.\n", \
3142 DUMP_WIDTH, #field); \
3143 } \
3144 catch_memory_errors = 0; \
3145} while (0)
3146
Michael Ellerman437a0702006-11-23 00:46:39 +01003147#define DUMP_FIELD(obj, format, field) \
3148 DUMP_VALUE(format, field, obj->field)
3149
Michael Ellermana8984972006-10-24 18:31:28 +02003150static void dump_spu_fields(struct spu *spu)
3151{
3152 printf("Dumping spu fields at address %p:\n", spu);
3153
3154 DUMP_FIELD(spu, "0x%x", number);
3155 DUMP_FIELD(spu, "%s", name);
Michael Ellermana8984972006-10-24 18:31:28 +02003156 DUMP_FIELD(spu, "0x%lx", local_store_phys);
3157 DUMP_FIELD(spu, "0x%p", local_store);
3158 DUMP_FIELD(spu, "0x%lx", ls_size);
3159 DUMP_FIELD(spu, "0x%x", node);
3160 DUMP_FIELD(spu, "0x%lx", flags);
Michael Ellermana8984972006-10-24 18:31:28 +02003161 DUMP_FIELD(spu, "%d", class_0_pending);
Luke Browningf3d69e02008-04-27 18:41:55 +00003162 DUMP_FIELD(spu, "0x%lx", class_0_dar);
Luke Browningf3d69e02008-04-27 18:41:55 +00003163 DUMP_FIELD(spu, "0x%lx", class_1_dar);
3164 DUMP_FIELD(spu, "0x%lx", class_1_dsisr);
Michael Ellermana8984972006-10-24 18:31:28 +02003165 DUMP_FIELD(spu, "0x%lx", irqs[0]);
3166 DUMP_FIELD(spu, "0x%lx", irqs[1]);
3167 DUMP_FIELD(spu, "0x%lx", irqs[2]);
3168 DUMP_FIELD(spu, "0x%x", slb_replace);
3169 DUMP_FIELD(spu, "%d", pid);
Michael Ellermana8984972006-10-24 18:31:28 +02003170 DUMP_FIELD(spu, "0x%p", mm);
3171 DUMP_FIELD(spu, "0x%p", ctx);
3172 DUMP_FIELD(spu, "0x%p", rq);
3173 DUMP_FIELD(spu, "0x%p", timestamp);
3174 DUMP_FIELD(spu, "0x%lx", problem_phys);
3175 DUMP_FIELD(spu, "0x%p", problem);
Michael Ellerman437a0702006-11-23 00:46:39 +01003176 DUMP_VALUE("0x%x", problem->spu_runcntl_RW,
3177 in_be32(&spu->problem->spu_runcntl_RW));
3178 DUMP_VALUE("0x%x", problem->spu_status_R,
3179 in_be32(&spu->problem->spu_status_R));
3180 DUMP_VALUE("0x%x", problem->spu_npc_RW,
3181 in_be32(&spu->problem->spu_npc_RW));
Michael Ellermana8984972006-10-24 18:31:28 +02003182 DUMP_FIELD(spu, "0x%p", priv2);
Michael Ellermana9852392006-11-23 00:46:50 +01003183 DUMP_FIELD(spu, "0x%p", pdata);
Michael Ellermana8984972006-10-24 18:31:28 +02003184}
3185
Michael Ellermanaf89fb82006-11-23 00:46:44 +01003186int
3187spu_inst_dump(unsigned long adr, long count, int praddr)
3188{
3189 return generic_inst_dump(adr, count, praddr, print_insn_spu);
3190}
3191
3192static void dump_spu_ls(unsigned long num, int subcmd)
Michael Ellerman24a24c82006-11-23 00:46:41 +01003193{
3194 unsigned long offset, addr, ls_addr;
3195
3196 if (setjmp(bus_error_jmp) == 0) {
3197 catch_memory_errors = 1;
3198 sync();
3199 ls_addr = (unsigned long)spu_info[num].spu->local_store;
3200 sync();
3201 __delay(200);
3202 } else {
3203 catch_memory_errors = 0;
3204 printf("*** Error: accessing spu info for spu %d\n", num);
3205 return;
3206 }
3207 catch_memory_errors = 0;
3208
3209 if (scanhex(&offset))
3210 addr = ls_addr + offset;
3211 else
3212 addr = spu_info[num].dump_addr;
3213
3214 if (addr >= ls_addr + LS_SIZE) {
3215 printf("*** Error: address outside of local store\n");
3216 return;
3217 }
3218
Michael Ellermanaf89fb82006-11-23 00:46:44 +01003219 switch (subcmd) {
3220 case 'i':
3221 addr += spu_inst_dump(addr, 16, 1);
3222 last_cmd = "sdi\n";
3223 break;
3224 default:
3225 prdump(addr, 64);
3226 addr += 64;
3227 last_cmd = "sd\n";
3228 break;
3229 }
Michael Ellerman24a24c82006-11-23 00:46:41 +01003230
3231 spu_info[num].dump_addr = addr;
3232}
3233
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003234static int do_spu_cmd(void)
3235{
Michael Ellerman24a24c82006-11-23 00:46:41 +01003236 static unsigned long num = 0;
Michael Ellermanaf89fb82006-11-23 00:46:44 +01003237 int cmd, subcmd = 0;
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003238
3239 cmd = inchar();
3240 switch (cmd) {
3241 case 's':
3242 stop_spus();
3243 break;
3244 case 'r':
3245 restart_spus();
3246 break;
Michael Ellerman24a24c82006-11-23 00:46:41 +01003247 case 'd':
Michael Ellermanaf89fb82006-11-23 00:46:44 +01003248 subcmd = inchar();
3249 if (isxdigit(subcmd) || subcmd == '\n')
3250 termch = subcmd;
3251 case 'f':
Michael Ellerman24a24c82006-11-23 00:46:41 +01003252 scanhex(&num);
3253 if (num >= XMON_NUM_SPUS || !spu_info[num].spu) {
Michael Ellermana8984972006-10-24 18:31:28 +02003254 printf("*** Error: invalid spu number\n");
Michael Ellerman24a24c82006-11-23 00:46:41 +01003255 return 0;
3256 }
3257
3258 switch (cmd) {
3259 case 'f':
3260 dump_spu_fields(spu_info[num].spu);
3261 break;
3262 default:
Michael Ellermanaf89fb82006-11-23 00:46:44 +01003263 dump_spu_ls(num, subcmd);
Michael Ellerman24a24c82006-11-23 00:46:41 +01003264 break;
3265 }
3266
Michael Ellermana8984972006-10-24 18:31:28 +02003267 break;
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003268 default:
3269 return -1;
3270 }
3271
3272 return 0;
3273}
Arnd Bergmanne0555952006-11-27 19:18:55 +01003274#else /* ! CONFIG_SPU_BASE */
Michael Ellermanff8a8f22006-10-24 18:31:27 +02003275static int do_spu_cmd(void)
3276{
3277 return -1;
3278}
3279#endif