/* * Copyright (C) 2001 - 2003 Jeff Dike (jdike@addtoit.com) * Licensed under the GPL */ #include "linux/kernel.h" #include "linux/smp.h" #include "linux/sched.h" #include "linux/kallsyms.h" #include "asm/ptrace.h" #include "sysrq.h" /* This is declared by */ void show_regs(struct pt_regs *regs) { printk("\n"); printk("EIP: %04lx:[<%08lx>] CPU: %d %s", 0xffff & PT_REGS_CS(regs), PT_REGS_IP(regs), smp_processor_id(), print_tainted()); if (PT_REGS_CS(regs) & 3) printk(" ESP: %04lx:%08lx", 0xffff & PT_REGS_SS(regs), PT_REGS_SP(regs)); printk(" EFLAGS: %08lx\n %s\n", PT_REGS_EFLAGS(regs), print_tainted()); printk("EAX: %08lx EBX: %08lx ECX: %08lx EDX: %08lx\n", PT_REGS_EAX(regs), PT_REGS_EBX(regs), PT_REGS_ECX(regs), PT_REGS_EDX(regs)); printk("ESI: %08lx EDI: %08lx EBP: %08lx", PT_REGS_ESI(regs), PT_REGS_EDI(regs), PT_REGS_EBP(regs)); printk(" DS: %04lx ES: %04lx\n", 0xffff & PT_REGS_DS(regs), 0xffff & PT_REGS_ES(regs)); show_trace(NULL, (unsigned long *) ®s); } /* Copied from i386. */ static inline int valid_stack_ptr(struct thread_info *tinfo, void *p) { return p > (void *)tinfo && p < (void *)tinfo + THREAD_SIZE - 3; } /* Adapted from i386 (we also print the address we read from). */ static inline unsigned long print_context_stack(struct thread_info *tinfo, unsigned long *stack, unsigned long ebp) { unsigned long addr; #ifdef CONFIG_FRAME_POINTER while (valid_stack_ptr(tinfo, (void *)ebp)) { addr = *(unsigned long *)(ebp + 4); printk("%08lx: [<%08lx>]", ebp + 4, addr); print_symbol(" %s", addr); printk("\n"); ebp = *(unsigned long *)ebp; } #else while (valid_stack_ptr(tinfo, stack)) { addr = *stack; if (__kernel_text_address(addr)) { printk("%08lx: [<%08lx>]", (unsigned long) stack, addr); print_symbol(" %s", addr); printk("\n"); } stack++; } #endif return ebp; } void show_trace(struct task_struct* task, unsigned long * stack) { unsigned long ebp; struct thread_info *context; /* Turn this into BUG_ON if possible. */ if (!stack) { stack = (unsigned long*) &stack; printk("show_trace: got NULL stack, implicit assumption task == current"); WARN_ON(1); } if (!task) task = current; if (task != current) { ebp = (unsigned long) KSTK_EBP(task); } else { asm ("movl %%ebp, %0" : "=r" (ebp) : ); } context = (struct thread_info *) ((unsigned long)stack & (~(THREAD_SIZE - 1))); print_context_stack(context, stack, ebp); printk("\n"); }