sysrq.c 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  1. /*
  2. * Copyright (C) 2001 - 2003 Jeff Dike (jdike@addtoit.com)
  3. * Licensed under the GPL
  4. */
  5. #include "linux/config.h"
  6. #include "linux/kernel.h"
  7. #include "linux/smp.h"
  8. #include "linux/sched.h"
  9. #include "linux/kallsyms.h"
  10. #include "asm/ptrace.h"
  11. #include "sysrq.h"
  12. /* This is declared by <linux/sched.h> */
  13. void show_regs(struct pt_regs *regs)
  14. {
  15. printk("\n");
  16. printk("EIP: %04lx:[<%08lx>] CPU: %d %s",
  17. 0xffff & PT_REGS_CS(regs), PT_REGS_IP(regs),
  18. smp_processor_id(), print_tainted());
  19. if (PT_REGS_CS(regs) & 3)
  20. printk(" ESP: %04lx:%08lx", 0xffff & PT_REGS_SS(regs),
  21. PT_REGS_SP(regs));
  22. printk(" EFLAGS: %08lx\n %s\n", PT_REGS_EFLAGS(regs),
  23. print_tainted());
  24. printk("EAX: %08lx EBX: %08lx ECX: %08lx EDX: %08lx\n",
  25. PT_REGS_EAX(regs), PT_REGS_EBX(regs),
  26. PT_REGS_ECX(regs),
  27. PT_REGS_EDX(regs));
  28. printk("ESI: %08lx EDI: %08lx EBP: %08lx",
  29. PT_REGS_ESI(regs), PT_REGS_EDI(regs),
  30. PT_REGS_EBP(regs));
  31. printk(" DS: %04lx ES: %04lx\n",
  32. 0xffff & PT_REGS_DS(regs),
  33. 0xffff & PT_REGS_ES(regs));
  34. show_trace(NULL, (unsigned long *) &regs);
  35. }
  36. /* Copied from i386. */
  37. static inline int valid_stack_ptr(struct thread_info *tinfo, void *p)
  38. {
  39. return p > (void *)tinfo &&
  40. p < (void *)tinfo + THREAD_SIZE - 3;
  41. }
  42. /* Adapted from i386 (we also print the address we read from). */
  43. static inline unsigned long print_context_stack(struct thread_info *tinfo,
  44. unsigned long *stack, unsigned long ebp)
  45. {
  46. unsigned long addr;
  47. #ifdef CONFIG_FRAME_POINTER
  48. while (valid_stack_ptr(tinfo, (void *)ebp)) {
  49. addr = *(unsigned long *)(ebp + 4);
  50. printk("%08lx: [<%08lx>]", ebp + 4, addr);
  51. print_symbol(" %s", addr);
  52. printk("\n");
  53. ebp = *(unsigned long *)ebp;
  54. }
  55. #else
  56. while (valid_stack_ptr(tinfo, stack)) {
  57. addr = *stack;
  58. if (__kernel_text_address(addr)) {
  59. printk("%08lx: [<%08lx>]", (unsigned long) stack, addr);
  60. print_symbol(" %s", addr);
  61. printk("\n");
  62. }
  63. stack++;
  64. }
  65. #endif
  66. return ebp;
  67. }
  68. void show_trace(struct task_struct* task, unsigned long * stack)
  69. {
  70. unsigned long ebp;
  71. struct thread_info *context;
  72. /* Turn this into BUG_ON if possible. */
  73. if (!stack) {
  74. stack = (unsigned long*) &stack;
  75. printk("show_trace: got NULL stack, implicit assumption task == current");
  76. WARN_ON(1);
  77. }
  78. if (!task)
  79. task = current;
  80. if (task != current) {
  81. //ebp = (unsigned long) KSTK_EBP(task);
  82. /* Which one? No actual difference - just coding style.*/
  83. ebp = (unsigned long) PT_REGS_EBP(&task->thread.regs);
  84. } else {
  85. asm ("movl %%ebp, %0" : "=r" (ebp) : );
  86. }
  87. context = (struct thread_info *)
  88. ((unsigned long)stack & (~(THREAD_SIZE - 1)));
  89. print_context_stack(context, stack, ebp);
  90. /*while (((long) stack & (THREAD_SIZE-1)) != 0) {
  91. addr = *stack;
  92. if (__kernel_text_address(addr)) {
  93. printk("%08lx: [<%08lx>]", (unsigned long) stack, addr);
  94. print_symbol(" %s", addr);
  95. printk("\n");
  96. }
  97. stack++;
  98. }*/
  99. printk("\n");
  100. }