aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--arch/x86/kernel/setup.c132
-rw-r--r--arch/x86/mm/numa_64.c6
-rw-r--r--include/asm-x86/topology.h25
3 files changed, 144 insertions, 19 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 */
diff --git a/arch/x86/mm/numa_64.c b/arch/x86/mm/numa_64.c
index 970f86775c41..14c7ab417ec7 100644
--- a/arch/x86/mm/numa_64.c
+++ b/arch/x86/mm/numa_64.c
@@ -35,9 +35,6 @@ s16 apicid_to_node[MAX_LOCAL_APIC] __cpuinitdata = {
35 [0 ... MAX_LOCAL_APIC-1] = NUMA_NO_NODE 35 [0 ... MAX_LOCAL_APIC-1] = NUMA_NO_NODE
36}; 36};
37 37
38cpumask_t node_to_cpumask_map[MAX_NUMNODES] __read_mostly;
39EXPORT_SYMBOL(node_to_cpumask_map);
40
41int numa_off __initdata; 38int numa_off __initdata;
42unsigned long __initdata nodemap_addr; 39unsigned long __initdata nodemap_addr;
43unsigned long __initdata nodemap_size; 40unsigned long __initdata nodemap_size;
@@ -560,9 +557,6 @@ void __init numa_initmem_init(unsigned long start_pfn, unsigned long end_pfn)
560 node_set(0, node_possible_map); 557 node_set(0, node_possible_map);
561 for (i = 0; i < NR_CPUS; i++) 558 for (i = 0; i < NR_CPUS; i++)
562 numa_set_node(i, 0); 559 numa_set_node(i, 0);
563 /* cpumask_of_cpu() may not be available during early startup */
564 memset(&node_to_cpumask_map[0], 0, sizeof(node_to_cpumask_map[0]));
565 cpu_set(0, node_to_cpumask_map[0]);
566 e820_register_active_regions(0, start_pfn, end_pfn); 560 e820_register_active_regions(0, start_pfn, end_pfn);
567 setup_node_bootmem(0, start_pfn << PAGE_SHIFT, end_pfn << PAGE_SHIFT); 561 setup_node_bootmem(0, start_pfn << PAGE_SHIFT, end_pfn << PAGE_SHIFT);
568} 562}
diff --git a/include/asm-x86/topology.h b/include/asm-x86/topology.h
index c0e6ff7671ea..1f97758de4ab 100644
--- a/include/asm-x86/topology.h
+++ b/include/asm-x86/topology.h
@@ -57,10 +57,16 @@ static inline int cpu_to_node(int cpu)
57} 57}
58#define early_cpu_to_node(cpu) cpu_to_node(cpu) 58#define early_cpu_to_node(cpu) cpu_to_node(cpu)
59 59
60/* Returns a bitmask of CPUs on Node 'node'. */
61static inline cpumask_t node_to_cpumask(int node)
62{
63 return node_to_cpumask_map[node];
64}
65
60#else /* CONFIG_X86_64 */ 66#else /* CONFIG_X86_64 */
61 67
62/* Mappings between node number and cpus on that node. */ 68/* Mappings between node number and cpus on that node. */
63extern cpumask_t node_to_cpumask_map[]; 69extern cpumask_t *node_to_cpumask_map;
64 70
65/* Mappings between logical cpu number and node number */ 71/* Mappings between logical cpu number and node number */
66DECLARE_EARLY_PER_CPU(int, x86_cpu_to_node_map); 72DECLARE_EARLY_PER_CPU(int, x86_cpu_to_node_map);
@@ -104,7 +110,6 @@ static inline cpumask_t node_to_cpumask(int node)
104} 110}
105 111
106#endif /* !CONFIG_DEBUG_PER_CPU_MAPS */ 112#endif /* !CONFIG_DEBUG_PER_CPU_MAPS */
107#endif /* CONFIG_X86_64 */
108 113
109/* Replace default node_to_cpumask_ptr with optimized version */ 114/* Replace default node_to_cpumask_ptr with optimized version */
110#define node_to_cpumask_ptr(v, node) \ 115#define node_to_cpumask_ptr(v, node) \
@@ -113,12 +118,7 @@ static inline cpumask_t node_to_cpumask(int node)
113#define node_to_cpumask_ptr_next(v, node) \ 118#define node_to_cpumask_ptr_next(v, node) \
114 v = _node_to_cpumask_ptr(node) 119 v = _node_to_cpumask_ptr(node)
115 120
116/* Returns the number of the first CPU on Node 'node'. */ 121#endif /* CONFIG_X86_64 */
117static inline int node_to_first_cpu(int node)
118{
119 node_to_cpumask_ptr(mask, node);
120 return first_cpu(*mask);
121}
122 122
123/* 123/*
124 * Returns the number of the node containing Node 'node'. This 124 * Returns the number of the node containing Node 'node'. This
@@ -204,6 +204,15 @@ static inline int node_to_first_cpu(int node)
204 204
205#include <asm-generic/topology.h> 205#include <asm-generic/topology.h>
206 206
207#ifdef CONFIG_NUMA
208/* Returns the number of the first CPU on Node 'node'. */
209static inline int node_to_first_cpu(int node)
210{
211 node_to_cpumask_ptr(mask, node);
212 return first_cpu(*mask);
213}
214#endif
215
207extern cpumask_t cpu_coregroup_map(int cpu); 216extern cpumask_t cpu_coregroup_map(int cpu);
208 217
209#ifdef ENABLE_TOPO_DEFINES 218#ifdef ENABLE_TOPO_DEFINES