diff options
Diffstat (limited to 'include/linux/kprobes.h')
-rw-r--r-- | include/linux/kprobes.h | 100 |
1 files changed, 98 insertions, 2 deletions
diff --git a/include/linux/kprobes.h b/include/linux/kprobes.h index 99ddba5a4e00..5e1a7b0d7b3f 100644 --- a/include/linux/kprobes.h +++ b/include/linux/kprobes.h | |||
@@ -25,27 +25,45 @@ | |||
25 | * Rusty Russell). | 25 | * Rusty Russell). |
26 | * 2004-July Suparna Bhattacharya <suparna@in.ibm.com> added jumper probes | 26 | * 2004-July Suparna Bhattacharya <suparna@in.ibm.com> added jumper probes |
27 | * interface to access function arguments. | 27 | * interface to access function arguments. |
28 | * 2005-May Hien Nguyen <hien@us.ibm.com> and Jim Keniston | ||
29 | * <jkenisto@us.ibm.com> and Prasanna S Panchamukhi | ||
30 | * <prasanna@in.ibm.com> added function-return probes. | ||
28 | */ | 31 | */ |
29 | #include <linux/config.h> | 32 | #include <linux/config.h> |
30 | #include <linux/list.h> | 33 | #include <linux/list.h> |
31 | #include <linux/notifier.h> | 34 | #include <linux/notifier.h> |
32 | #include <linux/smp.h> | 35 | #include <linux/smp.h> |
36 | |||
33 | #include <asm/kprobes.h> | 37 | #include <asm/kprobes.h> |
34 | 38 | ||
39 | /* kprobe_status settings */ | ||
40 | #define KPROBE_HIT_ACTIVE 0x00000001 | ||
41 | #define KPROBE_HIT_SS 0x00000002 | ||
42 | #define KPROBE_REENTER 0x00000004 | ||
43 | #define KPROBE_HIT_SSDONE 0x00000008 | ||
44 | |||
35 | struct kprobe; | 45 | struct kprobe; |
36 | struct pt_regs; | 46 | struct pt_regs; |
47 | struct kretprobe; | ||
48 | struct kretprobe_instance; | ||
37 | typedef int (*kprobe_pre_handler_t) (struct kprobe *, struct pt_regs *); | 49 | typedef int (*kprobe_pre_handler_t) (struct kprobe *, struct pt_regs *); |
38 | typedef int (*kprobe_break_handler_t) (struct kprobe *, struct pt_regs *); | 50 | typedef int (*kprobe_break_handler_t) (struct kprobe *, struct pt_regs *); |
39 | typedef void (*kprobe_post_handler_t) (struct kprobe *, struct pt_regs *, | 51 | typedef void (*kprobe_post_handler_t) (struct kprobe *, struct pt_regs *, |
40 | unsigned long flags); | 52 | unsigned long flags); |
41 | typedef int (*kprobe_fault_handler_t) (struct kprobe *, struct pt_regs *, | 53 | typedef int (*kprobe_fault_handler_t) (struct kprobe *, struct pt_regs *, |
42 | int trapnr); | 54 | int trapnr); |
55 | typedef int (*kretprobe_handler_t) (struct kretprobe_instance *, | ||
56 | struct pt_regs *); | ||
57 | |||
43 | struct kprobe { | 58 | struct kprobe { |
44 | struct hlist_node hlist; | 59 | struct hlist_node hlist; |
45 | 60 | ||
46 | /* list of kprobes for multi-handler support */ | 61 | /* list of kprobes for multi-handler support */ |
47 | struct list_head list; | 62 | struct list_head list; |
48 | 63 | ||
64 | /*count the number of times this probe was temporarily disarmed */ | ||
65 | unsigned long nmissed; | ||
66 | |||
49 | /* location of the probe point */ | 67 | /* location of the probe point */ |
50 | kprobe_opcode_t *addr; | 68 | kprobe_opcode_t *addr; |
51 | 69 | ||
@@ -85,6 +103,62 @@ struct jprobe { | |||
85 | kprobe_opcode_t *entry; /* probe handling code to jump to */ | 103 | kprobe_opcode_t *entry; /* probe handling code to jump to */ |
86 | }; | 104 | }; |
87 | 105 | ||
106 | #ifdef ARCH_SUPPORTS_KRETPROBES | ||
107 | extern int trampoline_probe_handler(struct kprobe *p, struct pt_regs *regs); | ||
108 | extern void trampoline_post_handler(struct kprobe *p, struct pt_regs *regs, | ||
109 | unsigned long flags); | ||
110 | extern struct task_struct *arch_get_kprobe_task(void *ptr); | ||
111 | extern void arch_prepare_kretprobe(struct kretprobe *rp, struct pt_regs *regs); | ||
112 | extern void arch_kprobe_flush_task(struct task_struct *tk); | ||
113 | #else /* ARCH_SUPPORTS_KRETPROBES */ | ||
114 | static inline void kretprobe_trampoline(void) | ||
115 | { | ||
116 | } | ||
117 | static inline int trampoline_probe_handler(struct kprobe *p, | ||
118 | struct pt_regs *regs) | ||
119 | { | ||
120 | return 0; | ||
121 | } | ||
122 | static inline void trampoline_post_handler(struct kprobe *p, | ||
123 | struct pt_regs *regs, unsigned long flags) | ||
124 | { | ||
125 | } | ||
126 | static inline void arch_prepare_kretprobe(struct kretprobe *rp, | ||
127 | struct pt_regs *regs) | ||
128 | { | ||
129 | } | ||
130 | static inline void arch_kprobe_flush_task(struct task_struct *tk) | ||
131 | { | ||
132 | } | ||
133 | #define arch_get_kprobe_task(ptr) ((struct task_struct *)NULL) | ||
134 | #endif /* ARCH_SUPPORTS_KRETPROBES */ | ||
135 | /* | ||
136 | * Function-return probe - | ||
137 | * Note: | ||
138 | * User needs to provide a handler function, and initialize maxactive. | ||
139 | * maxactive - The maximum number of instances of the probed function that | ||
140 | * can be active concurrently. | ||
141 | * nmissed - tracks the number of times the probed function's return was | ||
142 | * ignored, due to maxactive being too low. | ||
143 | * | ||
144 | */ | ||
145 | struct kretprobe { | ||
146 | struct kprobe kp; | ||
147 | kretprobe_handler_t handler; | ||
148 | int maxactive; | ||
149 | int nmissed; | ||
150 | struct hlist_head free_instances; | ||
151 | struct hlist_head used_instances; | ||
152 | }; | ||
153 | |||
154 | struct kretprobe_instance { | ||
155 | struct hlist_node uflist; /* either on free list or used list */ | ||
156 | struct hlist_node hlist; | ||
157 | struct kretprobe *rp; | ||
158 | void *ret_addr; | ||
159 | void *stack_addr; | ||
160 | }; | ||
161 | |||
88 | #ifdef CONFIG_KPROBES | 162 | #ifdef CONFIG_KPROBES |
89 | /* Locks kprobe: irq must be disabled */ | 163 | /* Locks kprobe: irq must be disabled */ |
90 | void lock_kprobes(void); | 164 | void lock_kprobes(void); |
@@ -99,11 +173,14 @@ static inline int kprobe_running(void) | |||
99 | 173 | ||
100 | extern int arch_prepare_kprobe(struct kprobe *p); | 174 | extern int arch_prepare_kprobe(struct kprobe *p); |
101 | extern void arch_copy_kprobe(struct kprobe *p); | 175 | extern void arch_copy_kprobe(struct kprobe *p); |
176 | extern void arch_arm_kprobe(struct kprobe *p); | ||
177 | extern void arch_disarm_kprobe(struct kprobe *p); | ||
102 | extern void arch_remove_kprobe(struct kprobe *p); | 178 | extern void arch_remove_kprobe(struct kprobe *p); |
103 | extern void show_registers(struct pt_regs *regs); | 179 | extern void show_registers(struct pt_regs *regs); |
104 | 180 | ||
105 | /* Get the kprobe at this addr (if any). Must have called lock_kprobes */ | 181 | /* Get the kprobe at this addr (if any). Must have called lock_kprobes */ |
106 | struct kprobe *get_kprobe(void *addr); | 182 | struct kprobe *get_kprobe(void *addr); |
183 | struct hlist_head * kretprobe_inst_table_head(struct task_struct *tsk); | ||
107 | 184 | ||
108 | int register_kprobe(struct kprobe *p); | 185 | int register_kprobe(struct kprobe *p); |
109 | void unregister_kprobe(struct kprobe *p); | 186 | void unregister_kprobe(struct kprobe *p); |
@@ -113,7 +190,16 @@ int register_jprobe(struct jprobe *p); | |||
113 | void unregister_jprobe(struct jprobe *p); | 190 | void unregister_jprobe(struct jprobe *p); |
114 | void jprobe_return(void); | 191 | void jprobe_return(void); |
115 | 192 | ||
116 | #else | 193 | int register_kretprobe(struct kretprobe *rp); |
194 | void unregister_kretprobe(struct kretprobe *rp); | ||
195 | |||
196 | struct kretprobe_instance *get_free_rp_inst(struct kretprobe *rp); | ||
197 | struct kretprobe_instance *get_rp_inst(void *sara); | ||
198 | struct kretprobe_instance *get_rp_inst_tsk(struct task_struct *tk); | ||
199 | void add_rp_inst(struct kretprobe_instance *ri); | ||
200 | void kprobe_flush_task(struct task_struct *tk); | ||
201 | void recycle_rp_inst(struct kretprobe_instance *ri); | ||
202 | #else /* CONFIG_KPROBES */ | ||
117 | static inline int kprobe_running(void) | 203 | static inline int kprobe_running(void) |
118 | { | 204 | { |
119 | return 0; | 205 | return 0; |
@@ -135,5 +221,15 @@ static inline void unregister_jprobe(struct jprobe *p) | |||
135 | static inline void jprobe_return(void) | 221 | static inline void jprobe_return(void) |
136 | { | 222 | { |
137 | } | 223 | } |
138 | #endif | 224 | static inline int register_kretprobe(struct kretprobe *rp) |
225 | { | ||
226 | return -ENOSYS; | ||
227 | } | ||
228 | static inline void unregister_kretprobe(struct kretprobe *rp) | ||
229 | { | ||
230 | } | ||
231 | static inline void kprobe_flush_task(struct task_struct *tk) | ||
232 | { | ||
233 | } | ||
234 | #endif /* CONFIG_KPROBES */ | ||
139 | #endif /* _LINUX_KPROBES_H */ | 235 | #endif /* _LINUX_KPROBES_H */ |