diff options
| author | Wang Nan <wangnan0@huawei.com> | 2015-02-13 17:40:26 -0500 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2015-02-14 00:21:42 -0500 |
| commit | 69d54b916d83872a0a327778a01af2a096923f59 (patch) | |
| tree | ee62261a484b8e939ad763e2cf4e8039ca9658ea | |
| parent | 977ad481b66ca91e1f6492b3c5c4748c68fdee9c (diff) | |
kprobes: makes kprobes/enabled works correctly for optimized kprobes.
debugfs/kprobes/enabled doesn't work correctly on optimized kprobes.
Masami Hiramatsu has a test report on x86_64 platform:
https://lkml.org/lkml/2015/1/19/274
This patch forces it to unoptimize kprobe if kprobes_all_disarmed is set.
It also checks the flag in unregistering path for skipping unneeded
disarming process when kprobes globally disarmed.
Signed-off-by: Wang Nan <wangnan0@huawei.com>
Signed-off-by: Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com>
Cc: Ingo Molnar <mingo@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
| -rw-r--r-- | kernel/kprobes.c | 11 |
1 files changed, 9 insertions, 2 deletions
diff --git a/kernel/kprobes.c b/kernel/kprobes.c index c39790001854..c90e417bb963 100644 --- a/kernel/kprobes.c +++ b/kernel/kprobes.c | |||
| @@ -869,7 +869,8 @@ static void __disarm_kprobe(struct kprobe *p, bool reopt) | |||
| 869 | { | 869 | { |
| 870 | struct kprobe *_p; | 870 | struct kprobe *_p; |
| 871 | 871 | ||
| 872 | unoptimize_kprobe(p, false); /* Try to unoptimize */ | 872 | /* Try to unoptimize */ |
| 873 | unoptimize_kprobe(p, kprobes_all_disarmed); | ||
| 873 | 874 | ||
| 874 | if (!kprobe_queued(p)) { | 875 | if (!kprobe_queued(p)) { |
| 875 | arch_disarm_kprobe(p); | 876 | arch_disarm_kprobe(p); |
| @@ -1571,7 +1572,13 @@ static struct kprobe *__disable_kprobe(struct kprobe *p) | |||
| 1571 | 1572 | ||
| 1572 | /* Try to disarm and disable this/parent probe */ | 1573 | /* Try to disarm and disable this/parent probe */ |
| 1573 | if (p == orig_p || aggr_kprobe_disabled(orig_p)) { | 1574 | if (p == orig_p || aggr_kprobe_disabled(orig_p)) { |
| 1574 | disarm_kprobe(orig_p, true); | 1575 | /* |
| 1576 | * If kprobes_all_disarmed is set, orig_p | ||
| 1577 | * should have already been disarmed, so | ||
| 1578 | * skip unneed disarming process. | ||
| 1579 | */ | ||
| 1580 | if (!kprobes_all_disarmed) | ||
| 1581 | disarm_kprobe(orig_p, true); | ||
| 1575 | orig_p->flags |= KPROBE_FLAG_DISABLED; | 1582 | orig_p->flags |= KPROBE_FLAG_DISABLED; |
| 1576 | } | 1583 | } |
| 1577 | } | 1584 | } |
