please check the patch.
YH
[PATCH]x86_64: build and use GDT on copied compressed kernel
Build and use GDT on copied compressed kernel postion, instead of using GDT
in data segment of loaded compressed kernel. Otherwise decompressing
compressed kernel image later in 64bit longmode, will overwrite GDT.
Signed-off-by: Yinghai Lu <[email protected]>
diff --git a/arch/x86_64/boot/compressed/head.S b/arch/x86_64/boot/compressed/head.S
index f9d5692..1f5ac7c 100644
--- a/arch/x86_64/boot/compressed/head.S
+++ b/arch/x86_64/boot/compressed/head.S
@@ -94,10 +94,23 @@ startup_32:
* Prepare for entering 64 bit mode
*/
+/*
+ * Build early GDT table on copied compressed kernel image position, So GDT
+ * is not be overwriten when decompressing the compressed kernel.
+ */
+ movw $0x0020, gdt+0x00(%ebx)
+ movw $0x0000, gdt+0x06(%ebx)
+ movl $0x00000000, gdt+0x08(%ebx) /* NULL descriptor */
+ movl $0x00000000, gdt+0x0c(%ebx)
+ movl $0x0000ffff, gdt+0x10(%ebx) /* __KERNEL_CS */
+ movl $0x00af9a00, gdt+0x14(%ebx)
+ movl $0x0000ffff, gdt+0x18(%ebx) /* __KERNEL_DS */
+ movl $0x00cf9200, gdt+0x1c(%ebx)
+
/* Load new GDT with the 64bit segments using 32bit descriptor */
- leal gdt(%ebp), %eax
- movl %eax, gdt+2(%ebp)
- lgdt gdt(%ebp)
+ leal gdt(%ebx), %eax
+ movl %eax, gdt+2(%ebx)
+ lgdt gdt(%ebx)
/* Enable PAE mode */
xorl %eax, %eax
@@ -182,7 +195,7 @@ no_longmode:
* it may change in the future.
*/
.code64
- .org 0x200
+ .org 0x400
ENTRY(startup_64)
/* We come here either from startup_32 or directly from a
* 64bit bootloader. If we come here from a bootloader we depend on
@@ -287,15 +300,6 @@ relocated:
*/
jmp *%rbp
- .data
-gdt:
- .word gdt_end - gdt
- .long gdt
- .word 0
- .quad 0x0000000000000000 /* NULL descriptor */
- .quad 0x00af9a000000ffff /* __KERNEL_CS */
- .quad 0x00cf92000000ffff /* __KERNEL_DS */
-gdt_end:
.bss
/* Stack for uncompression */
.balign 4
diff --git a/arch/x86_64/boot/compressed/vmlinux.lds b/arch/x86_64/boot/compressed/vmlinux.lds
index 94c13e5..6e041ef 100644
--- a/arch/x86_64/boot/compressed/vmlinux.lds
+++ b/arch/x86_64/boot/compressed/vmlinux.lds
@@ -36,6 +36,8 @@ SECTIONS
*(COMMON)
. = ALIGN(8);
_end = . ;
+ gdt = . ;
+ . = . + 8 * 4;
. = ALIGN(4096);
pgtable = . ;
. = . + 4096 * 6;
[Index of Archives]
[Kernel Newbies]
[Netfilter]
[Bugtraq]
[Photo]
[Stuff]
[Gimp]
[Yosemite News]
[MIPS Linux]
[ARM Linux]
[Linux Security]
[Linux RAID]
[Video 4 Linux]
[Linux for the blind]
[Linux Resources]