sysrq.c 2.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798
  1. /*
  2. * Copyright (C) 2001 - 2007 Jeff Dike (jdike@{addtoit,linux.intel}.com)
  3. * Copyright (C) 2013 Richard Weinberger <richrd@nod.at>
  4. *
  5. * This program is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License version 2 as
  7. * published by the Free Software Foundation.
  8. */
  9. #include <linux/kallsyms.h>
  10. #include <linux/kernel.h>
  11. #include <linux/module.h>
  12. #include <linux/sched.h>
  13. #include <asm/sysrq.h>
  14. #include <os.h>
  15. struct stack_frame {
  16. struct stack_frame *next_frame;
  17. unsigned long return_address;
  18. };
  19. static void print_stack_trace(unsigned long *sp, unsigned long bp)
  20. {
  21. int reliable;
  22. unsigned long addr;
  23. struct stack_frame *frame = (struct stack_frame *)bp;
  24. printk(KERN_INFO "Call Trace:\n");
  25. while (((long) sp & (THREAD_SIZE-1)) != 0) {
  26. addr = *sp;
  27. if (__kernel_text_address(addr)) {
  28. reliable = 0;
  29. if ((unsigned long) sp == bp + sizeof(long)) {
  30. frame = frame ? frame->next_frame : NULL;
  31. bp = (unsigned long)frame;
  32. reliable = 1;
  33. }
  34. printk(KERN_INFO " [<%08lx>]", addr);
  35. printk(KERN_CONT " %s", reliable ? "" : "? ");
  36. print_symbol(KERN_CONT "%s", addr);
  37. printk(KERN_CONT "\n");
  38. }
  39. sp++;
  40. }
  41. printk(KERN_INFO "\n");
  42. }
  43. static unsigned long get_frame_pointer(struct task_struct *task,
  44. struct pt_regs *segv_regs)
  45. {
  46. if (!task || task == current)
  47. return segv_regs ? PT_REGS_BP(segv_regs) : current_bp();
  48. else
  49. return KSTK_EBP(task);
  50. }
  51. static unsigned long *get_stack_pointer(struct task_struct *task,
  52. struct pt_regs *segv_regs)
  53. {
  54. if (!task || task == current)
  55. return segv_regs ? (unsigned long *)PT_REGS_SP(segv_regs) : current_sp();
  56. else
  57. return (unsigned long *)KSTK_ESP(task);
  58. }
  59. void show_stack(struct task_struct *task, unsigned long *stack)
  60. {
  61. unsigned long *sp = stack, bp = 0;
  62. struct pt_regs *segv_regs = current->thread.segv_regs;
  63. int i;
  64. if (!segv_regs && os_is_signal_stack()) {
  65. printk(KERN_ERR "Received SIGSEGV in SIGSEGV handler,"
  66. " aborting stack trace!\n");
  67. return;
  68. }
  69. #ifdef CONFIG_FRAME_POINTER
  70. bp = get_frame_pointer(task, segv_regs);
  71. #endif
  72. if (!stack)
  73. sp = get_stack_pointer(task, segv_regs);
  74. printk(KERN_INFO "Stack:\n");
  75. stack = sp;
  76. for (i = 0; i < 3 * STACKSLOTS_PER_LINE; i++) {
  77. if (kstack_end(stack))
  78. break;
  79. if (i && ((i % STACKSLOTS_PER_LINE) == 0))
  80. printk(KERN_CONT "\n");
  81. printk(KERN_CONT " %08lx", *stack++);
  82. }
  83. printk(KERN_CONT "\n");
  84. print_stack_trace(sp, bp);
  85. }