stacktrace.c 2.0 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485
  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 (!reliable)
  25. return;
  26. if (trace->skip > 0) {
  27. trace->skip--;
  28. return;
  29. }
  30. if (trace->nr_entries < trace->max_entries)
  31. trace->entries[trace->nr_entries++] = addr;
  32. }
  33. static void
  34. save_stack_address_nosched(void *data, unsigned long addr, int reliable)
  35. {
  36. struct stack_trace *trace = (struct stack_trace *)data;
  37. if (!reliable)
  38. return;
  39. if (in_sched_functions(addr))
  40. return;
  41. if (trace->skip > 0) {
  42. trace->skip--;
  43. return;
  44. }
  45. if (trace->nr_entries < trace->max_entries)
  46. trace->entries[trace->nr_entries++] = addr;
  47. }
  48. static const struct stacktrace_ops save_stack_ops = {
  49. .warning = save_stack_warning,
  50. .warning_symbol = save_stack_warning_symbol,
  51. .stack = save_stack_stack,
  52. .address = save_stack_address,
  53. };
  54. static const struct stacktrace_ops save_stack_ops_nosched = {
  55. .warning = save_stack_warning,
  56. .warning_symbol = save_stack_warning_symbol,
  57. .stack = save_stack_stack,
  58. .address = save_stack_address_nosched,
  59. };
  60. /*
  61. * Save stack-backtrace addresses into a stack_trace buffer.
  62. */
  63. void save_stack_trace(struct stack_trace *trace)
  64. {
  65. dump_trace(current, NULL, NULL, 0, &save_stack_ops, trace);
  66. if (trace->nr_entries < trace->max_entries)
  67. trace->entries[trace->nr_entries++] = ULONG_MAX;
  68. }
  69. EXPORT_SYMBOL_GPL(save_stack_trace);
  70. void save_stack_trace_tsk(struct task_struct *tsk, struct stack_trace *trace)
  71. {
  72. dump_trace(tsk, NULL, NULL, 0, &save_stack_ops_nosched, trace);
  73. if (trace->nr_entries < trace->max_entries)
  74. trace->entries[trace->nr_entries++] = ULONG_MAX;
  75. }
  76. EXPORT_SYMBOL_GPL(save_stack_trace_tsk);