1f7eb23eeSEd Maste#- 2f7eb23eeSEd Maste# Copyright (c) 2007 Yahoo!, Inc. 3f7eb23eeSEd Maste# All rights reserved. 4f7eb23eeSEd Maste# Written by: John Baldwin <jhb@FreeBSD.org> 5f7eb23eeSEd Maste# 6f7eb23eeSEd Maste# Redistribution and use in source and binary forms, with or without 7f7eb23eeSEd Maste# modification, are permitted provided that the following conditions 8f7eb23eeSEd Maste# are met: 9f7eb23eeSEd Maste# 1. Redistributions of source code must retain the above copyright 10f7eb23eeSEd Maste# notice, this list of conditions and the following disclaimer. 11f7eb23eeSEd Maste# 2. Redistributions in binary form must reproduce the above copyright 12f7eb23eeSEd Maste# notice, this list of conditions and the following disclaimer in the 13f7eb23eeSEd Maste# documentation and/or other materials provided with the distribution. 14f7eb23eeSEd Maste# 3. Neither the name of the author nor the names of any co-contributors 15f7eb23eeSEd Maste# may be used to endorse or promote products derived from this software 16f7eb23eeSEd Maste# without specific prior written permission. 17f7eb23eeSEd Maste# 18f7eb23eeSEd Maste# THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 19f7eb23eeSEd Maste# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 20f7eb23eeSEd Maste# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 21f7eb23eeSEd Maste# ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 22f7eb23eeSEd Maste# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 23f7eb23eeSEd Maste# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 24f7eb23eeSEd Maste# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 25f7eb23eeSEd Maste# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 26f7eb23eeSEd Maste# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 27f7eb23eeSEd Maste# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 28f7eb23eeSEd Maste# SUCH DAMAGE. 29f7eb23eeSEd Maste# 30f7eb23eeSEd Maste# 31f7eb23eeSEd Maste# Partly from: src/sys/boot/i386/mbr/mbr.s 1.7 32f7eb23eeSEd Maste 33f7eb23eeSEd Maste# A 512 byte PMBR boot manager that looks for a FreeBSD boot GPT partition 34f7eb23eeSEd Maste# and boots it. 35f7eb23eeSEd Maste 36f7eb23eeSEd Maste .set LOAD,0x7c00 # Load address 37f7eb23eeSEd Maste .set EXEC,0x600 # Execution address 38f7eb23eeSEd Maste .set MAGIC,0xaa55 # Magic: bootable 39f7eb23eeSEd Maste .set SECSIZE,0x200 # Size of a single disk sector 40f7eb23eeSEd Maste .set DISKSIG,440 # Disk signature offset 41f7eb23eeSEd Maste .set STACK,EXEC+SECSIZE*4 # Stack address 42f7eb23eeSEd Maste .set GPT_ADDR,STACK # GPT header address 43f7eb23eeSEd Maste .set GPT_SIG,0 44f7eb23eeSEd Maste .set GPT_SIG_0,0x20494645 # "EFI " 45f7eb23eeSEd Maste .set GPT_SIG_1,0x54524150 # "PART" 46f7eb23eeSEd Maste .set GPT_MYLBA,24 47f7eb23eeSEd Maste .set GPT_PART_LBA,72 48f7eb23eeSEd Maste .set GPT_NPART,80 49f7eb23eeSEd Maste .set GPT_PART_SIZE,84 50f7eb23eeSEd Maste .set PART_ADDR,GPT_ADDR+SECSIZE # GPT partition array address 51f7eb23eeSEd Maste .set PART_TYPE,0 52f7eb23eeSEd Maste .set PART_START_LBA,32 53f7eb23eeSEd Maste .set PART_END_LBA,40 54f7eb23eeSEd Maste .set DPBUF,PART_ADDR+SECSIZE 55f7eb23eeSEd Maste .set DPBUF_SEC,0x10 # Number of sectors 56f7eb23eeSEd Maste 57f7eb23eeSEd Maste .set NHRDRV,0x475 # Number of hard drives 58f7eb23eeSEd Maste 59f7eb23eeSEd Maste .globl start # Entry point 60f7eb23eeSEd Maste .code16 61f7eb23eeSEd Maste 62f7eb23eeSEd Maste# 63f7eb23eeSEd Maste# Setup the segment registers for flat addressing and setup the stack. 64f7eb23eeSEd Maste# 65f7eb23eeSEd Mastestart: cld # String ops inc 66f7eb23eeSEd Maste xorw %ax,%ax # Zero 67f7eb23eeSEd Maste movw %ax,%es # Address 68f7eb23eeSEd Maste movw %ax,%ds # data 69f7eb23eeSEd Maste movw %ax,%ss # Set up 70f7eb23eeSEd Maste movw $STACK,%sp # stack 71f7eb23eeSEd Maste# 72f7eb23eeSEd Maste# Relocate ourself to a lower address so that we have more room to load 73f7eb23eeSEd Maste# other sectors. 74f7eb23eeSEd Maste# 75f7eb23eeSEd Maste movw $main-EXEC+LOAD,%si # Source 76f7eb23eeSEd Maste movw $main,%di # Destination 77f7eb23eeSEd Maste movw $SECSIZE-(main-start),%cx # Byte count 78f7eb23eeSEd Maste rep # Relocate 79f7eb23eeSEd Maste movsb # code 80f7eb23eeSEd Maste# 81f7eb23eeSEd Maste# Jump to the relocated code. 82f7eb23eeSEd Maste# 83f7eb23eeSEd Maste jmp main-LOAD+EXEC # To relocated code 84f7eb23eeSEd Maste# 85f7eb23eeSEd Maste# Validate drive number in %dl. 86f7eb23eeSEd Maste# 87f7eb23eeSEd Mastemain: cmpb $0x80,%dl # Drive valid? 88f7eb23eeSEd Maste jb main.1 # No 89f7eb23eeSEd Maste movb NHRDRV,%dh # Calculate the highest 90f7eb23eeSEd Maste addb $0x80,%dh # drive number available 91f7eb23eeSEd Maste cmpb %dh,%dl # Within range? 92f7eb23eeSEd Maste jb main.2 # Yes 93f7eb23eeSEd Mastemain.1: movb $0x80,%dl # Assume drive 0x80 94f7eb23eeSEd Maste# 95f7eb23eeSEd Maste# Load the GPT header and verify signature. Try LBA 1 for the primary one and 96f7eb23eeSEd Maste# the last LBA for the backup if it is broken. 97f7eb23eeSEd Maste# 98f7eb23eeSEd Mastemain.2: call getdrvparams # Read drive parameters 99f7eb23eeSEd Maste movb $1,%dh # %dh := 1 (reading primary) 100f7eb23eeSEd Mastemain.2a: movw $GPT_ADDR,%bx 101f7eb23eeSEd Maste movw $lba,%si 102f7eb23eeSEd Maste call read # Read header and check GPT sig 103f7eb23eeSEd Maste cmpl $GPT_SIG_0,GPT_ADDR+GPT_SIG 104f7eb23eeSEd Maste jnz main.2b 105f7eb23eeSEd Maste cmpl $GPT_SIG_1,GPT_ADDR+GPT_SIG+4 106f7eb23eeSEd Maste jnz main.2b 107f7eb23eeSEd Maste jmp load_part 108f7eb23eeSEd Mastemain.2b: cmpb $1,%dh # Reading primary? 109f7eb23eeSEd Maste jne err_pt # If no - invalid table found 110f7eb23eeSEd Maste# 111f7eb23eeSEd Maste# Try alternative LBAs from the last sector for the GPT header. 112f7eb23eeSEd Maste# 113f7eb23eeSEd Mastemain.3: movb $0,%dh # %dh := 0 (reading backup) 114f7eb23eeSEd Maste movw $DPBUF+DPBUF_SEC,%si # %si = last sector + 1 115f7eb23eeSEd Maste movw $lba,%di # %di = $lba 1160ca9f1d4SEmrionmain.3a: subl $1, (%si) # 0x0(%si) = last sec (0-31) 1170ca9f1d4SEmrion sbbl $0, 4(%si) 1180ca9f1d4SEmrion movw $4,%cx 119f7eb23eeSEd Maste rep 120f7eb23eeSEd Maste movsw # $lastsec--, copy it to $lba 121f7eb23eeSEd Maste jmp main.2a # Read the next sector 122f7eb23eeSEd Maste# 123f7eb23eeSEd Maste# Load a partition table sector from disk and look for a FreeBSD boot 124f7eb23eeSEd Maste# partition. 125f7eb23eeSEd Maste# 126f7eb23eeSEd Masteload_part: movw $GPT_ADDR+GPT_PART_LBA,%si 127f7eb23eeSEd Maste movw $PART_ADDR,%bx 128f7eb23eeSEd Maste call read 129f7eb23eeSEd Mastescan: movw %bx,%si # Compare partition UUID 130f7eb23eeSEd Maste movw $boot_uuid,%di # with FreeBSD boot UUID 1310ca9f1d4SEmrion movw $0x10,%cx 132f7eb23eeSEd Maste repe cmpsb 133f7eb23eeSEd Maste jnz next_part # Didn't match, next partition 134f7eb23eeSEd Maste# 135f7eb23eeSEd Maste# We found a boot partition. Load it into RAM starting at 0x7c00. 136f7eb23eeSEd Maste# 137f7eb23eeSEd Maste movw %bx,%di # Save partition pointer in %di 138f7eb23eeSEd Maste leaw PART_START_LBA(%di),%si 139f7eb23eeSEd Maste movw $LOAD/16,%bx 140f7eb23eeSEd Maste movw %bx,%es 141f7eb23eeSEd Maste xorw %bx,%bx 142f7eb23eeSEd Masteload_boot: push %si # Save %si 143f7eb23eeSEd Maste call read 144f7eb23eeSEd Maste pop %si # Restore 145f7eb23eeSEd Maste movl PART_END_LBA(%di),%eax # See if this was the last LBA 146f7eb23eeSEd Maste cmpl (%si),%eax 147f7eb23eeSEd Maste jnz next_boot 148f7eb23eeSEd Maste movl PART_END_LBA+4(%di),%eax 149f7eb23eeSEd Maste cmpl 4(%si),%eax 150f7eb23eeSEd Maste jnz next_boot 151f7eb23eeSEd Maste mov %bx,%es # Reset %es to zero 152f7eb23eeSEd Maste jmp LOAD # Jump to boot code 1530ca9f1d4SEmrionnext_boot: addl $1,(%si) # Next LBA 154f7eb23eeSEd Maste adcl $0,4(%si) 155f7eb23eeSEd Maste mov %es,%ax # Adjust segment for next 156f7eb23eeSEd Maste addw $SECSIZE/16,%ax # sector 157f7eb23eeSEd Maste cmp $0x9000,%ax # Don't load past 0x90000, 158*c596126aSWarner Losh jb sz_ok # 545k should be enough for 159*c596126aSWarner Losh call err_big # any boot code, but warn 160*c596126aSWarner Losh mov $0x9000-SECSIZE/16,%ax # and truncate 161*c596126aSWarner Loshsz_ok: mov %ax,%es 162f7eb23eeSEd Maste jmp load_boot 163f7eb23eeSEd Maste# 164f7eb23eeSEd Maste# Move to the next partition. If we walk off the end of the sector, load 165f7eb23eeSEd Maste# the next sector. We assume that partition entries are smaller than 64k 166f7eb23eeSEd Maste# and that they won't span a sector boundary. 167f7eb23eeSEd Maste# 168f7eb23eeSEd Maste# XXX: Should we int 0x18 instead of err_noboot if we hit the end of the table? 169f7eb23eeSEd Maste# 170f7eb23eeSEd Mastenext_part: decl GPT_ADDR+GPT_NPART # Was this the last partition? 171f7eb23eeSEd Maste jz err_noboot 172f7eb23eeSEd Maste movw GPT_ADDR+GPT_PART_SIZE,%ax 173f7eb23eeSEd Maste addw %ax,%bx # Next partition 174f7eb23eeSEd Maste cmpw $PART_ADDR+0x200,%bx # Still in sector? 175f7eb23eeSEd Maste jb scan 1760ca9f1d4SEmrion addl $1, GPT_ADDR+GPT_PART_LBA # Next sector 177f7eb23eeSEd Maste adcl $0,GPT_ADDR+GPT_PART_LBA+4 178f7eb23eeSEd Maste jmp load_part 179f7eb23eeSEd Maste# 180f7eb23eeSEd Maste# Load a sector (64-bit LBA at %si) from disk %dl into %es:%bx by creating 181f7eb23eeSEd Maste# a EDD packet on the stack and passing it to the BIOS. Trashes %ax and %si. 182f7eb23eeSEd Maste# 183f7eb23eeSEd Masteread: pushl 0x4(%si) # Set the LBA 184f7eb23eeSEd Maste pushl 0x0(%si) # address 185f7eb23eeSEd Maste pushw %es # Set the address of 186f7eb23eeSEd Maste pushw %bx # the transfer buffer 187f7eb23eeSEd Maste pushw $0x1 # Read 1 sector 188f7eb23eeSEd Maste pushw $0x10 # Packet length 189f7eb23eeSEd Maste movw %sp,%si # Packer pointer 190f7eb23eeSEd Maste movw $0x4200,%ax # BIOS: LBA Read from disk 191f7eb23eeSEd Maste int $0x13 # Call the BIOS 192f7eb23eeSEd Maste add $0x10,%sp # Restore stack 193f7eb23eeSEd Maste jc err_rd # If error 194f7eb23eeSEd Maste ret 195f7eb23eeSEd Maste# 196f7eb23eeSEd Maste# Check the number of LBAs on the drive index %dx. Trashes %ax and %si. 197f7eb23eeSEd Maste# 198f7eb23eeSEd Mastegetdrvparams: 199f7eb23eeSEd Maste movw $DPBUF,%si # Set the address of result buf 200f7eb23eeSEd Maste movw $0x001e,(%si) # len 201f7eb23eeSEd Maste movw $0x4800,%ax # BIOS: Read Drive Parameters 202f7eb23eeSEd Maste int $0x13 # Call the BIOS 203f7eb23eeSEd Maste jc err_rd # "I/O error" if error 204f7eb23eeSEd Maste ret 205f7eb23eeSEd Maste# 206f7eb23eeSEd Maste# Various error message entry points. 207f7eb23eeSEd Maste# 208*c596126aSWarner Losherr_big: movw $msg_big,%si # "Truncated 209*c596126aSWarner Losh call putstr # to 545k" 210*c596126aSWarner Losh ret 211f7eb23eeSEd Maste 212f7eb23eeSEd Masteerr_pt: movw $msg_pt,%si # "Invalid partition 213*c596126aSWarner Losh call putstr # table" 214*c596126aSWarner Losherr_pt.1: jmp err_pt.1 # Await reset 215f7eb23eeSEd Maste 216f7eb23eeSEd Masteerr_rd: movw $msg_rd,%si # "I/O error loading 217*c596126aSWarner Losh call putstr # boot loader" 218*c596126aSWarner Losh jmp err_pt.1 219f7eb23eeSEd Maste 220f7eb23eeSEd Masteerr_noboot: movw $msg_noboot,%si # "Missing boot 221*c596126aSWarner Losh call putstr # loader" 222*c596126aSWarner Losh jmp err_pt.1 223f7eb23eeSEd Maste# 224f7eb23eeSEd Maste# Output an ASCIZ string to the console via the BIOS. 225f7eb23eeSEd Maste# 226f7eb23eeSEd Masteputstr.0: movw $0x7,%bx # Page:attribute 227f7eb23eeSEd Maste movb $0xe,%ah # BIOS: Display 228f7eb23eeSEd Maste int $0x10 # character 229f7eb23eeSEd Masteputstr: lodsb # Get character 230f7eb23eeSEd Maste testb %al,%al # End of string? 231f7eb23eeSEd Maste jnz putstr.0 # No 232*c596126aSWarner Losh ret 233f7eb23eeSEd Maste 234*c596126aSWarner Loshmsg_big: .asciz "Loaded only 545k" 235f7eb23eeSEd Mastemsg_pt: .asciz "Invalid partition table" 236f7eb23eeSEd Mastemsg_rd: .asciz "I/O error loading boot loader" 237f7eb23eeSEd Mastemsg_noboot: .asciz "Missing boot loader" 238f7eb23eeSEd Maste 239f7eb23eeSEd Mastelba: .quad 1 # LBA of GPT header 240f7eb23eeSEd Maste 241f7eb23eeSEd Masteboot_uuid: .long 0x83bd6b9d 242f7eb23eeSEd Maste .word 0x7f41 243f7eb23eeSEd Maste .word 0x11dc 244f7eb23eeSEd Maste .byte 0xbe 245f7eb23eeSEd Maste .byte 0x0b 246f7eb23eeSEd Maste .byte 0x00 247f7eb23eeSEd Maste .byte 0x15 248f7eb23eeSEd Maste .byte 0x60 249f7eb23eeSEd Maste .byte 0xb8 250f7eb23eeSEd Maste .byte 0x4f 251f7eb23eeSEd Maste .byte 0x0f 252f7eb23eeSEd Maste 253f7eb23eeSEd Maste .org DISKSIG,0x90 254f7eb23eeSEd Mastesig: .long 0 # OS Disk Signature 255f7eb23eeSEd Maste .word 0 # "Unknown" in PMBR 256f7eb23eeSEd Maste 257f7eb23eeSEd Mastepartbl: .fill 0x10,0x4,0x0 # Partition table 258f7eb23eeSEd Maste .word MAGIC # Magic number 259