summaryrefslogtreecommitdiff
path: root/sys/arch/i386
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch/i386')
-rw-r--r--sys/arch/i386/stand/mbr/mbr.S245
1 files changed, 157 insertions, 88 deletions
diff --git a/sys/arch/i386/stand/mbr/mbr.S b/sys/arch/i386/stand/mbr/mbr.S
index 2b48f580e96..f7b3e1407d3 100644
--- a/sys/arch/i386/stand/mbr/mbr.S
+++ b/sys/arch/i386/stand/mbr/mbr.S
@@ -1,7 +1,7 @@
-/* $OpenBSD: mbr.S,v 1.1 1997/03/31 03:12:19 weingart Exp $ */
+/* $OpenBSD: mbr.S,v 1.2 1997/08/01 18:39:45 weingart Exp $ */
/*
- * Copyright (c) 1997 Michael Shalayeff
+ * Copyright (c) 1997 Michael Shalayeff and Tobias Weingartner
*/
/* Copyright (c) 1996 VaX#n8 (vax@linkdead.paranoia.com)
@@ -13,22 +13,7 @@
* this copyright notice in the file.
*/
-/* $OpenBSD: mbr.S,v 1.1 1997/03/31 03:12:19 weingart Exp $ */
-
-/*
- * Copyright (c) 1996 Michael Shalayeff
- */
-
-/* Copyright (c) 1996 VaX#n8 (vax@linkdead.paranoia.com)
- * last edited 9 July 1996
- * many thanks to Erich Boleyn (erich@uruk.org) for putting up with
- * all my questions, and for his work on GRUB
- * You may use this code or fragments thereof in a manner consistent
- * with the other copyrights as long as you retain my pseudonym and
- * this copyright notice in the file.
- */
-
- .file "start.S"
+ .file "mbr.S"
#include <machine/asm.h>
#include <machine/specialreg.h>
@@ -42,14 +27,12 @@
#define NUMPART 4 /* number of partitions in partition table */
#define PARTSZ 16 /* each partition table entry is 16 bytes */
#define BSDPART 0xA6 /* OpenBSD partition */
-#define OLDBSDPART 0xA5 /* {386,Net,Free}BSD partition */
#define BOOTABLE 0x80 /* bootable partition */
#ifdef DEBUG
#define CHAR_S 'S' /* started */
-#define CHAR_L 'L' /* looking up bootable partition */
-#define CHAR_P 'P' /* partition table corrupted */
-#define CHAR_O 'O' /* force OpenBSD load */
+#define CHAR_R 'R' /* relocated */
+#define CHAR_L 'L' /* looking for bootable partition */
#define CHAR_B 'B' /* loading boot */
#define CHAR_G 'G' /* jumping to boot */
@@ -65,131 +48,203 @@
.globl start
start:
+ /* Adjust %cs to be right */
+ data32
+ ljmp $BOOTBIOS, $1f
+1:
+ /* Set up stack */
movl %cs, %ax
- # set up stack(%ss:%esp)
- cli /* disable interrupts w/o stack */
+ cli
movl %ax, %ss
data32
movl $0xfffc, %esp
- sti /* we have stack, do ints */
- /* setup %ds */
+ sti
+
+ /* Set up data segment */
movl %ax, %ds
+ DBGMSG(CHAR_S)
- /* relocate the code to leave the space for boot stage */
+ /* Relocate 512 bytes so we can load PBS here */
data32
movl $BOOTRELOC, %eax
movl %ax, %es
+ data32
xorl %si, %si
+ data32
xorl %di, %di
- xorl %cx, %cx
- incb %ch /* movl $0x100, %ecx */
+ data32
+ movl $0x200, %ecx
cld
rep
- movsw
+ movsb
- # jump to the relocated code
+ /* Jump to relocated self */
data32
- ljmp $BOOTRELOC, $1f
-1:
- DBGMSG(CHAR_S)
+ ljmp $BOOTRELOC, $reloc
+reloc:
+ DBGMSG(CHAR_R)
- /* setup %es, %ds */
+ /* Set up %es and %ds */
pushl %ds
popl %es /* next boot is at the same place as we were loaded */
pushl %cs
popl %ds /* and %ds is at the %cs */
#ifdef SERIAL
- # Initialize the serial port to 9600 baud, 8N1.
+ /* Initialize the serial port to 9600 baud, 8N1. */
+ * Do we need to do this? Most things at this level
+ * do not know or care (on a PC) where the output is
+ * happening to go. I think if we are headless,
+ * /boot should figure (as it does now) that out.
+ *
+ * If there is a problem with this stage of the boot
+ * process, connect up a monitor and kbd, and see what
+ * is going on. Left here for the time being.
+ *
+ * --Toby.
+ */
xorl %ax, %ax
- movb $0xe3, %ax
+ movb $0xe3, %ax
data32
movl $SERIAL, %dx
int $0x14
#endif
- /* bootstrap passes us drive number in %dl */
+ /* bootstrap passes us drive number in %dl
+ *
+ * XXX - This is not always true. We currently
+ * check if %dl points to a HD, and if not we
+ * complain, and set it to point to the first
+ * HDD. Note, this is not 100% correct, since
+ * there is a possibility that you boot of of
+ * HD #2, and still get (%dl & 0x80) == 0x00,
+ * these type of systems will loose. I don't
+ * know of any like this, but I've come to the
+ * conclusion, that if it can exists, it will,
+ * someplace in the PC world. If anyone knows
+ * how to fix this, speak up!
+ *
+ * Toby - Thu Jul 31 21:01:00 CDT 1997
+ */
testb $0x80, %dl
jnz 1f
- /* mbr on floppy ??? */
+ /* MBR on floppy or old BIOS
+ * Note: MBR (this code) should never
+ * be on a floppy. It does not belong
+ * there, so %dl should never be 0x00.
+ *
+ * Here we simply complain (should we?),
+ * and then hardcode the boot drive to
+ * 0x80.
+ */
data32
movl $fdmbr, %esi
data32
call message
+ /* If we are passed bogus data, set it to HD #1.
+ * We should load the value from a hard coded
+ * location in this sector. Maybe I'll write
+ * that next, since my machines seem to be one
+ * of the weird ones...
+ */
+ movb $0x80, %dl
+
+ /* Do we need to check our signature? The BIOS will
+ * check it for us, I doubt there is a need for us to
+ * do the same thing over again. If we fail here,
+ * something terrible is wrong. However, I doubt we
+ * can recover anyways. The message might be nice
+ * for the (l)user though.
+ */
1: xorl %bx, %bx
# cmpw $SIGNATURE, (%bx)
.byte 0x81, 0xbf
.word signature
.word SIGNATURE
- je 1f
- DBGMSG(CHAR_P)
+ je sigok
+ data32
+ movl $esig, %esi
+ data32
+ call message
- /* find the first active partition */
-1: DBGMSG(CHAR_L)
+ /* find the first active partition
+ * Note: this should be the only active
+ * partition. We currently don't check
+ * for that, but we really should. If
+ * and when I feel up to it, I'll add
+ * that code.
+ */
+sigok:
data32
movl $pt, %esi
data32
movl $NUMPART, %cx
1:
+ DBGMSG(CHAR_L)
# movb (%si), %al
.byte 0x8a, 0x44, 0x00
cmpb $BOOTABLE, %al
- je 2f
+ je found
data32
addl $PARTSZ, %esi
loop 1b
- jmp 1f /* no bootable -- find $BSDPART */
-2: # test if bootable is ours
- # movb 4(%si), %al # partition type
- .byte 0x8a, 0x44, 0x04
- cmpb $BSDPART, %al
- je found
-
- /* else: ask! */
+ /* No bootable partition */
+no_part:
data32
- movl $prompt, %esi
+ movl $noboot, %esi
data32
call message
+err_stop:
+ cli
+ hlt
+ /* Just to make sure */
+ jmp err_stop
- # BIOS call: read char from kbd
- # return: %ah == scancode, %al == ascii
- xorl %ax, %ax
- int $0x16
+ /* Found bootable partition */
+found:
+ DBGMSG(CHAR_B)
+ pushl %eax
+ pushl %edx
+ pushl %esi
+ pushl %ecx
- /* load active, if not */
- cmpb 'n', %al
- je found
- cmpb 'N', %al
- je found
+ /* Print out drive and partition */
+ data32
+ movl $drive, %esi
+ data32
+ call message
-1:
- /* else: find OpenBSD partition */
- DBGMSG(CHAR_O)
+ movl %dx, %ax
+ andl $0x0F, %ax
+ orl $0x30, %ax
data32
- movl $pt, %esi
+ call chr
+
data32
- movl $NUMPART, %ecx
-1:
- # movb (%si), %al
- .byte 0x8a, 0x44, 0x00
- cmpb $BSDPART, %al
- je found
+ movl $partn, %esi
data32
- addl $PARTSZ, %esi
- loop 1b
+ call message
+
+ popl %eax
+ andl $0x0F, %ax
+ orl $0x30, %ax
data32
- movl $enoboot, %esi
-err_stop:
+ call chr
+
+ data32
+ movl $newln, %esi
data32
call message
- cli
- hlt
-found:
- DBGMSG(CHAR_B)
+ /* Restore %esi and %edx */
+ popl %esi
+ popl %edx
+ popl %eax
+
+ /* Load values from active partition table entry */
# movb 1(%si), %dh # head
.byte 0x8a, 0x74, 0x01
# movw 2(%si), %cx # sect, cyl
@@ -213,10 +268,15 @@ found:
movl $0x200 | 1, %eax /* number of blocks */
xorl %bx, %bx /* put it at %es:0 */
int $0x13
+ jnc 1f
data32
movl $eread, %si
- jc err_stop
+ data32
+ call message
+ jmp err_stop
+
+1:
DBGMSG(CHAR_G)
# jump to the new code (%ds:%si is at he right point)
@@ -259,7 +319,7 @@ chr:
int $0x14
popl %edx
#endif
- popl %ax
+ popl %eax
data32
ret
@@ -280,17 +340,25 @@ message:
popl %ax
ret
-/* error messages */
+/* Info messages */
+drive: .asciz "Using Drive: "
+partn: .asciz " Partition: "
+newln: .asciz "\r\n"
-fdmbr: .asciz "MBR on fd?\r\n"
+/* Error messages */
+fdmbr: .asciz "MBR on floppy or old BIOS\r\n"
eread: .asciz "Read error\r\n"
-enoboot: .asciz "No partition to boot\r\n"
-prompt: .asciz "OpenBSD? "
+noboot: .asciz "No active partition\r\n"
+esig: .asciz "Invalid Signature\r\n"
endofcode:
nop
-/* throw in a partition table */
+/* (MBR) NT registry offset */
+ . = 0x1b8
+ .space 4, 0
+
+/* partition table */
/* flag, head, sec, cyl, type, ehead, esect, ecyl, start, len */
. = 0x1be # starting address of partition table
pt:
@@ -301,9 +369,10 @@ pt:
.byte 0x0,0,0,0,0,0,0,0
.long 0,0
.byte BOOTABLE,0,1,0,BSDPART,255,255,255
- .long 0,50000
+ .long 0,0x7FFFFFFF
/* the last 2 bytes in the sector 0 contain the signature */
. = 0x1fe
signature:
.short SIGNATURE
. = 0x200
+