diff options
Diffstat (limited to 'arch/arm/kernel/smp.c')
-rw-r--r-- | arch/arm/kernel/smp.c | 85 |
1 files changed, 85 insertions, 0 deletions
diff --git a/arch/arm/kernel/smp.c b/arch/arm/kernel/smp.c index b40915dcd533..edb5a406922f 100644 --- a/arch/arm/kernel/smp.c +++ b/arch/arm/kernel/smp.c | |||
@@ -159,6 +159,91 @@ int __cpuinit __cpu_up(unsigned int cpu) | |||
159 | return ret; | 159 | return ret; |
160 | } | 160 | } |
161 | 161 | ||
162 | #ifdef CONFIG_HOTPLUG_CPU | ||
163 | /* | ||
164 | * __cpu_disable runs on the processor to be shutdown. | ||
165 | */ | ||
166 | int __cpuexit __cpu_disable(void) | ||
167 | { | ||
168 | unsigned int cpu = smp_processor_id(); | ||
169 | struct task_struct *p; | ||
170 | int ret; | ||
171 | |||
172 | ret = mach_cpu_disable(cpu); | ||
173 | if (ret) | ||
174 | return ret; | ||
175 | |||
176 | /* | ||
177 | * Take this CPU offline. Once we clear this, we can't return, | ||
178 | * and we must not schedule until we're ready to give up the cpu. | ||
179 | */ | ||
180 | cpu_clear(cpu, cpu_online_map); | ||
181 | |||
182 | /* | ||
183 | * OK - migrate IRQs away from this CPU | ||
184 | */ | ||
185 | migrate_irqs(); | ||
186 | |||
187 | /* | ||
188 | * Flush user cache and TLB mappings, and then remove this CPU | ||
189 | * from the vm mask set of all processes. | ||
190 | */ | ||
191 | flush_cache_all(); | ||
192 | local_flush_tlb_all(); | ||
193 | |||
194 | read_lock(&tasklist_lock); | ||
195 | for_each_process(p) { | ||
196 | if (p->mm) | ||
197 | cpu_clear(cpu, p->mm->cpu_vm_mask); | ||
198 | } | ||
199 | read_unlock(&tasklist_lock); | ||
200 | |||
201 | return 0; | ||
202 | } | ||
203 | |||
204 | /* | ||
205 | * called on the thread which is asking for a CPU to be shutdown - | ||
206 | * waits until shutdown has completed, or it is timed out. | ||
207 | */ | ||
208 | void __cpuexit __cpu_die(unsigned int cpu) | ||
209 | { | ||
210 | if (!platform_cpu_kill(cpu)) | ||
211 | printk("CPU%u: unable to kill\n", cpu); | ||
212 | } | ||
213 | |||
214 | /* | ||
215 | * Called from the idle thread for the CPU which has been shutdown. | ||
216 | * | ||
217 | * Note that we disable IRQs here, but do not re-enable them | ||
218 | * before returning to the caller. This is also the behaviour | ||
219 | * of the other hotplug-cpu capable cores, so presumably coming | ||
220 | * out of idle fixes this. | ||
221 | */ | ||
222 | void __cpuexit cpu_die(void) | ||
223 | { | ||
224 | unsigned int cpu = smp_processor_id(); | ||
225 | |||
226 | local_irq_disable(); | ||
227 | idle_task_exit(); | ||
228 | |||
229 | /* | ||
230 | * actual CPU shutdown procedure is at least platform (if not | ||
231 | * CPU) specific | ||
232 | */ | ||
233 | platform_cpu_die(cpu); | ||
234 | |||
235 | /* | ||
236 | * Do not return to the idle loop - jump back to the secondary | ||
237 | * cpu initialisation. There's some initialisation which needs | ||
238 | * to be repeated to undo the effects of taking the CPU offline. | ||
239 | */ | ||
240 | __asm__("mov sp, %0\n" | ||
241 | " b secondary_start_kernel" | ||
242 | : | ||
243 | : "r" ((void *)current->thread_info + THREAD_SIZE - 8)); | ||
244 | } | ||
245 | #endif /* CONFIG_HOTPLUG_CPU */ | ||
246 | |||
162 | /* | 247 | /* |
163 | * This is the secondary CPU boot entry. We're using this CPUs | 248 | * This is the secondary CPU boot entry. We're using this CPUs |
164 | * idle thread stack, but a set of temporary page tables. | 249 | * idle thread stack, but a set of temporary page tables. |