stacktrace.c 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158
  1. // SPDX-License-Identifier: GPL-2.0
  2. #include <linux/sched/debug.h>
  3. #include <linux/sched/task_stack.h>
  4. #include <linux/stacktrace.h>
  5. #include <linux/ftrace.h>
  6. #include <linux/ptrace.h>
  7. #ifdef CONFIG_FRAME_POINTER
  8. struct stackframe {
  9. unsigned long fp;
  10. unsigned long ra;
  11. };
  12. void notrace walk_stackframe(struct task_struct *task, struct pt_regs *regs,
  13. bool (*fn)(unsigned long, void *), void *arg)
  14. {
  15. unsigned long fp, sp, pc;
  16. if (regs) {
  17. fp = frame_pointer(regs);
  18. sp = user_stack_pointer(regs);
  19. pc = instruction_pointer(regs);
  20. } else if (task == NULL || task == current) {
  21. const register unsigned long current_sp __asm__ ("sp");
  22. const register unsigned long current_fp __asm__ ("r8");
  23. fp = current_fp;
  24. sp = current_sp;
  25. pc = (unsigned long)walk_stackframe;
  26. } else {
  27. /* task blocked in __switch_to */
  28. fp = thread_saved_fp(task);
  29. sp = thread_saved_sp(task);
  30. pc = thread_saved_lr(task);
  31. }
  32. for (;;) {
  33. unsigned long low, high;
  34. struct stackframe *frame;
  35. if (unlikely(!__kernel_text_address(pc) || fn(pc, arg)))
  36. break;
  37. /* Validate frame pointer */
  38. low = sp;
  39. high = ALIGN(sp, THREAD_SIZE);
  40. if (unlikely(fp < low || fp > high || fp & 0x3))
  41. break;
  42. /* Unwind stack frame */
  43. frame = (struct stackframe *)fp;
  44. sp = fp;
  45. fp = frame->fp;
  46. pc = ftrace_graph_ret_addr(current, NULL, frame->ra,
  47. (unsigned long *)(fp - 8));
  48. }
  49. }
  50. #else /* !CONFIG_FRAME_POINTER */
  51. static void notrace walk_stackframe(struct task_struct *task,
  52. struct pt_regs *regs, bool (*fn)(unsigned long, void *), void *arg)
  53. {
  54. unsigned long sp, pc;
  55. unsigned long *ksp;
  56. if (regs) {
  57. sp = user_stack_pointer(regs);
  58. pc = instruction_pointer(regs);
  59. } else if (task == NULL || task == current) {
  60. const register unsigned long current_sp __asm__ ("sp");
  61. sp = current_sp;
  62. pc = (unsigned long)walk_stackframe;
  63. } else {
  64. /* task blocked in __switch_to */
  65. sp = thread_saved_sp(task);
  66. pc = thread_saved_lr(task);
  67. }
  68. if (unlikely(sp & 0x3))
  69. return;
  70. ksp = (unsigned long *)sp;
  71. while (!kstack_end(ksp)) {
  72. if (__kernel_text_address(pc) && unlikely(fn(pc, arg)))
  73. break;
  74. pc = (*ksp++) - 0x4;
  75. }
  76. }
  77. #endif /* CONFIG_FRAME_POINTER */
  78. static bool print_trace_address(unsigned long pc, void *arg)
  79. {
  80. print_ip_sym((const char *)arg, pc);
  81. return false;
  82. }
  83. void show_stack(struct task_struct *task, unsigned long *sp, const char *loglvl)
  84. {
  85. pr_cont("Call Trace:\n");
  86. walk_stackframe(task, NULL, print_trace_address, (void *)loglvl);
  87. }
  88. static bool save_wchan(unsigned long pc, void *arg)
  89. {
  90. if (!in_sched_functions(pc)) {
  91. unsigned long *p = arg;
  92. *p = pc;
  93. return true;
  94. }
  95. return false;
  96. }
  97. unsigned long __get_wchan(struct task_struct *task)
  98. {
  99. unsigned long pc = 0;
  100. walk_stackframe(task, NULL, save_wchan, &pc);
  101. return pc;
  102. }
  103. #ifdef CONFIG_STACKTRACE
  104. static bool __save_trace(unsigned long pc, void *arg, bool nosched)
  105. {
  106. struct stack_trace *trace = arg;
  107. if (unlikely(nosched && in_sched_functions(pc)))
  108. return false;
  109. if (unlikely(trace->skip > 0)) {
  110. trace->skip--;
  111. return false;
  112. }
  113. trace->entries[trace->nr_entries++] = pc;
  114. return (trace->nr_entries >= trace->max_entries);
  115. }
  116. static bool save_trace(unsigned long pc, void *arg)
  117. {
  118. return __save_trace(pc, arg, false);
  119. }
  120. /*
  121. * Save stack-backtrace addresses into a stack_trace buffer.
  122. */
  123. void save_stack_trace_tsk(struct task_struct *tsk, struct stack_trace *trace)
  124. {
  125. walk_stackframe(tsk, NULL, save_trace, trace);
  126. }
  127. EXPORT_SYMBOL_GPL(save_stack_trace_tsk);
  128. void save_stack_trace(struct stack_trace *trace)
  129. {
  130. save_stack_trace_tsk(NULL, trace);
  131. }
  132. EXPORT_SYMBOL_GPL(save_stack_trace);
  133. #endif /* CONFIG_STACKTRACE */