diff options
Diffstat (limited to 'arch/x86_64/kernel/kprobes.c')
-rw-r--r-- | arch/x86_64/kernel/kprobes.c | 98 |
1 files changed, 97 insertions, 1 deletions
diff --git a/arch/x86_64/kernel/kprobes.c b/arch/x86_64/kernel/kprobes.c index f77f8a0ff187..203672ca7401 100644 --- a/arch/x86_64/kernel/kprobes.c +++ b/arch/x86_64/kernel/kprobes.c | |||
@@ -27,6 +27,8 @@ | |||
27 | * <prasanna@in.ibm.com> adapted for x86_64 | 27 | * <prasanna@in.ibm.com> adapted for x86_64 |
28 | * 2005-Mar Roland McGrath <roland@redhat.com> | 28 | * 2005-Mar Roland McGrath <roland@redhat.com> |
29 | * Fixed to handle %rip-relative addressing mode correctly. | 29 | * Fixed to handle %rip-relative addressing mode correctly. |
30 | * 2005-May Rusty Lynch <rusty.lynch@intel.com> | ||
31 | * Added function return probes functionality | ||
30 | */ | 32 | */ |
31 | 33 | ||
32 | #include <linux/config.h> | 34 | #include <linux/config.h> |
@@ -240,6 +242,50 @@ static void prepare_singlestep(struct kprobe *p, struct pt_regs *regs) | |||
240 | regs->rip = (unsigned long)p->ainsn.insn; | 242 | regs->rip = (unsigned long)p->ainsn.insn; |
241 | } | 243 | } |
242 | 244 | ||
245 | struct task_struct *arch_get_kprobe_task(void *ptr) | ||
246 | { | ||
247 | return ((struct thread_info *) (((unsigned long) ptr) & | ||
248 | (~(THREAD_SIZE -1))))->task; | ||
249 | } | ||
250 | |||
251 | void arch_prepare_kretprobe(struct kretprobe *rp, struct pt_regs *regs) | ||
252 | { | ||
253 | unsigned long *sara = (unsigned long *)regs->rsp; | ||
254 | struct kretprobe_instance *ri; | ||
255 | static void *orig_ret_addr; | ||
256 | |||
257 | /* | ||
258 | * Save the return address when the return probe hits | ||
259 | * the first time, and use it to populate the (krprobe | ||
260 | * instance)->ret_addr for subsequent return probes at | ||
261 | * the same addrress since stack address would have | ||
262 | * the kretprobe_trampoline by then. | ||
263 | */ | ||
264 | if (((void*) *sara) != kretprobe_trampoline) | ||
265 | orig_ret_addr = (void*) *sara; | ||
266 | |||
267 | if ((ri = get_free_rp_inst(rp)) != NULL) { | ||
268 | ri->rp = rp; | ||
269 | ri->stack_addr = sara; | ||
270 | ri->ret_addr = orig_ret_addr; | ||
271 | add_rp_inst(ri); | ||
272 | /* Replace the return addr with trampoline addr */ | ||
273 | *sara = (unsigned long) &kretprobe_trampoline; | ||
274 | } else { | ||
275 | rp->nmissed++; | ||
276 | } | ||
277 | } | ||
278 | |||
279 | void arch_kprobe_flush_task(struct task_struct *tk) | ||
280 | { | ||
281 | struct kretprobe_instance *ri; | ||
282 | while ((ri = get_rp_inst_tsk(tk)) != NULL) { | ||
283 | *((unsigned long *)(ri->stack_addr)) = | ||
284 | (unsigned long) ri->ret_addr; | ||
285 | recycle_rp_inst(ri); | ||
286 | } | ||
287 | } | ||
288 | |||
243 | /* | 289 | /* |
244 | * Interrupts are disabled on entry as trap3 is an interrupt gate and they | 290 | * Interrupts are disabled on entry as trap3 is an interrupt gate and they |
245 | * remain disabled thorough out this function. | 291 | * remain disabled thorough out this function. |
@@ -317,6 +363,55 @@ no_kprobe: | |||
317 | } | 363 | } |
318 | 364 | ||
319 | /* | 365 | /* |
366 | * For function-return probes, init_kprobes() establishes a probepoint | ||
367 | * here. When a retprobed function returns, this probe is hit and | ||
368 | * trampoline_probe_handler() runs, calling the kretprobe's handler. | ||
369 | */ | ||
370 | void kretprobe_trampoline_holder(void) | ||
371 | { | ||
372 | asm volatile ( ".global kretprobe_trampoline\n" | ||
373 | "kretprobe_trampoline: \n" | ||
374 | "nop\n"); | ||
375 | } | ||
376 | |||
377 | /* | ||
378 | * Called when we hit the probe point at kretprobe_trampoline | ||
379 | */ | ||
380 | int trampoline_probe_handler(struct kprobe *p, struct pt_regs *regs) | ||
381 | { | ||
382 | struct task_struct *tsk; | ||
383 | struct kretprobe_instance *ri; | ||
384 | struct hlist_head *head; | ||
385 | struct hlist_node *node; | ||
386 | unsigned long *sara = (unsigned long *)regs->rsp - 1; | ||
387 | |||
388 | tsk = arch_get_kprobe_task(sara); | ||
389 | head = kretprobe_inst_table_head(tsk); | ||
390 | |||
391 | hlist_for_each_entry(ri, node, head, hlist) { | ||
392 | if (ri->stack_addr == sara && ri->rp) { | ||
393 | if (ri->rp->handler) | ||
394 | ri->rp->handler(ri, regs); | ||
395 | } | ||
396 | } | ||
397 | return 0; | ||
398 | } | ||
399 | |||
400 | void trampoline_post_handler(struct kprobe *p, struct pt_regs *regs, | ||
401 | unsigned long flags) | ||
402 | { | ||
403 | struct kretprobe_instance *ri; | ||
404 | /* RA already popped */ | ||
405 | unsigned long *sara = ((unsigned long *)regs->rsp) - 1; | ||
406 | |||
407 | while ((ri = get_rp_inst(sara))) { | ||
408 | regs->rip = (unsigned long)ri->ret_addr; | ||
409 | recycle_rp_inst(ri); | ||
410 | } | ||
411 | regs->eflags &= ~TF_MASK; | ||
412 | } | ||
413 | |||
414 | /* | ||
320 | * Called after single-stepping. p->addr is the address of the | 415 | * Called after single-stepping. p->addr is the address of the |
321 | * instruction whose first byte has been replaced by the "int 3" | 416 | * instruction whose first byte has been replaced by the "int 3" |
322 | * instruction. To avoid the SMP problems that can occur when we | 417 | * instruction. To avoid the SMP problems that can occur when we |
@@ -404,7 +499,8 @@ int post_kprobe_handler(struct pt_regs *regs) | |||
404 | if (current_kprobe->post_handler) | 499 | if (current_kprobe->post_handler) |
405 | current_kprobe->post_handler(current_kprobe, regs, 0); | 500 | current_kprobe->post_handler(current_kprobe, regs, 0); |
406 | 501 | ||
407 | resume_execution(current_kprobe, regs); | 502 | if (current_kprobe->post_handler != trampoline_post_handler) |
503 | resume_execution(current_kprobe, regs); | ||
408 | regs->eflags |= kprobe_saved_rflags; | 504 | regs->eflags |= kprobe_saved_rflags; |
409 | 505 | ||
410 | unlock_kprobes(); | 506 | unlock_kprobes(); |