diff options
-rw-r--r-- | arch/x86_64/kernel/head64.c | 3 | ||||
-rw-r--r-- | arch/x86_64/kernel/setup64.c | 3 | ||||
-rw-r--r-- | arch/x86_64/kernel/smpboot.c | 17 | ||||
-rw-r--r-- | include/asm-x86_64/pda.h | 5 |
4 files changed, 25 insertions, 3 deletions
diff --git a/arch/x86_64/kernel/head64.c b/arch/x86_64/kernel/head64.c index b675c5add01e..2ea42ceb08f3 100644 --- a/arch/x86_64/kernel/head64.c +++ b/arch/x86_64/kernel/head64.c | |||
@@ -92,6 +92,9 @@ void __init x86_64_start_kernel(char * real_mode_data) | |||
92 | memcpy(init_level4_pgt, boot_level4_pgt, PTRS_PER_PGD*sizeof(pgd_t)); | 92 | memcpy(init_level4_pgt, boot_level4_pgt, PTRS_PER_PGD*sizeof(pgd_t)); |
93 | asm volatile("movq %0,%%cr3" :: "r" (__pa_symbol(&init_level4_pgt))); | 93 | asm volatile("movq %0,%%cr3" :: "r" (__pa_symbol(&init_level4_pgt))); |
94 | 94 | ||
95 | for (i = 0; i < NR_CPUS; i++) | ||
96 | cpu_pda(i) = &boot_cpu_pda[i]; | ||
97 | |||
95 | pda_init(0); | 98 | pda_init(0); |
96 | copy_bootdata(real_mode_data); | 99 | copy_bootdata(real_mode_data); |
97 | #ifdef CONFIG_SMP | 100 | #ifdef CONFIG_SMP |
diff --git a/arch/x86_64/kernel/setup64.c b/arch/x86_64/kernel/setup64.c index 66325a79ed53..6eff51e9400c 100644 --- a/arch/x86_64/kernel/setup64.c +++ b/arch/x86_64/kernel/setup64.c | |||
@@ -30,7 +30,8 @@ char x86_boot_params[BOOT_PARAM_SIZE] __initdata = {0,}; | |||
30 | 30 | ||
31 | cpumask_t cpu_initialized __cpuinitdata = CPU_MASK_NONE; | 31 | cpumask_t cpu_initialized __cpuinitdata = CPU_MASK_NONE; |
32 | 32 | ||
33 | struct x8664_pda _cpu_pda[NR_CPUS] __cacheline_aligned; | 33 | struct x8664_pda *_cpu_pda[NR_CPUS] __read_mostly; |
34 | struct x8664_pda boot_cpu_pda[NR_CPUS] __cacheline_aligned; | ||
34 | 35 | ||
35 | struct desc_ptr idt_descr = { 256 * 16, (unsigned long) idt_table }; | 36 | struct desc_ptr idt_descr = { 256 * 16, (unsigned long) idt_table }; |
36 | 37 | ||
diff --git a/arch/x86_64/kernel/smpboot.c b/arch/x86_64/kernel/smpboot.c index 4884ca12444a..c8169d066cbd 100644 --- a/arch/x86_64/kernel/smpboot.c +++ b/arch/x86_64/kernel/smpboot.c | |||
@@ -757,6 +757,23 @@ static int __cpuinit do_boot_cpu(int cpu, int apicid) | |||
757 | return -1; | 757 | return -1; |
758 | } | 758 | } |
759 | 759 | ||
760 | /* Allocate node local memory for AP pdas */ | ||
761 | if (cpu_pda(cpu) == &boot_cpu_pda[cpu]) { | ||
762 | struct x8664_pda *newpda, *pda; | ||
763 | int node = cpu_to_node(cpu); | ||
764 | pda = cpu_pda(cpu); | ||
765 | newpda = kmalloc_node(sizeof (struct x8664_pda), GFP_ATOMIC, | ||
766 | node); | ||
767 | if (newpda) { | ||
768 | memcpy(newpda, pda, sizeof (struct x8664_pda)); | ||
769 | cpu_pda(cpu) = newpda; | ||
770 | } else | ||
771 | printk(KERN_ERR | ||
772 | "Could not allocate node local PDA for CPU %d on node %d\n", | ||
773 | cpu, node); | ||
774 | } | ||
775 | |||
776 | |||
760 | c_idle.idle = get_idle_for_cpu(cpu); | 777 | c_idle.idle = get_idle_for_cpu(cpu); |
761 | 778 | ||
762 | if (c_idle.idle) { | 779 | if (c_idle.idle) { |
diff --git a/include/asm-x86_64/pda.h b/include/asm-x86_64/pda.h index dc33d28a72dd..c7ab38a601af 100644 --- a/include/asm-x86_64/pda.h +++ b/include/asm-x86_64/pda.h | |||
@@ -27,9 +27,10 @@ struct x8664_pda { | |||
27 | unsigned apic_timer_irqs; | 27 | unsigned apic_timer_irqs; |
28 | } ____cacheline_aligned_in_smp; | 28 | } ____cacheline_aligned_in_smp; |
29 | 29 | ||
30 | extern struct x8664_pda _cpu_pda[]; | 30 | extern struct x8664_pda *_cpu_pda[]; |
31 | extern struct x8664_pda boot_cpu_pda[]; | ||
31 | 32 | ||
32 | #define cpu_pda(i) (&_cpu_pda[i]) | 33 | #define cpu_pda(i) (_cpu_pda[i]) |
33 | 34 | ||
34 | /* | 35 | /* |
35 | * There is no fast way to get the base address of the PDA, all the accesses | 36 | * There is no fast way to get the base address of the PDA, all the accesses |