diff options
author | Linus Torvalds <torvalds@g5.osdl.org> | 2005-11-02 21:38:22 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@g5.osdl.org> | 2005-11-02 21:38:22 -0500 |
commit | cfa024f4e45562c50b9eccb23649ab103578037b (patch) | |
tree | d49992521230a4e302c6d4bef9191e885220b82e /arch | |
parent | 3a7142371efdc95f4c5b5ffc188b18efdc4e64dd (diff) | |
parent | a054a811597a17ffbe92bc4db04a4dc2f1b1ea55 (diff) |
Merge master.kernel.org:/home/rmk/linux-2.6-arm
Diffstat (limited to 'arch')
-rw-r--r-- | arch/arm/Kconfig | 7 | ||||
-rw-r--r-- | arch/arm/kernel/irq.c | 31 | ||||
-rw-r--r-- | arch/arm/kernel/process.c | 9 | ||||
-rw-r--r-- | arch/arm/kernel/smp.c | 109 | ||||
-rw-r--r-- | arch/arm/mach-realview/realview_eb.c | 44 | ||||
-rw-r--r-- | arch/arm/mm/init.c | 9 |
6 files changed, 192 insertions, 17 deletions
diff --git a/arch/arm/Kconfig b/arch/arm/Kconfig index dc6d8342e5e6..6b12d71978de 100644 --- a/arch/arm/Kconfig +++ b/arch/arm/Kconfig | |||
@@ -349,6 +349,13 @@ config NR_CPUS | |||
349 | depends on SMP | 349 | depends on SMP |
350 | default "4" | 350 | default "4" |
351 | 351 | ||
352 | config HOTPLUG_CPU | ||
353 | bool "Support for hot-pluggable CPUs (EXPERIMENTAL)" | ||
354 | depends on SMP && HOTPLUG && EXPERIMENTAL | ||
355 | help | ||
356 | Say Y here to experiment with turning CPUs off and on. CPUs | ||
357 | can be controlled through /sys/devices/system/cpu. | ||
358 | |||
352 | config PREEMPT | 359 | config PREEMPT |
353 | bool "Preemptible Kernel (EXPERIMENTAL)" | 360 | bool "Preemptible Kernel (EXPERIMENTAL)" |
354 | depends on EXPERIMENTAL | 361 | depends on EXPERIMENTAL |
diff --git a/arch/arm/kernel/irq.c b/arch/arm/kernel/irq.c index 3284118f356b..9def4404e1f2 100644 --- a/arch/arm/kernel/irq.c +++ b/arch/arm/kernel/irq.c | |||
@@ -1050,3 +1050,34 @@ static int __init noirqdebug_setup(char *str) | |||
1050 | } | 1050 | } |
1051 | 1051 | ||
1052 | __setup("noirqdebug", noirqdebug_setup); | 1052 | __setup("noirqdebug", noirqdebug_setup); |
1053 | |||
1054 | #ifdef CONFIG_HOTPLUG_CPU | ||
1055 | /* | ||
1056 | * The CPU has been marked offline. Migrate IRQs off this CPU. If | ||
1057 | * the affinity settings do not allow other CPUs, force them onto any | ||
1058 | * available CPU. | ||
1059 | */ | ||
1060 | void migrate_irqs(void) | ||
1061 | { | ||
1062 | unsigned int i, cpu = smp_processor_id(); | ||
1063 | |||
1064 | for (i = 0; i < NR_IRQS; i++) { | ||
1065 | struct irqdesc *desc = irq_desc + i; | ||
1066 | |||
1067 | if (desc->cpu == cpu) { | ||
1068 | unsigned int newcpu = any_online_cpu(desc->affinity); | ||
1069 | |||
1070 | if (newcpu == NR_CPUS) { | ||
1071 | if (printk_ratelimit()) | ||
1072 | printk(KERN_INFO "IRQ%u no longer affine to CPU%u\n", | ||
1073 | i, cpu); | ||
1074 | |||
1075 | cpus_setall(desc->affinity); | ||
1076 | newcpu = any_online_cpu(desc->affinity); | ||
1077 | } | ||
1078 | |||
1079 | route_irq(desc, i, newcpu); | ||
1080 | } | ||
1081 | } | ||
1082 | } | ||
1083 | #endif /* CONFIG_HOTPLUG_CPU */ | ||
diff --git a/arch/arm/kernel/process.c b/arch/arm/kernel/process.c index 409db6d5ec99..ba298277becd 100644 --- a/arch/arm/kernel/process.c +++ b/arch/arm/kernel/process.c | |||
@@ -26,6 +26,7 @@ | |||
26 | #include <linux/interrupt.h> | 26 | #include <linux/interrupt.h> |
27 | #include <linux/kallsyms.h> | 27 | #include <linux/kallsyms.h> |
28 | #include <linux/init.h> | 28 | #include <linux/init.h> |
29 | #include <linux/cpu.h> | ||
29 | 30 | ||
30 | #include <asm/system.h> | 31 | #include <asm/system.h> |
31 | #include <asm/io.h> | 32 | #include <asm/io.h> |
@@ -105,6 +106,14 @@ void cpu_idle(void) | |||
105 | /* endless idle loop with no priority at all */ | 106 | /* endless idle loop with no priority at all */ |
106 | while (1) { | 107 | while (1) { |
107 | void (*idle)(void) = pm_idle; | 108 | void (*idle)(void) = pm_idle; |
109 | |||
110 | #ifdef CONFIG_HOTPLUG_CPU | ||
111 | if (cpu_is_offline(smp_processor_id())) { | ||
112 | leds_event(led_idle_start); | ||
113 | cpu_die(); | ||
114 | } | ||
115 | #endif | ||
116 | |||
108 | if (!idle) | 117 | if (!idle) |
109 | idle = default_idle; | 118 | idle = default_idle; |
110 | preempt_disable(); | 119 | preempt_disable(); |
diff --git a/arch/arm/kernel/smp.c b/arch/arm/kernel/smp.c index 826164945747..edb5a406922f 100644 --- a/arch/arm/kernel/smp.c +++ b/arch/arm/kernel/smp.c | |||
@@ -80,19 +80,23 @@ static DEFINE_SPINLOCK(smp_call_function_lock); | |||
80 | 80 | ||
81 | int __cpuinit __cpu_up(unsigned int cpu) | 81 | int __cpuinit __cpu_up(unsigned int cpu) |
82 | { | 82 | { |
83 | struct task_struct *idle; | 83 | struct cpuinfo_arm *ci = &per_cpu(cpu_data, cpu); |
84 | struct task_struct *idle = ci->idle; | ||
84 | pgd_t *pgd; | 85 | pgd_t *pgd; |
85 | pmd_t *pmd; | 86 | pmd_t *pmd; |
86 | int ret; | 87 | int ret; |
87 | 88 | ||
88 | /* | 89 | /* |
89 | * Spawn a new process manually. Grab a pointer to | 90 | * Spawn a new process manually, if not already done. |
90 | * its task struct so we can mess with it | 91 | * Grab a pointer to its task struct so we can mess with it |
91 | */ | 92 | */ |
92 | idle = fork_idle(cpu); | 93 | if (!idle) { |
93 | if (IS_ERR(idle)) { | 94 | idle = fork_idle(cpu); |
94 | printk(KERN_ERR "CPU%u: fork() failed\n", cpu); | 95 | if (IS_ERR(idle)) { |
95 | return PTR_ERR(idle); | 96 | printk(KERN_ERR "CPU%u: fork() failed\n", cpu); |
97 | return PTR_ERR(idle); | ||
98 | } | ||
99 | ci->idle = idle; | ||
96 | } | 100 | } |
97 | 101 | ||
98 | /* | 102 | /* |
@@ -155,6 +159,91 @@ int __cpuinit __cpu_up(unsigned int cpu) | |||
155 | return ret; | 159 | return ret; |
156 | } | 160 | } |
157 | 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 | |||
158 | /* | 247 | /* |
159 | * 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 |
160 | * idle thread stack, but a set of temporary page tables. | 249 | * idle thread stack, but a set of temporary page tables. |
@@ -236,6 +325,8 @@ void __init smp_prepare_boot_cpu(void) | |||
236 | { | 325 | { |
237 | unsigned int cpu = smp_processor_id(); | 326 | unsigned int cpu = smp_processor_id(); |
238 | 327 | ||
328 | per_cpu(cpu_data, cpu).idle = current; | ||
329 | |||
239 | cpu_set(cpu, cpu_possible_map); | 330 | cpu_set(cpu, cpu_possible_map); |
240 | cpu_set(cpu, cpu_present_map); | 331 | cpu_set(cpu, cpu_present_map); |
241 | cpu_set(cpu, cpu_online_map); | 332 | cpu_set(cpu, cpu_online_map); |
@@ -309,8 +400,8 @@ int smp_call_function_on_cpu(void (*func)(void *info), void *info, int retry, | |||
309 | printk(KERN_CRIT | 400 | printk(KERN_CRIT |
310 | "CPU%u: smp_call_function timeout for %p(%p)\n" | 401 | "CPU%u: smp_call_function timeout for %p(%p)\n" |
311 | " callmap %lx pending %lx, %swait\n", | 402 | " callmap %lx pending %lx, %swait\n", |
312 | smp_processor_id(), func, info, callmap, data.pending, | 403 | smp_processor_id(), func, info, *cpus_addr(callmap), |
313 | wait ? "" : "no "); | 404 | *cpus_addr(data.pending), wait ? "" : "no "); |
314 | 405 | ||
315 | /* | 406 | /* |
316 | * TRACE | 407 | * TRACE |
diff --git a/arch/arm/mach-realview/realview_eb.c b/arch/arm/mach-realview/realview_eb.c index 01b264be5029..267bb07e39b7 100644 --- a/arch/arm/mach-realview/realview_eb.c +++ b/arch/arm/mach-realview/realview_eb.c | |||
@@ -43,14 +43,44 @@ | |||
43 | #include "clock.h" | 43 | #include "clock.h" |
44 | 44 | ||
45 | static struct map_desc realview_eb_io_desc[] __initdata = { | 45 | static struct map_desc realview_eb_io_desc[] __initdata = { |
46 | { IO_ADDRESS(REALVIEW_SYS_BASE), REALVIEW_SYS_BASE, SZ_4K, MT_DEVICE }, | 46 | { |
47 | { IO_ADDRESS(REALVIEW_GIC_CPU_BASE), REALVIEW_GIC_CPU_BASE, SZ_4K, MT_DEVICE }, | 47 | .virtual = IO_ADDRESS(REALVIEW_SYS_BASE), |
48 | { IO_ADDRESS(REALVIEW_GIC_DIST_BASE), REALVIEW_GIC_DIST_BASE, SZ_4K, MT_DEVICE }, | 48 | .pfn = __phys_to_pfn(REALVIEW_SYS_BASE), |
49 | { IO_ADDRESS(REALVIEW_SCTL_BASE), REALVIEW_SCTL_BASE, SZ_4K, MT_DEVICE }, | 49 | .length = SZ_4K, |
50 | { IO_ADDRESS(REALVIEW_TIMER0_1_BASE), REALVIEW_TIMER0_1_BASE, SZ_4K, MT_DEVICE }, | 50 | .type = MT_DEVICE, |
51 | { IO_ADDRESS(REALVIEW_TIMER2_3_BASE), REALVIEW_TIMER2_3_BASE, SZ_4K, MT_DEVICE }, | 51 | }, { |
52 | .virtual = IO_ADDRESS(REALVIEW_GIC_CPU_BASE), | ||
53 | .pfn = __phys_to_pfn(REALVIEW_GIC_CPU_BASE), | ||
54 | .length = SZ_4K, | ||
55 | .type = MT_DEVICE, | ||
56 | }, { | ||
57 | .virtual = IO_ADDRESS(REALVIEW_GIC_DIST_BASE), | ||
58 | .pfn = __phys_to_pfn(REALVIEW_GIC_DIST_BASE), | ||
59 | .length = SZ_4K, | ||
60 | .type = MT_DEVICE, | ||
61 | }, { | ||
62 | .virtual = IO_ADDRESS(REALVIEW_SCTL_BASE), | ||
63 | .pfn = __phys_to_pfn(REALVIEW_SCTL_BASE), | ||
64 | .length = SZ_4K, | ||
65 | .type = MT_DEVICE, | ||
66 | }, { | ||
67 | .virtual = IO_ADDRESS(REALVIEW_TIMER0_1_BASE), | ||
68 | .pfn = __phys_to_pfn(REALVIEW_TIMER0_1_BASE), | ||
69 | .length = SZ_4K, | ||
70 | .type = MT_DEVICE, | ||
71 | }, { | ||
72 | .virtual = IO_ADDRESS(REALVIEW_TIMER2_3_BASE), | ||
73 | .pfn = __phys_to_pfn(REALVIEW_TIMER2_3_BASE), | ||
74 | .length = SZ_4K, | ||
75 | .type = MT_DEVICE, | ||
76 | }, | ||
52 | #ifdef CONFIG_DEBUG_LL | 77 | #ifdef CONFIG_DEBUG_LL |
53 | { IO_ADDRESS(REALVIEW_UART0_BASE), REALVIEW_UART0_BASE, SZ_4K, MT_DEVICE }, | 78 | { |
79 | .virtual = IO_ADDRESS(REALVIEW_UART0_BASE), | ||
80 | .pfn = __phys_to_pfn(REALVIEW_UART0_BASE), | ||
81 | .length = SZ_4K, | ||
82 | .type = MT_DEVICE, | ||
83 | } | ||
54 | #endif | 84 | #endif |
55 | }; | 85 | }; |
56 | 86 | ||
diff --git a/arch/arm/mm/init.c b/arch/arm/mm/init.c index fd079ff1fc53..c168f322ef8c 100644 --- a/arch/arm/mm/init.c +++ b/arch/arm/mm/init.c | |||
@@ -486,10 +486,17 @@ static void __init devicemaps_init(struct machine_desc *mdesc) | |||
486 | 486 | ||
487 | /* | 487 | /* |
488 | * Ask the machine support to map in the statically mapped devices. | 488 | * Ask the machine support to map in the statically mapped devices. |
489 | * After this point, we can start to touch devices again. | ||
490 | */ | 489 | */ |
491 | if (mdesc->map_io) | 490 | if (mdesc->map_io) |
492 | mdesc->map_io(); | 491 | mdesc->map_io(); |
492 | |||
493 | /* | ||
494 | * Finally flush the tlb again - this ensures that we're in a | ||
495 | * consistent state wrt the writebuffer if the writebuffer needs | ||
496 | * draining. After this point, we can start to touch devices | ||
497 | * again. | ||
498 | */ | ||
499 | local_flush_tlb_all(); | ||
493 | } | 500 | } |
494 | 501 | ||
495 | /* | 502 | /* |