11a59d1b8SThomas Gleixner /* SPDX-License-Identifier: GPL-2.0-or-later */ 21da177e4SLinus Torvalds #ifndef _LINUX_KPROBES_H 31da177e4SLinus Torvalds #define _LINUX_KPROBES_H 41da177e4SLinus Torvalds /* 51da177e4SLinus Torvalds * Kernel Probes (KProbes) 61da177e4SLinus Torvalds * 71da177e4SLinus Torvalds * Copyright (C) IBM Corporation, 2002, 2004 81da177e4SLinus Torvalds * 91da177e4SLinus Torvalds * 2002-Oct Created by Vamsi Krishna S <vamsi_krishna@in.ibm.com> Kernel 101da177e4SLinus Torvalds * Probes initial implementation ( includes suggestions from 111da177e4SLinus Torvalds * Rusty Russell). 121da177e4SLinus Torvalds * 2004-July Suparna Bhattacharya <suparna@in.ibm.com> added jumper probes 131da177e4SLinus Torvalds * interface to access function arguments. 14b94cce92SHien Nguyen * 2005-May Hien Nguyen <hien@us.ibm.com> and Jim Keniston 15b94cce92SHien Nguyen * <jkenisto@us.ibm.com> and Prasanna S Panchamukhi 16b94cce92SHien Nguyen * <prasanna@in.ibm.com> added function-return probes. 171da177e4SLinus Torvalds */ 187d134b2cSLuis R. Rodriguez #include <linux/compiler.h> 1936dcd67aSIngo Molnar #include <linux/linkage.h> 201da177e4SLinus Torvalds #include <linux/list.h> 211da177e4SLinus Torvalds #include <linux/notifier.h> 221da177e4SLinus Torvalds #include <linux/smp.h> 23187f1882SPaul Gortmaker #include <linux/bug.h> 24e6584523SAnanth N Mavinakayanahalli #include <linux/percpu.h> 253516a460SAnanth N Mavinakayanahalli #include <linux/spinlock.h> 263516a460SAnanth N Mavinakayanahalli #include <linux/rcupdate.h> 277a7d1cf9SIngo Molnar #include <linux/mutex.h> 28ae6aa16fSMasami Hiramatsu #include <linux/ftrace.h> 29d741bf41SPeter Zijlstra #include <linux/refcount.h> 306e426e0fSPeter Zijlstra #include <linux/freelist.h> 317d134b2cSLuis R. Rodriguez #include <asm/kprobes.h> 32b94cce92SHien Nguyen 3300d7c05aSKeshavamurthy Anil S #ifdef CONFIG_KPROBES 341da177e4SLinus Torvalds 35ea32c65cSPrasanna S Panchamukhi /* kprobe_status settings */ 36ea32c65cSPrasanna S Panchamukhi #define KPROBE_HIT_ACTIVE 0x00000001 37ea32c65cSPrasanna S Panchamukhi #define KPROBE_HIT_SS 0x00000002 38ea32c65cSPrasanna S Panchamukhi #define KPROBE_REENTER 0x00000004 39ea32c65cSPrasanna S Panchamukhi #define KPROBE_HIT_SSDONE 0x00000008 40ea32c65cSPrasanna S Panchamukhi 41223a76b2SMasami Hiramatsu #else /* !CONFIG_KPROBES */ 427d134b2cSLuis R. Rodriguez #include <asm-generic/kprobes.h> 43dc19835dSMasami Hiramatsu typedef int kprobe_opcode_t; 44dc19835dSMasami Hiramatsu struct arch_specific_insn { 45dc19835dSMasami Hiramatsu int dummy; 46dc19835dSMasami Hiramatsu }; 47dc19835dSMasami Hiramatsu #endif /* CONFIG_KPROBES */ 48d0aaff97SPrasanna S Panchamukhi 491da177e4SLinus Torvalds struct kprobe; 501da177e4SLinus Torvalds struct pt_regs; 51b94cce92SHien Nguyen struct kretprobe; 52b94cce92SHien Nguyen struct kretprobe_instance; 531da177e4SLinus Torvalds typedef int (*kprobe_pre_handler_t) (struct kprobe *, struct pt_regs *); 541da177e4SLinus Torvalds typedef void (*kprobe_post_handler_t) (struct kprobe *, struct pt_regs *, 551da177e4SLinus Torvalds unsigned long flags); 56b94cce92SHien Nguyen typedef int (*kretprobe_handler_t) (struct kretprobe_instance *, 57b94cce92SHien Nguyen struct pt_regs *); 58b94cce92SHien Nguyen 591da177e4SLinus Torvalds struct kprobe { 601da177e4SLinus Torvalds struct hlist_node hlist; 611da177e4SLinus Torvalds 6264f562c6SAnanth N Mavinakayanahalli /* list of kprobes for multi-handler support */ 6364f562c6SAnanth N Mavinakayanahalli struct list_head list; 6464f562c6SAnanth N Mavinakayanahalli 65ea32c65cSPrasanna S Panchamukhi /*count the number of times this probe was temporarily disarmed */ 66ea32c65cSPrasanna S Panchamukhi unsigned long nmissed; 67ea32c65cSPrasanna S Panchamukhi 681da177e4SLinus Torvalds /* location of the probe point */ 691da177e4SLinus Torvalds kprobe_opcode_t *addr; 701da177e4SLinus Torvalds 713a872d89SAnanth N Mavinakayanahalli /* Allow user to indicate symbol name of the probe point */ 729b3af29bSAnanth N Mavinakayanahalli const char *symbol_name; 733a872d89SAnanth N Mavinakayanahalli 743a872d89SAnanth N Mavinakayanahalli /* Offset into the symbol */ 753a872d89SAnanth N Mavinakayanahalli unsigned int offset; 763a872d89SAnanth N Mavinakayanahalli 771da177e4SLinus Torvalds /* Called before addr is executed. */ 781da177e4SLinus Torvalds kprobe_pre_handler_t pre_handler; 791da177e4SLinus Torvalds 801da177e4SLinus Torvalds /* Called after addr is executed, unless... */ 811da177e4SLinus Torvalds kprobe_post_handler_t post_handler; 821da177e4SLinus Torvalds 831da177e4SLinus Torvalds /* Saved opcode (which has been replaced with breakpoint) */ 841da177e4SLinus Torvalds kprobe_opcode_t opcode; 851da177e4SLinus Torvalds 861da177e4SLinus Torvalds /* copy of the original instruction */ 871da177e4SLinus Torvalds struct arch_specific_insn ainsn; 88e8386a0cSMasami Hiramatsu 89de5bd88dSMasami Hiramatsu /* 90de5bd88dSMasami Hiramatsu * Indicates various status flags. 91de5bd88dSMasami Hiramatsu * Protected by kprobe_mutex after this kprobe is registered. 92de5bd88dSMasami Hiramatsu */ 93e8386a0cSMasami Hiramatsu u32 flags; 941da177e4SLinus Torvalds }; 951da177e4SLinus Torvalds 96e8386a0cSMasami Hiramatsu /* Kprobe status flags */ 97e8386a0cSMasami Hiramatsu #define KPROBE_FLAG_GONE 1 /* breakpoint has already gone */ 98de5bd88dSMasami Hiramatsu #define KPROBE_FLAG_DISABLED 2 /* probe is temporarily disabled */ 99afd66255SMasami Hiramatsu #define KPROBE_FLAG_OPTIMIZED 4 /* 100afd66255SMasami Hiramatsu * probe is really optimized. 101afd66255SMasami Hiramatsu * NOTE: 102afd66255SMasami Hiramatsu * this flag is only for optimized_kprobe. 103afd66255SMasami Hiramatsu */ 104ae6aa16fSMasami Hiramatsu #define KPROBE_FLAG_FTRACE 8 /* probe is using ftrace */ 105e8386a0cSMasami Hiramatsu 106de5bd88dSMasami Hiramatsu /* Has this kprobe gone ? */ 10729e8077aSMasami Hiramatsu static inline bool kprobe_gone(struct kprobe *p) 108e8386a0cSMasami Hiramatsu { 109e8386a0cSMasami Hiramatsu return p->flags & KPROBE_FLAG_GONE; 110e8386a0cSMasami Hiramatsu } 111e8386a0cSMasami Hiramatsu 112de5bd88dSMasami Hiramatsu /* Is this kprobe disabled ? */ 11329e8077aSMasami Hiramatsu static inline bool kprobe_disabled(struct kprobe *p) 114de5bd88dSMasami Hiramatsu { 115de5bd88dSMasami Hiramatsu return p->flags & (KPROBE_FLAG_DISABLED | KPROBE_FLAG_GONE); 116de5bd88dSMasami Hiramatsu } 117afd66255SMasami Hiramatsu 118afd66255SMasami Hiramatsu /* Is this kprobe really running optimized path ? */ 11929e8077aSMasami Hiramatsu static inline bool kprobe_optimized(struct kprobe *p) 120afd66255SMasami Hiramatsu { 121afd66255SMasami Hiramatsu return p->flags & KPROBE_FLAG_OPTIMIZED; 122afd66255SMasami Hiramatsu } 123ae6aa16fSMasami Hiramatsu 124ae6aa16fSMasami Hiramatsu /* Is this kprobe uses ftrace ? */ 12529e8077aSMasami Hiramatsu static inline bool kprobe_ftrace(struct kprobe *p) 126ae6aa16fSMasami Hiramatsu { 127ae6aa16fSMasami Hiramatsu return p->flags & KPROBE_FLAG_FTRACE; 128ae6aa16fSMasami Hiramatsu } 129ae6aa16fSMasami Hiramatsu 1301da177e4SLinus Torvalds /* 131b94cce92SHien Nguyen * Function-return probe - 132b94cce92SHien Nguyen * Note: 133b94cce92SHien Nguyen * User needs to provide a handler function, and initialize maxactive. 134b94cce92SHien Nguyen * maxactive - The maximum number of instances of the probed function that 135b94cce92SHien Nguyen * can be active concurrently. 136b94cce92SHien Nguyen * nmissed - tracks the number of times the probed function's return was 137b94cce92SHien Nguyen * ignored, due to maxactive being too low. 138b94cce92SHien Nguyen * 139b94cce92SHien Nguyen */ 140d741bf41SPeter Zijlstra struct kretprobe_holder { 141d741bf41SPeter Zijlstra struct kretprobe *rp; 142d741bf41SPeter Zijlstra refcount_t ref; 143d741bf41SPeter Zijlstra }; 144d741bf41SPeter Zijlstra 145b94cce92SHien Nguyen struct kretprobe { 146b94cce92SHien Nguyen struct kprobe kp; 147b94cce92SHien Nguyen kretprobe_handler_t handler; 148f47cd9b5SAbhishek Sagar kretprobe_handler_t entry_handler; 149b94cce92SHien Nguyen int maxactive; 150b94cce92SHien Nguyen int nmissed; 151f47cd9b5SAbhishek Sagar size_t data_size; 1526e426e0fSPeter Zijlstra struct freelist_head freelist; 153d741bf41SPeter Zijlstra struct kretprobe_holder *rph; 154b94cce92SHien Nguyen }; 155b94cce92SHien Nguyen 1566bbfa441SMasami Hiramatsu #define KRETPROBE_MAX_DATA_SIZE 4096 1576bbfa441SMasami Hiramatsu 158b94cce92SHien Nguyen struct kretprobe_instance { 159b3388178SMasami Hiramatsu union { 1606e426e0fSPeter Zijlstra struct freelist_node freelist; 161b3388178SMasami Hiramatsu struct rcu_head rcu; 162b3388178SMasami Hiramatsu }; 1636e426e0fSPeter Zijlstra struct llist_node llist; 164d741bf41SPeter Zijlstra struct kretprobe_holder *rph; 165802eae7cSRusty Lynch kprobe_opcode_t *ret_addr; 1663ff9c075SMasami Hiramatsu void *fp; 16767a862a9SGustavo A. R. Silva char data[]; 168b94cce92SHien Nguyen }; 169b94cce92SHien Nguyen 170f438d914SMasami Hiramatsu struct kretprobe_blackpoint { 171f438d914SMasami Hiramatsu const char *name; 172f438d914SMasami Hiramatsu void *addr; 173f438d914SMasami Hiramatsu }; 1743d8d996eSSrinivasa Ds 175376e2424SMasami Hiramatsu struct kprobe_blacklist_entry { 176376e2424SMasami Hiramatsu struct list_head list; 1773d8d996eSSrinivasa Ds unsigned long start_addr; 178376e2424SMasami Hiramatsu unsigned long end_addr; 1793d8d996eSSrinivasa Ds }; 1803d8d996eSSrinivasa Ds 181dc19835dSMasami Hiramatsu #ifdef CONFIG_KPROBES 182dc19835dSMasami Hiramatsu DECLARE_PER_CPU(struct kprobe *, current_kprobe); 183dc19835dSMasami Hiramatsu DECLARE_PER_CPU(struct kprobe_ctlblk, kprobe_ctlblk); 184dc19835dSMasami Hiramatsu 18566ada2ccSMasami Hiramatsu extern void kprobe_busy_begin(void); 18666ada2ccSMasami Hiramatsu extern void kprobe_busy_end(void); 18766ada2ccSMasami Hiramatsu 188dc19835dSMasami Hiramatsu #ifdef CONFIG_KRETPROBES 189dc19835dSMasami Hiramatsu extern void arch_prepare_kretprobe(struct kretprobe_instance *ri, 190dc19835dSMasami Hiramatsu struct pt_regs *regs); 191dc19835dSMasami Hiramatsu extern int arch_trampoline_kprobe(struct kprobe *p); 19266ada2ccSMasami Hiramatsu 193bf094cffSMasami Hiramatsu void arch_kretprobe_fixup_return(struct pt_regs *regs, 194bf094cffSMasami Hiramatsu kprobe_opcode_t *correct_ret_addr); 195bf094cffSMasami Hiramatsu 196adf8a61aSMasami Hiramatsu void __kretprobe_trampoline(void); 19796fed8acSMasami Hiramatsu /* 19896fed8acSMasami Hiramatsu * Since some architecture uses structured function pointer, 19996fed8acSMasami Hiramatsu * use dereference_function_descriptor() to get real function address. 20096fed8acSMasami Hiramatsu */ 20196fed8acSMasami Hiramatsu static nokprobe_inline void *kretprobe_trampoline_addr(void) 20296fed8acSMasami Hiramatsu { 203adf8a61aSMasami Hiramatsu return dereference_kernel_function_descriptor(__kretprobe_trampoline); 20496fed8acSMasami Hiramatsu } 20596fed8acSMasami Hiramatsu 20666ada2ccSMasami Hiramatsu /* If the trampoline handler called from a kprobe, use this version */ 20766ada2ccSMasami Hiramatsu unsigned long __kretprobe_trampoline_handler(struct pt_regs *regs, 20866ada2ccSMasami Hiramatsu void *frame_pointer); 20966ada2ccSMasami Hiramatsu 21066ada2ccSMasami Hiramatsu static nokprobe_inline 21166ada2ccSMasami Hiramatsu unsigned long kretprobe_trampoline_handler(struct pt_regs *regs, 21266ada2ccSMasami Hiramatsu void *frame_pointer) 21366ada2ccSMasami Hiramatsu { 21466ada2ccSMasami Hiramatsu unsigned long ret; 21566ada2ccSMasami Hiramatsu /* 21666ada2ccSMasami Hiramatsu * Set a dummy kprobe for avoiding kretprobe recursion. 21766ada2ccSMasami Hiramatsu * Since kretprobe never runs in kprobe handler, no kprobe must 21866ada2ccSMasami Hiramatsu * be running at this point. 21966ada2ccSMasami Hiramatsu */ 22066ada2ccSMasami Hiramatsu kprobe_busy_begin(); 22196fed8acSMasami Hiramatsu ret = __kretprobe_trampoline_handler(regs, frame_pointer); 22266ada2ccSMasami Hiramatsu kprobe_busy_end(); 22366ada2ccSMasami Hiramatsu 22466ada2ccSMasami Hiramatsu return ret; 22566ada2ccSMasami Hiramatsu } 22666ada2ccSMasami Hiramatsu 227d741bf41SPeter Zijlstra static nokprobe_inline struct kretprobe *get_kretprobe(struct kretprobe_instance *ri) 228d741bf41SPeter Zijlstra { 229d741bf41SPeter Zijlstra RCU_LOCKDEP_WARN(!rcu_read_lock_any_held(), 230d741bf41SPeter Zijlstra "Kretprobe is accessed from instance under preemptive context"); 231d741bf41SPeter Zijlstra 232d741bf41SPeter Zijlstra return READ_ONCE(ri->rph->rp); 233d741bf41SPeter Zijlstra } 234d741bf41SPeter Zijlstra 235223a76b2SMasami Hiramatsu #else /* !CONFIG_KRETPROBES */ 236dc19835dSMasami Hiramatsu static inline void arch_prepare_kretprobe(struct kretprobe *rp, 237dc19835dSMasami Hiramatsu struct pt_regs *regs) 238dc19835dSMasami Hiramatsu { 239dc19835dSMasami Hiramatsu } 240dc19835dSMasami Hiramatsu static inline int arch_trampoline_kprobe(struct kprobe *p) 241dc19835dSMasami Hiramatsu { 242dc19835dSMasami Hiramatsu return 0; 243dc19835dSMasami Hiramatsu } 244dc19835dSMasami Hiramatsu #endif /* CONFIG_KRETPROBES */ 245dc19835dSMasami Hiramatsu 246223a76b2SMasami Hiramatsu /* Markers of '_kprobe_blacklist' section */ 247223a76b2SMasami Hiramatsu extern unsigned long __start_kprobe_blacklist[]; 248223a76b2SMasami Hiramatsu extern unsigned long __stop_kprobe_blacklist[]; 249223a76b2SMasami Hiramatsu 250f438d914SMasami Hiramatsu extern struct kretprobe_blackpoint kretprobe_blacklist[]; 251f438d914SMasami Hiramatsu 2528c1c9356SAnanth N Mavinakayanahalli #ifdef CONFIG_KPROBES_SANITY_TEST 2538c1c9356SAnanth N Mavinakayanahalli extern int init_test_probes(void); 254223a76b2SMasami Hiramatsu #else /* !CONFIG_KPROBES_SANITY_TEST */ 2558c1c9356SAnanth N Mavinakayanahalli static inline int init_test_probes(void) 2568c1c9356SAnanth N Mavinakayanahalli { 2578c1c9356SAnanth N Mavinakayanahalli return 0; 2588c1c9356SAnanth N Mavinakayanahalli } 2598c1c9356SAnanth N Mavinakayanahalli #endif /* CONFIG_KPROBES_SANITY_TEST */ 2608c1c9356SAnanth N Mavinakayanahalli 2611da177e4SLinus Torvalds extern int arch_prepare_kprobe(struct kprobe *p); 2627e1048b1SRusty Lynch extern void arch_arm_kprobe(struct kprobe *p); 2637e1048b1SRusty Lynch extern void arch_disarm_kprobe(struct kprobe *p); 2646772926bSRusty Lynch extern int arch_init_kprobes(void); 265bf8d5c52SKeshavamurthy Anil S extern void kprobes_inc_nmissed_count(struct kprobe *p); 266be8f2743SMasami Hiramatsu extern bool arch_within_kprobe_blacklist(unsigned long addr); 267fb1a59faSMasami Hiramatsu extern int arch_populate_kprobe_blacklist(void); 268659b957fSNaveen N. Rao extern bool arch_kprobe_on_func_entry(unsigned long offset); 26997c753e6SMasami Hiramatsu extern int kprobe_on_func_entry(kprobe_opcode_t *addr, const char *sym, unsigned long offset); 2701da177e4SLinus Torvalds 271e5779e8eSAndy Lutomirski extern bool within_kprobe_blacklist(unsigned long addr); 272fb1a59faSMasami Hiramatsu extern int kprobe_add_ksym_blacklist(unsigned long entry); 273fb1a59faSMasami Hiramatsu extern int kprobe_add_area_blacklist(unsigned long start, unsigned long end); 274e5779e8eSAndy Lutomirski 275c802d64aSHeiko Carstens struct kprobe_insn_cache { 276c802d64aSHeiko Carstens struct mutex mutex; 277af96397dSHeiko Carstens void *(*alloc)(void); /* allocate insn page */ 278af96397dSHeiko Carstens void (*free)(void *); /* free insn page */ 279d002b8bcSAdrian Hunter const char *sym; /* symbol for insn pages */ 280c802d64aSHeiko Carstens struct list_head pages; /* list of kprobe_insn_page */ 281c802d64aSHeiko Carstens size_t insn_size; /* size of instruction slot */ 282c802d64aSHeiko Carstens int nr_garbage; 283c802d64aSHeiko Carstens }; 284c802d64aSHeiko Carstens 2855b485629SMasami Hiramatsu #ifdef __ARCH_WANT_KPROBES_INSN_SLOT 286c802d64aSHeiko Carstens extern kprobe_opcode_t *__get_insn_slot(struct kprobe_insn_cache *c); 287c802d64aSHeiko Carstens extern void __free_insn_slot(struct kprobe_insn_cache *c, 288c802d64aSHeiko Carstens kprobe_opcode_t *slot, int dirty); 2895b485629SMasami Hiramatsu /* sleep-less address checking routine */ 2905b485629SMasami Hiramatsu extern bool __is_insn_slot_addr(struct kprobe_insn_cache *c, 2915b485629SMasami Hiramatsu unsigned long addr); 292c802d64aSHeiko Carstens 293c802d64aSHeiko Carstens #define DEFINE_INSN_CACHE_OPS(__name) \ 294c802d64aSHeiko Carstens extern struct kprobe_insn_cache kprobe_##__name##_slots; \ 295c802d64aSHeiko Carstens \ 296c802d64aSHeiko Carstens static inline kprobe_opcode_t *get_##__name##_slot(void) \ 297c802d64aSHeiko Carstens { \ 298c802d64aSHeiko Carstens return __get_insn_slot(&kprobe_##__name##_slots); \ 299c802d64aSHeiko Carstens } \ 300c802d64aSHeiko Carstens \ 301c802d64aSHeiko Carstens static inline void free_##__name##_slot(kprobe_opcode_t *slot, int dirty)\ 302c802d64aSHeiko Carstens { \ 303c802d64aSHeiko Carstens __free_insn_slot(&kprobe_##__name##_slots, slot, dirty); \ 304c802d64aSHeiko Carstens } \ 3055b485629SMasami Hiramatsu \ 3065b485629SMasami Hiramatsu static inline bool is_kprobe_##__name##_slot(unsigned long addr) \ 3075b485629SMasami Hiramatsu { \ 3085b485629SMasami Hiramatsu return __is_insn_slot_addr(&kprobe_##__name##_slots, addr); \ 3095b485629SMasami Hiramatsu } 310d002b8bcSAdrian Hunter #define KPROBE_INSN_PAGE_SYM "kprobe_insn_page" 311d002b8bcSAdrian Hunter #define KPROBE_OPTINSN_PAGE_SYM "kprobe_optinsn_page" 312d002b8bcSAdrian Hunter int kprobe_cache_get_kallsym(struct kprobe_insn_cache *c, unsigned int *symnum, 313d002b8bcSAdrian Hunter unsigned long *value, char *type, char *sym); 314223a76b2SMasami Hiramatsu #else /* !__ARCH_WANT_KPROBES_INSN_SLOT */ 3155b485629SMasami Hiramatsu #define DEFINE_INSN_CACHE_OPS(__name) \ 3165b485629SMasami Hiramatsu static inline bool is_kprobe_##__name##_slot(unsigned long addr) \ 3175b485629SMasami Hiramatsu { \ 3185b485629SMasami Hiramatsu return 0; \ 3195b485629SMasami Hiramatsu } 3205b485629SMasami Hiramatsu #endif 321c802d64aSHeiko Carstens 322c802d64aSHeiko Carstens DEFINE_INSN_CACHE_OPS(insn); 323c802d64aSHeiko Carstens 324afd66255SMasami Hiramatsu #ifdef CONFIG_OPTPROBES 325afd66255SMasami Hiramatsu /* 326afd66255SMasami Hiramatsu * Internal structure for direct jump optimized probe 327afd66255SMasami Hiramatsu */ 328afd66255SMasami Hiramatsu struct optimized_kprobe { 329afd66255SMasami Hiramatsu struct kprobe kp; 330afd66255SMasami Hiramatsu struct list_head list; /* list for optimizing queue */ 331afd66255SMasami Hiramatsu struct arch_optimized_insn optinsn; 332afd66255SMasami Hiramatsu }; 333afd66255SMasami Hiramatsu 334afd66255SMasami Hiramatsu /* Architecture dependent functions for direct jump optimization */ 335afd66255SMasami Hiramatsu extern int arch_prepared_optinsn(struct arch_optimized_insn *optinsn); 336afd66255SMasami Hiramatsu extern int arch_check_optimized_kprobe(struct optimized_kprobe *op); 337cbf6ab52SMasami Hiramatsu extern int arch_prepare_optimized_kprobe(struct optimized_kprobe *op, 338cbf6ab52SMasami Hiramatsu struct kprobe *orig); 339afd66255SMasami Hiramatsu extern void arch_remove_optimized_kprobe(struct optimized_kprobe *op); 340cd7ebe22SMasami Hiramatsu extern void arch_optimize_kprobes(struct list_head *oplist); 341f984ba4eSMasami Hiramatsu extern void arch_unoptimize_kprobes(struct list_head *oplist, 342f984ba4eSMasami Hiramatsu struct list_head *done_list); 343afd66255SMasami Hiramatsu extern void arch_unoptimize_kprobe(struct optimized_kprobe *op); 344afd66255SMasami Hiramatsu extern int arch_within_optimized_kprobe(struct optimized_kprobe *op, 345c42421e2SMasami Hiramatsu kprobe_opcode_t *addr); 346afd66255SMasami Hiramatsu 347afd66255SMasami Hiramatsu extern void opt_pre_handler(struct kprobe *p, struct pt_regs *regs); 348b2be84dfSMasami Hiramatsu 349c802d64aSHeiko Carstens DEFINE_INSN_CACHE_OPS(optinsn); 350c802d64aSHeiko Carstens 35130e7d894SThomas Gleixner extern void wait_for_kprobe_optimizer(void); 352223a76b2SMasami Hiramatsu #else /* !CONFIG_OPTPROBES */ 35330e7d894SThomas Gleixner static inline void wait_for_kprobe_optimizer(void) { } 354afd66255SMasami Hiramatsu #endif /* CONFIG_OPTPROBES */ 355223a76b2SMasami Hiramatsu 356e7dbfe34SMasami Hiramatsu #ifdef CONFIG_KPROBES_ON_FTRACE 357ae6aa16fSMasami Hiramatsu extern void kprobe_ftrace_handler(unsigned long ip, unsigned long parent_ip, 358d19ad077SSteven Rostedt (VMware) struct ftrace_ops *ops, struct ftrace_regs *fregs); 359ae6aa16fSMasami Hiramatsu extern int arch_prepare_kprobe_ftrace(struct kprobe *p); 36002afb8d6SPunit Agrawal #else 36102afb8d6SPunit Agrawal static inline int arch_prepare_kprobe_ftrace(struct kprobe *p) 36202afb8d6SPunit Agrawal { 36302afb8d6SPunit Agrawal return -EINVAL; 36402afb8d6SPunit Agrawal } 365223a76b2SMasami Hiramatsu #endif /* CONFIG_KPROBES_ON_FTRACE */ 366ae6aa16fSMasami Hiramatsu 367d217d545SAnanth N Mavinakayanahalli /* Get the kprobe at this addr (if any) - called with preemption disabled */ 3681da177e4SLinus Torvalds struct kprobe *get_kprobe(void *addr); 3691da177e4SLinus Torvalds 370e6584523SAnanth N Mavinakayanahalli /* kprobe_running() will just return the current_kprobe on this CPU */ 371e6584523SAnanth N Mavinakayanahalli static inline struct kprobe *kprobe_running(void) 372e6584523SAnanth N Mavinakayanahalli { 373223a76b2SMasami Hiramatsu return __this_cpu_read(current_kprobe); 374e6584523SAnanth N Mavinakayanahalli } 375e6584523SAnanth N Mavinakayanahalli 376e6584523SAnanth N Mavinakayanahalli static inline void reset_current_kprobe(void) 377e6584523SAnanth N Mavinakayanahalli { 378b76834bcSChristoph Lameter __this_cpu_write(current_kprobe, NULL); 379e6584523SAnanth N Mavinakayanahalli } 380e6584523SAnanth N Mavinakayanahalli 381e6584523SAnanth N Mavinakayanahalli static inline struct kprobe_ctlblk *get_kprobe_ctlblk(void) 382e6584523SAnanth N Mavinakayanahalli { 383bdffd893SChristoph Lameter return this_cpu_ptr(&kprobe_ctlblk); 384e6584523SAnanth N Mavinakayanahalli } 385e6584523SAnanth N Mavinakayanahalli 386290e3070SNaveen N. Rao kprobe_opcode_t *kprobe_lookup_name(const char *name, unsigned int offset); 3871da177e4SLinus Torvalds int register_kprobe(struct kprobe *p); 3881da177e4SLinus Torvalds void unregister_kprobe(struct kprobe *p); 3899861668fSMasami Hiramatsu int register_kprobes(struct kprobe **kps, int num); 3909861668fSMasami Hiramatsu void unregister_kprobes(struct kprobe **kps, int num); 3911da177e4SLinus Torvalds 392b94cce92SHien Nguyen int register_kretprobe(struct kretprobe *rp); 393b94cce92SHien Nguyen void unregister_kretprobe(struct kretprobe *rp); 3944a296e07SMasami Hiramatsu int register_kretprobes(struct kretprobe **rps, int num); 3954a296e07SMasami Hiramatsu void unregister_kretprobes(struct kretprobe **rps, int num); 396b94cce92SHien Nguyen 397b94cce92SHien Nguyen void kprobe_flush_task(struct task_struct *tk); 3988c1c9356SAnanth N Mavinakayanahalli 39982d083abSMasami Hiramatsu void kprobe_free_init_mem(void); 40082d083abSMasami Hiramatsu 401de5bd88dSMasami Hiramatsu int disable_kprobe(struct kprobe *kp); 402de5bd88dSMasami Hiramatsu int enable_kprobe(struct kprobe *kp); 403de5bd88dSMasami Hiramatsu 40424851d24SFrederic Weisbecker void dump_kprobe(struct kprobe *kp); 40524851d24SFrederic Weisbecker 406ad3bc25aSBorislav Petkov void *alloc_insn_page(void); 407ad3bc25aSBorislav Petkov 4087ee3e97eSChristophe Leroy void *alloc_optinsn_page(void); 4097ee3e97eSChristophe Leroy void free_optinsn_page(void *page); 4107ee3e97eSChristophe Leroy 411d002b8bcSAdrian Hunter int kprobe_get_kallsym(unsigned int symnum, unsigned long *value, char *type, 412d002b8bcSAdrian Hunter char *sym); 413d002b8bcSAdrian Hunter 414d002b8bcSAdrian Hunter int arch_kprobe_get_kallsym(unsigned int *symnum, unsigned long *value, 415d002b8bcSAdrian Hunter char *type, char *sym); 416b1801812SIngo Molnar #else /* !CONFIG_KPROBES: */ 41700d7c05aSKeshavamurthy Anil S 418b1801812SIngo Molnar static inline int kprobe_fault_handler(struct pt_regs *regs, int trapnr) 419b1801812SIngo Molnar { 420b1801812SIngo Molnar return 0; 421b1801812SIngo Molnar } 422785656a4SAbhishek Sagar static inline struct kprobe *get_kprobe(void *addr) 423785656a4SAbhishek Sagar { 424785656a4SAbhishek Sagar return NULL; 425785656a4SAbhishek Sagar } 426e6584523SAnanth N Mavinakayanahalli static inline struct kprobe *kprobe_running(void) 4271da177e4SLinus Torvalds { 428e6584523SAnanth N Mavinakayanahalli return NULL; 4291da177e4SLinus Torvalds } 430*ab51e15dSMasami Hiramatsu #define kprobe_busy_begin() do {} while (0) 431*ab51e15dSMasami Hiramatsu #define kprobe_busy_end() do {} while (0) 432*ab51e15dSMasami Hiramatsu 4331da177e4SLinus Torvalds static inline int register_kprobe(struct kprobe *p) 4341da177e4SLinus Torvalds { 435223a76b2SMasami Hiramatsu return -EOPNOTSUPP; 4361da177e4SLinus Torvalds } 4379861668fSMasami Hiramatsu static inline int register_kprobes(struct kprobe **kps, int num) 4389861668fSMasami Hiramatsu { 439223a76b2SMasami Hiramatsu return -EOPNOTSUPP; 4409861668fSMasami Hiramatsu } 4411da177e4SLinus Torvalds static inline void unregister_kprobe(struct kprobe *p) 4421da177e4SLinus Torvalds { 4431da177e4SLinus Torvalds } 4449861668fSMasami Hiramatsu static inline void unregister_kprobes(struct kprobe **kps, int num) 4459861668fSMasami Hiramatsu { 4469861668fSMasami Hiramatsu } 447b94cce92SHien Nguyen static inline int register_kretprobe(struct kretprobe *rp) 448b94cce92SHien Nguyen { 449223a76b2SMasami Hiramatsu return -EOPNOTSUPP; 450b94cce92SHien Nguyen } 4514a296e07SMasami Hiramatsu static inline int register_kretprobes(struct kretprobe **rps, int num) 4524a296e07SMasami Hiramatsu { 453223a76b2SMasami Hiramatsu return -EOPNOTSUPP; 4544a296e07SMasami Hiramatsu } 455b94cce92SHien Nguyen static inline void unregister_kretprobe(struct kretprobe *rp) 456b94cce92SHien Nguyen { 457b94cce92SHien Nguyen } 4584a296e07SMasami Hiramatsu static inline void unregister_kretprobes(struct kretprobe **rps, int num) 4594a296e07SMasami Hiramatsu { 4604a296e07SMasami Hiramatsu } 461b94cce92SHien Nguyen static inline void kprobe_flush_task(struct task_struct *tk) 462b94cce92SHien Nguyen { 463b94cce92SHien Nguyen } 46482d083abSMasami Hiramatsu static inline void kprobe_free_init_mem(void) 46582d083abSMasami Hiramatsu { 46682d083abSMasami Hiramatsu } 467de5bd88dSMasami Hiramatsu static inline int disable_kprobe(struct kprobe *kp) 468de5bd88dSMasami Hiramatsu { 469223a76b2SMasami Hiramatsu return -EOPNOTSUPP; 470de5bd88dSMasami Hiramatsu } 471de5bd88dSMasami Hiramatsu static inline int enable_kprobe(struct kprobe *kp) 472de5bd88dSMasami Hiramatsu { 473223a76b2SMasami Hiramatsu return -EOPNOTSUPP; 474de5bd88dSMasami Hiramatsu } 475fab94075SBorislav Petkov 476fab94075SBorislav Petkov static inline bool within_kprobe_blacklist(unsigned long addr) 477fab94075SBorislav Petkov { 478fab94075SBorislav Petkov return true; 479fab94075SBorislav Petkov } 480d002b8bcSAdrian Hunter static inline int kprobe_get_kallsym(unsigned int symnum, unsigned long *value, 481d002b8bcSAdrian Hunter char *type, char *sym) 482d002b8bcSAdrian Hunter { 483d002b8bcSAdrian Hunter return -ERANGE; 484d002b8bcSAdrian Hunter } 485b94cce92SHien Nguyen #endif /* CONFIG_KPROBES */ 486223a76b2SMasami Hiramatsu 4878f9b1528SMasami Hiramatsu static inline int disable_kretprobe(struct kretprobe *rp) 4888f9b1528SMasami Hiramatsu { 4898f9b1528SMasami Hiramatsu return disable_kprobe(&rp->kp); 4908f9b1528SMasami Hiramatsu } 4918f9b1528SMasami Hiramatsu static inline int enable_kretprobe(struct kretprobe *rp) 4928f9b1528SMasami Hiramatsu { 4938f9b1528SMasami Hiramatsu return enable_kprobe(&rp->kp); 4948f9b1528SMasami Hiramatsu } 4958f9b1528SMasami Hiramatsu 4965b485629SMasami Hiramatsu #ifndef CONFIG_KPROBES 4975b485629SMasami Hiramatsu static inline bool is_kprobe_insn_slot(unsigned long addr) 4985b485629SMasami Hiramatsu { 4995b485629SMasami Hiramatsu return false; 5005b485629SMasami Hiramatsu } 501223a76b2SMasami Hiramatsu #endif /* !CONFIG_KPROBES */ 502223a76b2SMasami Hiramatsu 5035b485629SMasami Hiramatsu #ifndef CONFIG_OPTPROBES 5045b485629SMasami Hiramatsu static inline bool is_kprobe_optinsn_slot(unsigned long addr) 5055b485629SMasami Hiramatsu { 5065b485629SMasami Hiramatsu return false; 5075b485629SMasami Hiramatsu } 508223a76b2SMasami Hiramatsu #endif /* !CONFIG_OPTPROBES */ 5095b485629SMasami Hiramatsu 51003bac0dfSMasami Hiramatsu #ifdef CONFIG_KRETPROBES 51103bac0dfSMasami Hiramatsu static nokprobe_inline bool is_kretprobe_trampoline(unsigned long addr) 51203bac0dfSMasami Hiramatsu { 51303bac0dfSMasami Hiramatsu return (void *)addr == kretprobe_trampoline_addr(); 51403bac0dfSMasami Hiramatsu } 51503bac0dfSMasami Hiramatsu 51603bac0dfSMasami Hiramatsu unsigned long kretprobe_find_ret_addr(struct task_struct *tsk, void *fp, 51703bac0dfSMasami Hiramatsu struct llist_node **cur); 51803bac0dfSMasami Hiramatsu #else 51903bac0dfSMasami Hiramatsu static nokprobe_inline bool is_kretprobe_trampoline(unsigned long addr) 52003bac0dfSMasami Hiramatsu { 52103bac0dfSMasami Hiramatsu return false; 52203bac0dfSMasami Hiramatsu } 52303bac0dfSMasami Hiramatsu 52403bac0dfSMasami Hiramatsu static nokprobe_inline 52503bac0dfSMasami Hiramatsu unsigned long kretprobe_find_ret_addr(struct task_struct *tsk, void *fp, 52603bac0dfSMasami Hiramatsu struct llist_node **cur) 52703bac0dfSMasami Hiramatsu { 52803bac0dfSMasami Hiramatsu return 0; 52903bac0dfSMasami Hiramatsu } 53003bac0dfSMasami Hiramatsu #endif 53103bac0dfSMasami Hiramatsu 532b98cca44SAnshuman Khandual /* Returns true if kprobes handled the fault */ 533b98cca44SAnshuman Khandual static nokprobe_inline bool kprobe_page_fault(struct pt_regs *regs, 534b98cca44SAnshuman Khandual unsigned int trap) 535b98cca44SAnshuman Khandual { 536dfc05b55SMasami Hiramatsu if (!IS_ENABLED(CONFIG_KPROBES)) 537b98cca44SAnshuman Khandual return false; 538b98cca44SAnshuman Khandual if (user_mode(regs)) 539b98cca44SAnshuman Khandual return false; 540b98cca44SAnshuman Khandual /* 541b98cca44SAnshuman Khandual * To be potentially processing a kprobe fault and to be allowed 542b98cca44SAnshuman Khandual * to call kprobe_running(), we have to be non-preemptible. 543b98cca44SAnshuman Khandual */ 544b98cca44SAnshuman Khandual if (preemptible()) 545b98cca44SAnshuman Khandual return false; 546b98cca44SAnshuman Khandual if (!kprobe_running()) 547b98cca44SAnshuman Khandual return false; 548b98cca44SAnshuman Khandual return kprobe_fault_handler(regs, trap); 549b98cca44SAnshuman Khandual } 550b98cca44SAnshuman Khandual 5511da177e4SLinus Torvalds #endif /* _LINUX_KPROBES_H */ 552