1*f7eb23eeSEd Maste# 2*f7eb23eeSEd Maste# Copyright (c) 1999 Robert Nordier 3*f7eb23eeSEd Maste# All rights reserved. 4*f7eb23eeSEd Maste# 5*f7eb23eeSEd Maste# Redistribution and use in source and binary forms are freely 6*f7eb23eeSEd Maste# permitted provided that the above copyright notice and this 7*f7eb23eeSEd Maste# paragraph and the following disclaimer are duplicated in all 8*f7eb23eeSEd Maste# such forms. 9*f7eb23eeSEd Maste# 10*f7eb23eeSEd Maste# This software is provided "AS IS" and without any express or 11*f7eb23eeSEd Maste# implied warranties, including, without limitation, the implied 12*f7eb23eeSEd Maste# warranties of merchantability and fitness for a particular 13*f7eb23eeSEd Maste# purpose. 14*f7eb23eeSEd Maste# 15*f7eb23eeSEd Maste 16*f7eb23eeSEd Maste# A 512 byte MBR boot manager that simply boots the active partition. 17*f7eb23eeSEd Maste 18*f7eb23eeSEd Maste .set LOAD,0x7c00 # Load address 19*f7eb23eeSEd Maste .set EXEC,0x600 # Execution address 20*f7eb23eeSEd Maste .set PT_OFF,0x1be # Partition table 21*f7eb23eeSEd Maste .set MAGIC,0xaa55 # Magic: bootable 22*f7eb23eeSEd Maste .set FL_PACKET,0x80 # Flag: try EDD 23*f7eb23eeSEd Maste 24*f7eb23eeSEd Maste .set NHRDRV,0x475 # Number of hard drives 25*f7eb23eeSEd Maste 26*f7eb23eeSEd Maste .globl start # Entry point 27*f7eb23eeSEd Maste .code16 28*f7eb23eeSEd Maste 29*f7eb23eeSEd Maste# 30*f7eb23eeSEd Maste# Setup the segment registers for flat addressing and setup the stack. 31*f7eb23eeSEd Maste# 32*f7eb23eeSEd Mastestart: cld # String ops inc 33*f7eb23eeSEd Maste xorw %ax,%ax # Zero 34*f7eb23eeSEd Maste movw %ax,%es # Address 35*f7eb23eeSEd Maste movw %ax,%ds # data 36*f7eb23eeSEd Maste movw %ax,%ss # Set up 37*f7eb23eeSEd Maste movw $LOAD,%sp # stack 38*f7eb23eeSEd Maste# 39*f7eb23eeSEd Maste# Relocate ourself to a lower address so that we are out of the way when 40*f7eb23eeSEd Maste# we load in the bootstrap from the partition to boot. 41*f7eb23eeSEd Maste# 42*f7eb23eeSEd Maste movw $main-EXEC+LOAD,%si # Source 43*f7eb23eeSEd Maste movw $main,%di # Destination 44*f7eb23eeSEd Maste movw $0x200-(main-start),%cx # Byte count 45*f7eb23eeSEd Maste rep # Relocate 46*f7eb23eeSEd Maste movsb # code 47*f7eb23eeSEd Maste# 48*f7eb23eeSEd Maste# Jump to the relocated code. 49*f7eb23eeSEd Maste# 50*f7eb23eeSEd Maste jmp main-LOAD+EXEC # To relocated code 51*f7eb23eeSEd Maste# 52*f7eb23eeSEd Maste# Scan the partition table looking for an active entry. Note that %ch is 53*f7eb23eeSEd Maste# zero from the repeated string instruction above. We save the offset of 54*f7eb23eeSEd Maste# the active partition in %si and scan the entire table to ensure that only 55*f7eb23eeSEd Maste# one partition is marked active. 56*f7eb23eeSEd Maste# 57*f7eb23eeSEd Mastemain: xorw %si,%si # No active partition 58*f7eb23eeSEd Maste movw $partbl,%bx # Partition table 59*f7eb23eeSEd Maste movb $0x4,%cl # Number of entries 60*f7eb23eeSEd Mastemain.1: cmpb %ch,(%bx) # Null entry? 61*f7eb23eeSEd Maste je main.2 # Yes 62*f7eb23eeSEd Maste jg err_pt # If 0x1..0x7f 63*f7eb23eeSEd Maste testw %si,%si # Active already found? 64*f7eb23eeSEd Maste jnz err_pt # Yes 65*f7eb23eeSEd Maste movw %bx,%si # Point to active 66*f7eb23eeSEd Mastemain.2: addb $0x10,%bl # Till 67*f7eb23eeSEd Maste loop main.1 # done 68*f7eb23eeSEd Maste testw %si,%si # Active found? 69*f7eb23eeSEd Maste jnz main.3 # Yes 70*f7eb23eeSEd Maste int $0x18 # BIOS: Diskless boot 71*f7eb23eeSEd Maste# 72*f7eb23eeSEd Maste# Ok, we've found a possible active partition. Check to see that the drive 73*f7eb23eeSEd Maste# is a valid hard drive number. 74*f7eb23eeSEd Maste# 75*f7eb23eeSEd Mastemain.3: cmpb $0x80,%dl # Drive valid? 76*f7eb23eeSEd Maste jb main.4 # No 77*f7eb23eeSEd Maste movb NHRDRV,%dh # Calculate the highest 78*f7eb23eeSEd Maste addb $0x80,%dh # drive number available 79*f7eb23eeSEd Maste cmpb %dh,%dl # Within range? 80*f7eb23eeSEd Maste jb main.5 # Yes 81*f7eb23eeSEd Mastemain.4: movb (%si),%dl # Load drive 82*f7eb23eeSEd Maste# 83*f7eb23eeSEd Maste# Ok, now that we have a valid drive and partition entry, load the CHS from 84*f7eb23eeSEd Maste# the partition entry and read the sector from the disk. 85*f7eb23eeSEd Maste# 86*f7eb23eeSEd Mastemain.5: movw %sp,%di # Save stack pointer 87*f7eb23eeSEd Maste movb 0x1(%si),%dh # Load head 88*f7eb23eeSEd Maste movw 0x2(%si),%cx # Load cylinder:sector 89*f7eb23eeSEd Maste movw $LOAD,%bx # Transfer buffer 90*f7eb23eeSEd Maste testb $FL_PACKET,flags # Try EDD? 91*f7eb23eeSEd Maste jz main.7 # No. 92*f7eb23eeSEd Maste pushw %cx # Save %cx 93*f7eb23eeSEd Maste pushw %bx # Save %bx 94*f7eb23eeSEd Maste movw $0x55aa,%bx # Magic 95*f7eb23eeSEd Maste movb $0x41,%ah # BIOS: EDD extensions 96*f7eb23eeSEd Maste int $0x13 # present? 97*f7eb23eeSEd Maste jc main.6 # No. 98*f7eb23eeSEd Maste cmpw $0xaa55,%bx # Magic ok? 99*f7eb23eeSEd Maste jne main.6 # No. 100*f7eb23eeSEd Maste testb $0x1,%cl # Packet mode present? 101*f7eb23eeSEd Maste jz main.6 # No. 102*f7eb23eeSEd Maste popw %bx # Restore %bx 103*f7eb23eeSEd Maste pushl $0x0 # Set the LBA 104*f7eb23eeSEd Maste pushl 0x8(%si) # address 105*f7eb23eeSEd Maste pushw %es # Set the address of 106*f7eb23eeSEd Maste pushw %bx # the transfer buffer 107*f7eb23eeSEd Maste pushw $0x1 # Read 1 sector 108*f7eb23eeSEd Maste pushw $0x10 # Packet length 109*f7eb23eeSEd Maste movw %sp,%si # Packer pointer 110*f7eb23eeSEd Maste movw $0x4200,%ax # BIOS: LBA Read from disk 111*f7eb23eeSEd Maste jmp main.8 # Skip the CHS setup 112*f7eb23eeSEd Mastemain.6: popw %bx # Restore %bx 113*f7eb23eeSEd Maste popw %cx # Restore %cx 114*f7eb23eeSEd Mastemain.7: movw $0x201,%ax # BIOS: Read from disk 115*f7eb23eeSEd Mastemain.8: int $0x13 # Call the BIOS 116*f7eb23eeSEd Maste movw %di,%sp # Restore stack 117*f7eb23eeSEd Maste jc err_rd # If error 118*f7eb23eeSEd Maste# 119*f7eb23eeSEd Maste# Now that we've loaded the bootstrap, check for the 0xaa55 signature. If it 120*f7eb23eeSEd Maste# is present, execute the bootstrap we just loaded. 121*f7eb23eeSEd Maste# 122*f7eb23eeSEd Maste cmpw $MAGIC,0x1fe(%bx) # Bootable? 123*f7eb23eeSEd Maste jne err_os # No 124*f7eb23eeSEd Maste jmp *%bx # Invoke bootstrap 125*f7eb23eeSEd Maste# 126*f7eb23eeSEd Maste# Various error message entry points. 127*f7eb23eeSEd Maste# 128*f7eb23eeSEd Masteerr_pt: movw $msg_pt,%si # "Invalid partition 129*f7eb23eeSEd Maste jmp putstr # table" 130*f7eb23eeSEd Maste 131*f7eb23eeSEd Masteerr_rd: movw $msg_rd,%si # "Error loading 132*f7eb23eeSEd Maste jmp putstr # operating system" 133*f7eb23eeSEd Maste 134*f7eb23eeSEd Masteerr_os: movw $msg_os,%si # "Missing operating 135*f7eb23eeSEd Maste jmp putstr # system" 136*f7eb23eeSEd Maste# 137*f7eb23eeSEd Maste# Output an ASCIZ string to the console via the BIOS. 138*f7eb23eeSEd Maste# 139*f7eb23eeSEd Masteputstr.0: movw $0x7,%bx # Page:attribute 140*f7eb23eeSEd Maste movb $0xe,%ah # BIOS: Display 141*f7eb23eeSEd Maste int $0x10 # character 142*f7eb23eeSEd Masteputstr: lodsb # Get character 143*f7eb23eeSEd Maste testb %al,%al # End of string? 144*f7eb23eeSEd Maste jnz putstr.0 # No 145*f7eb23eeSEd Masteputstr.1: jmp putstr.1 # Await reset 146*f7eb23eeSEd Maste 147*f7eb23eeSEd Mastemsg_pt: .asciz "Invalid partition table" 148*f7eb23eeSEd Mastemsg_rd: .asciz "Error loading operating system" 149*f7eb23eeSEd Mastemsg_os: .asciz "Missing operating system" 150*f7eb23eeSEd Maste 151*f7eb23eeSEd Maste .org PT_OFF-1,0x90 152*f7eb23eeSEd Masteflags: .byte FLAGS # Flags 153*f7eb23eeSEd Maste 154*f7eb23eeSEd Mastepartbl: .fill 0x10,0x4,0x0 # Partition table 155*f7eb23eeSEd Maste .word MAGIC # Magic number 156