aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/linux/stop_machine.h14
-rw-r--r--kernel/stop_machine.c62
2 files changed, 73 insertions, 3 deletions
diff --git a/include/linux/stop_machine.h b/include/linux/stop_machine.h
index 14d3524d1274..e0f2da25d751 100644
--- a/include/linux/stop_machine.h
+++ b/include/linux/stop_machine.h
@@ -126,15 +126,19 @@ int stop_machine(int (*fn)(void *), void *data, const struct cpumask *cpus);
126 */ 126 */
127int __stop_machine(int (*fn)(void *), void *data, const struct cpumask *cpus); 127int __stop_machine(int (*fn)(void *), void *data, const struct cpumask *cpus);
128 128
129int stop_machine_from_inactive_cpu(int (*fn)(void *), void *data,
130 const struct cpumask *cpus);
131
129#else /* CONFIG_STOP_MACHINE && CONFIG_SMP */ 132#else /* CONFIG_STOP_MACHINE && CONFIG_SMP */
130 133
131static inline int __stop_machine(int (*fn)(void *), void *data, 134static inline int __stop_machine(int (*fn)(void *), void *data,
132 const struct cpumask *cpus) 135 const struct cpumask *cpus)
133{ 136{
137 unsigned long flags;
134 int ret; 138 int ret;
135 local_irq_disable(); 139 local_irq_save(flags);
136 ret = fn(data); 140 ret = fn(data);
137 local_irq_enable(); 141 local_irq_restore(flags);
138 return ret; 142 return ret;
139} 143}
140 144
@@ -144,5 +148,11 @@ static inline int stop_machine(int (*fn)(void *), void *data,
144 return __stop_machine(fn, data, cpus); 148 return __stop_machine(fn, data, cpus);
145} 149}
146 150
151static inline int stop_machine_from_inactive_cpu(int (*fn)(void *), void *data,
152 const struct cpumask *cpus)
153{
154 return __stop_machine(fn, data, cpus);
155}
156
147#endif /* CONFIG_STOP_MACHINE && CONFIG_SMP */ 157#endif /* CONFIG_STOP_MACHINE && CONFIG_SMP */
148#endif /* _LINUX_STOP_MACHINE */ 158#endif /* _LINUX_STOP_MACHINE */
diff --git a/kernel/stop_machine.c b/kernel/stop_machine.c
index 4c89ee9fc56b..e8f05b14cd43 100644
--- a/kernel/stop_machine.c
+++ b/kernel/stop_machine.c
@@ -439,8 +439,15 @@ static int stop_machine_cpu_stop(void *data)
439 struct stop_machine_data *smdata = data; 439 struct stop_machine_data *smdata = data;
440 enum stopmachine_state curstate = STOPMACHINE_NONE; 440 enum stopmachine_state curstate = STOPMACHINE_NONE;
441 int cpu = smp_processor_id(), err = 0; 441 int cpu = smp_processor_id(), err = 0;
442 unsigned long flags;
442 bool is_active; 443 bool is_active;
443 444
445 /*
446 * When called from stop_machine_from_inactive_cpu(), irq might
447 * already be disabled. Save the state and restore it on exit.
448 */
449 local_save_flags(flags);
450
444 if (!smdata->active_cpus) 451 if (!smdata->active_cpus)
445 is_active = cpu == cpumask_first(cpu_online_mask); 452 is_active = cpu == cpumask_first(cpu_online_mask);
446 else 453 else
@@ -468,7 +475,7 @@ static int stop_machine_cpu_stop(void *data)
468 } 475 }
469 } while (curstate != STOPMACHINE_EXIT); 476 } while (curstate != STOPMACHINE_EXIT);
470 477
471 local_irq_enable(); 478 local_irq_restore(flags);
472 return err; 479 return err;
473} 480}
474 481
@@ -495,4 +502,57 @@ int stop_machine(int (*fn)(void *), void *data, const struct cpumask *cpus)
495} 502}
496EXPORT_SYMBOL_GPL(stop_machine); 503EXPORT_SYMBOL_GPL(stop_machine);
497 504
505/**
506 * stop_machine_from_inactive_cpu - stop_machine() from inactive CPU
507 * @fn: the function to run
508 * @data: the data ptr for the @fn()
509 * @cpus: the cpus to run the @fn() on (NULL = any online cpu)
510 *
511 * This is identical to stop_machine() but can be called from a CPU which
512 * is not active. The local CPU is in the process of hotplug (so no other
513 * CPU hotplug can start) and not marked active and doesn't have enough
514 * context to sleep.
515 *
516 * This function provides stop_machine() functionality for such state by
517 * using busy-wait for synchronization and executing @fn directly for local
518 * CPU.
519 *
520 * CONTEXT:
521 * Local CPU is inactive. Temporarily stops all active CPUs.
522 *
523 * RETURNS:
524 * 0 if all executions of @fn returned 0, any non zero return value if any
525 * returned non zero.
526 */
527int stop_machine_from_inactive_cpu(int (*fn)(void *), void *data,
528 const struct cpumask *cpus)
529{
530 struct stop_machine_data smdata = { .fn = fn, .data = data,
531 .active_cpus = cpus };
532 struct cpu_stop_done done;
533 int ret;
534
535 /* Local CPU must be inactive and CPU hotplug in progress. */
536 BUG_ON(cpu_active(raw_smp_processor_id()));
537 smdata.num_threads = num_active_cpus() + 1; /* +1 for local */
538
539 /* No proper task established and can't sleep - busy wait for lock. */
540 while (!mutex_trylock(&stop_cpus_mutex))
541 cpu_relax();
542
543 /* Schedule work on other CPUs and execute directly for local CPU */
544 set_state(&smdata, STOPMACHINE_PREPARE);
545 cpu_stop_init_done(&done, num_active_cpus());
546 queue_stop_cpus_work(cpu_active_mask, stop_machine_cpu_stop, &smdata,
547 &done);
548 ret = stop_machine_cpu_stop(&smdata);
549
550 /* Busy wait for completion. */
551 while (!completion_done(&done.completion))
552 cpu_relax();
553
554 mutex_unlock(&stop_cpus_mutex);
555 return ret ?: done.ret;
556}
557
498#endif /* CONFIG_STOP_MACHINE */ 558#endif /* CONFIG_STOP_MACHINE */