Commit 3c030bea authored by Russell King's avatar Russell King

ARM: CPU hotplug: move cpu_killed completion to core code

We always need to wait for the dying CPU to reach a safe state before
taking it down, irrespective of the requirements of the platform.
Move the completion code into the ARM SMP hotplug code rather than
having each platform re-implement this.
Signed-off-by: default avatarRussell King <rmk+kernel@arm.linux.org.uk>
parent 2c0136db
...@@ -24,6 +24,7 @@ ...@@ -24,6 +24,7 @@
#include <linux/irq.h> #include <linux/irq.h>
#include <linux/percpu.h> #include <linux/percpu.h>
#include <linux/clockchips.h> #include <linux/clockchips.h>
#include <linux/completion.h>
#include <asm/atomic.h> #include <asm/atomic.h>
#include <asm/cacheflush.h> #include <asm/cacheflush.h>
...@@ -238,12 +239,20 @@ int __cpu_disable(void) ...@@ -238,12 +239,20 @@ int __cpu_disable(void)
return 0; return 0;
} }
static DECLARE_COMPLETION(cpu_died);
/* /*
* called on the thread which is asking for a CPU to be shutdown - * called on the thread which is asking for a CPU to be shutdown -
* waits until shutdown has completed, or it is timed out. * waits until shutdown has completed, or it is timed out.
*/ */
void __cpu_die(unsigned int cpu) void __cpu_die(unsigned int cpu)
{ {
if (!wait_for_completion_timeout(&cpu_died, msecs_to_jiffies(5000))) {
pr_err("CPU%u: cpu didn't die\n", cpu);
return;
}
printk(KERN_NOTICE "CPU%u: shutdown\n", cpu);
if (!platform_cpu_kill(cpu)) if (!platform_cpu_kill(cpu))
printk("CPU%u: unable to kill\n", cpu); printk("CPU%u: unable to kill\n", cpu);
} }
...@@ -263,9 +272,12 @@ void __ref cpu_die(void) ...@@ -263,9 +272,12 @@ void __ref cpu_die(void)
local_irq_disable(); local_irq_disable();
idle_task_exit(); idle_task_exit();
/* Tell __cpu_die() that this CPU is now safe to dispose of */
complete(&cpu_died);
/* /*
* actual CPU shutdown procedure is at least platform (if not * actual CPU shutdown procedure is at least platform (if not
* CPU) specific * CPU) specific.
*/ */
platform_cpu_die(cpu); platform_cpu_die(cpu);
......
...@@ -17,16 +17,13 @@ ...@@ -17,16 +17,13 @@
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/errno.h> #include <linux/errno.h>
#include <linux/smp.h> #include <linux/smp.h>
#include <linux/completion.h>
#include <asm/cacheflush.h> #include <asm/cacheflush.h>
#include <mach/omap4-common.h> #include <mach/omap4-common.h>
static DECLARE_COMPLETION(cpu_killed);
int platform_cpu_kill(unsigned int cpu) int platform_cpu_kill(unsigned int cpu)
{ {
return wait_for_completion_timeout(&cpu_killed, 5000); return 1;
} }
/* /*
...@@ -42,8 +39,7 @@ void platform_cpu_die(unsigned int cpu) ...@@ -42,8 +39,7 @@ void platform_cpu_die(unsigned int cpu)
this_cpu, cpu); this_cpu, cpu);
BUG(); BUG();
} }
pr_notice("CPU%u: shutdown\n", cpu);
complete(&cpu_killed);
flush_cache_all(); flush_cache_all();
dsb(); dsb();
......
...@@ -11,14 +11,11 @@ ...@@ -11,14 +11,11 @@
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/errno.h> #include <linux/errno.h>
#include <linux/smp.h> #include <linux/smp.h>
#include <linux/completion.h>
#include <asm/cacheflush.h> #include <asm/cacheflush.h>
extern volatile int pen_release; extern volatile int pen_release;
static DECLARE_COMPLETION(cpu_killed);
static inline void cpu_enter_lowpower(void) static inline void cpu_enter_lowpower(void)
{ {
unsigned int v; unsigned int v;
...@@ -95,7 +92,7 @@ static inline void platform_do_lowpower(unsigned int cpu) ...@@ -95,7 +92,7 @@ static inline void platform_do_lowpower(unsigned int cpu)
int platform_cpu_kill(unsigned int cpu) int platform_cpu_kill(unsigned int cpu)
{ {
return wait_for_completion_timeout(&cpu_killed, 5000); return 1;
} }
/* /*
...@@ -115,9 +112,6 @@ void platform_cpu_die(unsigned int cpu) ...@@ -115,9 +112,6 @@ void platform_cpu_die(unsigned int cpu)
} }
#endif #endif
printk(KERN_NOTICE "CPU%u: shutdown\n", cpu);
complete(&cpu_killed);
/* /*
* we're ready for shutdown now, so do it * we're ready for shutdown now, so do it
*/ */
......
...@@ -13,14 +13,11 @@ ...@@ -13,14 +13,11 @@
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/errno.h> #include <linux/errno.h>
#include <linux/smp.h> #include <linux/smp.h>
#include <linux/completion.h>
#include <asm/cacheflush.h> #include <asm/cacheflush.h>
extern volatile int pen_release; extern volatile int pen_release;
static DECLARE_COMPLETION(cpu_killed);
static inline void cpu_enter_lowpower(void) static inline void cpu_enter_lowpower(void)
{ {
unsigned int v; unsigned int v;
...@@ -98,7 +95,7 @@ static inline void platform_do_lowpower(unsigned int cpu) ...@@ -98,7 +95,7 @@ static inline void platform_do_lowpower(unsigned int cpu)
int platform_cpu_kill(unsigned int cpu) int platform_cpu_kill(unsigned int cpu)
{ {
return wait_for_completion_timeout(&cpu_killed, 5000); return 1;
} }
/* /*
...@@ -118,9 +115,6 @@ void platform_cpu_die(unsigned int cpu) ...@@ -118,9 +115,6 @@ void platform_cpu_die(unsigned int cpu)
} }
#endif #endif
printk(KERN_NOTICE "CPU%u: shutdown\n", cpu);
complete(&cpu_killed);
/* /*
* we're ready for shutdown now, so do it * we're ready for shutdown now, so do it
*/ */
......
...@@ -11,12 +11,9 @@ ...@@ -11,12 +11,9 @@
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/errno.h> #include <linux/errno.h>
#include <linux/smp.h> #include <linux/smp.h>
#include <linux/completion.h>
#include <asm/cacheflush.h> #include <asm/cacheflush.h>
static DECLARE_COMPLETION(cpu_killed);
static inline void cpu_enter_lowpower(void) static inline void cpu_enter_lowpower(void)
{ {
unsigned int v; unsigned int v;
...@@ -94,7 +91,7 @@ static inline void platform_do_lowpower(unsigned int cpu) ...@@ -94,7 +91,7 @@ static inline void platform_do_lowpower(unsigned int cpu)
int platform_cpu_kill(unsigned int cpu) int platform_cpu_kill(unsigned int cpu)
{ {
return wait_for_completion_timeout(&cpu_killed, 5000); return 1;
} }
/* /*
...@@ -114,9 +111,6 @@ void platform_cpu_die(unsigned int cpu) ...@@ -114,9 +111,6 @@ void platform_cpu_die(unsigned int cpu)
} }
#endif #endif
printk(KERN_NOTICE "CPU%u: shutdown\n", cpu);
complete(&cpu_killed);
/* /*
* we're ready for shutdown now, so do it * we're ready for shutdown now, so do it
*/ */
......
...@@ -11,14 +11,11 @@ ...@@ -11,14 +11,11 @@
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/errno.h> #include <linux/errno.h>
#include <linux/smp.h> #include <linux/smp.h>
#include <linux/completion.h>
#include <asm/cacheflush.h> #include <asm/cacheflush.h>
extern volatile int pen_release; extern volatile int pen_release;
static DECLARE_COMPLETION(cpu_killed);
static inline void platform_do_lowpower(unsigned int cpu) static inline void platform_do_lowpower(unsigned int cpu)
{ {
flush_cache_all(); flush_cache_all();
...@@ -38,7 +35,7 @@ static inline void platform_do_lowpower(unsigned int cpu) ...@@ -38,7 +35,7 @@ static inline void platform_do_lowpower(unsigned int cpu)
int platform_cpu_kill(unsigned int cpu) int platform_cpu_kill(unsigned int cpu)
{ {
return wait_for_completion_timeout(&cpu_killed, 5000); return 1;
} }
/* /*
...@@ -58,9 +55,6 @@ void platform_cpu_die(unsigned int cpu) ...@@ -58,9 +55,6 @@ void platform_cpu_die(unsigned int cpu)
} }
#endif #endif
printk(KERN_NOTICE "CPU%u: shutdown\n", cpu);
complete(&cpu_killed);
/* directly enter low power state, skipping secure registers */ /* directly enter low power state, skipping secure registers */
platform_do_lowpower(cpu); platform_do_lowpower(cpu);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment