aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/kernel/setup.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/x86/kernel/setup.c')
-rw-r--r--arch/x86/kernel/setup.c132
1 files changed, 127 insertions, 5 deletions
diff --git a/arch/x86/kernel/setup.c b/arch/x86/kernel/setup.c
index 0dff17ee3d73..913af838c3c5 100644
--- a/arch/x86/kernel/setup.c
+++ b/arch/x86/kernel/setup.c
@@ -35,6 +35,16 @@ EXPORT_EARLY_PER_CPU_SYMBOL(x86_bios_cpu_apicid);
35/* map cpu index to node index */ 35/* map cpu index to node index */
36DEFINE_EARLY_PER_CPU(int, x86_cpu_to_node_map, NUMA_NO_NODE); 36DEFINE_EARLY_PER_CPU(int, x86_cpu_to_node_map, NUMA_NO_NODE);
37EXPORT_EARLY_PER_CPU_SYMBOL(x86_cpu_to_node_map); 37EXPORT_EARLY_PER_CPU_SYMBOL(x86_cpu_to_node_map);
38
39/* which logical CPUs are on which nodes */
40cpumask_t *node_to_cpumask_map;
41EXPORT_SYMBOL(node_to_cpumask_map);
42
43/* setup node_to_cpumask_map */
44static void __init setup_node_to_cpumask_map(void);
45
46#else
47static inline void setup_node_to_cpumask_map(void) { }
38#endif 48#endif
39 49
40#if defined(CONFIG_HAVE_SETUP_PER_CPU_AREA) && defined(CONFIG_X86_SMP) 50#if defined(CONFIG_HAVE_SETUP_PER_CPU_AREA) && defined(CONFIG_X86_SMP)
@@ -140,11 +150,15 @@ void __init setup_per_cpu_areas(void)
140 } 150 }
141 151
142 nr_cpu_ids = highest_cpu + 1; 152 nr_cpu_ids = highest_cpu + 1;
143 printk(KERN_DEBUG "NR_CPUS: %d, nr_cpu_ids: %d\n", NR_CPUS, nr_cpu_ids); 153 printk(KERN_DEBUG "NR_CPUS: %d, nr_cpu_ids: %d, nr_node_ids %d\n",
154 NR_CPUS, nr_cpu_ids, nr_node_ids);
144 155
145 /* Setup percpu data maps */ 156 /* Setup percpu data maps */
146 setup_per_cpu_maps(); 157 setup_per_cpu_maps();
147 158
159 /* Setup node to cpumask map */
160 setup_node_to_cpumask_map();
161
148 /* Setup cpumask_of_cpu map */ 162 /* Setup cpumask_of_cpu map */
149 setup_cpumask_of_cpu(); 163 setup_cpumask_of_cpu();
150} 164}
@@ -152,6 +166,35 @@ void __init setup_per_cpu_areas(void)
152#endif 166#endif
153 167
154#ifdef X86_64_NUMA 168#ifdef X86_64_NUMA
169
170/*
171 * Allocate node_to_cpumask_map based on number of available nodes
172 * Requires node_possible_map to be valid.
173 *
174 * Note: node_to_cpumask() is not valid until after this is done.
175 */
176static void __init setup_node_to_cpumask_map(void)
177{
178 unsigned int node, num = 0;
179 cpumask_t *map;
180
181 /* setup nr_node_ids if not done yet */
182 if (nr_node_ids == MAX_NUMNODES) {
183 for_each_node_mask(node, node_possible_map)
184 num = node;
185 nr_node_ids = num + 1;
186 }
187
188 /* allocate the map */
189 map = alloc_bootmem_low(nr_node_ids * sizeof(cpumask_t));
190
191 Dprintk(KERN_DEBUG "Node to cpumask map at %p for %d nodes\n",
192 map, nr_node_ids);
193
194 /* node_to_cpumask() will now work */
195 node_to_cpumask_map = map;
196}
197
155void __cpuinit numa_set_node(int cpu, int node) 198void __cpuinit numa_set_node(int cpu, int node)
156{ 199{
157 int *cpu_to_node_map = early_per_cpu_ptr(x86_cpu_to_node_map); 200 int *cpu_to_node_map = early_per_cpu_ptr(x86_cpu_to_node_map);
@@ -174,6 +217,8 @@ void __cpuinit numa_clear_node(int cpu)
174 numa_set_node(cpu, NUMA_NO_NODE); 217 numa_set_node(cpu, NUMA_NO_NODE);
175} 218}
176 219
220#ifndef CONFIG_DEBUG_PER_CPU_MAPS
221
177void __cpuinit numa_add_cpu(int cpu) 222void __cpuinit numa_add_cpu(int cpu)
178{ 223{
179 cpu_set(cpu, node_to_cpumask_map[early_cpu_to_node(cpu)]); 224 cpu_set(cpu, node_to_cpumask_map[early_cpu_to_node(cpu)]);
@@ -183,9 +228,44 @@ void __cpuinit numa_remove_cpu(int cpu)
183{ 228{
184 cpu_clear(cpu, node_to_cpumask_map[cpu_to_node(cpu)]); 229 cpu_clear(cpu, node_to_cpumask_map[cpu_to_node(cpu)]);
185} 230}
186#endif /* CONFIG_NUMA */
187 231
188#if defined(CONFIG_DEBUG_PER_CPU_MAPS) && defined(CONFIG_X86_64) 232#else /* CONFIG_DEBUG_PER_CPU_MAPS */
233
234/*
235 * --------- debug versions of the numa functions ---------
236 */
237static void __cpuinit numa_set_cpumask(int cpu, int enable)
238{
239 int node = cpu_to_node(cpu);
240 cpumask_t *mask;
241 char buf[64];
242
243 if (node_to_cpumask_map == NULL) {
244 printk(KERN_ERR "node_to_cpumask_map NULL\n");
245 dump_stack();
246 return;
247 }
248
249 mask = &node_to_cpumask_map[node];
250 if (enable)
251 cpu_set(cpu, *mask);
252 else
253 cpu_clear(cpu, *mask);
254
255 cpulist_scnprintf(buf, sizeof(buf), *mask);
256 printk(KERN_DEBUG "%s cpu %d node %d: mask now %s\n",
257 enable? "numa_add_cpu":"numa_remove_cpu", cpu, node, buf);
258 }
259
260void __cpuinit numa_add_cpu(int cpu)
261{
262 numa_set_cpumask(cpu, 1);
263}
264
265void __cpuinit numa_remove_cpu(int cpu)
266{
267 numa_set_cpumask(cpu, 0);
268}
189 269
190int cpu_to_node(int cpu) 270int cpu_to_node(int cpu)
191{ 271{
@@ -199,6 +279,10 @@ int cpu_to_node(int cpu)
199} 279}
200EXPORT_SYMBOL(cpu_to_node); 280EXPORT_SYMBOL(cpu_to_node);
201 281
282/*
283 * Same function as cpu_to_node() but used if called before the
284 * per_cpu areas are setup.
285 */
202int early_cpu_to_node(int cpu) 286int early_cpu_to_node(int cpu)
203{ 287{
204 if (early_per_cpu_ptr(x86_cpu_to_node_map)) 288 if (early_per_cpu_ptr(x86_cpu_to_node_map))
@@ -207,9 +291,47 @@ int early_cpu_to_node(int cpu)
207 if (!per_cpu_offset(cpu)) { 291 if (!per_cpu_offset(cpu)) {
208 printk(KERN_WARNING 292 printk(KERN_WARNING
209 "early_cpu_to_node(%d): no per_cpu area!\n", cpu); 293 "early_cpu_to_node(%d): no per_cpu area!\n", cpu);
210 dump_stack(); 294 dump_stack();
211 return NUMA_NO_NODE; 295 return NUMA_NO_NODE;
212 } 296 }
213 return per_cpu(x86_cpu_to_node_map, cpu); 297 return per_cpu(x86_cpu_to_node_map, cpu);
214} 298}
215#endif 299
300/*
301 * Returns a pointer to the bitmask of CPUs on Node 'node'.
302 */
303cpumask_t *_node_to_cpumask_ptr(int node)
304{
305 if (node_to_cpumask_map == NULL) {
306 printk(KERN_WARNING
307 "_node_to_cpumask_ptr(%d): no node_to_cpumask_map!\n",
308 node);
309 dump_stack();
310 return &cpu_online_map;
311 }
312 return &node_to_cpumask_map[node];
313}
314EXPORT_SYMBOL(_node_to_cpumask_ptr);
315
316/*
317 * Returns a bitmask of CPUs on Node 'node'.
318 */
319cpumask_t node_to_cpumask(int node)
320{
321 if (node_to_cpumask_map == NULL) {
322 printk(KERN_WARNING
323 "node_to_cpumask(%d): no node_to_cpumask_map!\n", node);
324 dump_stack();
325 return cpu_online_map;
326 }
327 return node_to_cpumask_map[node];
328}
329EXPORT_SYMBOL(node_to_cpumask);
330
331/*
332 * --------- end of debug versions of the numa functions ---------
333 */
334
335#endif /* CONFIG_DEBUG_PER_CPU_MAPS */
336
337#endif /* X86_64_NUMA */