sysrq.c 1.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (C) 2001 - 2007 Jeff Dike (jdike@{addtoit,linux.intel}.com)
  4. * Copyright (C) 2013 Richard Weinberger <[email protected]>
  5. */
  6. #include <linux/kallsyms.h>
  7. #include <linux/kernel.h>
  8. #include <linux/module.h>
  9. #include <linux/sched.h>
  10. #include <linux/sched/debug.h>
  11. #include <linux/sched/task_stack.h>
  12. #include <asm/sysrq.h>
  13. #include <asm/stacktrace.h>
  14. #include <os.h>
  15. static void _print_addr(void *data, unsigned long address, int reliable)
  16. {
  17. const char *loglvl = data;
  18. printk("%s [<%08lx>] %s%pS\n", loglvl, address, reliable ? "" : "? ",
  19. (void *)address);
  20. }
  21. static const struct stacktrace_ops stackops = {
  22. .address = _print_addr
  23. };
  24. void show_stack(struct task_struct *task, unsigned long *stack,
  25. const char *loglvl)
  26. {
  27. struct pt_regs *segv_regs = current->thread.segv_regs;
  28. int i;
  29. if (!segv_regs && os_is_signal_stack()) {
  30. pr_err("Received SIGSEGV in SIGSEGV handler,"
  31. " aborting stack trace!\n");
  32. return;
  33. }
  34. if (!stack)
  35. stack = get_stack_pointer(task, segv_regs);
  36. printk("%sStack:\n", loglvl);
  37. for (i = 0; i < 3 * STACKSLOTS_PER_LINE; i++) {
  38. if (kstack_end(stack))
  39. break;
  40. if (i && ((i % STACKSLOTS_PER_LINE) == 0))
  41. pr_cont("\n");
  42. pr_cont(" %08lx", READ_ONCE_NOCHECK(*stack));
  43. stack++;
  44. }
  45. printk("%sCall Trace:\n", loglvl);
  46. dump_trace(current, &stackops, (void *)loglvl);
  47. }