diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2015-04-18 11:23:42 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2015-04-18 11:23:42 -0400 |
commit | 396c9df2231865ef55aa031e3f5df9d99e036869 (patch) | |
tree | 566c2a9e9cab77720e1dc41fd16ef00fd6a662c7 /kernel/smp.c | |
parent | 8f502d5b9e3362971f58dad5d468f070340336e1 (diff) | |
parent | 8053871d0f7f67c7efb7f226ef031f78877d6625 (diff) |
Merge branch 'locking-urgent-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip
Pull locking fixes from Ingo Molnar:
"Two fixes: an smp-call fix and a lockdep fix"
* 'locking-urgent-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip:
smp: Fix smp_call_function_single_async() locking
lockdep: Make print_lock() robust against concurrent release
Diffstat (limited to 'kernel/smp.c')
-rw-r--r-- | kernel/smp.c | 78 |
1 files changed, 47 insertions, 31 deletions
diff --git a/kernel/smp.c b/kernel/smp.c index f38a1e692259..2aaac2c47683 100644 --- a/kernel/smp.c +++ b/kernel/smp.c | |||
@@ -19,7 +19,7 @@ | |||
19 | 19 | ||
20 | enum { | 20 | enum { |
21 | CSD_FLAG_LOCK = 0x01, | 21 | CSD_FLAG_LOCK = 0x01, |
22 | CSD_FLAG_WAIT = 0x02, | 22 | CSD_FLAG_SYNCHRONOUS = 0x02, |
23 | }; | 23 | }; |
24 | 24 | ||
25 | struct call_function_data { | 25 | struct call_function_data { |
@@ -107,7 +107,7 @@ void __init call_function_init(void) | |||
107 | */ | 107 | */ |
108 | static void csd_lock_wait(struct call_single_data *csd) | 108 | static void csd_lock_wait(struct call_single_data *csd) |
109 | { | 109 | { |
110 | while (csd->flags & CSD_FLAG_LOCK) | 110 | while (smp_load_acquire(&csd->flags) & CSD_FLAG_LOCK) |
111 | cpu_relax(); | 111 | cpu_relax(); |
112 | } | 112 | } |
113 | 113 | ||
@@ -121,19 +121,17 @@ static void csd_lock(struct call_single_data *csd) | |||
121 | * to ->flags with any subsequent assignments to other | 121 | * to ->flags with any subsequent assignments to other |
122 | * fields of the specified call_single_data structure: | 122 | * fields of the specified call_single_data structure: |
123 | */ | 123 | */ |
124 | smp_mb(); | 124 | smp_wmb(); |
125 | } | 125 | } |
126 | 126 | ||
127 | static void csd_unlock(struct call_single_data *csd) | 127 | static void csd_unlock(struct call_single_data *csd) |
128 | { | 128 | { |
129 | WARN_ON((csd->flags & CSD_FLAG_WAIT) && !(csd->flags & CSD_FLAG_LOCK)); | 129 | WARN_ON(!(csd->flags & CSD_FLAG_LOCK)); |
130 | 130 | ||
131 | /* | 131 | /* |
132 | * ensure we're all done before releasing data: | 132 | * ensure we're all done before releasing data: |
133 | */ | 133 | */ |
134 | smp_mb(); | 134 | smp_store_release(&csd->flags, 0); |
135 | |||
136 | csd->flags &= ~CSD_FLAG_LOCK; | ||
137 | } | 135 | } |
138 | 136 | ||
139 | static DEFINE_PER_CPU_SHARED_ALIGNED(struct call_single_data, csd_data); | 137 | static DEFINE_PER_CPU_SHARED_ALIGNED(struct call_single_data, csd_data); |
@@ -144,13 +142,16 @@ static DEFINE_PER_CPU_SHARED_ALIGNED(struct call_single_data, csd_data); | |||
144 | * ->func, ->info, and ->flags set. | 142 | * ->func, ->info, and ->flags set. |
145 | */ | 143 | */ |
146 | static int generic_exec_single(int cpu, struct call_single_data *csd, | 144 | static int generic_exec_single(int cpu, struct call_single_data *csd, |
147 | smp_call_func_t func, void *info, int wait) | 145 | smp_call_func_t func, void *info) |
148 | { | 146 | { |
149 | struct call_single_data csd_stack = { .flags = 0 }; | ||
150 | unsigned long flags; | ||
151 | |||
152 | |||
153 | if (cpu == smp_processor_id()) { | 147 | if (cpu == smp_processor_id()) { |
148 | unsigned long flags; | ||
149 | |||
150 | /* | ||
151 | * We can unlock early even for the synchronous on-stack case, | ||
152 | * since we're doing this from the same CPU.. | ||
153 | */ | ||
154 | csd_unlock(csd); | ||
154 | local_irq_save(flags); | 155 | local_irq_save(flags); |
155 | func(info); | 156 | func(info); |
156 | local_irq_restore(flags); | 157 | local_irq_restore(flags); |
@@ -161,21 +162,9 @@ static int generic_exec_single(int cpu, struct call_single_data *csd, | |||
161 | if ((unsigned)cpu >= nr_cpu_ids || !cpu_online(cpu)) | 162 | if ((unsigned)cpu >= nr_cpu_ids || !cpu_online(cpu)) |
162 | return -ENXIO; | 163 | return -ENXIO; |
163 | 164 | ||
164 | |||
165 | if (!csd) { | ||
166 | csd = &csd_stack; | ||
167 | if (!wait) | ||
168 | csd = this_cpu_ptr(&csd_data); | ||
169 | } | ||
170 | |||
171 | csd_lock(csd); | ||
172 | |||
173 | csd->func = func; | 165 | csd->func = func; |
174 | csd->info = info; | 166 | csd->info = info; |
175 | 167 | ||
176 | if (wait) | ||
177 | csd->flags |= CSD_FLAG_WAIT; | ||
178 | |||
179 | /* | 168 | /* |
180 | * The list addition should be visible before sending the IPI | 169 | * The list addition should be visible before sending the IPI |
181 | * handler locks the list to pull the entry off it because of | 170 | * handler locks the list to pull the entry off it because of |
@@ -190,9 +179,6 @@ static int generic_exec_single(int cpu, struct call_single_data *csd, | |||
190 | if (llist_add(&csd->llist, &per_cpu(call_single_queue, cpu))) | 179 | if (llist_add(&csd->llist, &per_cpu(call_single_queue, cpu))) |
191 | arch_send_call_function_single_ipi(cpu); | 180 | arch_send_call_function_single_ipi(cpu); |
192 | 181 | ||
193 | if (wait) | ||
194 | csd_lock_wait(csd); | ||
195 | |||
196 | return 0; | 182 | return 0; |
197 | } | 183 | } |
198 | 184 | ||
@@ -250,8 +236,17 @@ static void flush_smp_call_function_queue(bool warn_cpu_offline) | |||
250 | } | 236 | } |
251 | 237 | ||
252 | llist_for_each_entry_safe(csd, csd_next, entry, llist) { | 238 | llist_for_each_entry_safe(csd, csd_next, entry, llist) { |
253 | csd->func(csd->info); | 239 | smp_call_func_t func = csd->func; |
254 | csd_unlock(csd); | 240 | void *info = csd->info; |
241 | |||
242 | /* Do we wait until *after* callback? */ | ||
243 | if (csd->flags & CSD_FLAG_SYNCHRONOUS) { | ||
244 | func(info); | ||
245 | csd_unlock(csd); | ||
246 | } else { | ||
247 | csd_unlock(csd); | ||
248 | func(info); | ||
249 | } | ||
255 | } | 250 | } |
256 | 251 | ||
257 | /* | 252 | /* |
@@ -274,6 +269,8 @@ static void flush_smp_call_function_queue(bool warn_cpu_offline) | |||
274 | int smp_call_function_single(int cpu, smp_call_func_t func, void *info, | 269 | int smp_call_function_single(int cpu, smp_call_func_t func, void *info, |
275 | int wait) | 270 | int wait) |
276 | { | 271 | { |
272 | struct call_single_data *csd; | ||
273 | struct call_single_data csd_stack = { .flags = CSD_FLAG_LOCK | CSD_FLAG_SYNCHRONOUS }; | ||
277 | int this_cpu; | 274 | int this_cpu; |
278 | int err; | 275 | int err; |
279 | 276 | ||
@@ -292,7 +289,16 @@ int smp_call_function_single(int cpu, smp_call_func_t func, void *info, | |||
292 | WARN_ON_ONCE(cpu_online(this_cpu) && irqs_disabled() | 289 | WARN_ON_ONCE(cpu_online(this_cpu) && irqs_disabled() |
293 | && !oops_in_progress); | 290 | && !oops_in_progress); |
294 | 291 | ||
295 | err = generic_exec_single(cpu, NULL, func, info, wait); | 292 | csd = &csd_stack; |
293 | if (!wait) { | ||
294 | csd = this_cpu_ptr(&csd_data); | ||
295 | csd_lock(csd); | ||
296 | } | ||
297 | |||
298 | err = generic_exec_single(cpu, csd, func, info); | ||
299 | |||
300 | if (wait) | ||
301 | csd_lock_wait(csd); | ||
296 | 302 | ||
297 | put_cpu(); | 303 | put_cpu(); |
298 | 304 | ||
@@ -321,7 +327,15 @@ int smp_call_function_single_async(int cpu, struct call_single_data *csd) | |||
321 | int err = 0; | 327 | int err = 0; |
322 | 328 | ||
323 | preempt_disable(); | 329 | preempt_disable(); |
324 | err = generic_exec_single(cpu, csd, csd->func, csd->info, 0); | 330 | |
331 | /* We could deadlock if we have to wait here with interrupts disabled! */ | ||
332 | if (WARN_ON_ONCE(csd->flags & CSD_FLAG_LOCK)) | ||
333 | csd_lock_wait(csd); | ||
334 | |||
335 | csd->flags = CSD_FLAG_LOCK; | ||
336 | smp_wmb(); | ||
337 | |||
338 | err = generic_exec_single(cpu, csd, csd->func, csd->info); | ||
325 | preempt_enable(); | 339 | preempt_enable(); |
326 | 340 | ||
327 | return err; | 341 | return err; |
@@ -433,6 +447,8 @@ void smp_call_function_many(const struct cpumask *mask, | |||
433 | struct call_single_data *csd = per_cpu_ptr(cfd->csd, cpu); | 447 | struct call_single_data *csd = per_cpu_ptr(cfd->csd, cpu); |
434 | 448 | ||
435 | csd_lock(csd); | 449 | csd_lock(csd); |
450 | if (wait) | ||
451 | csd->flags |= CSD_FLAG_SYNCHRONOUS; | ||
436 | csd->func = func; | 452 | csd->func = func; |
437 | csd->info = info; | 453 | csd->info = info; |
438 | llist_add(&csd->llist, &per_cpu(call_single_queue, cpu)); | 454 | llist_add(&csd->llist, &per_cpu(call_single_queue, cpu)); |