Commit 4d1c8cd7 authored by Zwane Mwaikambo's avatar Zwane Mwaikambo Committed by Linus Torvalds

[PATCH] ppc64: generic hotplug cpu support

Patch provides a generic hotplug cpu implementation, with the only current
user being pmac.  This doesn't replace real hotplug code as is currently
used by LPAR systems.  Ben i can add the additional pmac specific code to
put the processor into a sleeping state seperately.  Thanks to Nathan for
testing.
Signed-off-by: default avatarZwane Mwaikambo <zwane@arm.linux.org.uk>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 0c8c1108
...@@ -313,7 +313,7 @@ source "drivers/pci/Kconfig" ...@@ -313,7 +313,7 @@ source "drivers/pci/Kconfig"
config HOTPLUG_CPU config HOTPLUG_CPU
bool "Support for hot-pluggable CPUs" bool "Support for hot-pluggable CPUs"
depends on SMP && EXPERIMENTAL && PPC_PSERIES depends on SMP && EXPERIMENTAL && (PPC_PSERIES || PPC_PMAC)
select HOTPLUG select HOTPLUG
---help--- ---help---
Say Y here to be able to turn CPUs off and on. Say Y here to be able to turn CPUs off and on.
......
...@@ -293,6 +293,10 @@ static int native_idle(void) ...@@ -293,6 +293,10 @@ static int native_idle(void)
power4_idle(); power4_idle();
if (need_resched()) if (need_resched())
schedule(); schedule();
if (cpu_is_offline(smp_processor_id()) &&
system_state == SYSTEM_RUNNING)
cpu_die();
} }
return 0; return 0;
} }
......
...@@ -116,6 +116,35 @@ int show_interrupts(struct seq_file *p, void *v) ...@@ -116,6 +116,35 @@ int show_interrupts(struct seq_file *p, void *v)
return 0; return 0;
} }
#ifdef CONFIG_HOTPLUG_CPU
void fixup_irqs(cpumask_t map)
{
unsigned int irq;
static int warned;
for_each_irq(irq) {
cpumask_t mask;
if (irq_desc[irq].status & IRQ_PER_CPU)
continue;
cpus_and(mask, irq_affinity[irq], map);
if (any_online_cpu(mask) == NR_CPUS) {
printk("Breaking affinity for irq %i\n", irq);
mask = map;
}
if (irq_desc[irq].handler->set_affinity)
irq_desc[irq].handler->set_affinity(irq, mask);
else if (irq_desc[irq].action && !(warned++))
printk("Cannot set affinity for irq %i\n", irq);
}
local_irq_enable();
mdelay(1);
local_irq_disable();
}
#endif
extern int noirqdebug; extern int noirqdebug;
/* /*
......
...@@ -320,8 +320,9 @@ static void __init pSeries_discover_pic(void) ...@@ -320,8 +320,9 @@ static void __init pSeries_discover_pic(void)
} }
} }
static void pSeries_cpu_die(void) static void pSeries_mach_cpu_die(void)
{ {
idle_task_exit();
local_irq_disable(); local_irq_disable();
/* Some hardware requires clearing the CPPR, while other hardware does not /* Some hardware requires clearing the CPPR, while other hardware does not
* it is safe either way * it is safe either way
...@@ -599,7 +600,7 @@ struct machdep_calls __initdata pSeries_md = { ...@@ -599,7 +600,7 @@ struct machdep_calls __initdata pSeries_md = {
.power_off = rtas_power_off, .power_off = rtas_power_off,
.halt = rtas_halt, .halt = rtas_halt,
.panic = rtas_os_term, .panic = rtas_os_term,
.cpu_die = pSeries_cpu_die, .cpu_die = pSeries_mach_cpu_die,
.get_boot_time = pSeries_get_boot_time, .get_boot_time = pSeries_get_boot_time,
.get_rtc_time = pSeries_get_rtc_time, .get_rtc_time = pSeries_get_rtc_time,
.set_rtc_time = pSeries_set_rtc_time, .set_rtc_time = pSeries_set_rtc_time,
......
...@@ -439,6 +439,9 @@ static int __init pmac_probe(int platform) ...@@ -439,6 +439,9 @@ static int __init pmac_probe(int platform)
} }
struct machdep_calls __initdata pmac_md = { struct machdep_calls __initdata pmac_md = {
#ifdef CONFIG_HOTPLUG_CPU
.cpu_die = generic_mach_cpu_die,
#endif
.probe = pmac_probe, .probe = pmac_probe,
.setup_arch = pmac_setup_arch, .setup_arch = pmac_setup_arch,
.init_early = pmac_init_early, .init_early = pmac_init_early,
......
...@@ -308,4 +308,9 @@ struct smp_ops_t core99_smp_ops __pmacdata = { ...@@ -308,4 +308,9 @@ struct smp_ops_t core99_smp_ops __pmacdata = {
void __init pmac_setup_smp(void) void __init pmac_setup_smp(void)
{ {
smp_ops = &core99_smp_ops; smp_ops = &core99_smp_ops;
#ifdef CONFIG_HOTPLUG_CPU
smp_ops->cpu_enable = generic_cpu_enable;
smp_ops->cpu_disable = generic_cpu_disable;
smp_ops->cpu_die = generic_cpu_die;
#endif
} }
...@@ -1377,9 +1377,6 @@ early_param("xmon", early_xmon); ...@@ -1377,9 +1377,6 @@ early_param("xmon", early_xmon);
void cpu_die(void) void cpu_die(void)
{ {
idle_task_exit();
if (ppc_md.cpu_die) if (ppc_md.cpu_die)
ppc_md.cpu_die(); ppc_md.cpu_die();
local_irq_disable();
for (;;);
} }
...@@ -30,6 +30,7 @@ ...@@ -30,6 +30,7 @@
#include <linux/err.h> #include <linux/err.h>
#include <linux/sysdev.h> #include <linux/sysdev.h>
#include <linux/cpu.h> #include <linux/cpu.h>
#include <linux/notifier.h>
#include <asm/ptrace.h> #include <asm/ptrace.h>
#include <asm/atomic.h> #include <asm/atomic.h>
...@@ -406,10 +407,89 @@ void __devinit smp_prepare_boot_cpu(void) ...@@ -406,10 +407,89 @@ void __devinit smp_prepare_boot_cpu(void)
current_set[boot_cpuid] = current->thread_info; current_set[boot_cpuid] = current->thread_info;
} }
#ifdef CONFIG_HOTPLUG_CPU
/* State of each CPU during hotplug phases */
DEFINE_PER_CPU(int, cpu_state) = { 0 };
int generic_cpu_disable(void)
{
unsigned int cpu = smp_processor_id();
if (cpu == boot_cpuid)
return -EBUSY;
systemcfg->processorCount--;
cpu_clear(cpu, cpu_online_map);
fixup_irqs(cpu_online_map);
return 0;
}
int generic_cpu_enable(unsigned int cpu)
{
/* Do the normal bootup if we haven't
* already bootstrapped. */
if (system_state != SYSTEM_RUNNING)
return -ENOSYS;
/* get the target out of it's holding state */
per_cpu(cpu_state, cpu) = CPU_UP_PREPARE;
wmb();
while (!cpu_online(cpu))
cpu_relax();
fixup_irqs(cpu_online_map);
/* counter the irq disable in fixup_irqs */
local_irq_enable();
return 0;
}
void generic_cpu_die(unsigned int cpu)
{
int i;
for (i = 0; i < 100; i++) {
rmb();
if (per_cpu(cpu_state, cpu) == CPU_DEAD)
return;
msleep(100);
}
printk(KERN_ERR "CPU%d didn't die...\n", cpu);
}
void generic_mach_cpu_die(void)
{
unsigned int cpu;
local_irq_disable();
cpu = smp_processor_id();
printk(KERN_DEBUG "CPU%d offline\n", cpu);
__get_cpu_var(cpu_state) = CPU_DEAD;
wmb();
while (__get_cpu_var(cpu_state) != CPU_UP_PREPARE)
cpu_relax();
flush_tlb_pending();
cpu_set(cpu, cpu_online_map);
local_irq_enable();
}
#endif
static int __devinit cpu_enable(unsigned int cpu)
{
if (smp_ops->cpu_enable)
return smp_ops->cpu_enable(cpu);
return -ENOSYS;
}
int __devinit __cpu_up(unsigned int cpu) int __devinit __cpu_up(unsigned int cpu)
{ {
int c; int c;
if (!cpu_enable(cpu))
return 0;
/* At boot, don't bother with non-present cpus -JSCHOPP */ /* At boot, don't bother with non-present cpus -JSCHOPP */
if (system_state < SYSTEM_RUNNING && !cpu_present(cpu)) if (system_state < SYSTEM_RUNNING && !cpu_present(cpu))
return -ENOENT; return -ENOENT;
......
...@@ -18,7 +18,7 @@ ...@@ -18,7 +18,7 @@
#include <asm/systemcfg.h> #include <asm/systemcfg.h>
#include <asm/paca.h> #include <asm/paca.h>
#include <asm/lppaca.h> #include <asm/lppaca.h>
#include <asm/machdep.h>
static DEFINE_PER_CPU(struct cpu, cpu_devices); static DEFINE_PER_CPU(struct cpu, cpu_devices);
...@@ -413,9 +413,7 @@ static int __init topology_init(void) ...@@ -413,9 +413,7 @@ static int __init topology_init(void)
* CPU. For instance, the boot cpu might never be valid * CPU. For instance, the boot cpu might never be valid
* for hotplugging. * for hotplugging.
*/ */
#ifdef CONFIG_HOTPLUG_CPU if (!ppc_md.cpu_die)
if (systemcfg->platform != PLATFORM_PSERIES_LPAR)
#endif
c->no_control = 1; c->no_control = 1;
if (cpu_online(cpu) || (c->no_control == 0)) { if (cpu_online(cpu) || (c->no_control == 0)) {
......
...@@ -30,6 +30,7 @@ struct smp_ops_t { ...@@ -30,6 +30,7 @@ struct smp_ops_t {
void (*setup_cpu)(int nr); void (*setup_cpu)(int nr);
void (*take_timebase)(void); void (*take_timebase)(void);
void (*give_timebase)(void); void (*give_timebase)(void);
int (*cpu_enable)(unsigned int nr);
int (*cpu_disable)(void); int (*cpu_disable)(void);
void (*cpu_die)(unsigned int nr); void (*cpu_die)(unsigned int nr);
}; };
......
...@@ -29,7 +29,7 @@ ...@@ -29,7 +29,7 @@
extern int boot_cpuid; extern int boot_cpuid;
extern int boot_cpuid_phys; extern int boot_cpuid_phys;
extern void cpu_die(void) __attribute__((noreturn)); extern void cpu_die(void);
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
...@@ -37,6 +37,13 @@ extern void smp_send_debugger_break(int cpu); ...@@ -37,6 +37,13 @@ extern void smp_send_debugger_break(int cpu);
struct pt_regs; struct pt_regs;
extern void smp_message_recv(int, struct pt_regs *); extern void smp_message_recv(int, struct pt_regs *);
#ifdef CONFIG_HOTPLUG_CPU
extern void fixup_irqs(cpumask_t map);
int generic_cpu_disable(void);
int generic_cpu_enable(unsigned int cpu);
void generic_cpu_die(unsigned int cpu);
void generic_mach_cpu_die(void);
#endif
#define __smp_processor_id() (get_paca()->paca_index) #define __smp_processor_id() (get_paca()->paca_index)
#define hard_smp_processor_id() (get_paca()->hw_cpu_id) #define hard_smp_processor_id() (get_paca()->hw_cpu_id)
......
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