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