diff options
-rw-r--r-- | include/linux/stop_machine.h | 14 | ||||
-rw-r--r-- | kernel/stop_machine.c | 62 |
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 | */ |
127 | int __stop_machine(int (*fn)(void *), void *data, const struct cpumask *cpus); | 127 | int __stop_machine(int (*fn)(void *), void *data, const struct cpumask *cpus); |
128 | 128 | ||
129 | int 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 | ||
131 | static inline int __stop_machine(int (*fn)(void *), void *data, | 134 | static 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 | ||
151 | static 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 | } |
496 | EXPORT_SYMBOL_GPL(stop_machine); | 503 | EXPORT_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 | */ | ||
527 | int 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 */ |