ftrace.c 1.7 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667
  1. // SPDX-License-Identifier: GPL-2.0
  2. #include <linux/kprobes.h>
  3. /* Ftrace callback handler for kprobes -- called under preepmt disabled */
  4. void kprobe_ftrace_handler(unsigned long ip, unsigned long parent_ip,
  5. struct ftrace_ops *ops, struct ftrace_regs *fregs)
  6. {
  7. int bit;
  8. bool lr_saver = false;
  9. struct kprobe *p;
  10. struct kprobe_ctlblk *kcb;
  11. struct pt_regs *regs;
  12. bit = ftrace_test_recursion_trylock(ip, parent_ip);
  13. if (bit < 0)
  14. return;
  15. regs = ftrace_get_regs(fregs);
  16. p = get_kprobe((kprobe_opcode_t *)ip);
  17. if (!p) {
  18. p = get_kprobe((kprobe_opcode_t *)(ip - MCOUNT_INSN_SIZE));
  19. if (unlikely(!p) || kprobe_disabled(p))
  20. goto out;
  21. lr_saver = true;
  22. }
  23. kcb = get_kprobe_ctlblk();
  24. if (kprobe_running()) {
  25. kprobes_inc_nmissed_count(p);
  26. } else {
  27. unsigned long orig_ip = instruction_pointer(regs);
  28. if (lr_saver)
  29. ip -= MCOUNT_INSN_SIZE;
  30. instruction_pointer_set(regs, ip);
  31. __this_cpu_write(current_kprobe, p);
  32. kcb->kprobe_status = KPROBE_HIT_ACTIVE;
  33. if (!p->pre_handler || !p->pre_handler(p, regs)) {
  34. /*
  35. * Emulate singlestep (and also recover regs->pc)
  36. * as if there is a nop
  37. */
  38. instruction_pointer_set(regs,
  39. (unsigned long)p->addr + MCOUNT_INSN_SIZE);
  40. if (unlikely(p->post_handler)) {
  41. kcb->kprobe_status = KPROBE_HIT_SSDONE;
  42. p->post_handler(p, regs, 0);
  43. }
  44. instruction_pointer_set(regs, orig_ip);
  45. }
  46. /*
  47. * If pre_handler returns !0, it changes regs->pc. We have to
  48. * skip emulating post_handler.
  49. */
  50. __this_cpu_write(current_kprobe, NULL);
  51. }
  52. out:
  53. ftrace_test_recursion_unlock(bit);
  54. }
  55. NOKPROBE_SYMBOL(kprobe_ftrace_handler);
  56. int arch_prepare_kprobe_ftrace(struct kprobe *p)
  57. {
  58. p->ainsn.api.insn = NULL;
  59. return 0;
  60. }