perf_regs.c 1.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162
  1. // SPDX-License-Identifier: GPL-2.0
  2. #include <linux/compat.h>
  3. #include <linux/errno.h>
  4. #include <linux/kernel.h>
  5. #include <linux/perf_event.h>
  6. #include <linux/bug.h>
  7. #include <linux/sched/task_stack.h>
  8. #include <asm/perf_regs.h>
  9. #include <asm/ptrace.h>
  10. u64 perf_reg_value(struct pt_regs *regs, int idx)
  11. {
  12. if (WARN_ON_ONCE((u32)idx >= PERF_REG_ARM64_MAX))
  13. return 0;
  14. /*
  15. * Compat (i.e. 32 bit) mode:
  16. * - PC has been set in the pt_regs struct in kernel_entry,
  17. * - Handle SP and LR here.
  18. */
  19. if (compat_user_mode(regs)) {
  20. if ((u32)idx == PERF_REG_ARM64_SP)
  21. return regs->compat_sp;
  22. if ((u32)idx == PERF_REG_ARM64_LR)
  23. return regs->compat_lr;
  24. }
  25. if ((u32)idx == PERF_REG_ARM64_SP)
  26. return regs->sp;
  27. if ((u32)idx == PERF_REG_ARM64_PC)
  28. return regs->pc;
  29. return regs->regs[idx];
  30. }
  31. #define REG_RESERVED (~((1ULL << PERF_REG_ARM64_MAX) - 1))
  32. int perf_reg_validate(u64 mask)
  33. {
  34. if (!mask || mask & REG_RESERVED)
  35. return -EINVAL;
  36. return 0;
  37. }
  38. u64 perf_reg_abi(struct task_struct *task)
  39. {
  40. if (is_compat_thread(task_thread_info(task)))
  41. return PERF_SAMPLE_REGS_ABI_32;
  42. else
  43. return PERF_SAMPLE_REGS_ABI_64;
  44. }
  45. void perf_get_regs_user(struct perf_regs *regs_user,
  46. struct pt_regs *regs,
  47. struct pt_regs *regs_user_copy)
  48. {
  49. regs_user->regs = task_pt_regs(current);
  50. regs_user->abi = perf_reg_abi(current);
  51. }