1 /* Copyright 2002,2003 Andi Kleen, SuSE Labs */
3 /* vsyscall handling for 32bit processes. Map a stub page into it
4 on demand because 32bit cannot reach the kernel's fixmaps */
7 #include <linux/string.h>
8 #include <linux/kernel.h>
10 #include <linux/init.h>
11 #include <linux/stringify.h>
12 #include <linux/security.h>
13 #include <asm/proto.h>
14 #include <asm/tlbflush.h>
15 #include <asm/ia32_unistd.h>
17 extern unsigned char syscall32_syscall
[], syscall32_syscall_end
[];
18 extern unsigned char syscall32_sysenter
[], syscall32_sysenter_end
[];
19 extern int sysctl_vsyscall32
;
22 static int use_sysenter
= -1;
25 syscall32_nopage(struct vm_area_struct
*vma
, unsigned long adr
, int *type
)
27 struct page
*p
= virt_to_page(adr
- vma
->vm_start
+ syscall32_page
);
32 /* Prevent VMA merging */
33 static void syscall32_vma_close(struct vm_area_struct
*vma
)
37 static struct vm_operations_struct syscall32_vm_ops
= {
38 .close
= syscall32_vma_close
,
39 .nopage
= syscall32_nopage
,
44 /* Setup a VMA at program startup for the vsyscall page */
45 int syscall32_setup_pages(struct linux_binprm
*bprm
, int exstack
)
47 int npages
= (VSYSCALL32_END
- VSYSCALL32_BASE
) >> PAGE_SHIFT
;
48 struct vm_area_struct
*vma
;
49 struct mm_struct
*mm
= current
->mm
;
52 vma
= kmem_cache_alloc(vm_area_cachep
, SLAB_KERNEL
);
55 if (security_vm_enough_memory(npages
)) {
56 kmem_cache_free(vm_area_cachep
, vma
);
60 memset(vma
, 0, sizeof(struct vm_area_struct
));
61 /* Could randomize here */
62 vma
->vm_start
= VSYSCALL32_BASE
;
63 vma
->vm_end
= VSYSCALL32_END
;
64 /* MAYWRITE to allow gdb to COW and set breakpoints */
65 vma
->vm_flags
= VM_READ
|VM_EXEC
|VM_MAYREAD
|VM_MAYEXEC
|VM_MAYEXEC
|VM_MAYWRITE
;
66 vma
->vm_flags
|= mm
->def_flags
;
67 vma
->vm_page_prot
= protection_map
[vma
->vm_flags
& 7];
68 vma
->vm_ops
= &syscall32_vm_ops
;
71 down_write(&mm
->mmap_sem
);
72 if ((ret
= insert_vm_struct(mm
, vma
))) {
73 up_write(&mm
->mmap_sem
);
74 kmem_cache_free(vm_area_cachep
, vma
);
77 mm
->total_vm
+= npages
;
78 up_write(&mm
->mmap_sem
);
82 static int __init
init_syscall32(void)
84 syscall32_page
= (void *)get_zeroed_page(GFP_KERNEL
);
86 panic("Cannot allocate syscall32 page");
87 if (use_sysenter
> 0) {
88 memcpy(syscall32_page
, syscall32_sysenter
,
89 syscall32_sysenter_end
- syscall32_sysenter
);
91 memcpy(syscall32_page
, syscall32_syscall
,
92 syscall32_syscall_end
- syscall32_syscall
);
97 __initcall(init_syscall32
);
99 /* May not be __init: called during resume */
100 void syscall32_cpu_init(void)
102 if (use_sysenter
< 0)
103 use_sysenter
= (boot_cpu_data
.x86_vendor
== X86_VENDOR_INTEL
);
105 /* Load these always in case some future AMD CPU supports
106 SYSENTER from compat mode too. */
107 checking_wrmsrl(MSR_IA32_SYSENTER_CS
, (u64
)__KERNEL_CS
);
108 checking_wrmsrl(MSR_IA32_SYSENTER_ESP
, 0ULL);
109 checking_wrmsrl(MSR_IA32_SYSENTER_EIP
, (u64
)ia32_sysenter_target
);
111 wrmsrl(MSR_CSTAR
, ia32_cstar_target
);