stacktrace.c 1.9 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879
  1. /*
  2. * Stack trace management functions
  3. *
  4. * Copyright (C) 2006 Red Hat, Inc., Ingo Molnar <mingo@redhat.com>
  5. */
  6. #include <linux/sched.h>
  7. #include <linux/stacktrace.h>
  8. #include <linux/module.h>
  9. #include <asm/stacktrace.h>
  10. static void save_stack_warning(void *data, char *msg)
  11. {
  12. }
  13. static void
  14. save_stack_warning_symbol(void *data, char *msg, unsigned long symbol)
  15. {
  16. }
  17. static int save_stack_stack(void *data, char *name)
  18. {
  19. return -1;
  20. }
  21. static void save_stack_address(void *data, unsigned long addr, int reliable)
  22. {
  23. struct stack_trace *trace = data;
  24. if (trace->skip > 0) {
  25. trace->skip--;
  26. return;
  27. }
  28. if (trace->nr_entries < trace->max_entries)
  29. trace->entries[trace->nr_entries++] = addr;
  30. }
  31. static void
  32. save_stack_address_nosched(void *data, unsigned long addr, int reliable)
  33. {
  34. struct stack_trace *trace = (struct stack_trace *)data;
  35. if (in_sched_functions(addr))
  36. return;
  37. if (trace->skip > 0) {
  38. trace->skip--;
  39. return;
  40. }
  41. if (trace->nr_entries < trace->max_entries)
  42. trace->entries[trace->nr_entries++] = addr;
  43. }
  44. static const struct stacktrace_ops save_stack_ops = {
  45. .warning = save_stack_warning,
  46. .warning_symbol = save_stack_warning_symbol,
  47. .stack = save_stack_stack,
  48. .address = save_stack_address,
  49. };
  50. static const struct stacktrace_ops save_stack_ops_nosched = {
  51. .warning = save_stack_warning,
  52. .warning_symbol = save_stack_warning_symbol,
  53. .stack = save_stack_stack,
  54. .address = save_stack_address_nosched,
  55. };
  56. /*
  57. * Save stack-backtrace addresses into a stack_trace buffer.
  58. */
  59. void save_stack_trace(struct stack_trace *trace)
  60. {
  61. dump_trace(current, NULL, NULL, 0, &save_stack_ops, trace);
  62. if (trace->nr_entries < trace->max_entries)
  63. trace->entries[trace->nr_entries++] = ULONG_MAX;
  64. }
  65. void save_stack_trace_tsk(struct task_struct *tsk, struct stack_trace *trace)
  66. {
  67. dump_trace(tsk, NULL, NULL, 0, &save_stack_ops_nosched, trace);
  68. if (trace->nr_entries < trace->max_entries)
  69. trace->entries[trace->nr_entries++] = ULONG_MAX;
  70. }