diff options
-rw-r--r-- | arch/x86/kernel/setup.c | 132 | ||||
-rw-r--r-- | arch/x86/mm/numa_64.c | 6 | ||||
-rw-r--r-- | include/asm-x86/topology.h | 25 |
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 */ |
36 | DEFINE_EARLY_PER_CPU(int, x86_cpu_to_node_map, NUMA_NO_NODE); | 36 | DEFINE_EARLY_PER_CPU(int, x86_cpu_to_node_map, NUMA_NO_NODE); |
37 | EXPORT_EARLY_PER_CPU_SYMBOL(x86_cpu_to_node_map); | 37 | EXPORT_EARLY_PER_CPU_SYMBOL(x86_cpu_to_node_map); |
38 | |||
39 | /* which logical CPUs are on which nodes */ | ||
40 | cpumask_t *node_to_cpumask_map; | ||
41 | EXPORT_SYMBOL(node_to_cpumask_map); | ||
42 | |||
43 | /* setup node_to_cpumask_map */ | ||
44 | static void __init setup_node_to_cpumask_map(void); | ||
45 | |||
46 | #else | ||
47 | static 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 | */ | ||
176 | static 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 | |||
155 | void __cpuinit numa_set_node(int cpu, int node) | 198 | void __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 | |||
177 | void __cpuinit numa_add_cpu(int cpu) | 222 | void __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 | */ | ||
237 | static 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 | |||
260 | void __cpuinit numa_add_cpu(int cpu) | ||
261 | { | ||
262 | numa_set_cpumask(cpu, 1); | ||
263 | } | ||
264 | |||
265 | void __cpuinit numa_remove_cpu(int cpu) | ||
266 | { | ||
267 | numa_set_cpumask(cpu, 0); | ||
268 | } | ||
189 | 269 | ||
190 | int cpu_to_node(int cpu) | 270 | int cpu_to_node(int cpu) |
191 | { | 271 | { |
@@ -199,6 +279,10 @@ int cpu_to_node(int cpu) | |||
199 | } | 279 | } |
200 | EXPORT_SYMBOL(cpu_to_node); | 280 | EXPORT_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 | */ | ||
202 | int early_cpu_to_node(int cpu) | 286 | int 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 | */ | ||
303 | cpumask_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 | } | ||
314 | EXPORT_SYMBOL(_node_to_cpumask_ptr); | ||
315 | |||
316 | /* | ||
317 | * Returns a bitmask of CPUs on Node 'node'. | ||
318 | */ | ||
319 | cpumask_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 | } | ||
329 | EXPORT_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 | ||
38 | cpumask_t node_to_cpumask_map[MAX_NUMNODES] __read_mostly; | ||
39 | EXPORT_SYMBOL(node_to_cpumask_map); | ||
40 | |||
41 | int numa_off __initdata; | 38 | int numa_off __initdata; |
42 | unsigned long __initdata nodemap_addr; | 39 | unsigned long __initdata nodemap_addr; |
43 | unsigned long __initdata nodemap_size; | 40 | unsigned 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'. */ | ||
61 | static 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. */ |
63 | extern cpumask_t node_to_cpumask_map[]; | 69 | extern 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 */ |
66 | DECLARE_EARLY_PER_CPU(int, x86_cpu_to_node_map); | 72 | DECLARE_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 */ |
117 | static 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'. */ | ||
209 | static 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 | |||
207 | extern cpumask_t cpu_coregroup_map(int cpu); | 216 | extern cpumask_t cpu_coregroup_map(int cpu); |
208 | 217 | ||
209 | #ifdef ENABLE_TOPO_DEFINES | 218 | #ifdef ENABLE_TOPO_DEFINES |