summaryrefslogtreecommitdiff
path: root/sys/arch/i386/stand/mbr/mbr.S
blob: 2b48f580e961fa28c06536712ece09269d83cbb2 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
/*	$OpenBSD: mbr.S,v 1.1 1997/03/31 03:12:19 weingart Exp $	*/

/*
 * Copyright (c) 1997 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.
 */

/*	$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"

#include <machine/asm.h>
#include <machine/specialreg.h>

#define data32	.byte 0x66
#define addr32	.byte 0x67

#define BOOTBIOS	0x7c0	/* segment where we are loaded */
#define BOOTRELOC	0x7a0	/* segment where to relocate */
#define SIGNATURE	0xaa55	/* MBR signature */
#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_B		'B'	/* loading boot */
#define CHAR_G		'G'	/* jumping to boot */

#define DBGMSG(msg)		\
	movb    $msg, %al;	\
	data32;			\
	call    chr
#else /* !DEBUG */
#define DBGMSG(msg)
#endif /* !DEBUG */

	.text

	.globl	start
start:
	movl	%cs, %ax
	# set up stack(%ss:%esp)
	cli			/* disable interrupts w/o stack */
	movl	%ax, %ss
	data32
	movl	$0xfffc, %esp
	sti			/* we have stack, do ints */
	/* setup %ds */
	movl	%ax, %ds

	/* relocate the code to leave the space for boot stage */
	data32
	movl	$BOOTRELOC, %eax
	movl	%ax, %es
	xorl	%si, %si
	xorl	%di, %di
	xorl	%cx, %cx
	incb	%ch	/* movl $0x100, %ecx */
	cld
	rep
	movsw

	# jump to the relocated code
	data32
	ljmp $BOOTRELOC, $1f
1:
	DBGMSG(CHAR_S)

	/* setup %es, %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.
	xorl	%ax, %ax
       	movb	$0xe3, %ax
	data32
	movl	$SERIAL, %dx
	int	$0x14
#endif

	/* bootstrap passes us drive number in %dl */
	testb	$0x80, %dl
	jnz	1f

	/* mbr on floppy ??? */
	data32
	movl	$fdmbr, %esi
	data32
	call	message

1:	xorl	%bx, %bx
	# cmpw	$SIGNATURE, (%bx)
	.byte	0x81, 0xbf
	.word	signature
	.word	SIGNATURE
	je	1f
	DBGMSG(CHAR_P)

	/* find the first active partition */
1:	DBGMSG(CHAR_L)
	data32
	movl	$pt, %esi
	data32
	movl	$NUMPART, %cx
1:
	# movb	(%si), %al
	.byte	0x8a, 0x44, 0x00
	cmpb	$BOOTABLE, %al
	je	2f
	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! */
	data32
	movl	$prompt, %esi
	data32
	call	message

	# BIOS call: read char from kbd
	#	return: %ah == scancode, %al == ascii
	xorl	%ax, %ax
	int	$0x16

	/* load active, if not */
	cmpb	'n', %al
	je	found
	cmpb	'N', %al
	je	found

1:
	/* else: find OpenBSD partition */
	DBGMSG(CHAR_O)
	data32
	movl	$pt, %esi
	data32
	movl	$NUMPART, %ecx
1:
	# movb	(%si), %al
	.byte	0x8a, 0x44, 0x00
	cmpb	$BSDPART, %al
	je	found
	data32
	addl	$PARTSZ, %esi
	loop	1b
	data32
	movl	$enoboot, %esi
err_stop:
	data32
	call	message
	cli
	hlt

found:
	DBGMSG(CHAR_B)
	# movb	1(%si), %dh	# head
	.byte   0x8a, 0x74, 0x01
	# movw	2(%si), %cx	# sect, cyl
	.byte   0x8b, 0x4c, 0x02
	# movb	4(%si), %al	# partition type
	.byte   0x8a, 0x44, 0x04

/*
# BIOS call "INT 0x13 Function 0x2" to read sectors from disk into memory
#       Call with       %ah = 0x2
#                       %al = number of sectors
#                       %ch = cylinder
#                       %cl = sector
#                       %dh = head
#                       %dl = drive (0x80 for hard disk, 0x0 for floppy disk)
#                       %es:%bx = segment:offset of buffer
#       Return:
#                       %al = 0x0 on success; err code on failure
*/
	data32
	movl	$0x200 | 1, %eax	/* number of blocks */
	xorl	%bx, %bx		/* put it at %es:0 */
	int	$0x13
	data32
	movl    $eread, %si
	jc	err_stop

	DBGMSG(CHAR_G)

	# jump to the new code (%ds:%si is at he right point)
	data32
	ljmp	$0, $BOOTBIOS << 4
	/* not reached */

#
#	message: write the error message in %ds:%si to console
#
chr:
/*
#ifndef SERIAL
# BIOS call "INT 10H Function 0Eh" to write character to console
#	Call with	%ah = 0x0e
#			%al = character
#			%bh = page
#			%bl = foreground color
#else
# BIOS call "INT 14H Function 01h" to write character to console
#	Call with	%ah = 0x01
#			%al = character
#			%dx = port number
#endif
*/
	pushl	%eax

#ifndef SERIAL
	pushl	%ebx
	movb	$0x0e, %ah
	xorl	%bx, %bx
	incl	%bx		/* movw $0x01, %bx */
	int	$0x10
	popl	%ebx
#else
	pushl	%edx
	movb	$0x01, %ah
	data32
	movl	SERIAL, %dx
	int	$0x14
	popl	%edx
#endif
	popl	%ax
	data32
	ret

/*
 * Display string
 */
message:
	pushl	%ax
	cld
1:
	lodsb			# load a byte into %al
	testb	%al, %al
	jz	1f
	data32
	call	chr
	jmp	1b
1:
	popl	%ax
	ret

/* error messages */

fdmbr:	.asciz		"MBR on fd?\r\n"
eread:	.asciz		"Read error\r\n"
enoboot: .asciz		"No partition to boot\r\n"
prompt:	.asciz		"OpenBSD? "

endofcode:
	nop

/* throw in a partition table */
/* flag, head, sec, cyl, type, ehead, esect, ecyl, start, len */
	. = 0x1be	# starting address of partition table
pt:
	.byte	0x0,0,0,0,0,0,0,0
	.long	0,0
	.byte	0x0,0,0,0,0,0,0,0
	.long	0,0
	.byte	0x0,0,0,0,0,0,0,0
	.long	0,0
	.byte	BOOTABLE,0,1,0,BSDPART,255,255,255
	.long	0,50000
/* the last 2 bytes in the sector 0 contain the signature */
	. = 0x1fe
signature:
	.short	SIGNATURE
	. = 0x200