stacktrace.c 1.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849
  1. // SPDX-License-Identifier: GPL-2.0
  2. // Copyright (C) 2005-2017 Andes Technology Corporation
  3. #include <linux/sched/debug.h>
  4. #include <linux/sched/task_stack.h>
  5. #include <linux/stacktrace.h>
  6. void save_stack_trace(struct stack_trace *trace)
  7. {
  8. save_stack_trace_tsk(current, trace);
  9. }
  10. EXPORT_SYMBOL_GPL(save_stack_trace);
  11. void save_stack_trace_tsk(struct task_struct *tsk, struct stack_trace *trace)
  12. {
  13. unsigned long *fpn;
  14. int skip = trace->skip;
  15. int savesched;
  16. if (tsk == current) {
  17. __asm__ __volatile__("\tori\t%0, $fp, #0\n":"=r"(fpn));
  18. savesched = 1;
  19. } else {
  20. fpn = (unsigned long *)thread_saved_fp(tsk);
  21. savesched = 0;
  22. }
  23. while (!kstack_end(fpn) && !((unsigned long)fpn & 0x3)
  24. && (fpn >= (unsigned long *)TASK_SIZE)) {
  25. unsigned long lpp, fpp;
  26. lpp = fpn[-1];
  27. fpp = fpn[FP_OFFSET];
  28. if (!__kernel_text_address(lpp))
  29. break;
  30. if (savesched || !in_sched_functions(lpp)) {
  31. if (skip) {
  32. skip--;
  33. } else {
  34. trace->entries[trace->nr_entries++] = lpp;
  35. if (trace->nr_entries >= trace->max_entries)
  36. break;
  37. }
  38. }
  39. fpn = (unsigned long *)fpp;
  40. }
  41. }
  42. EXPORT_SYMBOL_GPL(save_stack_trace_tsk);