stacktrace.c 1.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869
  1. #include <linux/module.h>
  2. #include <linux/sched.h>
  3. #include <linux/stacktrace.h>
  4. #include "stacktrace.h"
  5. int walk_stackframe(unsigned long fp, unsigned long low, unsigned long high,
  6. int (*fn)(struct stackframe *, void *), void *data)
  7. {
  8. struct stackframe *frame;
  9. do {
  10. /*
  11. * Check current frame pointer is within bounds
  12. */
  13. if (fp < (low + 12) || fp + 4 >= high)
  14. break;
  15. frame = (struct stackframe *)(fp - 12);
  16. if (fn(frame, data))
  17. break;
  18. /*
  19. * Update the low bound - the next frame must always
  20. * be at a higher address than the current frame.
  21. */
  22. low = fp + 4;
  23. fp = frame->fp;
  24. } while (fp);
  25. return 0;
  26. }
  27. EXPORT_SYMBOL(walk_stackframe);
  28. #ifdef CONFIG_STACKTRACE
  29. struct stack_trace_data {
  30. struct stack_trace *trace;
  31. unsigned int skip;
  32. };
  33. static int save_trace(struct stackframe *frame, void *d)
  34. {
  35. struct stack_trace_data *data = d;
  36. struct stack_trace *trace = data->trace;
  37. if (data->skip) {
  38. data->skip--;
  39. return 0;
  40. }
  41. trace->entries[trace->nr_entries++] = frame->lr;
  42. return trace->nr_entries >= trace->max_entries;
  43. }
  44. void save_stack_trace(struct stack_trace *trace)
  45. {
  46. struct stack_trace_data data;
  47. unsigned long fp, base;
  48. data.trace = trace;
  49. data.skip = trace->skip;
  50. base = (unsigned long)task_stack_page(current);
  51. asm("mov %0, fp" : "=r" (fp));
  52. walk_stackframe(fp, base, base + THREAD_SIZE, save_trace, &data);
  53. }
  54. #endif