diff options
author | Ravikiran G Thirumalai <kiran@scalex86.org> | 2006-01-11 16:43:57 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@g5.osdl.org> | 2006-01-11 22:04:53 -0500 |
commit | c11efdf94d3152443c11334720824bb6c7f6c655 (patch) | |
tree | d4185d8de8b855aca51b6bec730aeab3342e2a0f /arch | |
parent | bb33421dde79f9a36d5485c56335ff178ac7d268 (diff) |
[PATCH] x86_64: Align and pad x86_64 GDT on page boundary
This patch is on the same lines as Zachary Amsden's i386 GDT page alignemnt
patch in -mm, but for x86_64.
Patch to align and pad x86_64 GDT on page boundries.
[AK: some minor cleanups and fixed incorrect TLS initialization
in CPU init.]
Signed-off-by: Nippun Goel <nippung@calsoftinc.com>
Signed-off-by: Ravikiran Thirumalai <kiran@scalex86.org>
Signed-off-by: Shai Fultheim <shai@scalex86.org>
Signed-off-by: Andi Kleen <ak@suse.de>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Diffstat (limited to 'arch')
-rw-r--r-- | arch/x86_64/kernel/head.S | 9 | ||||
-rw-r--r-- | arch/x86_64/kernel/setup64.c | 8 | ||||
-rw-r--r-- | arch/x86_64/kernel/smpboot.c | 7 | ||||
-rw-r--r-- | arch/x86_64/kernel/suspend.c | 2 |
4 files changed, 16 insertions, 10 deletions
diff --git a/arch/x86_64/kernel/head.S b/arch/x86_64/kernel/head.S index 15290968e49d..1d216a9fc6d8 100644 --- a/arch/x86_64/kernel/head.S +++ b/arch/x86_64/kernel/head.S | |||
@@ -379,7 +379,7 @@ gdt: | |||
379 | * Also sysret mandates a special GDT layout | 379 | * Also sysret mandates a special GDT layout |
380 | */ | 380 | */ |
381 | 381 | ||
382 | .align L1_CACHE_BYTES | 382 | .align PAGE_SIZE |
383 | 383 | ||
384 | /* The TLS descriptors are currently at a different place compared to i386. | 384 | /* The TLS descriptors are currently at a different place compared to i386. |
385 | Hopefully nobody expects them at a fixed place (Wine?) */ | 385 | Hopefully nobody expects them at a fixed place (Wine?) */ |
@@ -401,10 +401,11 @@ ENTRY(cpu_gdt_table) | |||
401 | gdt_end: | 401 | gdt_end: |
402 | /* asm/segment.h:GDT_ENTRIES must match this */ | 402 | /* asm/segment.h:GDT_ENTRIES must match this */ |
403 | /* This should be a multiple of the cache line size */ | 403 | /* This should be a multiple of the cache line size */ |
404 | /* GDTs of other CPUs: */ | 404 | /* GDTs of other CPUs are now dynamically allocated */ |
405 | .fill (GDT_SIZE * NR_CPUS) - (gdt_end - cpu_gdt_table) | 405 | |
406 | /* zero the remaining page */ | ||
407 | .fill PAGE_SIZE / 8 - GDT_ENTRIES,8,0 | ||
406 | 408 | ||
407 | .align L1_CACHE_BYTES | ||
408 | ENTRY(idt_table) | 409 | ENTRY(idt_table) |
409 | .rept 256 | 410 | .rept 256 |
410 | .quad 0 | 411 | .quad 0 |
diff --git a/arch/x86_64/kernel/setup64.c b/arch/x86_64/kernel/setup64.c index 39e728cfe6a0..972f5d4ce823 100644 --- a/arch/x86_64/kernel/setup64.c +++ b/arch/x86_64/kernel/setup64.c | |||
@@ -213,16 +213,14 @@ void __cpuinit cpu_init (void) | |||
213 | * Initialize the per-CPU GDT with the boot GDT, | 213 | * Initialize the per-CPU GDT with the boot GDT, |
214 | * and set up the GDT descriptor: | 214 | * and set up the GDT descriptor: |
215 | */ | 215 | */ |
216 | if (cpu) { | 216 | if (cpu) |
217 | memcpy(cpu_gdt_table[cpu], cpu_gdt_table[0], GDT_SIZE); | 217 | memcpy(cpu_gdt(cpu), cpu_gdt_table, GDT_SIZE); |
218 | } | ||
219 | 218 | ||
220 | cpu_gdt_descr[cpu].size = GDT_SIZE; | 219 | cpu_gdt_descr[cpu].size = GDT_SIZE; |
221 | cpu_gdt_descr[cpu].address = (unsigned long)cpu_gdt_table[cpu]; | ||
222 | asm volatile("lgdt %0" :: "m" (cpu_gdt_descr[cpu])); | 220 | asm volatile("lgdt %0" :: "m" (cpu_gdt_descr[cpu])); |
223 | asm volatile("lidt %0" :: "m" (idt_descr)); | 221 | asm volatile("lidt %0" :: "m" (idt_descr)); |
224 | 222 | ||
225 | memcpy(me->thread.tls_array, cpu_gdt_table[cpu], GDT_ENTRY_TLS_ENTRIES * 8); | 223 | memset(me->thread.tls_array, 0, GDT_ENTRY_TLS_ENTRIES * 8); |
226 | syscall_init(); | 224 | syscall_init(); |
227 | 225 | ||
228 | wrmsrl(MSR_FS_BASE, 0); | 226 | wrmsrl(MSR_FS_BASE, 0); |
diff --git a/arch/x86_64/kernel/smpboot.c b/arch/x86_64/kernel/smpboot.c index f513dacc177f..e6af93b51dce 100644 --- a/arch/x86_64/kernel/smpboot.c +++ b/arch/x86_64/kernel/smpboot.c | |||
@@ -744,6 +744,13 @@ static int __cpuinit do_boot_cpu(int cpu, int apicid) | |||
744 | }; | 744 | }; |
745 | DECLARE_WORK(work, do_fork_idle, &c_idle); | 745 | DECLARE_WORK(work, do_fork_idle, &c_idle); |
746 | 746 | ||
747 | /* allocate memory for gdts of secondary cpus. Hotplug is considered */ | ||
748 | if (!cpu_gdt_descr[cpu].address && | ||
749 | !(cpu_gdt_descr[cpu].address = get_zeroed_page(GFP_KERNEL))) { | ||
750 | printk(KERN_ERR "Failed to allocate GDT for CPU %d\n", cpu); | ||
751 | return -1; | ||
752 | } | ||
753 | |||
747 | c_idle.idle = get_idle_for_cpu(cpu); | 754 | c_idle.idle = get_idle_for_cpu(cpu); |
748 | 755 | ||
749 | if (c_idle.idle) { | 756 | if (c_idle.idle) { |
diff --git a/arch/x86_64/kernel/suspend.c b/arch/x86_64/kernel/suspend.c index fd2bef780882..ecbd34c1093d 100644 --- a/arch/x86_64/kernel/suspend.c +++ b/arch/x86_64/kernel/suspend.c | |||
@@ -120,7 +120,7 @@ void fix_processor_context(void) | |||
120 | 120 | ||
121 | set_tss_desc(cpu,t); /* This just modifies memory; should not be neccessary. But... This is neccessary, because 386 hardware has concept of busy TSS or some similar stupidity. */ | 121 | set_tss_desc(cpu,t); /* This just modifies memory; should not be neccessary. But... This is neccessary, because 386 hardware has concept of busy TSS or some similar stupidity. */ |
122 | 122 | ||
123 | cpu_gdt_table[cpu][GDT_ENTRY_TSS].type = 9; | 123 | cpu_gdt(cpu)[GDT_ENTRY_TSS].type = 9; |
124 | 124 | ||
125 | syscall_init(); /* This sets MSR_*STAR and related */ | 125 | syscall_init(); /* This sets MSR_*STAR and related */ |
126 | load_TR_desc(); /* This does ltr */ | 126 | load_TR_desc(); /* This does ltr */ |