xref: /freebsd-src/sys/powerpc/powerpc/db_trace.c (revision 1027e838a8eedf21682bbc29a636b83e87728b62)
145b4eca5SBenno Rice /*	$NetBSD: db_trace.c,v 1.20 2002/05/13 20:30:09 matt Exp $	*/
245b4eca5SBenno Rice /*	$OpenBSD: db_trace.c,v 1.3 1997/03/21 02:10:48 niklas Exp $	*/
345b4eca5SBenno Rice 
460727d8bSWarner Losh /*-
545b4eca5SBenno Rice  * Mach Operating System
645b4eca5SBenno Rice  * Copyright (c) 1992 Carnegie Mellon University
745b4eca5SBenno Rice  * All Rights Reserved.
845b4eca5SBenno Rice  *
945b4eca5SBenno Rice  * Permission to use, copy, modify and distribute this software and its
1045b4eca5SBenno Rice  * documentation is hereby granted, provided that both the copyright
1145b4eca5SBenno Rice  * notice and this permission notice appear in all copies of the
1245b4eca5SBenno Rice  * software, derivative works or modified versions, and any portions
1345b4eca5SBenno Rice  * thereof, and that both notices appear in supporting documentation.
1445b4eca5SBenno Rice  *
1545b4eca5SBenno Rice  * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
1645b4eca5SBenno Rice  * CONDITION.  CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
1745b4eca5SBenno Rice  * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
1845b4eca5SBenno Rice  *
1945b4eca5SBenno Rice  * Carnegie Mellon requests users of this software to return to
2045b4eca5SBenno Rice  *
2145b4eca5SBenno Rice  *  Software Distribution Coordinator  or  Software.Distribution@CS.CMU.EDU
2245b4eca5SBenno Rice  *  School of Computer Science
2345b4eca5SBenno Rice  *  Carnegie Mellon University
2445b4eca5SBenno Rice  *  Pittsburgh PA 15213-3890
2545b4eca5SBenno Rice  *
2645b4eca5SBenno Rice  * any improvements or extensions that they make and grant Carnegie Mellon
2745b4eca5SBenno Rice  * the rights to redistribute these changes.
2845b4eca5SBenno Rice  */
2945b4eca5SBenno Rice 
3045b4eca5SBenno Rice #include <sys/param.h>
31d13782efSPeter Grehan #include <sys/systm.h>
32d8570880SPeter Grehan #include <sys/kdb.h>
3345b4eca5SBenno Rice #include <sys/proc.h>
348d511e2aSJeff Roberson #include <sys/stack.h>
3545b4eca5SBenno Rice 
3645b4eca5SBenno Rice #include <vm/vm.h>
3745b4eca5SBenno Rice #include <vm/pmap.h>
3845b4eca5SBenno Rice #include <vm/vm_extern.h>
3945b4eca5SBenno Rice 
4045b4eca5SBenno Rice #include <machine/db_machdep.h>
416004362eSDavid Schultz #include <machine/pcb.h>
4245b4eca5SBenno Rice #include <machine/spr.h>
433c90d1eaSRobert Watson #include <machine/stack.h>
4445b4eca5SBenno Rice #include <machine/trap.h>
4545b4eca5SBenno Rice 
4645b4eca5SBenno Rice #include <ddb/ddb.h>
4745b4eca5SBenno Rice #include <ddb/db_access.h>
4845b4eca5SBenno Rice #include <ddb/db_sym.h>
4945b4eca5SBenno Rice #include <ddb/db_variables.h>
5045b4eca5SBenno Rice 
51d8570880SPeter Grehan static db_varfcn_t db_frame;
52d8570880SPeter Grehan 
53d8570880SPeter Grehan #define DB_OFFSET(x)	(db_expr_t *)offsetof(struct trapframe, x)
54d8570880SPeter Grehan 
55c3e289e1SNathan Whitehorn #ifdef __powerpc64__
56c3e289e1SNathan Whitehorn #define	CALLOFFSET	8	/* Include TOC reload slot */
57c3e289e1SNathan Whitehorn #else
58c3e289e1SNathan Whitehorn #define	CALLOFFSET	4
59c3e289e1SNathan Whitehorn #endif
60c3e289e1SNathan Whitehorn 
6145b4eca5SBenno Rice struct db_variable db_regs[] = {
62d8570880SPeter Grehan 	{ "r0",	 DB_OFFSET(fixreg[0]),	db_frame },
63d8570880SPeter Grehan 	{ "r1",	 DB_OFFSET(fixreg[1]),	db_frame },
64d8570880SPeter Grehan 	{ "r2",	 DB_OFFSET(fixreg[2]),	db_frame },
65d8570880SPeter Grehan 	{ "r3",	 DB_OFFSET(fixreg[3]),	db_frame },
66d8570880SPeter Grehan 	{ "r4",	 DB_OFFSET(fixreg[4]),	db_frame },
67d8570880SPeter Grehan 	{ "r5",	 DB_OFFSET(fixreg[5]),	db_frame },
68d8570880SPeter Grehan 	{ "r6",	 DB_OFFSET(fixreg[6]),	db_frame },
69d8570880SPeter Grehan 	{ "r7",	 DB_OFFSET(fixreg[7]),	db_frame },
70d8570880SPeter Grehan 	{ "r8",	 DB_OFFSET(fixreg[8]),	db_frame },
71d8570880SPeter Grehan 	{ "r9",	 DB_OFFSET(fixreg[9]),	db_frame },
72d8570880SPeter Grehan 	{ "r10", DB_OFFSET(fixreg[10]),	db_frame },
73d8570880SPeter Grehan 	{ "r11", DB_OFFSET(fixreg[11]),	db_frame },
74d8570880SPeter Grehan 	{ "r12", DB_OFFSET(fixreg[12]),	db_frame },
75d8570880SPeter Grehan 	{ "r13", DB_OFFSET(fixreg[13]),	db_frame },
76d8570880SPeter Grehan 	{ "r14", DB_OFFSET(fixreg[14]),	db_frame },
77d8570880SPeter Grehan 	{ "r15", DB_OFFSET(fixreg[15]),	db_frame },
78d8570880SPeter Grehan 	{ "r16", DB_OFFSET(fixreg[16]),	db_frame },
79d8570880SPeter Grehan 	{ "r17", DB_OFFSET(fixreg[17]),	db_frame },
80d8570880SPeter Grehan 	{ "r18", DB_OFFSET(fixreg[18]),	db_frame },
81d8570880SPeter Grehan 	{ "r19", DB_OFFSET(fixreg[19]),	db_frame },
82d8570880SPeter Grehan 	{ "r20", DB_OFFSET(fixreg[20]),	db_frame },
83d8570880SPeter Grehan 	{ "r21", DB_OFFSET(fixreg[21]),	db_frame },
84d8570880SPeter Grehan 	{ "r22", DB_OFFSET(fixreg[22]),	db_frame },
85d8570880SPeter Grehan 	{ "r23", DB_OFFSET(fixreg[23]),	db_frame },
86d8570880SPeter Grehan 	{ "r24", DB_OFFSET(fixreg[24]),	db_frame },
87d8570880SPeter Grehan 	{ "r25", DB_OFFSET(fixreg[25]),	db_frame },
88d8570880SPeter Grehan 	{ "r26", DB_OFFSET(fixreg[26]),	db_frame },
89d8570880SPeter Grehan 	{ "r27", DB_OFFSET(fixreg[27]),	db_frame },
90d8570880SPeter Grehan 	{ "r28", DB_OFFSET(fixreg[28]),	db_frame },
91d8570880SPeter Grehan 	{ "r29", DB_OFFSET(fixreg[29]),	db_frame },
92d8570880SPeter Grehan 	{ "r30", DB_OFFSET(fixreg[30]),	db_frame },
93d8570880SPeter Grehan 	{ "r31", DB_OFFSET(fixreg[31]),	db_frame },
94d8570880SPeter Grehan 	{ "srr0", DB_OFFSET(srr0),	db_frame },
95d8570880SPeter Grehan 	{ "srr1", DB_OFFSET(srr1),	db_frame },
96d8570880SPeter Grehan 	{ "lr",	 DB_OFFSET(lr),		db_frame },
97d8570880SPeter Grehan 	{ "ctr", DB_OFFSET(ctr),	db_frame },
98d8570880SPeter Grehan 	{ "cr",	 DB_OFFSET(cr),		db_frame },
99d8570880SPeter Grehan 	{ "xer", DB_OFFSET(xer),	db_frame },
100f14cf38dSNathan Whitehorn 	{ "dar", DB_OFFSET(dar),	db_frame },
101786e4a1bSRafal Jaworowski #ifdef AIM
102786e4a1bSRafal Jaworowski 	{ "dsisr", DB_OFFSET(cpu.aim.dsisr),	db_frame },
103786e4a1bSRafal Jaworowski #endif
10417f4cae4SRafal Jaworowski #if defined(BOOKE)
105786e4a1bSRafal Jaworowski 	{ "esr", DB_OFFSET(cpu.booke.esr),	db_frame },
106786e4a1bSRafal Jaworowski #endif
10745b4eca5SBenno Rice };
10833495e5dSPedro F. Giffuni struct db_variable *db_eregs = db_regs + nitems(db_regs);
10945b4eca5SBenno Rice 
11045b4eca5SBenno Rice /*
111d8570880SPeter Grehan  * register variable handling
112d8570880SPeter Grehan  */
113d8570880SPeter Grehan static int
db_frame(struct db_variable * vp,db_expr_t * valuep,int op)114d8570880SPeter Grehan db_frame(struct db_variable *vp, db_expr_t *valuep, int op)
115d8570880SPeter Grehan {
116c3e289e1SNathan Whitehorn 	register_t *reg;
117d8570880SPeter Grehan 
118d8570880SPeter Grehan 	if (kdb_frame == NULL)
119d8570880SPeter Grehan 		return (0);
120c3e289e1SNathan Whitehorn 	reg = (register_t*)((uintptr_t)kdb_frame + (uintptr_t)vp->valuep);
121d8570880SPeter Grehan 	if (op == DB_VAR_GET)
122d8570880SPeter Grehan 		*valuep = *reg;
123d8570880SPeter Grehan 	else
124d8570880SPeter Grehan 		*reg = *valuep;
125d8570880SPeter Grehan 	return (1);
126d8570880SPeter Grehan }
127d8570880SPeter Grehan 
128d8570880SPeter Grehan /*
12945b4eca5SBenno Rice  *	Frame tracing.
13045b4eca5SBenno Rice  */
131d8570880SPeter Grehan static int
db_backtrace(struct thread * td,db_addr_t fp,int count)132d8570880SPeter Grehan db_backtrace(struct thread *td, db_addr_t fp, int count)
13345b4eca5SBenno Rice {
134d8570880SPeter Grehan 	db_addr_t stackframe, lr, *args;
135*1027e838SJohn Baldwin 	bool kernel_only = true;
136*1027e838SJohn Baldwin 	bool full = false;
13745b4eca5SBenno Rice 
138d8570880SPeter Grehan #if 0
13945b4eca5SBenno Rice 	{
14045b4eca5SBenno Rice 		register char *cp = modif;
14145b4eca5SBenno Rice 		register char c;
14245b4eca5SBenno Rice 
14345b4eca5SBenno Rice 		while ((c = *cp++) != 0) {
14445b4eca5SBenno Rice 			if (c == 't')
145*1027e838SJohn Baldwin 				trace_thread = true;
14645b4eca5SBenno Rice 			if (c == 'u')
147*1027e838SJohn Baldwin 				kernel_only = false;
14845b4eca5SBenno Rice 			if (c == 'f')
149*1027e838SJohn Baldwin 				full = true;
15045b4eca5SBenno Rice 		}
15145b4eca5SBenno Rice 	}
15245b4eca5SBenno Rice #endif
153d8570880SPeter Grehan 
154d8570880SPeter Grehan 	stackframe = fp;
155d8570880SPeter Grehan 
15619e9205aSJohn Baldwin 	while (!db_pager_quit) {
157d8570880SPeter Grehan 		if (stackframe < PAGE_SIZE)
15845b4eca5SBenno Rice 			break;
159d8570880SPeter Grehan 
160d8570880SPeter Grehan 		/*
161d8570880SPeter Grehan 		 * Locate the next frame by grabbing the backchain ptr
162d8570880SPeter Grehan 		 * from frame[0]
163d8570880SPeter Grehan 		 */
164d8570880SPeter Grehan 		stackframe = *(db_addr_t *)stackframe;
165d8570880SPeter Grehan 
16645b4eca5SBenno Rice 	next_frame:
167c3e289e1SNathan Whitehorn 	    #ifdef __powerpc64__
168c3e289e1SNathan Whitehorn 		/* The saved arg values start at frame[6] */
169c3e289e1SNathan Whitehorn 		args = (db_addr_t *)(stackframe + 48);
170c3e289e1SNathan Whitehorn 	    #else
171d8570880SPeter Grehan 		/* The saved arg values start at frame[2] */
172d8570880SPeter Grehan 		args = (db_addr_t *)(stackframe + 8);
173c3e289e1SNathan Whitehorn 	    #endif
174d8570880SPeter Grehan 
175d8570880SPeter Grehan 		if (stackframe < PAGE_SIZE)
17645b4eca5SBenno Rice 			break;
177d8570880SPeter Grehan 
17845b4eca5SBenno Rice 	        if (count-- == 0)
17945b4eca5SBenno Rice 			break;
18045b4eca5SBenno Rice 
181d8570880SPeter Grehan 		/*
182d8570880SPeter Grehan 		 * Extract link register from frame and subtract
183d8570880SPeter Grehan 		 * 4 to convert into calling address (as opposed to
184d8570880SPeter Grehan 		 * return address)
185d8570880SPeter Grehan 		 */
186c3e289e1SNathan Whitehorn 	    #ifdef __powerpc64__
187c3e289e1SNathan Whitehorn 		lr = *(db_addr_t *)(stackframe + 16) - 4;
188c3e289e1SNathan Whitehorn 	    #else
189d8570880SPeter Grehan 		lr = *(db_addr_t *)(stackframe + 4) - 4;
190c3e289e1SNathan Whitehorn 	    #endif
19145b4eca5SBenno Rice 		if ((lr & 3) || (lr < 0x100)) {
192c3e289e1SNathan Whitehorn 			db_printf("saved LR(0x%zx) is invalid.", lr);
19345b4eca5SBenno Rice 			break;
19445b4eca5SBenno Rice 		}
19545b4eca5SBenno Rice 
196c3e289e1SNathan Whitehorn 		#ifdef __powerpc64__
197f7688a41SAndreas Tobler 		db_printf("0x%016lx: ", stackframe);
198c3e289e1SNathan Whitehorn 		#else
199d8570880SPeter Grehan 		db_printf("0x%08x: ", stackframe);
200c3e289e1SNathan Whitehorn 		#endif
201d8570880SPeter Grehan 
202d8570880SPeter Grehan 		/*
20378cb8214SPeter Grehan 		 * The trap code labels the return addresses from the
20478cb8214SPeter Grehan 		 * call to C code as 'trapexit' and 'asttrapexit. Use this
20578cb8214SPeter Grehan 		 * to determine if the callframe has to traverse a saved
20678cb8214SPeter Grehan 		 * trap context
207d8570880SPeter Grehan 		 */
208c3e289e1SNathan Whitehorn 		if ((lr + CALLOFFSET == (db_addr_t) &trapexit) ||
209c3e289e1SNathan Whitehorn 		    (lr + CALLOFFSET == (db_addr_t) &asttrapexit)) {
21045b4eca5SBenno Rice 			const char *trapstr;
211c3e289e1SNathan Whitehorn 			struct trapframe *tf = (struct trapframe *)(args);
212d8570880SPeter Grehan 			db_printf("%s ", tf->srr1 & PSL_PR ? "user" : "kernel");
21345b4eca5SBenno Rice 			switch (tf->exc) {
21445b4eca5SBenno Rice 			case EXC_DSI:
215786e4a1bSRafal Jaworowski 				/* XXX take advantage of the union. */
21615f9620eSJustin Hibbits #ifdef BOOKE
21715f9620eSJustin Hibbits 				db_printf("DSI %s trap @ %#zx by ",
21815f9620eSJustin Hibbits 				    (tf->cpu.booke.esr & ESR_ST) ? "write"
21915f9620eSJustin Hibbits 				    : "read", tf->dar);
22015f9620eSJustin Hibbits #else
221c3e289e1SNathan Whitehorn 				db_printf("DSI %s trap @ %#zx by ",
222786e4a1bSRafal Jaworowski 				    (tf->cpu.aim.dsisr & DSISR_STORE) ? "write"
223f14cf38dSNathan Whitehorn 				    : "read", tf->dar);
22415f9620eSJustin Hibbits #endif
225d8570880SPeter Grehan 				goto print_trap;
226d8570880SPeter Grehan 			case EXC_ALI:
227c3e289e1SNathan Whitehorn 				db_printf("ALI trap @ %#zx (xSR %#x) ",
228f14cf38dSNathan Whitehorn 				    tf->dar, (uint32_t)tf->cpu.aim.dsisr);
22945b4eca5SBenno Rice 				goto print_trap;
230c3e289e1SNathan Whitehorn #ifdef __powerpc64__
231c3e289e1SNathan Whitehorn 			case EXC_DSE:
232f14cf38dSNathan Whitehorn 				db_printf("DSE trap @ %#zx by ", tf->dar);
233c3e289e1SNathan Whitehorn 				goto print_trap;
234c3e289e1SNathan Whitehorn 			case EXC_ISE:
235c3e289e1SNathan Whitehorn 				db_printf("ISE trap @ %#zx by ", tf->srr0);
236c3e289e1SNathan Whitehorn 				goto print_trap;
237c3e289e1SNathan Whitehorn #endif
23845b4eca5SBenno Rice 			case EXC_ISI: trapstr = "ISI"; break;
23945b4eca5SBenno Rice 			case EXC_PGM: trapstr = "PGM"; break;
24045b4eca5SBenno Rice 			case EXC_SC: trapstr = "SC"; break;
24145b4eca5SBenno Rice 			case EXC_EXI: trapstr = "EXI"; break;
24245b4eca5SBenno Rice 			case EXC_MCHK: trapstr = "MCHK"; break;
24345b4eca5SBenno Rice 			case EXC_VEC: trapstr = "VEC"; break;
24415f9620eSJustin Hibbits #if !defined(BOOKE)
24545b4eca5SBenno Rice 			case EXC_FPA: trapstr = "FPA"; break;
24645b4eca5SBenno Rice 			case EXC_BPT: trapstr = "BPT"; break;
24745b4eca5SBenno Rice 			case EXC_TRC: trapstr = "TRC"; break;
24845b4eca5SBenno Rice 			case EXC_RUNMODETRC: trapstr = "RUNMODETRC"; break;
24945b4eca5SBenno Rice 			case EXC_SMI: trapstr = "SMI"; break;
25045b4eca5SBenno Rice 			case EXC_RST: trapstr = "RST"; break;
251786e4a1bSRafal Jaworowski #endif
252786e4a1bSRafal Jaworowski 			case EXC_FPU: trapstr = "FPU"; break;
253786e4a1bSRafal Jaworowski 			case EXC_DECR: trapstr = "DECR"; break;
254786e4a1bSRafal Jaworowski 			case EXC_PERF: trapstr = "PERF"; break;
25535f612b8SNathan Whitehorn 			case EXC_VSX: trapstr = "VSX"; break;
256be2bd024SLeandro Lupori 			case EXC_SOFT_PATCH: trapstr = "SOFT_PATCH"; break;
25745b4eca5SBenno Rice 			default: trapstr = NULL; break;
25845b4eca5SBenno Rice 			}
25945b4eca5SBenno Rice 			if (trapstr != NULL) {
260d8570880SPeter Grehan 				db_printf("%s trap by ", trapstr);
26145b4eca5SBenno Rice 			} else {
262c3e289e1SNathan Whitehorn 				db_printf("trap %#zx by ", tf->exc);
26345b4eca5SBenno Rice 			}
264d8570880SPeter Grehan 
26545b4eca5SBenno Rice 		   print_trap:
26645b4eca5SBenno Rice 			lr = (db_addr_t) tf->srr0;
2678f92db28SJustin Hibbits 			db_printsym(lr, DB_STGY_ANY);
26869e45c75SJustin Hibbits 			db_printf(": srr1=%#zx\n", tf->srr1);
269c3e289e1SNathan Whitehorn 			db_printf("%-10s  r1=%#zx cr=%#x xer=%#x ctr=%#zx",
270c3e289e1SNathan Whitehorn 			    "", tf->fixreg[1], (uint32_t)tf->cr,
271c3e289e1SNathan Whitehorn 			    (uint32_t)tf->xer, tf->ctr);
2721c8e60edSNathan Whitehorn #ifdef __powerpc64__
2731c8e60edSNathan Whitehorn 			db_printf(" r2=%#zx", tf->fixreg[2]);
2741c8e60edSNathan Whitehorn #endif
27545b4eca5SBenno Rice 			if (tf->exc == EXC_DSI)
276c3e289e1SNathan Whitehorn 				db_printf(" sr=%#x",
277c3e289e1SNathan Whitehorn 				    (uint32_t)tf->cpu.aim.dsisr);
278093f7de6SJustin Hibbits 			db_printf(" frame=%p\n", tf);
279d8570880SPeter Grehan 			stackframe = (db_addr_t) tf->fixreg[1];
28045b4eca5SBenno Rice 			if (kernel_only && (tf->srr1 & PSL_PR))
28145b4eca5SBenno Rice 				break;
28245b4eca5SBenno Rice 			goto next_frame;
28345b4eca5SBenno Rice 		}
28445b4eca5SBenno Rice 
2858f92db28SJustin Hibbits 		db_printf("at ");
2868f92db28SJustin Hibbits 		db_printsym(lr, DB_STGY_PROC);
28745b4eca5SBenno Rice 		if (full)
28845b4eca5SBenno Rice 			/* Print all the args stored in that stackframe. */
289c3e289e1SNathan Whitehorn 			db_printf("(%zx, %zx, %zx, %zx, %zx, %zx, %zx, %zx)",
29045b4eca5SBenno Rice 				args[0], args[1], args[2], args[3],
29145b4eca5SBenno Rice 				args[4], args[5], args[6], args[7]);
292d8570880SPeter Grehan 		db_printf("\n");
29345b4eca5SBenno Rice 	}
294d8570880SPeter Grehan 
295d8570880SPeter Grehan 	return (0);
29645b4eca5SBenno Rice }
29745b4eca5SBenno Rice 
29845b4eca5SBenno Rice void
db_trace_self(void)299d8570880SPeter Grehan db_trace_self(void)
300c79408a0SJohn Baldwin {
301d8570880SPeter Grehan 	db_addr_t addr;
302d8570880SPeter Grehan 
3038d2f50dbSJustin Hibbits 	addr = (db_addr_t)__builtin_frame_address(0);
3048d2f50dbSJustin Hibbits 	if (addr == 0) {
3058d2f50dbSJustin Hibbits 		db_printf("Null frame address\n");
3068d2f50dbSJustin Hibbits 		return;
3078d2f50dbSJustin Hibbits 	}
3088d2f50dbSJustin Hibbits 	db_backtrace(curthread, *(db_addr_t *)addr, -1);
309c79408a0SJohn Baldwin }
310d8570880SPeter Grehan 
311d8570880SPeter Grehan int
db_trace_thread(struct thread * td,int count)312d8570880SPeter Grehan db_trace_thread(struct thread *td, int count)
313d8570880SPeter Grehan {
314d8570880SPeter Grehan 	struct pcb *ctx;
315d8570880SPeter Grehan 
316d8570880SPeter Grehan 	ctx = kdb_thr_ctx(td);
317d8570880SPeter Grehan 	return (db_backtrace(td, (db_addr_t)ctx->pcb_sp, count));
318d8570880SPeter Grehan }
319