diff options
Diffstat (limited to 'arch/um/sys-i386/sysrq.c')
| -rw-r--r-- | arch/um/sys-i386/sysrq.c | 80 |
1 files changed, 79 insertions, 1 deletions
diff --git a/arch/um/sys-i386/sysrq.c b/arch/um/sys-i386/sysrq.c index 281fc7b8ca00..e3706d15c4f5 100644 --- a/arch/um/sys-i386/sysrq.c +++ b/arch/um/sys-i386/sysrq.c | |||
| @@ -3,12 +3,15 @@ | |||
| 3 | * Licensed under the GPL | 3 | * Licensed under the GPL |
| 4 | */ | 4 | */ |
| 5 | 5 | ||
| 6 | #include "linux/config.h" | ||
| 6 | #include "linux/kernel.h" | 7 | #include "linux/kernel.h" |
| 7 | #include "linux/smp.h" | 8 | #include "linux/smp.h" |
| 8 | #include "linux/sched.h" | 9 | #include "linux/sched.h" |
| 10 | #include "linux/kallsyms.h" | ||
| 9 | #include "asm/ptrace.h" | 11 | #include "asm/ptrace.h" |
| 10 | #include "sysrq.h" | 12 | #include "sysrq.h" |
| 11 | 13 | ||
| 14 | /* This is declared by <linux/sched.h> */ | ||
| 12 | void show_regs(struct pt_regs *regs) | 15 | void show_regs(struct pt_regs *regs) |
| 13 | { | 16 | { |
| 14 | printk("\n"); | 17 | printk("\n"); |
| @@ -31,5 +34,80 @@ void show_regs(struct pt_regs *regs) | |||
| 31 | 0xffff & PT_REGS_DS(regs), | 34 | 0xffff & PT_REGS_DS(regs), |
| 32 | 0xffff & PT_REGS_ES(regs)); | 35 | 0xffff & PT_REGS_ES(regs)); |
| 33 | 36 | ||
| 34 | show_trace((unsigned long *) ®s); | 37 | show_trace(NULL, (unsigned long *) ®s); |
| 35 | } | 38 | } |
| 39 | |||
| 40 | /* Copied from i386. */ | ||
| 41 | static inline int valid_stack_ptr(struct thread_info *tinfo, void *p) | ||
| 42 | { | ||
| 43 | return p > (void *)tinfo && | ||
| 44 | p < (void *)tinfo + THREAD_SIZE - 3; | ||
| 45 | } | ||
| 46 | |||
| 47 | /* Adapted from i386 (we also print the address we read from). */ | ||
| 48 | static inline unsigned long print_context_stack(struct thread_info *tinfo, | ||
| 49 | unsigned long *stack, unsigned long ebp) | ||
| 50 | { | ||
| 51 | unsigned long addr; | ||
| 52 | |||
| 53 | #ifdef CONFIG_FRAME_POINTER | ||
| 54 | while (valid_stack_ptr(tinfo, (void *)ebp)) { | ||
| 55 | addr = *(unsigned long *)(ebp + 4); | ||
| 56 | printk("%08lx: [<%08lx>]", ebp + 4, addr); | ||
| 57 | print_symbol(" %s", addr); | ||
| 58 | printk("\n"); | ||
| 59 | ebp = *(unsigned long *)ebp; | ||
| 60 | } | ||
| 61 | #else | ||
| 62 | while (valid_stack_ptr(tinfo, stack)) { | ||
| 63 | addr = *stack; | ||
| 64 | if (__kernel_text_address(addr)) { | ||
| 65 | printk("%08lx: [<%08lx>]", (unsigned long) stack, addr); | ||
| 66 | print_symbol(" %s", addr); | ||
| 67 | printk("\n"); | ||
| 68 | } | ||
| 69 | stack++; | ||
| 70 | } | ||
| 71 | #endif | ||
| 72 | return ebp; | ||
| 73 | } | ||
| 74 | |||
| 75 | void show_trace(struct task_struct* task, unsigned long * stack) | ||
| 76 | { | ||
| 77 | unsigned long ebp; | ||
| 78 | struct thread_info *context; | ||
| 79 | |||
| 80 | /* Turn this into BUG_ON if possible. */ | ||
| 81 | if (!stack) { | ||
| 82 | stack = (unsigned long*) &stack; | ||
| 83 | printk("show_trace: got NULL stack, implicit assumption task == current"); | ||
| 84 | WARN_ON(1); | ||
| 85 | } | ||
| 86 | |||
| 87 | if (!task) | ||
| 88 | task = current; | ||
| 89 | |||
| 90 | if (task != current) { | ||
| 91 | //ebp = (unsigned long) KSTK_EBP(task); | ||
| 92 | /* Which one? No actual difference - just coding style.*/ | ||
| 93 | ebp = (unsigned long) PT_REGS_EBP(&task->thread.regs); | ||
| 94 | } else { | ||
| 95 | asm ("movl %%ebp, %0" : "=r" (ebp) : ); | ||
| 96 | } | ||
| 97 | |||
| 98 | context = (struct thread_info *) | ||
| 99 | ((unsigned long)stack & (~(THREAD_SIZE - 1))); | ||
| 100 | print_context_stack(context, stack, ebp); | ||
| 101 | |||
| 102 | /*while (((long) stack & (THREAD_SIZE-1)) != 0) { | ||
| 103 | addr = *stack; | ||
| 104 | if (__kernel_text_address(addr)) { | ||
| 105 | printk("%08lx: [<%08lx>]", (unsigned long) stack, addr); | ||
| 106 | print_symbol(" %s", addr); | ||
| 107 | printk("\n"); | ||
| 108 | } | ||
| 109 | stack++; | ||
| 110 | }*/ | ||
| 111 | printk("\n"); | ||
| 112 | } | ||
| 113 | |||
