diff options
Diffstat (limited to 'kernel/kprobes.c')
| -rw-r--r-- | kernel/kprobes.c | 22 |
1 files changed, 14 insertions, 8 deletions
diff --git a/kernel/kprobes.c b/kernel/kprobes.c index 282035f3ae96..6dd5359e1f0e 100644 --- a/kernel/kprobes.c +++ b/kernel/kprobes.c | |||
| @@ -399,7 +399,7 @@ static inline int kprobe_optready(struct kprobe *p) | |||
| 399 | * Return an optimized kprobe whose optimizing code replaces | 399 | * Return an optimized kprobe whose optimizing code replaces |
| 400 | * instructions including addr (exclude breakpoint). | 400 | * instructions including addr (exclude breakpoint). |
| 401 | */ | 401 | */ |
| 402 | struct kprobe *__kprobes get_optimized_kprobe(unsigned long addr) | 402 | static struct kprobe *__kprobes get_optimized_kprobe(unsigned long addr) |
| 403 | { | 403 | { |
| 404 | int i; | 404 | int i; |
| 405 | struct kprobe *p = NULL; | 405 | struct kprobe *p = NULL; |
| @@ -831,6 +831,7 @@ void __kprobes recycle_rp_inst(struct kretprobe_instance *ri, | |||
| 831 | 831 | ||
| 832 | void __kprobes kretprobe_hash_lock(struct task_struct *tsk, | 832 | void __kprobes kretprobe_hash_lock(struct task_struct *tsk, |
| 833 | struct hlist_head **head, unsigned long *flags) | 833 | struct hlist_head **head, unsigned long *flags) |
| 834 | __acquires(hlist_lock) | ||
| 834 | { | 835 | { |
| 835 | unsigned long hash = hash_ptr(tsk, KPROBE_HASH_BITS); | 836 | unsigned long hash = hash_ptr(tsk, KPROBE_HASH_BITS); |
| 836 | spinlock_t *hlist_lock; | 837 | spinlock_t *hlist_lock; |
| @@ -842,6 +843,7 @@ void __kprobes kretprobe_hash_lock(struct task_struct *tsk, | |||
| 842 | 843 | ||
| 843 | static void __kprobes kretprobe_table_lock(unsigned long hash, | 844 | static void __kprobes kretprobe_table_lock(unsigned long hash, |
| 844 | unsigned long *flags) | 845 | unsigned long *flags) |
| 846 | __acquires(hlist_lock) | ||
| 845 | { | 847 | { |
| 846 | spinlock_t *hlist_lock = kretprobe_table_lock_ptr(hash); | 848 | spinlock_t *hlist_lock = kretprobe_table_lock_ptr(hash); |
| 847 | spin_lock_irqsave(hlist_lock, *flags); | 849 | spin_lock_irqsave(hlist_lock, *flags); |
| @@ -849,6 +851,7 @@ static void __kprobes kretprobe_table_lock(unsigned long hash, | |||
| 849 | 851 | ||
| 850 | void __kprobes kretprobe_hash_unlock(struct task_struct *tsk, | 852 | void __kprobes kretprobe_hash_unlock(struct task_struct *tsk, |
| 851 | unsigned long *flags) | 853 | unsigned long *flags) |
| 854 | __releases(hlist_lock) | ||
| 852 | { | 855 | { |
| 853 | unsigned long hash = hash_ptr(tsk, KPROBE_HASH_BITS); | 856 | unsigned long hash = hash_ptr(tsk, KPROBE_HASH_BITS); |
| 854 | spinlock_t *hlist_lock; | 857 | spinlock_t *hlist_lock; |
| @@ -857,7 +860,9 @@ void __kprobes kretprobe_hash_unlock(struct task_struct *tsk, | |||
| 857 | spin_unlock_irqrestore(hlist_lock, *flags); | 860 | spin_unlock_irqrestore(hlist_lock, *flags); |
| 858 | } | 861 | } |
| 859 | 862 | ||
| 860 | void __kprobes kretprobe_table_unlock(unsigned long hash, unsigned long *flags) | 863 | static void __kprobes kretprobe_table_unlock(unsigned long hash, |
| 864 | unsigned long *flags) | ||
| 865 | __releases(hlist_lock) | ||
| 861 | { | 866 | { |
| 862 | spinlock_t *hlist_lock = kretprobe_table_lock_ptr(hash); | 867 | spinlock_t *hlist_lock = kretprobe_table_lock_ptr(hash); |
| 863 | spin_unlock_irqrestore(hlist_lock, *flags); | 868 | spin_unlock_irqrestore(hlist_lock, *flags); |
| @@ -1339,18 +1344,19 @@ int __kprobes register_jprobes(struct jprobe **jps, int num) | |||
| 1339 | if (num <= 0) | 1344 | if (num <= 0) |
| 1340 | return -EINVAL; | 1345 | return -EINVAL; |
| 1341 | for (i = 0; i < num; i++) { | 1346 | for (i = 0; i < num; i++) { |
| 1342 | unsigned long addr; | 1347 | unsigned long addr, offset; |
| 1343 | jp = jps[i]; | 1348 | jp = jps[i]; |
| 1344 | addr = arch_deref_entry_point(jp->entry); | 1349 | addr = arch_deref_entry_point(jp->entry); |
| 1345 | 1350 | ||
| 1346 | if (!kernel_text_address(addr)) | 1351 | /* Verify probepoint is a function entry point */ |
| 1347 | ret = -EINVAL; | 1352 | if (kallsyms_lookup_size_offset(addr, NULL, &offset) && |
| 1348 | else { | 1353 | offset == 0) { |
| 1349 | /* Todo: Verify probepoint is a function entry point */ | ||
| 1350 | jp->kp.pre_handler = setjmp_pre_handler; | 1354 | jp->kp.pre_handler = setjmp_pre_handler; |
| 1351 | jp->kp.break_handler = longjmp_break_handler; | 1355 | jp->kp.break_handler = longjmp_break_handler; |
| 1352 | ret = register_kprobe(&jp->kp); | 1356 | ret = register_kprobe(&jp->kp); |
| 1353 | } | 1357 | } else |
| 1358 | ret = -EINVAL; | ||
| 1359 | |||
| 1354 | if (ret < 0) { | 1360 | if (ret < 0) { |
| 1355 | if (i > 0) | 1361 | if (i > 0) |
| 1356 | unregister_jprobes(jps, i); | 1362 | unregister_jprobes(jps, i); |
