diff options
Diffstat (limited to 'sys/arch')
-rw-r--r-- | sys/arch/sparc64/sparc64/db_interface.c | 1236 |
1 files changed, 1236 insertions, 0 deletions
diff --git a/sys/arch/sparc64/sparc64/db_interface.c b/sys/arch/sparc64/sparc64/db_interface.c new file mode 100644 index 00000000000..cd28194c02c --- /dev/null +++ b/sys/arch/sparc64/sparc64/db_interface.c @@ -0,0 +1,1236 @@ +/* $NetBSD: db_interface.c,v 1.61 2001/07/31 06:55:47 eeh Exp $ */ + +/* + * 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. + * + * From: db_interface.c,v 2.4 1991/02/05 17:11:13 mrt (CMU) + */ + +#include <sys/param.h> +#include <sys/proc.h> +#include <sys/user.h> +#include <sys/reboot.h> +#include <sys/systm.h> + +#include <uvm/uvm_extern.h> + +#include <dev/cons.h> + +#include <machine/db_machdep.h> +#include <ddb/db_command.h> +#include <ddb/db_sym.h> +#include <ddb/db_variables.h> +#include <ddb/db_extern.h> +#include <ddb/db_access.h> +#include <ddb/db_output.h> +#include <ddb/db_interface.h> + +#include <machine/instr.h> +#include <machine/cpu.h> +#include <machine/openfirm.h> +#include <machine/ctlreg.h> +#include <machine/pmap.h> + +#ifdef notyet +#include "fb.h" +#include "esp_sbus.h" +#endif + +extern void OF_enter __P((void)); + +extern struct traptrace { + unsigned short tl:3, /* Trap level */ + ns:4, /* PCB nsaved */ + tt:9; /* Trap type */ + unsigned short pid; /* PID */ + u_int tstate; /* tstate */ + u_int tsp; /* sp */ + u_int tpc; /* pc */ + u_int tfault; /* MMU tag access */ +} trap_trace[], trap_trace_end[]; + +static int nil; + +static int +db__char_value(struct db_variable *var, db_expr_t *expr, int mode) +{ + + switch (mode) { + case DB_VAR_SET: + *var->valuep = *(char *)expr; + break; + case DB_VAR_GET: + *expr = *(char *)var->valuep; + break; +#ifdef DIAGNOSTIC + default: + printf("db__char_value: mode %d\n", mode); + break; +#endif + } + + return 0; +} + +#ifdef notdef_yet +static int +db__short_value(struct db_variable *var, db_expr_t *expr, int mode) +{ + + switch (mode) { + case DB_VAR_SET: + *var->valuep = *(short *)expr; + break; + case DB_VAR_GET: + *expr = *(short *)var->valuep; + break; +#ifdef DIAGNOSTIC + default: + printf("db__short_value: mode %d\n", mode); + break; +#endif + } + + return 0; +} +#endif + +static int +db__int_value(struct db_variable *var, db_expr_t *expr, int mode) +{ + + switch (mode) { + case DB_VAR_SET: + *var->valuep = *(int *)expr; + break; + case DB_VAR_GET: + *expr = *(int *)var->valuep; + break; +#ifdef DIAGNOSTIC + default: + printf("db__int_value: mode %d\n", mode); + break; +#endif + } + + return 0; +} + +struct db_variable db_regs[] = { + { "tstate", (long *)&DDB_TF->tf_tstate, FCN_NULL, }, + { "pc", (long *)&DDB_TF->tf_pc, FCN_NULL, }, + { "npc", (long *)&DDB_TF->tf_npc, FCN_NULL, }, + { "ipl", (long *)&DDB_TF->tf_oldpil, db__char_value, }, + { "y", (long *)&DDB_TF->tf_y, db__int_value, }, + { "g0", (long *)&nil, FCN_NULL, }, + { "g1", (long *)&DDB_TF->tf_global[1], FCN_NULL, }, + { "g2", (long *)&DDB_TF->tf_global[2], FCN_NULL, }, + { "g3", (long *)&DDB_TF->tf_global[3], FCN_NULL, }, + { "g4", (long *)&DDB_TF->tf_global[4], FCN_NULL, }, + { "g5", (long *)&DDB_TF->tf_global[5], FCN_NULL, }, + { "g6", (long *)&DDB_TF->tf_global[6], FCN_NULL, }, + { "g7", (long *)&DDB_TF->tf_global[7], FCN_NULL, }, + { "o0", (long *)&DDB_TF->tf_out[0], FCN_NULL, }, + { "o1", (long *)&DDB_TF->tf_out[1], FCN_NULL, }, + { "o2", (long *)&DDB_TF->tf_out[2], FCN_NULL, }, + { "o3", (long *)&DDB_TF->tf_out[3], FCN_NULL, }, + { "o4", (long *)&DDB_TF->tf_out[4], FCN_NULL, }, + { "o5", (long *)&DDB_TF->tf_out[5], FCN_NULL, }, + { "o6", (long *)&DDB_TF->tf_out[6], FCN_NULL, }, + { "o7", (long *)&DDB_TF->tf_out[7], FCN_NULL, }, + { "l0", (long *)&DDB_TF->tf_local[0], FCN_NULL, }, + { "l1", (long *)&DDB_TF->tf_local[1], FCN_NULL, }, + { "l2", (long *)&DDB_TF->tf_local[2], FCN_NULL, }, + { "l3", (long *)&DDB_TF->tf_local[3], FCN_NULL, }, + { "l4", (long *)&DDB_TF->tf_local[4], FCN_NULL, }, + { "l5", (long *)&DDB_TF->tf_local[5], FCN_NULL, }, + { "l6", (long *)&DDB_TF->tf_local[6], FCN_NULL, }, + { "l7", (long *)&DDB_TF->tf_local[7], FCN_NULL, }, + { "i0", (long *)&DDB_FR->fr_arg[0], FCN_NULL, }, + { "i1", (long *)&DDB_FR->fr_arg[1], FCN_NULL, }, + { "i2", (long *)&DDB_FR->fr_arg[2], FCN_NULL, }, + { "i3", (long *)&DDB_FR->fr_arg[3], FCN_NULL, }, + { "i4", (long *)&DDB_FR->fr_arg[4], FCN_NULL, }, + { "i5", (long *)&DDB_FR->fr_arg[5], FCN_NULL, }, + { "i6", (long *)&DDB_FR->fr_arg[6], FCN_NULL, }, + { "i7", (long *)&DDB_FR->fr_arg[7], FCN_NULL, }, + { "f0", (long *)&DDB_FP->fs_regs[0], FCN_NULL, }, + { "f2", (long *)&DDB_FP->fs_regs[2], FCN_NULL, }, + { "f4", (long *)&DDB_FP->fs_regs[4], FCN_NULL, }, + { "f6", (long *)&DDB_FP->fs_regs[6], FCN_NULL, }, + { "f8", (long *)&DDB_FP->fs_regs[8], FCN_NULL, }, + { "f10", (long *)&DDB_FP->fs_regs[10], FCN_NULL, }, + { "f12", (long *)&DDB_FP->fs_regs[12], FCN_NULL, }, + { "f14", (long *)&DDB_FP->fs_regs[14], FCN_NULL, }, + { "f16", (long *)&DDB_FP->fs_regs[16], FCN_NULL, }, + { "f18", (long *)&DDB_FP->fs_regs[18], FCN_NULL, }, + { "f20", (long *)&DDB_FP->fs_regs[20], FCN_NULL, }, + { "f22", (long *)&DDB_FP->fs_regs[22], FCN_NULL, }, + { "f24", (long *)&DDB_FP->fs_regs[24], FCN_NULL, }, + { "f26", (long *)&DDB_FP->fs_regs[26], FCN_NULL, }, + { "f28", (long *)&DDB_FP->fs_regs[28], FCN_NULL, }, + { "f30", (long *)&DDB_FP->fs_regs[30], FCN_NULL, }, + { "f32", (long *)&DDB_FP->fs_regs[32], FCN_NULL, }, + { "f34", (long *)&DDB_FP->fs_regs[34], FCN_NULL, }, + { "f36", (long *)&DDB_FP->fs_regs[36], FCN_NULL, }, + { "f38", (long *)&DDB_FP->fs_regs[38], FCN_NULL, }, + { "f40", (long *)&DDB_FP->fs_regs[40], FCN_NULL, }, + { "f42", (long *)&DDB_FP->fs_regs[42], FCN_NULL, }, + { "f44", (long *)&DDB_FP->fs_regs[44], FCN_NULL, }, + { "f46", (long *)&DDB_FP->fs_regs[46], FCN_NULL, }, + { "f48", (long *)&DDB_FP->fs_regs[48], FCN_NULL, }, + { "f50", (long *)&DDB_FP->fs_regs[50], FCN_NULL, }, + { "f52", (long *)&DDB_FP->fs_regs[52], FCN_NULL, }, + { "f54", (long *)&DDB_FP->fs_regs[54], FCN_NULL, }, + { "f56", (long *)&DDB_FP->fs_regs[56], FCN_NULL, }, + { "f58", (long *)&DDB_FP->fs_regs[58], FCN_NULL, }, + { "f60", (long *)&DDB_FP->fs_regs[60], FCN_NULL, }, + { "f62", (long *)&DDB_FP->fs_regs[62], FCN_NULL, }, + { "fsr", (long *)&DDB_FP->fs_fsr, FCN_NULL, }, + { "gsr", (long *)&DDB_FP->fs_gsr, FCN_NULL, }, + +}; +struct db_variable *db_eregs = db_regs + sizeof(db_regs)/sizeof(db_regs[0]); + +extern label_t *db_recover; + +int db_active = 0; + +extern char *trap_type[]; + +void kdb_kbd_trap __P((struct trapframe64 *)); +void db_prom_cmd __P((db_expr_t, int, db_expr_t, char *)); +void db_proc_cmd __P((db_expr_t, int, db_expr_t, char *)); +void db_ctx_cmd __P((db_expr_t, int, db_expr_t, char *)); +void db_dump_window __P((db_expr_t, int, db_expr_t, char *)); +void db_dump_stack __P((db_expr_t, int, db_expr_t, char *)); +void db_dump_trap __P((db_expr_t, int, db_expr_t, char *)); +void db_dump_fpstate __P((db_expr_t, int, db_expr_t, char *)); +void db_dump_ts __P((db_expr_t, int, db_expr_t, char *)); +void db_dump_pcb __P((db_expr_t, int, db_expr_t, char *)); +void db_dump_pv __P((db_expr_t, int, db_expr_t, char *)); +void db_setpcb __P((db_expr_t, int, db_expr_t, char *)); +void db_dump_dtlb __P((db_expr_t, int, db_expr_t, char *)); +void db_dump_dtsb __P((db_expr_t, int, db_expr_t, char *)); +void db_pmap_kernel __P((db_expr_t, int, db_expr_t, char *)); +void db_pload_cmd __P((db_expr_t, int, db_expr_t, char *)); +void db_pmap_cmd __P((db_expr_t, int, db_expr_t, char *)); +void db_lock __P((db_expr_t, int, db_expr_t, char *)); +void db_traptrace __P((db_expr_t, int, db_expr_t, char *)); +void db_dump_buf __P((db_expr_t, int, db_expr_t, char *)); +void db_dump_espcmd __P((db_expr_t, int, db_expr_t, char *)); +void db_watch __P((db_expr_t, int, db_expr_t, char *)); + +static void db_dump_pmap __P((struct pmap*)); +static void db_print_trace_entry __P((struct traptrace *, int)); + + +/* + * Received keyboard interrupt sequence. + */ +void +kdb_kbd_trap(tf) + struct trapframe64 *tf; +{ + if (db_active == 0 /* && (boothowto & RB_KDB) */) { + printf("\n\nkernel: keyboard interrupt tf=%p\n", tf); + kdb_trap(-1, tf); + } +} + +/* + * kdb_trap - field a TRACE or BPT trap + */ +int +kdb_trap(type, tf) + int type; + register struct trapframe64 *tf; +{ + int s, tl; + struct trapstate *ts = &ddb_regs.ddb_ts[0]; + extern int savetstate(struct trapstate *ts); + extern void restoretstate(int tl, struct trapstate *ts); + extern int trap_trace_dis; + extern int doing_shutdown; + + trap_trace_dis++; + doing_shutdown++; +#if NFB > 0 + fb_unblank(); +#endif + switch (type) { + case T_BREAKPOINT: /* breakpoint */ + printf("kdb breakpoint at %llx\n", + (unsigned long long)tf->tf_pc); + break; + case -1: /* keyboard interrupt */ + printf("kdb tf=%p\n", tf); + break; + default: + printf("kernel trap %x: %s\n", type, trap_type[type & 0x1ff]); + if (db_recover != 0) { + OF_enter(); + db_error("Faulted in DDB; continuing...\n"); + OF_enter(); + /*NOTREACHED*/ + } + db_recover = (label_t *)1; + } + + /* Should switch to kdb`s own stack here. */ + write_all_windows(); + + ddb_regs.ddb_tf = *tf; + if (fpproc) { + savefpstate(fpproc->p_md.md_fpstate); + ddb_regs.ddb_fpstate = *fpproc->p_md.md_fpstate; + loadfpstate(fpproc->p_md.md_fpstate); + } + /* We should do a proper copyin and xlate 64-bit stack frames, but... */ +/* if (tf->tf_tstate & TSTATE_PRIV) { */ + +#if 0 + /* make sure this is not causing ddb problems. */ + if (tf->tf_out[6] & 1) { + if ((unsigned)(tf->tf_out[6] + BIAS) > (unsigned)KERNBASE) + ddb_regs.ddb_fr = *(struct frame64 *)(tf->tf_out[6] + BIAS); + else + copyin((caddr_t)(tf->tf_out[6] + BIAS), &ddb_regs.ddb_fr, sizeof(struct frame64)); + } else { + struct frame32 tfr; + + /* First get a local copy of the frame32 */ + if ((unsigned)(tf->tf_out[6]) > (unsigned)KERNBASE) + tfr = *(struct frame32 *)tf->tf_out[6]; + else + copyin((caddr_t)(tf->tf_out[6]), &tfr, sizeof(struct frame32)); + /* Now copy each field from the 32-bit value to the 64-bit value */ + for (i=0; i<8; i++) + ddb_regs.ddb_fr.fr_local[i] = tfr.fr_local[i]; + for (i=0; i<6; i++) + ddb_regs.ddb_fr.fr_arg[i] = tfr.fr_arg[i]; + ddb_regs.ddb_fr.fr_fp = (long)tfr.fr_fp; + ddb_regs.ddb_fr.fr_pc = tfr.fr_pc; + } +#endif + + s = splhigh(); + db_active++; + cnpollc(TRUE); + /* Need to do spl stuff till cnpollc works */ + tl = ddb_regs.ddb_tl = savetstate(ts); + db_dump_ts(0, 0, 0, 0); + db_trap(type, 0/*code*/); + restoretstate(tl,ts); + cnpollc(FALSE); + db_active--; + splx(s); + + if (fpproc) { + *fpproc->p_md.md_fpstate = ddb_regs.ddb_fpstate; + loadfpstate(fpproc->p_md.md_fpstate); + } +#if 0 + /* We will not alter the machine's running state until we get everything else working */ + *(struct frame *)tf->tf_out[6] = ddb_regs.ddb_fr; +#endif + *tf = ddb_regs.ddb_tf; + trap_trace_dis--; + doing_shutdown--; + + return (1); +} + +/* + * Read bytes from kernel address space for debugger. + */ +void +db_read_bytes(addr, size, data) + vaddr_t addr; + register size_t size; + register char *data; +{ + register char *src; + + src = (char *)addr; + while (size-- > 0) { + if (src >= (char *)VM_MIN_KERNEL_ADDRESS) + *data++ = probeget((paddr_t)(u_long)src++, ASI_P, 1); + else + *data++ = fubyte(src++); + } +} + + +/* + * Write bytes to kernel address space for debugger. + */ +void +db_write_bytes(addr, size, data) + vaddr_t addr; + register size_t size; + register char *data; +{ + register char *dst; + extern vaddr_t ktext; + extern paddr_t ktextp; + + dst = (char *)addr; + while (size-- > 0) { + if ((dst >= (char *)VM_MIN_KERNEL_ADDRESS+0x400000)) + *dst = *data; + else if ((dst >= (char *)VM_MIN_KERNEL_ADDRESS) && + (dst < (char *)VM_MIN_KERNEL_ADDRESS+0x400000)) + /* Read Only mapping -- need to do a bypass access */ + stba((u_long)dst - ktext + ktextp, ASI_PHYS_CACHED, *data); + else + subyte(dst, *data); + dst++, data++; + } + +} + +void +Debugger() +{ + /* We use the breakpoint to trap into DDB */ + asm("ta 1; nop"); +} + +void +db_prom_cmd(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ + OF_enter(); +} + +void +db_dump_dtlb(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ + extern void print_dtlb __P((void)); + + if (have_addr) { + int i; + int64_t* p = (int64_t*)addr; + static int64_t buf[128]; + extern void dump_dtlb(int64_t *); + + dump_dtlb(buf); + p = buf; + for (i=0; i<64;) { +#ifdef __arch64__ + db_printf("%2d:%16.16lx %16.16lx ", i++, p[0], p[1]); + p += 2; + db_printf("%2d:%16.16lx %16.16lx\n", i++, p[0], p[1]); + p += 2; +#else + db_printf("%2d:%16.16qx %16.16qx ", i++, p[0], p[1]); + p += 2; + db_printf("%2d:%16.16qx %16.16qx\n", i++, p[0], p[1]); + p += 2; +#endif + } + } else { +#ifdef DEBUG + print_dtlb(); +#endif + } +} + +void +db_pload_cmd(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ + static paddr_t oldaddr = -1; + int asi = ASI_PHYS_CACHED; + + if (!have_addr) { + addr = oldaddr; + } + if (addr == -1) { + db_printf("no address\n"); + return; + } + addr &= ~0x7; /* align */ + { + register char c, *cp = modif; + while ((c = *cp++) != 0) + if (c == 'u') + asi = ASI_AIUS; + } + while (count--) { + if (db_print_position() == 0) { + /* Always print the address. */ + db_printf("%16.16lx:\t", addr); + } + oldaddr=addr; + db_printf("%8.8lx\n", (long)ldxa(addr, asi)); + addr += 8; + if (db_print_position() != 0) + db_end_line(0); + } +} + +int64_t pseg_get __P((struct pmap *, vaddr_t)); + +void +db_dump_pmap(pm) +struct pmap* pm; +{ + /* print all valid pages in the kernel pmap */ + long i, j, k, n; + paddr_t *pdir, *ptbl; + /* Almost the same as pmap_collect() */ + + n = 0; + for (i=0; i<STSZ; i++) { + if((pdir = (paddr_t *)(u_long)ldxa((vaddr_t)&pm->pm_segs[i], ASI_PHYS_CACHED))) { + db_printf("pdir %ld at %lx:\n", i, (long)pdir); + for (k=0; k<PDSZ; k++) { + if ((ptbl = (paddr_t *)(u_long)ldxa((vaddr_t)&pdir[k], ASI_PHYS_CACHED))) { + db_printf("\tptable %ld:%ld at %lx:\n", i, k, (long)ptbl); + for (j=0; j<PTSZ; j++) { + int64_t data0, data1; + data0 = ldxa((vaddr_t)&ptbl[j], ASI_PHYS_CACHED); + j++; + data1 = ldxa((vaddr_t)&ptbl[j], ASI_PHYS_CACHED); + if (data0 || data1) { + db_printf("%llx: %llx\t", + (unsigned long long)(((u_int64_t)i<<STSHIFT)|(k<<PDSHIFT)|((j-1)<<PTSHIFT)), + (unsigned long long)(data0)); + db_printf("%llx: %llx\n", + (unsigned long long)(((u_int64_t)i<<STSHIFT)|(k<<PDSHIFT)|(j<<PTSHIFT)), + (unsigned long long)(data1)); + } + } + } + } + } + } +} + +void +db_pmap_kernel(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ + extern struct pmap kernel_pmap_; + int i, j, full = 0; + u_int64_t data; + + { + register char c, *cp = modif; + while ((c = *cp++) != 0) + if (c == 'f') + full = 1; + } + if (have_addr) { + /* lookup an entry for this VA */ + + if ((data = pseg_get(&kernel_pmap_, (vaddr_t)addr))) { + db_printf("pmap_kernel(%p)->pm_segs[%lx][%lx][%lx]=>%qx\n", + (void *)addr, (u_long)va_to_seg(addr), + (u_long)va_to_dir(addr), (u_long)va_to_pte(addr), + (unsigned long long)data); + } else { + db_printf("No mapping for %p\n", (void *)addr); + } + return; + } + + db_printf("pmap_kernel(%p) psegs %p phys %llx\n", + &kernel_pmap_, kernel_pmap_.pm_segs, + (unsigned long long)kernel_pmap_.pm_physaddr); + if (full) { + db_dump_pmap(&kernel_pmap_); + } else { + for (j=i=0; i<STSZ; i++) { + long seg = (long)ldxa((vaddr_t)&kernel_pmap_.pm_segs[i], ASI_PHYS_CACHED); + if (seg) + db_printf("seg %d => %lx%c", i, seg, (j++%4)?'\t':'\n'); + } + } +} + + +void +db_pmap_cmd(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ + struct pmap* pm=NULL; + int i, j=0, full = 0; + + { + register char c, *cp = modif; + if (modif) + while ((c = *cp++) != 0) + if (c == 'f') + full = 1; + } + if (curproc && curproc->p_vmspace) + pm = curproc->p_vmspace->vm_map.pmap; + if (have_addr) { + pm = (struct pmap*)addr; + } + + db_printf("pmap %p: ctx %x refs %d physaddr %llx psegs %p\n", + pm, pm->pm_ctx, pm->pm_refs, + (unsigned long long)pm->pm_physaddr, pm->pm_segs); + + if (full) { + db_dump_pmap(pm); + } else { + for (i=0; i<STSZ; i++) { + long seg = (long)ldxa((vaddr_t)&kernel_pmap_.pm_segs[i], ASI_PHYS_CACHED); + if (seg) + db_printf("seg %d => %lx%c", i, seg, (j++%4)?'\t':'\n'); + } + } +} + + +void +db_lock(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ +#if 0 + struct lock *l; + + if (!have_addr) { + db_printf("What lock address?\n"); + return; + } + + l = (struct lock *)addr; + db_printf("interlock=%x flags=%x\n waitcount=%x sharecount=%x " + "exclusivecount=%x\n wmesg=%s recurselevel=%x\n", + l->lk_interlock.lock_data, l->lk_flags, l->lk_waitcount, + l->lk_sharecount, l->lk_exclusivecount, l->lk_wmesg, + l->lk_recurselevel); +#else + db_printf("locks unsupported\n"); +#endif +} + +void +db_dump_dtsb(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ + extern pte_t *tsb; + extern int tsbsize; +#define TSBENTS (512<<tsbsize) + int i; + + db_printf("TSB:\n"); + for (i=0; i<TSBENTS; i++) { + db_printf("%4d:%4d:%08x %08x:%08x ", i, + (int)((tsb[i].tag&TSB_TAG_G)?-1:TSB_TAG_CTX(tsb[i].tag)), + (int)((i<<13)|TSB_TAG_VA(tsb[i].tag)), + (int)(tsb[i].data>>32), (int)tsb[i].data); + i++; + db_printf("%4d:%4d:%08x %08x:%08x\n", i, + (int)((tsb[i].tag&TSB_TAG_G)?-1:TSB_TAG_CTX(tsb[i].tag)), + (int)((i<<13)|TSB_TAG_VA(tsb[i].tag)), + (int)(tsb[i].data>>32), (int)tsb[i].data); + } +} + +void db_page_cmd __P((db_expr_t, int, db_expr_t, char *)); +void +db_page_cmd(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ + + if (!have_addr) { + db_printf("Need paddr for page\n"); + return; + } + + db_printf("pa %llx pg %p\n", (unsigned long long)addr, + PHYS_TO_VM_PAGE(addr)); +} + + +void +db_proc_cmd(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ + struct proc *p; + + p = curproc; + if (have_addr) + p = (struct proc*) addr; + if (p == NULL) { + db_printf("no current process\n"); + return; + } + db_printf("process %p:", p); + db_printf("pid:%d vmspace:%p pmap:%p ctx:%x wchan:%p pri:%d upri:%d\n", + p->p_pid, p->p_vmspace, p->p_vmspace->vm_map.pmap, + p->p_vmspace->vm_map.pmap->pm_ctx, + p->p_wchan, p->p_priority, p->p_usrpri); + db_printf("maxsaddr:%p ssiz:%dpg or %llxB\n", + p->p_vmspace->vm_maxsaddr, p->p_vmspace->vm_ssize, + (unsigned long long)ctob(p->p_vmspace->vm_ssize)); + db_printf("profile timer: %ld sec %ld usec\n", + p->p_stats->p_timer[ITIMER_PROF].it_value.tv_sec, + p->p_stats->p_timer[ITIMER_PROF].it_value.tv_usec); + db_printf("pcb: %p fpstate: %p\n", &p->p_addr->u_pcb, + p->p_md.md_fpstate); + return; +} + +void +db_ctx_cmd(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ + struct proc *p; + + /* XXX LOCKING XXX */ + for (p = allproc.lh_first; p != 0; p = p->p_list.le_next) { + if (p->p_stat) { + db_printf("process %p:", p); + db_printf("pid:%d pmap:%p ctx:%x tf:%p fpstate %p " + "lastcall:%s\n", + p->p_pid, p->p_vmspace->vm_map.pmap, + p->p_vmspace->vm_map.pmap->pm_ctx, + p->p_md.md_tf, p->p_md.md_fpstate, + (p->p_addr->u_pcb.lastcall)? + p->p_addr->u_pcb.lastcall : "Null"); + } + } + return; +} + +void +db_dump_pcb(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ + extern struct pcb *cpcb; + struct pcb *pcb; + int i; + + pcb = cpcb; + if (have_addr) + pcb = (struct pcb*) addr; + + db_printf("pcb@%p sp:%p pc:%p cwp:%d pil:%d nsaved:%x onfault:%p\nlastcall:%s\nfull windows:\n", + pcb, (void *)(long)pcb->pcb_sp, (void *)(long)pcb->pcb_pc, pcb->pcb_cwp, + pcb->pcb_pil, pcb->pcb_nsaved, (void *)pcb->pcb_onfault, + (pcb->lastcall)?pcb->lastcall:"Null"); + + for (i=0; i<pcb->pcb_nsaved; i++) { + db_printf("win %d: at %llx local, in\n", i, + (unsigned long long)pcb->pcb_rw[i+1].rw_in[6]); + db_printf("%16llx %16llx %16llx %16llx\n", + (unsigned long long)pcb->pcb_rw[i].rw_local[0], + (unsigned long long)pcb->pcb_rw[i].rw_local[1], + (unsigned long long)pcb->pcb_rw[i].rw_local[2], + (unsigned long long)pcb->pcb_rw[i].rw_local[3]); + db_printf("%16llx %16llx %16llx %16llx\n", + (unsigned long long)pcb->pcb_rw[i].rw_local[4], + (unsigned long long)pcb->pcb_rw[i].rw_local[5], + (unsigned long long)pcb->pcb_rw[i].rw_local[6], + (unsigned long long)pcb->pcb_rw[i].rw_local[7]); + db_printf("%16llx %16llx %16llx %16llx\n", + (unsigned long long)pcb->pcb_rw[i].rw_in[0], + (unsigned long long)pcb->pcb_rw[i].rw_in[1], + (unsigned long long)pcb->pcb_rw[i].rw_in[2], + (unsigned long long)pcb->pcb_rw[i].rw_in[3]); + db_printf("%16llx %16llx %16llx %16llx\n", + (unsigned long long)pcb->pcb_rw[i].rw_in[4], + (unsigned long long)pcb->pcb_rw[i].rw_in[5], + (unsigned long long)pcb->pcb_rw[i].rw_in[6], + (unsigned long long)pcb->pcb_rw[i].rw_in[7]); + } +} + + +void +db_setpcb(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ + struct proc *p, *pp; + + extern struct pcb *cpcb; + + if (!have_addr) { + db_printf("What PID do you want to map in?\n"); + return; + } + + for (p = allproc.lh_first; p != 0; p = p->p_list.le_next) { + pp = p->p_pptr; + if (p->p_stat && p->p_pid == addr) { + curproc = p; + cpcb = (struct pcb*)p->p_addr; + if (p->p_vmspace->vm_map.pmap->pm_ctx) { + switchtoctx(p->p_vmspace->vm_map.pmap->pm_ctx); + return; + } + db_printf("PID %ld has a null context.\n", addr); + return; + } + } + db_printf("PID %ld not found.\n", addr); +} + +static void +db_print_trace_entry(te, i) + struct traptrace *te; + int i; +{ + db_printf("%d:%d p:%d tt:%d:%llx:%llx %llx:%llx ", i, + (int)te->tl, (int)te->pid, + (int)te->tt, (unsigned long long)te->tstate, + (unsigned long long)te->tfault, (unsigned long long)te->tsp, + (unsigned long long)te->tpc); + db_printsym((u_long)te->tpc, DB_STGY_PROC); + db_printf(": "); + if ((te->tpc && !(te->tpc&0x3)) && + curproc && + (curproc->p_pid == te->pid)) { + db_disasm((u_long)te->tpc, 0); + } else db_printf("\n"); +} + +void +db_traptrace(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ + int i, start = 0, full = 0, reverse = 0; + struct traptrace *end; + + start = 0; + end = &trap_trace_end[0]; + + { + register char c, *cp = modif; + if (modif) + while ((c = *cp++) != 0) { + if (c == 'f') + full = 1; + if (c == 'r') + reverse = 1; + } + } + + if (have_addr) { + start = addr / (sizeof (struct traptrace)); + if (&trap_trace[start] > &trap_trace_end[0]) { + db_printf("Address out of range.\n"); + return; + } + if (!full) end = &trap_trace[start+1]; + } + + db_printf("#:tl p:pid tt:tt:tstate:tfault sp:pc\n"); + if (reverse) { + if (full && start) + for (i=start; --i;) { + db_print_trace_entry(&trap_trace[i], i); + } + i = (end - &trap_trace[0]); + while(--i > start) { + db_print_trace_entry(&trap_trace[i], i); + } + } else { + for (i=start; &trap_trace[i] < end ; i++) { + db_print_trace_entry(&trap_trace[i], i); + } + if (full && start) + for (i=0; i < start ; i++) { + db_print_trace_entry(&trap_trace[i], i); + } + } +} + +/* + * Use physical or virtual watchpoint registers -- ugh + */ +void +db_watch(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ + int phys = 0; + +#define WATCH_VR (1L<<22) +#define WATCH_VW (1L<<21) +#define WATCH_PR (1L<<24) +#define WATCH_PW (1L<<23) +#define WATCH_PM (((u_int64_t)0xffffL)<<33) +#define WATCH_VM (((u_int64_t)0xffffL)<<25) + + { + register char c, *cp = modif; + if (modif) + while ((c = *cp++) != 0) + if (c == 'p') + phys = 1; + } + if (have_addr) { + /* turn on the watchpoint */ + int64_t tmp = ldxa(0, ASI_MCCR); + + if (phys) { + tmp &= ~(WATCH_PM|WATCH_PR|WATCH_PW); + stxa(PHYSICAL_WATCHPOINT, ASI_DMMU, addr); + } else { + tmp &= ~(WATCH_VM|WATCH_VR|WATCH_VW); + stxa(VIRTUAL_WATCHPOINT, ASI_DMMU, addr); + } + stxa(0, ASI_MCCR, tmp); + } else { + /* turn off the watchpoint */ + int64_t tmp = ldxa(0, ASI_MCCR); + if (phys) tmp &= ~(WATCH_PM); + else tmp &= ~(WATCH_VM); + stxa(0, ASI_MCCR, tmp); + } +} + + +#include <uvm/uvm.h> + +void db_uvmhistdump __P((db_expr_t, int, db_expr_t, char *)); +extern void uvmhist_dump __P((struct uvm_history *)); +extern struct uvm_history_head uvm_histories; + +void +db_uvmhistdump(addr, have_addr, count, modif) + db_expr_t addr; + int have_addr; + db_expr_t count; + char *modif; +{ + + uvmhist_dump(uvm_histories.lh_first); +} + +#if NESP_SBUS +extern void db_esp(db_expr_t, int, db_expr_t, char*); +#endif + +const struct db_command db_machine_command_table[] = { + { "ctx", db_ctx_cmd, 0, 0 }, + { "dtlb", db_dump_dtlb, 0, 0 }, + { "dtsb", db_dump_dtsb, 0, 0 }, +#if NESP_SBUS + { "esp", db_esp, 0, 0 }, +#endif + { "fpstate", db_dump_fpstate,0, 0 }, + { "kmap", db_pmap_kernel, 0, 0 }, + { "lock", db_lock, 0, 0 }, + { "pcb", db_dump_pcb, 0, 0 }, + { "pctx", db_setpcb, 0, 0 }, + { "page", db_page_cmd, 0, 0 }, + { "phys", db_pload_cmd, 0, 0 }, + { "pmap", db_pmap_cmd, 0, 0 }, + { "proc", db_proc_cmd, 0, 0 }, + { "prom", db_prom_cmd, 0, 0 }, + { "pv", db_dump_pv, 0, 0 }, + { "stack", db_dump_stack, 0, 0 }, + { "tf", db_dump_trap, 0, 0 }, + { "ts", db_dump_ts, 0, 0 }, + { "traptrace", db_traptrace, 0, 0 }, + { "uvmdump", db_uvmhistdump, 0, 0 }, + { "watch", db_watch, 0, 0 }, + { "window", db_dump_window, 0, 0 }, + { (char *)0, } +}; + +/* + * support for SOFTWARE_SSTEP: + * return the next pc if the given branch is taken. + * + * note: in the case of conditional branches with annul, + * this actually returns the next pc in the "not taken" path, + * but in that case next_instr_address() will return the + * next pc in the "taken" path. so even tho the breakpoints + * are backwards, everything will still work, and the logic is + * much simpler this way. + */ +db_addr_t +db_branch_taken(inst, pc, regs) + int inst; + db_addr_t pc; + db_regs_t *regs; +{ + union instr insn; + db_addr_t npc = ddb_regs.ddb_tf.tf_npc; + + insn.i_int = inst; + + /* + * if this is not an annulled conditional branch, the next pc is "npc". + */ + + if (insn.i_any.i_op != IOP_OP2 || insn.i_branch.i_annul != 1) + return npc; + + switch (insn.i_op2.i_op2) { + case IOP2_Bicc: + case IOP2_FBfcc: + case IOP2_BPcc: + case IOP2_FBPfcc: + case IOP2_CBccc: + /* branch on some condition-code */ + switch (insn.i_branch.i_cond) + { + case Icc_A: /* always */ + return pc + ((inst << 10) >> 8); + + default: /* all other conditions */ + return npc + 4; + } + + case IOP2_BPr: + /* branch on register, always conditional */ + return npc + 4; + + default: + /* not a branch */ + panic("branch_taken() on non-branch"); + } +} + +boolean_t +db_inst_branch(inst) + int inst; +{ + union instr insn; + + insn.i_int = inst; + + if (insn.i_any.i_op != IOP_OP2) + return FALSE; + + switch (insn.i_op2.i_op2) { + case IOP2_BPcc: + case IOP2_Bicc: + case IOP2_BPr: + case IOP2_FBPfcc: + case IOP2_FBfcc: + case IOP2_CBccc: + return TRUE; + + default: + return FALSE; + } +} + + +boolean_t +db_inst_call(inst) + int inst; +{ + union instr insn; + + insn.i_int = inst; + + switch (insn.i_any.i_op) { + case IOP_CALL: + return TRUE; + + case IOP_reg: + return (insn.i_op3.i_op3 == IOP3_JMPL) && !db_inst_return(inst); + + default: + return FALSE; + } +} + + +boolean_t +db_inst_unconditional_flow_transfer(inst) + int inst; +{ + union instr insn; + + insn.i_int = inst; + + if (db_inst_call(inst)) + return TRUE; + + if (insn.i_any.i_op != IOP_OP2) + return FALSE; + + switch (insn.i_op2.i_op2) + { + case IOP2_BPcc: + case IOP2_Bicc: + case IOP2_FBPfcc: + case IOP2_FBfcc: + case IOP2_CBccc: + return insn.i_branch.i_cond == Icc_A; + + default: + return FALSE; + } +} + + +boolean_t +db_inst_return(inst) + int inst; +{ + return (inst == I_JMPLri(I_G0, I_O7, 8) || /* ret */ + inst == I_JMPLri(I_G0, I_I7, 8)); /* retl */ +} + +boolean_t +db_inst_trap_return(inst) + int inst; +{ + union instr insn; + + insn.i_int = inst; + + return (insn.i_any.i_op == IOP_reg && + insn.i_op3.i_op3 == IOP3_RETT); +} + + +int +db_inst_load(inst) + int inst; +{ + union instr insn; + + insn.i_int = inst; + + if (insn.i_any.i_op != IOP_mem) + return 0; + + switch (insn.i_op3.i_op3) { + case IOP3_LD: + case IOP3_LDUB: + case IOP3_LDUH: + case IOP3_LDD: + case IOP3_LDSB: + case IOP3_LDSH: + case IOP3_LDSTUB: + case IOP3_SWAP: + case IOP3_LDA: + case IOP3_LDUBA: + case IOP3_LDUHA: + case IOP3_LDDA: + case IOP3_LDSBA: + case IOP3_LDSHA: + case IOP3_LDSTUBA: + case IOP3_SWAPA: + case IOP3_LDF: + case IOP3_LDFSR: + case IOP3_LDDF: + case IOP3_LFC: + case IOP3_LDCSR: + case IOP3_LDDC: + return 1; + + default: + return 0; + } +} + +int +db_inst_store(inst) + int inst; +{ + union instr insn; + + insn.i_int = inst; + + if (insn.i_any.i_op != IOP_mem) + return 0; + + switch (insn.i_op3.i_op3) { + case IOP3_ST: + case IOP3_STB: + case IOP3_STH: + case IOP3_STD: + case IOP3_LDSTUB: + case IOP3_SWAP: + case IOP3_STA: + case IOP3_STBA: + case IOP3_STHA: + case IOP3_STDA: + case IOP3_LDSTUBA: + case IOP3_SWAPA: + case IOP3_STF: + case IOP3_STFSR: + case IOP3_STDFQ: + case IOP3_STDF: + case IOP3_STC: + case IOP3_STCSR: + case IOP3_STDCQ: + case IOP3_STDC: + return 1; + + default: + return 0; + } +} |