diff options
author | Michal Hocko <mhocko@suse.com> | 2016-12-07 08:54:38 -0500 |
---|---|---|
committer | Greg Kroah-Hartman <gregkh@linuxfoundation.org> | 2017-01-06 04:40:10 -0500 |
commit | 56eaecc8ecf398c7af3fa252eff6164102b0ace1 (patch) | |
tree | d63de7613fc528666b2bbc331bb3d4d8ed34e50f | |
parent | 69973b830859bc6529a7a0468ba0d80ee5117826 (diff) |
hotplug: Make register and unregister notifier API symmetric
commit 777c6e0daebb3fcefbbd6f620410a946b07ef6d0 upstream.
Yu Zhao has noticed that __unregister_cpu_notifier only unregisters its
notifiers when HOTPLUG_CPU=y while the registration might succeed even
when HOTPLUG_CPU=n if MODULE is enabled. This means that e.g. zswap
might keep a stale notifier on the list on the manual clean up during
the pool tear down and thus corrupt the list. Resulting in the following
[ 144.964346] BUG: unable to handle kernel paging request at ffff880658a2be78
[ 144.971337] IP: [<ffffffffa290b00b>] raw_notifier_chain_register+0x1b/0x40
<snipped>
[ 145.122628] Call Trace:
[ 145.125086] [<ffffffffa28e5cf8>] __register_cpu_notifier+0x18/0x20
[ 145.131350] [<ffffffffa2a5dd73>] zswap_pool_create+0x273/0x400
[ 145.137268] [<ffffffffa2a5e0fc>] __zswap_param_set+0x1fc/0x300
[ 145.143188] [<ffffffffa2944c1d>] ? trace_hardirqs_on+0xd/0x10
[ 145.149018] [<ffffffffa2908798>] ? kernel_param_lock+0x28/0x30
[ 145.154940] [<ffffffffa2a3e8cf>] ? __might_fault+0x4f/0xa0
[ 145.160511] [<ffffffffa2a5e237>] zswap_compressor_param_set+0x17/0x20
[ 145.167035] [<ffffffffa2908d3c>] param_attr_store+0x5c/0xb0
[ 145.172694] [<ffffffffa290848d>] module_attr_store+0x1d/0x30
[ 145.178443] [<ffffffffa2b2b41f>] sysfs_kf_write+0x4f/0x70
[ 145.183925] [<ffffffffa2b2a5b9>] kernfs_fop_write+0x149/0x180
[ 145.189761] [<ffffffffa2a99248>] __vfs_write+0x18/0x40
[ 145.194982] [<ffffffffa2a9a412>] vfs_write+0xb2/0x1a0
[ 145.200122] [<ffffffffa2a9a732>] SyS_write+0x52/0xa0
[ 145.205177] [<ffffffffa2ff4d97>] entry_SYSCALL_64_fastpath+0x12/0x17
This can be even triggered manually by changing
/sys/module/zswap/parameters/compressor multiple times.
Fix this issue by making unregister APIs symmetric to the register so
there are no surprises.
Fixes: 47e627bc8c9a ("[PATCH] hotplug: Allow modules to use the cpu hotplug notifiers even if !CONFIG_HOTPLUG_CPU")
Reported-and-tested-by: Yu Zhao <yuzhao@google.com>
Signed-off-by: Michal Hocko <mhocko@suse.com>
Cc: linux-mm@kvack.org
Cc: Andrew Morton <akpm@linux-foundation.org>
Cc: Dan Streetman <ddstreet@ieee.org>
Link: http://lkml.kernel.org/r/20161207135438.4310-1-mhocko@kernel.org
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
-rw-r--r-- | include/linux/cpu.h | 15 | ||||
-rw-r--r-- | kernel/cpu.c | 2 |
2 files changed, 5 insertions, 12 deletions
diff --git a/include/linux/cpu.h b/include/linux/cpu.h index b886dc17f2f3..e571128ad99a 100644 --- a/include/linux/cpu.h +++ b/include/linux/cpu.h | |||
@@ -93,22 +93,16 @@ extern bool cpuhp_tasks_frozen; | |||
93 | { .notifier_call = fn, .priority = pri }; \ | 93 | { .notifier_call = fn, .priority = pri }; \ |
94 | __register_cpu_notifier(&fn##_nb); \ | 94 | __register_cpu_notifier(&fn##_nb); \ |
95 | } | 95 | } |
96 | #else /* #if defined(CONFIG_HOTPLUG_CPU) || !defined(MODULE) */ | ||
97 | #define cpu_notifier(fn, pri) do { (void)(fn); } while (0) | ||
98 | #define __cpu_notifier(fn, pri) do { (void)(fn); } while (0) | ||
99 | #endif /* #else #if defined(CONFIG_HOTPLUG_CPU) || !defined(MODULE) */ | ||
100 | 96 | ||
101 | #ifdef CONFIG_HOTPLUG_CPU | ||
102 | extern int register_cpu_notifier(struct notifier_block *nb); | 97 | extern int register_cpu_notifier(struct notifier_block *nb); |
103 | extern int __register_cpu_notifier(struct notifier_block *nb); | 98 | extern int __register_cpu_notifier(struct notifier_block *nb); |
104 | extern void unregister_cpu_notifier(struct notifier_block *nb); | 99 | extern void unregister_cpu_notifier(struct notifier_block *nb); |
105 | extern void __unregister_cpu_notifier(struct notifier_block *nb); | 100 | extern void __unregister_cpu_notifier(struct notifier_block *nb); |
106 | #else | ||
107 | 101 | ||
108 | #ifndef MODULE | 102 | #else /* #if defined(CONFIG_HOTPLUG_CPU) || !defined(MODULE) */ |
109 | extern int register_cpu_notifier(struct notifier_block *nb); | 103 | #define cpu_notifier(fn, pri) do { (void)(fn); } while (0) |
110 | extern int __register_cpu_notifier(struct notifier_block *nb); | 104 | #define __cpu_notifier(fn, pri) do { (void)(fn); } while (0) |
111 | #else | 105 | |
112 | static inline int register_cpu_notifier(struct notifier_block *nb) | 106 | static inline int register_cpu_notifier(struct notifier_block *nb) |
113 | { | 107 | { |
114 | return 0; | 108 | return 0; |
@@ -118,7 +112,6 @@ static inline int __register_cpu_notifier(struct notifier_block *nb) | |||
118 | { | 112 | { |
119 | return 0; | 113 | return 0; |
120 | } | 114 | } |
121 | #endif | ||
122 | 115 | ||
123 | static inline void unregister_cpu_notifier(struct notifier_block *nb) | 116 | static inline void unregister_cpu_notifier(struct notifier_block *nb) |
124 | { | 117 | { |
diff --git a/kernel/cpu.c b/kernel/cpu.c index 29de1a9352c0..217fd2e7f435 100644 --- a/kernel/cpu.c +++ b/kernel/cpu.c | |||
@@ -659,7 +659,6 @@ void __init cpuhp_threads_init(void) | |||
659 | kthread_unpark(this_cpu_read(cpuhp_state.thread)); | 659 | kthread_unpark(this_cpu_read(cpuhp_state.thread)); |
660 | } | 660 | } |
661 | 661 | ||
662 | #ifdef CONFIG_HOTPLUG_CPU | ||
663 | EXPORT_SYMBOL(register_cpu_notifier); | 662 | EXPORT_SYMBOL(register_cpu_notifier); |
664 | EXPORT_SYMBOL(__register_cpu_notifier); | 663 | EXPORT_SYMBOL(__register_cpu_notifier); |
665 | void unregister_cpu_notifier(struct notifier_block *nb) | 664 | void unregister_cpu_notifier(struct notifier_block *nb) |
@@ -676,6 +675,7 @@ void __unregister_cpu_notifier(struct notifier_block *nb) | |||
676 | } | 675 | } |
677 | EXPORT_SYMBOL(__unregister_cpu_notifier); | 676 | EXPORT_SYMBOL(__unregister_cpu_notifier); |
678 | 677 | ||
678 | #ifdef CONFIG_HOTPLUG_CPU | ||
679 | /** | 679 | /** |
680 | * clear_tasks_mm_cpumask - Safely clear tasks' mm_cpumask for a CPU | 680 | * clear_tasks_mm_cpumask - Safely clear tasks' mm_cpumask for a CPU |
681 | * @cpu: a CPU id | 681 | * @cpu: a CPU id |