diff options
author | minskey guo <chaohong_guo@linux.intel.com> | 2010-05-24 17:32:41 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2010-05-25 11:07:00 -0400 |
commit | cf23422b9d76215316855253da491d4c9f294372 (patch) | |
tree | 5663d2519d83d830d24dffdf2571d58d3e55d3f6 /kernel | |
parent | 8b25c6d2231b978ccce9c401e771932bde79aa9f (diff) |
cpu/mem hotplug: enable CPUs online before local memory online
Enable users to online CPUs even if the CPUs belongs to a numa node which
doesn't have onlined local memory.
The zonlists(pg_data_t.node_zonelists[]) of a numa node are created either
in system boot/init period, or at the time of local memory online. For a
numa node without onlined local memory, its zonelists are not initialized
at present. As a result, any memory allocation operations executed by
CPUs within this node will fail. In fact, an out-of-memory error is
triggered when attempt to online CPUs before memory comes to online.
This patch tries to create zonelists for such numa nodes, so that the
memory allocation for this node can be fallback'ed to other nodes.
[akpm@linux-foundation.org: remove unneeded export]
[akpm@linux-foundation.org: coding-style fixes]
Signed-off-by: minskey guo<chaohong.guo@intel.com>
Cc: Minchan Kim <minchan.kim@gmail.com>
Cc: Yasunori Goto <y-goto@jp.fujitsu.com>
Cc: Andi Kleen <andi@firstfloor.org>
Cc: Christoph Lameter <cl@linux-foundation.org>
Cc: Tejun Heo <tj@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'kernel')
-rw-r--r-- | kernel/cpu.c | 25 |
1 files changed, 25 insertions, 0 deletions
diff --git a/kernel/cpu.c b/kernel/cpu.c index 545777574779..a3fbcc0a0abc 100644 --- a/kernel/cpu.c +++ b/kernel/cpu.c | |||
@@ -326,6 +326,12 @@ out_notify: | |||
326 | int __cpuinit cpu_up(unsigned int cpu) | 326 | int __cpuinit cpu_up(unsigned int cpu) |
327 | { | 327 | { |
328 | int err = 0; | 328 | int err = 0; |
329 | |||
330 | #ifdef CONFIG_MEMORY_HOTPLUG | ||
331 | int nid; | ||
332 | pg_data_t *pgdat; | ||
333 | #endif | ||
334 | |||
329 | if (!cpu_possible(cpu)) { | 335 | if (!cpu_possible(cpu)) { |
330 | printk(KERN_ERR "can't online cpu %d because it is not " | 336 | printk(KERN_ERR "can't online cpu %d because it is not " |
331 | "configured as may-hotadd at boot time\n", cpu); | 337 | "configured as may-hotadd at boot time\n", cpu); |
@@ -336,6 +342,25 @@ int __cpuinit cpu_up(unsigned int cpu) | |||
336 | return -EINVAL; | 342 | return -EINVAL; |
337 | } | 343 | } |
338 | 344 | ||
345 | #ifdef CONFIG_MEMORY_HOTPLUG | ||
346 | nid = cpu_to_node(cpu); | ||
347 | if (!node_online(nid)) { | ||
348 | err = mem_online_node(nid); | ||
349 | if (err) | ||
350 | return err; | ||
351 | } | ||
352 | |||
353 | pgdat = NODE_DATA(nid); | ||
354 | if (!pgdat) { | ||
355 | printk(KERN_ERR | ||
356 | "Can't online cpu %d due to NULL pgdat\n", cpu); | ||
357 | return -ENOMEM; | ||
358 | } | ||
359 | |||
360 | if (pgdat->node_zonelists->_zonerefs->zone == NULL) | ||
361 | build_all_zonelists(); | ||
362 | #endif | ||
363 | |||
339 | cpu_maps_update_begin(); | 364 | cpu_maps_update_begin(); |
340 | 365 | ||
341 | if (cpu_hotplug_disabled) { | 366 | if (cpu_hotplug_disabled) { |