aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/kprobes.c
diff options
context:
space:
mode:
Diffstat (limited to 'kernel/kprobes.c')
-rw-r--r--kernel/kprobes.c64
1 files changed, 21 insertions, 43 deletions
diff --git a/kernel/kprobes.c b/kernel/kprobes.c
index 22857003a65b..f58f171bd65f 100644
--- a/kernel/kprobes.c
+++ b/kernel/kprobes.c
@@ -358,46 +358,6 @@ void __kprobes kprobes_inc_nmissed_count(struct kprobe *p)
358} 358}
359 359
360/* Called with kretprobe_lock held */ 360/* Called with kretprobe_lock held */
361struct kretprobe_instance __kprobes *get_free_rp_inst(struct kretprobe *rp)
362{
363 struct hlist_node *node;
364 struct kretprobe_instance *ri;
365 hlist_for_each_entry(ri, node, &rp->free_instances, uflist)
366 return ri;
367 return NULL;
368}
369
370/* Called with kretprobe_lock held */
371static struct kretprobe_instance __kprobes *get_used_rp_inst(struct kretprobe
372 *rp)
373{
374 struct hlist_node *node;
375 struct kretprobe_instance *ri;
376 hlist_for_each_entry(ri, node, &rp->used_instances, uflist)
377 return ri;
378 return NULL;
379}
380
381/* Called with kretprobe_lock held */
382void __kprobes add_rp_inst(struct kretprobe_instance *ri)
383{
384 /*
385 * Remove rp inst off the free list -
386 * Add it back when probed function returns
387 */
388 hlist_del(&ri->uflist);
389
390 /* Add rp inst onto table */
391 INIT_HLIST_NODE(&ri->hlist);
392 hlist_add_head(&ri->hlist,
393 &kretprobe_inst_table[hash_ptr(ri->task, KPROBE_HASH_BITS)]);
394
395 /* Also add this rp inst to the used list. */
396 INIT_HLIST_NODE(&ri->uflist);
397 hlist_add_head(&ri->uflist, &ri->rp->used_instances);
398}
399
400/* Called with kretprobe_lock held */
401void __kprobes recycle_rp_inst(struct kretprobe_instance *ri, 361void __kprobes recycle_rp_inst(struct kretprobe_instance *ri,
402 struct hlist_head *head) 362 struct hlist_head *head)
403{ 363{
@@ -450,7 +410,9 @@ void __kprobes kprobe_flush_task(struct task_struct *tk)
450static inline void free_rp_inst(struct kretprobe *rp) 410static inline void free_rp_inst(struct kretprobe *rp)
451{ 411{
452 struct kretprobe_instance *ri; 412 struct kretprobe_instance *ri;
453 while ((ri = get_free_rp_inst(rp)) != NULL) { 413 struct hlist_node *pos, *next;
414
415 hlist_for_each_entry_safe(ri, pos, next, &rp->free_instances, uflist) {
454 hlist_del(&ri->uflist); 416 hlist_del(&ri->uflist);
455 kfree(ri); 417 kfree(ri);
456 } 418 }
@@ -732,7 +694,21 @@ static int __kprobes pre_handler_kretprobe(struct kprobe *p,
732 694
733 /*TODO: consider to only swap the RA after the last pre_handler fired */ 695 /*TODO: consider to only swap the RA after the last pre_handler fired */
734 spin_lock_irqsave(&kretprobe_lock, flags); 696 spin_lock_irqsave(&kretprobe_lock, flags);
735 arch_prepare_kretprobe(rp, regs); 697 if (!hlist_empty(&rp->free_instances)) {
698 struct kretprobe_instance *ri;
699
700 ri = hlist_entry(rp->free_instances.first,
701 struct kretprobe_instance, uflist);
702 ri->rp = rp;
703 ri->task = current;
704 arch_prepare_kretprobe(ri, regs);
705
706 /* XXX(hch): why is there no hlist_move_head? */
707 hlist_del(&ri->uflist);
708 hlist_add_head(&ri->uflist, &ri->rp->used_instances);
709 hlist_add_head(&ri->hlist, kretprobe_inst_table_head(ri->task));
710 } else
711 rp->nmissed++;
736 spin_unlock_irqrestore(&kretprobe_lock, flags); 712 spin_unlock_irqrestore(&kretprobe_lock, flags);
737 return 0; 713 return 0;
738} 714}
@@ -795,11 +771,13 @@ void __kprobes unregister_kretprobe(struct kretprobe *rp)
795{ 771{
796 unsigned long flags; 772 unsigned long flags;
797 struct kretprobe_instance *ri; 773 struct kretprobe_instance *ri;
774 struct hlist_node *pos, *next;
798 775
799 unregister_kprobe(&rp->kp); 776 unregister_kprobe(&rp->kp);
777
800 /* No race here */ 778 /* No race here */
801 spin_lock_irqsave(&kretprobe_lock, flags); 779 spin_lock_irqsave(&kretprobe_lock, flags);
802 while ((ri = get_used_rp_inst(rp)) != NULL) { 780 hlist_for_each_entry_safe(ri, pos, next, &rp->used_instances, uflist) {
803 ri->rp = NULL; 781 ri->rp = NULL;
804 hlist_del(&ri->uflist); 782 hlist_del(&ri->uflist);
805 } 783 }