/* $NetBSD: db_trace.c,v 1.3 2001/09/09 10:33:43 toshii Exp $ */ /* * Copyright (c) 1996 Scott K. Stevens * * Mach Operating System * Copyright (c) 1991,1990 Carnegie Mellon University * All Rights Reserved. * * Permission to use, copy, modify and distribute this software and its * documentation is hereby granted, provided that both the copyright * notice and this permission notice appear in all copies of the * software, derivative works or modified versions, and any portions * thereof, and that both notices appear in supporting documentation. * * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS" * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE. * * Carnegie Mellon requests users of this software to return to * * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU * School of Computer Science * Carnegie Mellon University * Pittsburgh PA 15213-3890 * * any improvements or extensions that they make and grant Carnegie the * rights to redistribute these changes. */ #include #include #include #include #include #include #include #define INKERNEL(va) (((vm_offset_t)(va)) >= VM_MIN_KERNEL_ADDRESS) void db_stack_trace_print(addr, have_addr, count, modif, pr) db_expr_t addr; int have_addr; db_expr_t count; char *modif; void (*pr) __P((const char *, ...)); { struct frame *frame, *lastframe; char c, *cp = modif; boolean_t kernel_only = TRUE; boolean_t trace_thread = FALSE; while ((c = *cp++) != 0) { if (c == 'u') kernel_only = FALSE; if (c == 't') trace_thread = TRUE; } /* * The frame pointer points to the top word of the stack frame so we * need to adjust it by sizeof(struct frame) - sizeof(u_int)) * to get the address of the start of the frame structure. */ if (!have_addr) frame = (struct frame *)(DDB_REGS->tf_r11 - (sizeof(struct frame) - sizeof(u_int))); else { if (trace_thread) { struct proc *p; struct user *u; (*pr)("trace: pid %d ", (int)addr); p = pfind(addr); if (p == NULL) { (*pr)("not found\n"); return; } if (!(p->p_flag & P_INMEM)) { (*pr)("swapped out\n"); return; } u = p->p_addr; frame = (struct frame *) (u->u_pcb.pcb_r11 - (sizeof(struct frame) - sizeof(u_int))); (*pr)("at %p\n", frame); } else frame = (struct frame *)(addr - (sizeof(struct frame) - sizeof(u_int))); } lastframe = NULL; while (count--) { db_expr_t offset; char *name; db_addr_t pc; /* (*pr)("fp=%08x: fp=%08x sp=%08x lr=%08x pc=%08x\n", (u_int)frame, frame->fr_fp, frame->fr_sp, frame->fr_lr, frame->fr_pc);*/ pc = frame->fr_pc; /* Adjust the PC so the same address is printed no matter what CPU */ if (cputype == CPU_ID_SA110 || cputype == CPU_ID_ARM810) pc += 4; if (!INKERNEL(pc)) break; db_find_sym_and_offset(pc, &name, &offset); if (name == NULL) name = "?"; (*pr)("%s(", name); db_printsym(pc, DB_STGY_PROC, pr); (*pr)(")"); (*pr)("\n"); /* * Switch to next frame up */ lastframe = frame; if (frame->fr_fp == NULL) break; frame = (struct frame *)(frame->fr_fp - (sizeof(struct frame) - sizeof(u_int))); if (INKERNEL((int)frame)) { /* staying in kernel */ if (frame <= lastframe) { (*pr)("Bad frame pointer: %p\n", frame); break; } } else if (INKERNEL((int)lastframe)) { /* switch from user to kernel */ if (kernel_only) break; /* kernel stack only */ } else { /* in user */ if (frame <= lastframe) { (*pr)("Bad user frame pointer: %p\n", frame); break; } } } }