⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 xmon.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 4 页
字号:
{	struct bpt *bp;	a &= ~3UL;	bp = at_breakpoint(a);	if (bp)		return bp;	for (bp = bpts; bp < &bpts[NBPTS]; ++bp) {		if (!bp->enabled && atomic_read(&bp->ref_count) == 0) {			bp->address = a;			bp->instr[1] = bpinstr;			store_inst(&bp->instr[1]);			return bp;		}	}	printf("Sorry, no free breakpoints.  Please clear one first.\n");	return NULL;}static void insert_bpts(void){	int i;	struct bpt *bp;	bp = bpts;	for (i = 0; i < NBPTS; ++i, ++bp) {		if ((bp->enabled & (BP_TRAP|BP_IABR)) == 0)			continue;		if (mread(bp->address, &bp->instr[0], 4) != 4) {			printf("Couldn't read instruction at %lx, "			       "disabling breakpoint there\n", bp->address);			bp->enabled = 0;			continue;		}		if (IS_MTMSRD(bp->instr[0]) || IS_RFID(bp->instr[0])) {			printf("Breakpoint at %lx is on an mtmsrd or rfid "			       "instruction, disabling it\n", bp->address);			bp->enabled = 0;			continue;		}		store_inst(&bp->instr[0]);		if (bp->enabled & BP_IABR)			continue;		if (mwrite(bp->address, &bpinstr, 4) != 4) {			printf("Couldn't write instruction at %lx, "			       "disabling breakpoint there\n", bp->address);			bp->enabled &= ~BP_TRAP;			continue;		}		store_inst((void *)bp->address);	}}static void insert_cpu_bpts(void){	if (dabr.enabled)		set_dabr(dabr.address | (dabr.enabled & 7));	if (iabr && cpu_has_feature(CPU_FTR_IABR))		mtspr(SPRN_IABR, iabr->address			 | (iabr->enabled & (BP_IABR|BP_IABR_TE)));}static void remove_bpts(void){	int i;	struct bpt *bp;	unsigned instr;	bp = bpts;	for (i = 0; i < NBPTS; ++i, ++bp) {		if ((bp->enabled & (BP_TRAP|BP_IABR)) != BP_TRAP)			continue;		if (mread(bp->address, &instr, 4) == 4		    && instr == bpinstr		    && mwrite(bp->address, &bp->instr, 4) != 4)			printf("Couldn't remove breakpoint at %lx\n",			       bp->address);		else			store_inst((void *)bp->address);	}}static void remove_cpu_bpts(void){	set_dabr(0);	if (cpu_has_feature(CPU_FTR_IABR))		mtspr(SPRN_IABR, 0);}/* Command interpreting routine */static char *last_cmd;static intcmds(struct pt_regs *excp){	int cmd = 0;	last_cmd = NULL;	xmon_regs = excp;	for(;;) {#ifdef CONFIG_SMP		printf("%x:", smp_processor_id());#endif /* CONFIG_SMP */		printf("mon> ");		flush_input();		termch = 0;		cmd = skipbl();		if( cmd == '\n' ) {			if (last_cmd == NULL)				continue;			take_input(last_cmd);			last_cmd = NULL;			cmd = inchar();		}		switch (cmd) {		case 'm':			cmd = inchar();			switch (cmd) {			case 'm':			case 's':			case 'd':				memops(cmd);				break;			case 'l':				memlocate();				break;			case 'z':				memzcan();				break;			case 'i':				show_mem();				break;			default:				termch = cmd;				memex();			}			break;		case 'd':			dump();			break;		case 'l':			symbol_lookup();			break;		case 'r':			prregs(excp);	/* print regs */			break;		case 'e':			excprint(excp);			break;		case 'S':			super_regs();			break;		case 't':			backtrace(excp);			break;		case 'f':			cacheflush();			break;		case 's':			if (do_step(excp))				return cmd;			break;		case 'x':		case 'X':		case EOF:			return cmd;		case '?':			printf(help_string);			break;		case 'b':			bpt_cmds();			break;		case 'C':			csum();			break;		case 'c':			if (cpu_cmd())				return 0;			break;		case 'z':			bootcmds();			break;		case 'p':			proccall();			break;#ifdef CONFIG_PPC_STD_MMU		case 'u':			dump_segments();			break;#endif		default:			printf("Unrecognized command: ");		        do {				if (' ' < cmd && cmd <= '~')					putchar(cmd);				else					printf("\\x%x", cmd);				cmd = inchar();		        } while (cmd != '\n'); 			printf(" (type ? for help)\n");			break;		}	}}/* * Step a single instruction. * Some instructions we emulate, others we execute with MSR_SE set. */static int do_step(struct pt_regs *regs){	unsigned int instr;	int stepped;	/* check we are in 64-bit kernel mode, translation enabled */	if ((regs->msr & (MSR_SF|MSR_PR|MSR_IR)) == (MSR_SF|MSR_IR)) {		if (mread(regs->nip, &instr, 4) == 4) {			stepped = emulate_step(regs, instr);			if (stepped < 0) {				printf("Couldn't single-step %s instruction\n",				       (IS_RFID(instr)? "rfid": "mtmsrd"));				return 0;			}			if (stepped > 0) {				regs->trap = 0xd00 | (regs->trap & 1);				printf("stepped to ");				xmon_print_symbol(regs->nip, " ", "\n");				ppc_inst_dump(regs->nip, 1, 0);				return 0;			}		}	}	regs->msr |= MSR_SE;	return 1;}static void bootcmds(void){	int cmd;	cmd = inchar();	if (cmd == 'r')		ppc_md.restart(NULL);	else if (cmd == 'h')		ppc_md.halt();	else if (cmd == 'p')		ppc_md.power_off();}static int cpu_cmd(void){#ifdef CONFIG_SMP	unsigned long cpu;	int timeout;	int count;	if (!scanhex(&cpu)) {		/* print cpus waiting or in xmon */		printf("cpus stopped:");		count = 0;		for (cpu = 0; cpu < NR_CPUS; ++cpu) {			if (cpu_isset(cpu, cpus_in_xmon)) {				if (count == 0)					printf(" %x", cpu);				++count;			} else {				if (count > 1)					printf("-%x", cpu - 1);				count = 0;			}		}		if (count > 1)			printf("-%x", NR_CPUS - 1);		printf("\n");		return 0;	}	/* try to switch to cpu specified */	if (!cpu_isset(cpu, cpus_in_xmon)) {		printf("cpu 0x%x isn't in xmon\n", cpu);		return 0;	}	xmon_taken = 0;	mb();	xmon_owner = cpu;	timeout = 10000000;	while (!xmon_taken) {		if (--timeout == 0) {			if (test_and_set_bit(0, &xmon_taken))				break;			/* take control back */			mb();			xmon_owner = smp_processor_id();			printf("cpu %u didn't take control\n", cpu);			return 0;		}		barrier();	}	return 1;#else	return 0;#endif /* CONFIG_SMP */}static unsigned short fcstab[256] = {	0x0000, 0x1189, 0x2312, 0x329b, 0x4624, 0x57ad, 0x6536, 0x74bf,	0x8c48, 0x9dc1, 0xaf5a, 0xbed3, 0xca6c, 0xdbe5, 0xe97e, 0xf8f7,	0x1081, 0x0108, 0x3393, 0x221a, 0x56a5, 0x472c, 0x75b7, 0x643e,	0x9cc9, 0x8d40, 0xbfdb, 0xae52, 0xdaed, 0xcb64, 0xf9ff, 0xe876,	0x2102, 0x308b, 0x0210, 0x1399, 0x6726, 0x76af, 0x4434, 0x55bd,	0xad4a, 0xbcc3, 0x8e58, 0x9fd1, 0xeb6e, 0xfae7, 0xc87c, 0xd9f5,	0x3183, 0x200a, 0x1291, 0x0318, 0x77a7, 0x662e, 0x54b5, 0x453c,	0xbdcb, 0xac42, 0x9ed9, 0x8f50, 0xfbef, 0xea66, 0xd8fd, 0xc974,	0x4204, 0x538d, 0x6116, 0x709f, 0x0420, 0x15a9, 0x2732, 0x36bb,	0xce4c, 0xdfc5, 0xed5e, 0xfcd7, 0x8868, 0x99e1, 0xab7a, 0xbaf3,	0x5285, 0x430c, 0x7197, 0x601e, 0x14a1, 0x0528, 0x37b3, 0x263a,	0xdecd, 0xcf44, 0xfddf, 0xec56, 0x98e9, 0x8960, 0xbbfb, 0xaa72,	0x6306, 0x728f, 0x4014, 0x519d, 0x2522, 0x34ab, 0x0630, 0x17b9,	0xef4e, 0xfec7, 0xcc5c, 0xddd5, 0xa96a, 0xb8e3, 0x8a78, 0x9bf1,	0x7387, 0x620e, 0x5095, 0x411c, 0x35a3, 0x242a, 0x16b1, 0x0738,	0xffcf, 0xee46, 0xdcdd, 0xcd54, 0xb9eb, 0xa862, 0x9af9, 0x8b70,	0x8408, 0x9581, 0xa71a, 0xb693, 0xc22c, 0xd3a5, 0xe13e, 0xf0b7,	0x0840, 0x19c9, 0x2b52, 0x3adb, 0x4e64, 0x5fed, 0x6d76, 0x7cff,	0x9489, 0x8500, 0xb79b, 0xa612, 0xd2ad, 0xc324, 0xf1bf, 0xe036,	0x18c1, 0x0948, 0x3bd3, 0x2a5a, 0x5ee5, 0x4f6c, 0x7df7, 0x6c7e,	0xa50a, 0xb483, 0x8618, 0x9791, 0xe32e, 0xf2a7, 0xc03c, 0xd1b5,	0x2942, 0x38cb, 0x0a50, 0x1bd9, 0x6f66, 0x7eef, 0x4c74, 0x5dfd,	0xb58b, 0xa402, 0x9699, 0x8710, 0xf3af, 0xe226, 0xd0bd, 0xc134,	0x39c3, 0x284a, 0x1ad1, 0x0b58, 0x7fe7, 0x6e6e, 0x5cf5, 0x4d7c,	0xc60c, 0xd785, 0xe51e, 0xf497, 0x8028, 0x91a1, 0xa33a, 0xb2b3,	0x4a44, 0x5bcd, 0x6956, 0x78df, 0x0c60, 0x1de9, 0x2f72, 0x3efb,	0xd68d, 0xc704, 0xf59f, 0xe416, 0x90a9, 0x8120, 0xb3bb, 0xa232,	0x5ac5, 0x4b4c, 0x79d7, 0x685e, 0x1ce1, 0x0d68, 0x3ff3, 0x2e7a,	0xe70e, 0xf687, 0xc41c, 0xd595, 0xa12a, 0xb0a3, 0x8238, 0x93b1,	0x6b46, 0x7acf, 0x4854, 0x59dd, 0x2d62, 0x3ceb, 0x0e70, 0x1ff9,	0xf78f, 0xe606, 0xd49d, 0xc514, 0xb1ab, 0xa022, 0x92b9, 0x8330,	0x7bc7, 0x6a4e, 0x58d5, 0x495c, 0x3de3, 0x2c6a, 0x1ef1, 0x0f78};#define FCS(fcs, c)	(((fcs) >> 8) ^ fcstab[((fcs) ^ (c)) & 0xff])static voidcsum(void){	unsigned int i;	unsigned short fcs;	unsigned char v;	if (!scanhex(&adrs))		return;	if (!scanhex(&ncsum))		return;	fcs = 0xffff;	for (i = 0; i < ncsum; ++i) {		if (mread(adrs+i, &v, 1) == 0) {			printf("csum stopped at %x\n", adrs+i);			break;		}		fcs = FCS(fcs, v);	}	printf("%x\n", fcs);}/* * Check if this is a suitable place to put a breakpoint. */static long check_bp_loc(unsigned long addr){	unsigned int instr;	addr &= ~3;	if (addr < KERNELBASE) {		printf("Breakpoints may only be placed at kernel addresses\n");		return 0;	}	if (!mread(addr, &instr, sizeof(instr))) {		printf("Can't read instruction at address %lx\n", addr);		return 0;	}	if (IS_MTMSRD(instr) || IS_RFID(instr)) {		printf("Breakpoints may not be placed on mtmsrd or rfid "		       "instructions\n");		return 0;	}	return 1;}static char *breakpoint_help_string =     "Breakpoint command usage:\n"    "b                show breakpoints\n"    "b <addr> [cnt]   set breakpoint at given instr addr\n"    "bc               clear all breakpoints\n"    "bc <n/addr>      clear breakpoint number n or at addr\n"    "bi <addr> [cnt]  set hardware instr breakpoint (POWER3/RS64 only)\n"    "bd <addr> [cnt]  set hardware data breakpoint\n"    "";static voidbpt_cmds(void){	int cmd;	unsigned long a;	int mode, i;	struct bpt *bp;	const char badaddr[] = "Only kernel addresses are permitted "		"for breakpoints\n";	cmd = inchar();	switch (cmd) {#ifndef CONFIG_8xx	case 'd':	/* bd - hardware data breakpoint */		mode = 7;		cmd = inchar();		if (cmd == 'r')			mode = 5;		else if (cmd == 'w')			mode = 6;		else			termch = cmd;		dabr.address = 0;		dabr.enabled = 0;		if (scanhex(&dabr.address)) {			if (dabr.address < KERNELBASE) {				printf(badaddr);				break;			}			dabr.address &= ~7;			dabr.enabled = mode | BP_DABR;		}		break;	case 'i':	/* bi - hardware instr breakpoint */		if (!cpu_has_feature(CPU_FTR_IABR)) {			printf("Hardware instruction breakpoint "			       "not supported on this cpu\n");			break;		}		if (iabr) {			iabr->enabled &= ~(BP_IABR | BP_IABR_TE);			iabr = NULL;		}		if (!scanhex(&a))			break;		if (!check_bp_loc(a))			break;		bp = new_breakpoint(a);		if (bp != NULL) {			bp->enabled |= BP_IABR | BP_IABR_TE;			iabr = bp;		}		break;#endif	case 'c':		if (!scanhex(&a)) {			/* clear all breakpoints */			for (i = 0; i < NBPTS; ++i)				bpts[i].enabled = 0;			iabr = NULL;			dabr.enabled = 0;			printf("All breakpoints cleared\n");			break;		}		if (a <= NBPTS && a >= 1) {			/* assume a breakpoint number */			bp = &bpts[a-1];	/* bp nums are 1 based */		} else {			/* assume a breakpoint address */			bp = at_breakpoint(a);			if (bp == 0) {				printf("No breakpoint at %x\n", a);				break;			}		}		printf("Cleared breakpoint %x (", BP_NUM(bp));		xmon_print_symbol(bp->address, " ", ")\n");		bp->enabled = 0;		break;	default:		termch = cmd;	        cmd = skipbl();		if (cmd == '?') {			printf(breakpoint_help_string);			break;		}		termch = cmd;		if (!scanhex(&a)) {			/* print all breakpoints */			printf("   type            address\n");			if (dabr.enabled) {				printf("   data   "REG"  [", dabr.address);				if (dabr.enabled & 1)					printf("r");				if (dabr.enabled & 2)					printf("w");				printf("]\n");			}			for (bp = bpts; bp < &bpts[NBPTS]; ++bp) {				if (!bp->enabled)					continue;				printf("%2x %s   ", BP_NUM(bp),				    (bp->enabled & BP_IABR)? "inst": "trap");				xmon_print_symbol(bp->address, "  ", "\n");			}			break;		}		if (!check_bp_loc(a))			break;		bp = new_breakpoint(a);		if (bp != NULL)			bp->enabled |= BP_TRAP;		break;	}}/* Very cheap human name for vector lookup. */staticconst char *getvecname(unsigned long vec){	char *ret;	switch (vec) {	case 0x100:	ret = "(System Reset)"; break;	case 0x200:	ret = "(Machine Check)"; break;	case 0x300:	ret = "(Data Access)"; break;	case 0x380:	ret = "(Data SLB Access)"; break;	case 0x400:	ret = "(Instruction Access)"; break;	case 0x480:	ret = "(Instruction SLB Access)"; break;	case 0x500:	ret = "(Hardware Interrupt)"; break;	case 0x600:	ret = "(Alignment)"; break;	case 0x700:	ret = "(Program Check)"; break;	case 0x800:	ret = "(FPU Unavailable)"; break;	case 0x900:	ret = "(Decrementer)"; break;	case 0xc00:	ret = "(System Call)"; break;	case 0xd00:	ret = "(Single Step)"; break;	case 0xf00:	ret = "(Performance Monitor)"; break;	case 0xf20:	ret = "(Altivec Unavailable)"; break;	case 0x1300:	ret = "(Instruction Breakpoint)"; break;	default: ret = "";	}	return ret;}static void get_function_bounds(unsigned long pc, unsigned long *startp,				unsigned long *endp){	unsigned long size, offset;	const char *name;	char *modname;	*startp = *endp = 0;	if (pc == 0)		return;	if (setjmp(bus_error_jmp) == 0) {		catch_memory_errors = 1;		sync();		name = kallsyms_lookup(pc, &size, &offset, &modname, tmpstr);		if (name != NULL) {			*startp = pc - offset;			*endp = pc - offset + size;		}		sync();	}	catch_memory_errors = 0;}static int xmon_depth_to_print = 64;#ifdef CONFIG_PPC64#define LRSAVE_OFFSET		0x10#define REG_FRAME_MARKER	0x7265677368657265ul	/* "regshere" */#define MARKER_OFFSET		0x60#define REGS_OFFSET		0x70#else#define LRSAVE_OFFSET		4#define REG_FRAME_MARKER	0x72656773#define MARKER_OFFSET		8#define REGS_OFFSET		16#endifstatic void xmon_show_stack(unsigned long sp, unsigned long lr,			    unsigned long pc){	unsigned long ip;	unsigned long newsp;	unsigned long marker;	int count = 0;	struct pt_regs regs;	do {		if (sp < PAGE_OFFSET) {			if (sp != 0)				printf("SP (%lx) is in userspace\n", sp);			break;		}		if (!mread(sp + LRSAVE_OFFSET, &ip, sizeof(unsigned long))		    || !mread(sp, &newsp, sizeof(unsigned long))) {			printf("Couldn't read stack frame at %lx\n", sp);			break;		}		/*		 * For the first stack frame, try to work out if		 * LR and/or the saved LR value in the bottommost		 * stack frame are valid.		 */		if ((pc | lr) != 0) {			unsigned long fnstart, fnend;			unsigned long nextip;			int printip = 1;			get_function_bounds(pc, &fnstart, &fnend);			nextip = 0;			if (newsp > sp)				mread(newsp + LRSAVE_OFFSET, &nextip,				      sizeof(unsigned long));			if (lr == ip) {				if (lr < PAGE_OFFSET				    || (fnstart <= lr && lr < fnend))					printip = 0;			} else if (lr == nextip) {				printip = 0;			} else if (lr >= PAGE_OFFSET				   && !(fnstart <= lr && lr < fnend)) {				printf("[link register   ] ");				xmon_print_symbol(lr, " ", "\n");			}			if (printip) {				printf("["REG"] ", sp);				xmon_print_symbol(ip, " ", " (unreliable)\n");			}			pc = lr = 0;		} else {			printf("["REG"] ", sp);

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -