perf_callchain.c 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * arm64 callchain support
  4. *
  5. * Copyright (C) 2015 ARM Limited
  6. */
  7. #include <linux/perf_event.h>
  8. #include <linux/stacktrace.h>
  9. #include <linux/uaccess.h>
  10. #include <asm/pointer_auth.h>
  11. struct frame_tail {
  12. struct frame_tail __user *fp;
  13. unsigned long lr;
  14. } __attribute__((packed));
  15. /*
  16. * Get the return address for a single stackframe and return a pointer to the
  17. * next frame tail.
  18. */
  19. static struct frame_tail __user *
  20. user_backtrace(struct frame_tail __user *tail,
  21. struct perf_callchain_entry_ctx *entry)
  22. {
  23. struct frame_tail buftail;
  24. unsigned long err;
  25. unsigned long lr;
  26. /* Also check accessibility of one struct frame_tail beyond */
  27. if (!access_ok(tail, sizeof(buftail)))
  28. return NULL;
  29. pagefault_disable();
  30. err = __copy_from_user_inatomic(&buftail, tail, sizeof(buftail));
  31. pagefault_enable();
  32. if (err)
  33. return NULL;
  34. lr = ptrauth_strip_insn_pac(buftail.lr);
  35. perf_callchain_store(entry, lr);
  36. /*
  37. * Frame pointers should strictly progress back up the stack
  38. * (towards higher addresses).
  39. */
  40. if (tail >= buftail.fp)
  41. return NULL;
  42. return buftail.fp;
  43. }
  44. #ifdef CONFIG_COMPAT
  45. /*
  46. * The registers we're interested in are at the end of the variable
  47. * length saved register structure. The fp points at the end of this
  48. * structure so the address of this struct is:
  49. * (struct compat_frame_tail *)(xxx->fp)-1
  50. *
  51. * This code has been adapted from the ARM OProfile support.
  52. */
  53. struct compat_frame_tail {
  54. compat_uptr_t fp; /* a (struct compat_frame_tail *) in compat mode */
  55. u32 sp;
  56. u32 lr;
  57. } __attribute__((packed));
  58. static struct compat_frame_tail __user *
  59. compat_user_backtrace(struct compat_frame_tail __user *tail,
  60. struct perf_callchain_entry_ctx *entry)
  61. {
  62. struct compat_frame_tail buftail;
  63. unsigned long err;
  64. /* Also check accessibility of one struct frame_tail beyond */
  65. if (!access_ok(tail, sizeof(buftail)))
  66. return NULL;
  67. pagefault_disable();
  68. err = __copy_from_user_inatomic(&buftail, tail, sizeof(buftail));
  69. pagefault_enable();
  70. if (err)
  71. return NULL;
  72. perf_callchain_store(entry, buftail.lr);
  73. /*
  74. * Frame pointers should strictly progress back up the stack
  75. * (towards higher addresses).
  76. */
  77. if (tail + 1 >= (struct compat_frame_tail __user *)
  78. compat_ptr(buftail.fp))
  79. return NULL;
  80. return (struct compat_frame_tail __user *)compat_ptr(buftail.fp) - 1;
  81. }
  82. #endif /* CONFIG_COMPAT */
  83. void perf_callchain_user(struct perf_callchain_entry_ctx *entry,
  84. struct pt_regs *regs)
  85. {
  86. if (perf_guest_state()) {
  87. /* We don't support guest os callchain now */
  88. return;
  89. }
  90. perf_callchain_store(entry, regs->pc);
  91. if (!compat_user_mode(regs)) {
  92. /* AARCH64 mode */
  93. struct frame_tail __user *tail;
  94. tail = (struct frame_tail __user *)regs->regs[29];
  95. while (entry->nr < entry->max_stack &&
  96. tail && !((unsigned long)tail & 0x7))
  97. tail = user_backtrace(tail, entry);
  98. } else {
  99. #ifdef CONFIG_COMPAT
  100. /* AARCH32 compat mode */
  101. struct compat_frame_tail __user *tail;
  102. tail = (struct compat_frame_tail __user *)regs->compat_fp - 1;
  103. while ((entry->nr < entry->max_stack) &&
  104. tail && !((unsigned long)tail & 0x3))
  105. tail = compat_user_backtrace(tail, entry);
  106. #endif
  107. }
  108. }
  109. static bool callchain_trace(void *data, unsigned long pc)
  110. {
  111. struct perf_callchain_entry_ctx *entry = data;
  112. return perf_callchain_store(entry, pc) == 0;
  113. }
  114. void perf_callchain_kernel(struct perf_callchain_entry_ctx *entry,
  115. struct pt_regs *regs)
  116. {
  117. if (perf_guest_state()) {
  118. /* We don't support guest os callchain now */
  119. return;
  120. }
  121. arch_stack_walk(callchain_trace, entry, current, regs);
  122. }
  123. unsigned long perf_instruction_pointer(struct pt_regs *regs)
  124. {
  125. if (perf_guest_state())
  126. return perf_guest_get_ip();
  127. return instruction_pointer(regs);
  128. }
  129. unsigned long perf_misc_flags(struct pt_regs *regs)
  130. {
  131. unsigned int guest_state = perf_guest_state();
  132. int misc = 0;
  133. if (guest_state) {
  134. if (guest_state & PERF_GUEST_USER)
  135. misc |= PERF_RECORD_MISC_GUEST_USER;
  136. else
  137. misc |= PERF_RECORD_MISC_GUEST_KERNEL;
  138. } else {
  139. if (user_mode(regs))
  140. misc |= PERF_RECORD_MISC_USER;
  141. else
  142. misc |= PERF_RECORD_MISC_KERNEL;
  143. }
  144. return misc;
  145. }