xref: /freebsd/stand/i386/mbr/mbr.S (revision d0b2dbfa0ecf2bbc9709efc5e20baf8e4b44bbbf)
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