stacktrace.c 2.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. /*
  2. * arch/s390/kernel/stacktrace.c
  3. *
  4. * Stack trace management functions
  5. *
  6. * Copyright (C) IBM Corp. 2006
  7. * Author(s): Heiko Carstens <heiko.carstens@de.ibm.com>
  8. */
  9. #include <linux/sched.h>
  10. #include <linux/stacktrace.h>
  11. #include <linux/kallsyms.h>
  12. static unsigned long save_context_stack(struct stack_trace *trace,
  13. unsigned long sp,
  14. unsigned long low,
  15. unsigned long high,
  16. int savesched)
  17. {
  18. struct stack_frame *sf;
  19. struct pt_regs *regs;
  20. unsigned long addr;
  21. while(1) {
  22. sp &= PSW_ADDR_INSN;
  23. if (sp < low || sp > high)
  24. return sp;
  25. sf = (struct stack_frame *)sp;
  26. while(1) {
  27. addr = sf->gprs[8] & PSW_ADDR_INSN;
  28. if (!trace->skip)
  29. trace->entries[trace->nr_entries++] = addr;
  30. else
  31. trace->skip--;
  32. if (trace->nr_entries >= trace->max_entries)
  33. return sp;
  34. low = sp;
  35. sp = sf->back_chain & PSW_ADDR_INSN;
  36. if (!sp)
  37. break;
  38. if (sp <= low || sp > high - sizeof(*sf))
  39. return sp;
  40. sf = (struct stack_frame *)sp;
  41. }
  42. /* Zero backchain detected, check for interrupt frame. */
  43. sp = (unsigned long)(sf + 1);
  44. if (sp <= low || sp > high - sizeof(*regs))
  45. return sp;
  46. regs = (struct pt_regs *)sp;
  47. addr = regs->psw.addr & PSW_ADDR_INSN;
  48. if (savesched || !in_sched_functions(addr)) {
  49. if (!trace->skip)
  50. trace->entries[trace->nr_entries++] = addr;
  51. else
  52. trace->skip--;
  53. }
  54. if (trace->nr_entries >= trace->max_entries)
  55. return sp;
  56. low = sp;
  57. sp = regs->gprs[15];
  58. }
  59. }
  60. void save_stack_trace(struct stack_trace *trace)
  61. {
  62. register unsigned long sp asm ("15");
  63. unsigned long orig_sp, new_sp;
  64. orig_sp = sp & PSW_ADDR_INSN;
  65. new_sp = save_context_stack(trace, orig_sp,
  66. S390_lowcore.panic_stack - PAGE_SIZE,
  67. S390_lowcore.panic_stack, 1);
  68. if (new_sp != orig_sp)
  69. return;
  70. new_sp = save_context_stack(trace, new_sp,
  71. S390_lowcore.async_stack - ASYNC_SIZE,
  72. S390_lowcore.async_stack, 1);
  73. if (new_sp != orig_sp)
  74. return;
  75. save_context_stack(trace, new_sp,
  76. S390_lowcore.thread_info,
  77. S390_lowcore.thread_info + THREAD_SIZE, 1);
  78. }
  79. void save_stack_trace_tsk(struct task_struct *tsk, struct stack_trace *trace)
  80. {
  81. unsigned long sp, low, high;
  82. sp = tsk->thread.ksp & PSW_ADDR_INSN;
  83. low = (unsigned long) task_stack_page(tsk);
  84. high = (unsigned long) task_pt_regs(tsk);
  85. save_context_stack(trace, sp, low, high, 0);
  86. if (trace->nr_entries < trace->max_entries)
  87. trace->entries[trace->nr_entries++] = ULONG_MAX;
  88. }