trace_kdb.c 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * kdb helper for dumping the ftrace buffer
  4. *
  5. * Copyright (C) 2010 Jason Wessel <[email protected]>
  6. *
  7. * ftrace_dump_buf based on ftrace_dump:
  8. * Copyright (C) 2007-2008 Steven Rostedt <[email protected]>
  9. * Copyright (C) 2008 Ingo Molnar <[email protected]>
  10. *
  11. */
  12. #include <linux/init.h>
  13. #include <linux/kgdb.h>
  14. #include <linux/kdb.h>
  15. #include <linux/ftrace.h>
  16. #include "trace.h"
  17. #include "trace_output.h"
  18. static struct trace_iterator iter;
  19. static struct ring_buffer_iter *buffer_iter[CONFIG_NR_CPUS];
  20. static void ftrace_dump_buf(int skip_entries, long cpu_file)
  21. {
  22. struct trace_array *tr;
  23. unsigned int old_userobj;
  24. int cnt = 0, cpu;
  25. tr = iter.tr;
  26. old_userobj = tr->trace_flags;
  27. /* don't look at user memory in panic mode */
  28. tr->trace_flags &= ~TRACE_ITER_SYM_USEROBJ;
  29. kdb_printf("Dumping ftrace buffer:\n");
  30. if (skip_entries)
  31. kdb_printf("(skipping %d entries)\n", skip_entries);
  32. trace_iterator_reset(&iter);
  33. iter.iter_flags |= TRACE_FILE_LAT_FMT;
  34. if (cpu_file == RING_BUFFER_ALL_CPUS) {
  35. for_each_tracing_cpu(cpu) {
  36. iter.buffer_iter[cpu] =
  37. ring_buffer_read_prepare(iter.array_buffer->buffer,
  38. cpu, GFP_ATOMIC);
  39. ring_buffer_read_start(iter.buffer_iter[cpu]);
  40. tracing_iter_reset(&iter, cpu);
  41. }
  42. } else {
  43. iter.cpu_file = cpu_file;
  44. iter.buffer_iter[cpu_file] =
  45. ring_buffer_read_prepare(iter.array_buffer->buffer,
  46. cpu_file, GFP_ATOMIC);
  47. ring_buffer_read_start(iter.buffer_iter[cpu_file]);
  48. tracing_iter_reset(&iter, cpu_file);
  49. }
  50. while (trace_find_next_entry_inc(&iter)) {
  51. if (!cnt)
  52. kdb_printf("---------------------------------\n");
  53. cnt++;
  54. if (!skip_entries) {
  55. print_trace_line(&iter);
  56. trace_printk_seq(&iter.seq);
  57. } else {
  58. skip_entries--;
  59. }
  60. if (KDB_FLAG(CMD_INTERRUPT))
  61. goto out;
  62. }
  63. if (!cnt)
  64. kdb_printf(" (ftrace buffer empty)\n");
  65. else
  66. kdb_printf("---------------------------------\n");
  67. out:
  68. tr->trace_flags = old_userobj;
  69. for_each_tracing_cpu(cpu) {
  70. if (iter.buffer_iter[cpu]) {
  71. ring_buffer_read_finish(iter.buffer_iter[cpu]);
  72. iter.buffer_iter[cpu] = NULL;
  73. }
  74. }
  75. }
  76. /*
  77. * kdb_ftdump - Dump the ftrace log buffer
  78. */
  79. static int kdb_ftdump(int argc, const char **argv)
  80. {
  81. int skip_entries = 0;
  82. long cpu_file;
  83. char *cp;
  84. int cnt;
  85. int cpu;
  86. if (argc > 2)
  87. return KDB_ARGCOUNT;
  88. if (argc) {
  89. skip_entries = simple_strtol(argv[1], &cp, 0);
  90. if (*cp)
  91. skip_entries = 0;
  92. }
  93. if (argc == 2) {
  94. cpu_file = simple_strtol(argv[2], &cp, 0);
  95. if (*cp || cpu_file >= NR_CPUS || cpu_file < 0 ||
  96. !cpu_online(cpu_file))
  97. return KDB_BADINT;
  98. } else {
  99. cpu_file = RING_BUFFER_ALL_CPUS;
  100. }
  101. kdb_trap_printk++;
  102. trace_init_global_iter(&iter);
  103. iter.buffer_iter = buffer_iter;
  104. for_each_tracing_cpu(cpu) {
  105. atomic_inc(&per_cpu_ptr(iter.array_buffer->data, cpu)->disabled);
  106. }
  107. /* A negative skip_entries means skip all but the last entries */
  108. if (skip_entries < 0) {
  109. if (cpu_file == RING_BUFFER_ALL_CPUS)
  110. cnt = trace_total_entries(NULL);
  111. else
  112. cnt = trace_total_entries_cpu(NULL, cpu_file);
  113. skip_entries = max(cnt + skip_entries, 0);
  114. }
  115. ftrace_dump_buf(skip_entries, cpu_file);
  116. for_each_tracing_cpu(cpu) {
  117. atomic_dec(&per_cpu_ptr(iter.array_buffer->data, cpu)->disabled);
  118. }
  119. kdb_trap_printk--;
  120. return 0;
  121. }
  122. static kdbtab_t ftdump_cmd = {
  123. .name = "ftdump",
  124. .func = kdb_ftdump,
  125. .usage = "[skip_#entries] [cpu]",
  126. .help = "Dump ftrace log; -skip dumps last #entries",
  127. .flags = KDB_ENABLE_ALWAYS_SAFE,
  128. };
  129. static __init int kdb_ftrace_register(void)
  130. {
  131. kdb_register(&ftdump_cmd);
  132. return 0;
  133. }
  134. late_initcall(kdb_ftrace_register);