sysrq.c 2.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687
  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. struct stack_frame {
  15. struct stack_frame *next_frame;
  16. unsigned long return_address;
  17. };
  18. static void print_stack_trace(unsigned long *sp, unsigned long bp)
  19. {
  20. int reliable;
  21. unsigned long addr;
  22. struct stack_frame *frame = (struct stack_frame *)bp;
  23. printk(KERN_INFO "Call Trace:\n");
  24. while (((long) sp & (THREAD_SIZE-1)) != 0) {
  25. addr = *sp;
  26. if (__kernel_text_address(addr)) {
  27. reliable = 0;
  28. if ((unsigned long) sp == bp + sizeof(long)) {
  29. frame = frame ? frame->next_frame : NULL;
  30. bp = (unsigned long)frame;
  31. reliable = 1;
  32. }
  33. printk(KERN_INFO " [<%08lx>]", addr);
  34. printk(KERN_CONT " %s", reliable ? "" : "? ");
  35. print_symbol(KERN_CONT "%s", addr);
  36. printk(KERN_CONT "\n");
  37. }
  38. sp++;
  39. }
  40. printk(KERN_INFO "\n");
  41. }
  42. /*Stolen from arch/i386/kernel/traps.c */
  43. static const int kstack_depth_to_print = 24;
  44. static unsigned long get_frame_pointer(struct task_struct *task)
  45. {
  46. if (!task || task == current)
  47. return current_bp();
  48. else
  49. return KSTK_EBP(task);
  50. }
  51. void show_stack(struct task_struct *task, unsigned long *stack)
  52. {
  53. unsigned long *sp = stack, bp = 0;
  54. int i;
  55. #ifdef CONFIG_FRAME_POINTER
  56. bp = get_frame_pointer(task);
  57. #endif
  58. if (!stack) {
  59. if (!task || task == current)
  60. sp = current_sp();
  61. else
  62. sp = (unsigned long *)KSTK_ESP(task);
  63. }
  64. printk(KERN_INFO "Stack:\n");
  65. stack = sp;
  66. for (i = 0; i < kstack_depth_to_print; i++) {
  67. if (kstack_end(stack))
  68. break;
  69. if (i && ((i % STACKSLOTS_PER_LINE) == 0))
  70. printk(KERN_CONT "\n");
  71. printk(KERN_CONT " %08lx", *stack++);
  72. }
  73. printk(KERN_CONT "\n");
  74. print_stack_trace(sp, bp);
  75. }