[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

[Xen-changelog] [qemu-upstream-unstable] linuxboot: fix loading old kernels



commit 51d703ff2e03285be7ba0433d814a688c923b1ea
Author:     Paolo Bonzini <pbonzini@xxxxxxxxxx>
AuthorDate: Thu Dec 11 02:17:03 2014 +0100
Commit:     Michael Roth <mdroth@xxxxxxxxxxxxxxxxxx>
CommitDate: Sun Feb 22 12:27:19 2015 -0600

    linuxboot: fix loading old kernels
    
    Old kernels that used high memory only allowed the initrd to be in the
    first 896MB of memory.  If you load the initrd above, they complain
    that "initrd extends beyond end of memory".
    
    In order to fix this, while not breaking machines with small amounts
    of memory fixed by cdebec5 (linuxboot: compute initrd loading address,
    2014-10-06), we need to distinguish two cases.  If pc.c placed the
    initrd at end of memory, use the new algorithm based on the e801
    memory map.  If instead pc.c placed the initrd at the maximum address
    specified by the bzImage, leave it there.
    
    The only interesting part is that the low-memory info block is now
    loaded very early, in real mode, and thus the 32-bit address has
    to be converted into a real mode segment.  The initrd address is
    also patched in the info block before entering real mode, it is
    simpler that way.
    
    This fixes booting the RHEL4.8 32-bit installation image with 1GB
    of RAM.
    
    Cc: qemu-stable@xxxxxxxxxx
    Cc: mst@xxxxxxxxxx
    Cc: jsnow@xxxxxxxxxx
    Signed-off-by: Paolo Bonzini <pbonzini@xxxxxxxxxx>
    (cherry picked from commit 269e2358492b674c50160553d037702e916b9f1b)
    Signed-off-by: Michael Roth <mdroth@xxxxxxxxxxxxxxxxxx>
---
 pc-bios/linuxboot.bin         |  Bin 1024 -> 1024 bytes
 pc-bios/optionrom/linuxboot.S |   37 +++++++++++++++++++++++++++----------
 2 files changed, 27 insertions(+), 10 deletions(-)

diff --git a/pc-bios/linuxboot.bin b/pc-bios/linuxboot.bin
index 130103f..923d179 100644
Binary files a/pc-bios/linuxboot.bin and b/pc-bios/linuxboot.bin differ
diff --git a/pc-bios/optionrom/linuxboot.S b/pc-bios/optionrom/linuxboot.S
index 5bc0af0..ba821ab 100644
--- a/pc-bios/optionrom/linuxboot.S
+++ b/pc-bios/optionrom/linuxboot.S
@@ -76,7 +76,31 @@ boot_kernel:
 
 
 copy_kernel:
-       /* Compute initrd address */
+       /* Read info block in low memory (0x10000 or 0x90000) */
+       read_fw         FW_CFG_SETUP_ADDR
+       shr             $4, %eax
+       mov             %eax, %es
+       xor             %edi, %edi
+       read_fw_blob_addr32_edi(FW_CFG_SETUP)
+
+       cmpw            $0x203, %es:0x206      // if protocol >= 0x203
+       jae             1f                     // have initrd_max
+       movl            $0x37ffffff, %es:0x22c // else assume 0x37ffffff
+1:
+
+       /* Check if using kernel-specified initrd address */
+       read_fw         FW_CFG_INITRD_ADDR
+       mov             %eax, %edi             // (load_kernel wants it in %edi)
+       read_fw         FW_CFG_INITRD_SIZE     // find end of initrd
+       add             %edi, %eax
+       xor             %es:0x22c, %eax        // if it matches es:0x22c
+       and             $-4096, %eax           // (apart from padding for page)
+       jz              load_kernel            // then initrd is not at top
+                                              // of memory
+
+       /* pc.c placed the initrd at end of memory.  Compute a better
+        * initrd address based on e801 data.
+        */
        mov             $0xe801, %ax
        xor             %cx, %cx
        xor             %dx, %dx
@@ -107,7 +131,9 @@ copy_kernel:
        read_fw         FW_CFG_INITRD_SIZE
        subl            %eax, %edi
        andl            $-4096, %edi          /* EDI = start of initrd */
+       movl            %edi, %es:0x218       /* put it in the header */
 
+load_kernel:
        /* We need to load the kernel into memory we can't access in 16 bit
           mode, so let's get into 32 bit mode, write the kernel and jump
           back again. */
@@ -139,19 +165,10 @@ copy_kernel:
        /* We're now running in 16-bit CS, but 32-bit ES! */
 
        /* Load kernel and initrd */
-       pushl           %edi
        read_fw_blob_addr32_edi(FW_CFG_INITRD)
        read_fw_blob_addr32(FW_CFG_KERNEL)
        read_fw_blob_addr32(FW_CFG_CMDLINE)
 
-       read_fw         FW_CFG_SETUP_ADDR
-       mov             %eax, %edi
-       mov             %eax, %ebx
-       read_fw_blob_addr32_edi(FW_CFG_SETUP)
-
-       /* Update the header with the initrd address we chose above */
-       popl            %es:0x218(%ebx)
-
        /* And now jump into Linux! */
        mov             $0, %eax
        mov             %eax, %cr0
--
generated by git-patchbot for /home/xen/git/qemu-upstream-unstable.git

_______________________________________________
Xen-changelog mailing list
Xen-changelog@xxxxxxxxxxxxx
http://lists.xensource.com/xen-changelog


 


Rackspace

Lists.xenproject.org is hosted with RackSpace, monitoring our
servers 24x7x365 and backed by RackSpace's Fanatical Support®.