Commit 3daa0dc8 authored by Andi Kleen's avatar Andi Kleen Committed by Linus Torvalds

[PATCH] x86_64: fix vsyscalls

Fix incorrect alignment in the vsyscall variables that caused
vsyscalls to be completely broken.

This change should decrease system time during TPC-* tests
considerably.

Clean up the vmlinux.lds to make it easier readable

Do some cleanups in the vsyscall code. 

Align cacheline_aligned correctly on 128 byte cacheline systems.
Signed-off-by: default avatarAndi Kleen <ak@suse.de>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 142ef9e6
...@@ -45,32 +45,31 @@ SECTIONS ...@@ -45,32 +45,31 @@ SECTIONS
} }
__bss_end = .; __bss_end = .;
. = ALIGN(64); . = ALIGN(CONFIG_X86_L1_CACHE_BYTES);
.data.cacheline_aligned : { *(.data.cacheline_aligned) } .data.cacheline_aligned : { *(.data.cacheline_aligned) }
#define AFTER(x) BINALIGN(LOADADDR(x) + SIZEOF(x), 16)
#define BINALIGN(x,y) (((x) + (y) - 1) & ~((y) - 1))
#define CACHE_ALIGN(x) BINALIGN(x, CONFIG_X86_L1_CACHE_BYTES)
.vsyscall_0 -10*1024*1024: AT ((LOADADDR(.data.cacheline_aligned) + SIZEOF(.data.cacheline_aligned) + 4095) & ~(4095)) { *(.vsyscall_0) } .vsyscall_0 -10*1024*1024: AT ((LOADADDR(.data.cacheline_aligned) + SIZEOF(.data.cacheline_aligned) + 4095) & ~(4095)) { *(.vsyscall_0) }
__vsyscall_0 = LOADADDR(.vsyscall_0); __vsyscall_0 = LOADADDR(.vsyscall_0);
. = ALIGN(64); . = ALIGN(CONFIG_X86_L1_CACHE_BYTES);
.xtime_lock : AT ((LOADADDR(.vsyscall_0) + SIZEOF(.vsyscall_0) + 63) & ~(63)) { *(.xtime_lock) } .xtime_lock : AT CACHE_ALIGN(AFTER(.vsyscall_0)) { *(.xtime_lock) }
xtime_lock = LOADADDR(.xtime_lock); xtime_lock = LOADADDR(.xtime_lock);
. = ALIGN(16); .vxtime : AT AFTER(.xtime_lock) { *(.vxtime) }
.vxtime : AT ((LOADADDR(.xtime_lock) + SIZEOF(.xtime_lock) + 15) & ~(15)) { *(.vxtime) }
vxtime = LOADADDR(.vxtime); vxtime = LOADADDR(.vxtime);
. = ALIGN(16); .wall_jiffies : AT AFTER(.vxtime) { *(.wall_jiffies) }
.wall_jiffies : AT ((LOADADDR(.vxtime) + SIZEOF(.vxtime) + 15) & ~(15)) { *(.wall_jiffies) }
wall_jiffies = LOADADDR(.wall_jiffies); wall_jiffies = LOADADDR(.wall_jiffies);
. = ALIGN(16); .sys_tz : AT AFTER(.wall_jiffies) { *(.sys_tz) }
.sys_tz : AT ((LOADADDR(.wall_jiffies) + SIZEOF(.wall_jiffies) + 15) & ~(15)) { *(.sys_tz) }
sys_tz = LOADADDR(.sys_tz); sys_tz = LOADADDR(.sys_tz);
. = ALIGN(16); .sysctl_vsyscall : AT AFTER(.sys_tz) { *(.sysctl_vsyscall) }
.sysctl_vsyscall : AT ((LOADADDR(.sys_tz) + SIZEOF(.sys_tz) + 15) & ~(15)) { *(.sysctl_vsyscall) }
sysctl_vsyscall = LOADADDR(.sysctl_vsyscall); sysctl_vsyscall = LOADADDR(.sysctl_vsyscall);
. = ALIGN(16); .xtime : AT AFTER(.sysctl_vsyscall) { *(.xtime) }
.jiffies : AT ((LOADADDR(.sysctl_vsyscall) + SIZEOF(.sysctl_vsyscall) + 15) & ~(15)) { *(.jiffies) }
jiffies = LOADADDR(.jiffies);
. = ALIGN(16);
.xtime : AT ((LOADADDR(.jiffies) + SIZEOF(.jiffies) + 15) & ~(15)) { *(.xtime) }
xtime = LOADADDR(.xtime); xtime = LOADADDR(.xtime);
. = ALIGN(CONFIG_X86_L1_CACHE_BYTES);
.jiffies : AT CACHE_ALIGN(AFTER(.xtime)) { *(.jiffies) }
jiffies = LOADADDR(.jiffies);
.vsyscall_1 ADDR(.vsyscall_0) + 1024: AT (LOADADDR(.vsyscall_0) + 1024) { *(.vsyscall_1) } .vsyscall_1 ADDR(.vsyscall_0) + 1024: AT (LOADADDR(.vsyscall_0) + 1024) { *(.vsyscall_1) }
. = LOADADDR(.vsyscall_0) + 4096; . = LOADADDR(.vsyscall_0) + 4096;
......
...@@ -166,14 +166,12 @@ static void __init map_vsyscall(void) ...@@ -166,14 +166,12 @@ static void __init map_vsyscall(void)
static int __init vsyscall_init(void) static int __init vsyscall_init(void)
{ {
if ((unsigned long) &vgettimeofday != VSYSCALL_ADDR(__NR_vgettimeofday)) BUG_ON(((unsigned long) &vgettimeofday !=
panic("vgettimeofday link addr broken"); VSYSCALL_ADDR(__NR_vgettimeofday)));
if ((unsigned long) &vtime != VSYSCALL_ADDR(__NR_vtime)) BUG_ON((unsigned long) &vtime != VSYSCALL_ADDR(__NR_vtime));
panic("vtime link addr broken"); BUG_ON((VSYSCALL_ADDR(0) != __fix_to_virt(VSYSCALL_FIRST_PAGE)));
if (VSYSCALL_ADDR(0) != __fix_to_virt(VSYSCALL_FIRST_PAGE))
panic("fixmap first vsyscall %lx should be %lx", __fix_to_virt(VSYSCALL_FIRST_PAGE),
VSYSCALL_ADDR(0));
map_vsyscall(); map_vsyscall();
sysctl_vsyscall = 1;
return 0; return 0;
} }
......
...@@ -21,7 +21,7 @@ enum vsyscall_num { ...@@ -21,7 +21,7 @@ enum vsyscall_num {
#define __section_sys_tz __attribute__ ((unused, __section__ (".sys_tz"), aligned(16))) #define __section_sys_tz __attribute__ ((unused, __section__ (".sys_tz"), aligned(16)))
#define __section_sysctl_vsyscall __attribute__ ((unused, __section__ (".sysctl_vsyscall"), aligned(16))) #define __section_sysctl_vsyscall __attribute__ ((unused, __section__ (".sysctl_vsyscall"), aligned(16)))
#define __section_xtime __attribute__ ((unused, __section__ (".xtime"), aligned(16))) #define __section_xtime __attribute__ ((unused, __section__ (".xtime"), aligned(16)))
#define __section_xtime_lock __attribute__ ((unused, __section__ (".xtime_lock"), aligned(L1_CACHE_BYTES))) #define __section_xtime_lock __attribute__ ((unused, __section__ (".xtime_lock"), aligned(16)))
#define VXTIME_TSC 1 #define VXTIME_TSC 1
#define VXTIME_HPET 2 #define VXTIME_HPET 2
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment