xref: /linux/include/linux/kprobes.h (revision ab51e15d535e07be9839e0df056a4ebe9c5bac83)
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