xref: /netbsd-src/sys/arch/landisk/stand/mbr/mbr.S (revision 60ab2ca5c0570c0013b39de285ddaa91fe27d029)
1/*	$NetBSD: mbr.S,v 1.1 2006/09/01 21:26:19 uwe Exp $	*/
2
3/*-
4 * Copyright (c) 2005 NONAKA Kimihiro
5 * All rights reserved.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 * 1. Redistributions of source code must retain the above copyright
11 *    notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 *    notice, this list of conditions and the following disclaimer in the
14 *    documentation and/or other materials provided with the distribution.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
17 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19 * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
20 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 * SUCH DAMAGE.
27 */
28
29#include <machine/asm.h>
30#include <sys/bootblock.h>
31
32ENTRY(start)
33	mova	mbr_end, r0
34	mov	r0, r11			/* r11: relocate address */
35
36	mov.w	mbr_size, r2
37	sub	r2, r0
38	mov	r0, r10			/* r10: loaded address */
39
40	mov.w	stack_offset, r1
41	add	r1, r0
42	mov	r0, r15			/* r15: stack pointer */
43
44	/* relocate code */
45	mova	jmp_start, r0
46	mov	r0, r13
47	add	r2, r13			/* calc jump address */
48
49	mov	r10, r0
50	mov	r11, r1
511:	mov.b	@r0+, r3
52	dt	r2
53	mov.b	r3, @r1
54	bf/s	1b
55	 add	#1, r1
56
57	jmp	@r13
58	 nop
59
60	.align	2
61jmp_start:
62	/* enable cache */
63	mov	#0, r4
64	mov	#6, r0
65	trapa	#0x3f
66
67#ifndef	NO_BANNER
68	/* print banner */
69	mova	banner, r0
70	bsr	message_crlf
71	 mov	r0, r4
72#endif
73
74	/* search bootable partition */
75	mov.w	part_offset, r12
76	add	r11, r12		/* r12: pointer to partition entry */
77	mov	#MBR_PART_COUNT, r8	/* r8: partition loop counter */
78loop_part:
79	mov.b	@(4, r12), r0
80	cmp/eq	#MBR_PTYPE_UNUSED, r0
81	bt	next_part
82
83	/* check active partition */
84	mov.b	@(0, r12), r0
85	cmp/eq	#0x80, r0
86	bf	next_part
87
88	/* found bootable partition */
89	mov.w	@(8, r12), r0		/* load unaligned 32bit data */
90	mov	r0, r1
91	mov.w	@(10, r12), r0
92	extu.w	r1, r1
93	shll16	r0
94	or	r1, r0
95
96	mov	r0, r3
97	mova	found_sector, r0
98	bra	boot_lba
99	 mov.l	r3, @r0
100
101next_part:
102	dt	r8
103	bf/s	loop_part
104	 add	#16, r12
105
106noos_error:
107	/* Not found bootable partition */
108	mova	ERR_NOOS, r0
109error:
110	bsr	message_crlf
111	 mov	r0, r4
11299:	bra	99b
113	 nop
114
115read_error:
116	bra	error
117	 mova	ERR_READ, r0
118
119message_crlf:
120	mov	#32, r0
121	trapa	#0x3f
122	mova	crlf, r0
123	mov	r0, r4
124	mov	#32, r0
125	trapa	#0x3f
126	rts
127	 nop
128
129read_sector_lba:
130	mov	#1, r7
131	mov	#2, r0
132	trapa	#0x3f
133	tst	r0, r0
134	bf	read_error
135	rts
136	 nop
137
138boot_lba:
139	/* read PBR sector */
140	mova	found_sector, r0
141	mov	#0x40, r4
142	mov.l	@r0, r5
143	bsr	read_sector_lba
144	 mov	r10, r6
145
146	/* flush cache */
147	mov	#0, r4
148	mov	#6, r0
149	trapa	#0x3f
150
151	/* check signature */
152	mov.b	@(0, r10), r0
153	tst	r0, r0
154	bt	noos_error		/* first byte non-zero */
155	mov.w	magic_offset, r0
156	mov.w	@(r0, r10), r1
157	mov.w	magic, r2
158	cmp/eq	r1, r2
159	bf	noos_error		/* magic */
160
161	/* now jump to PBR */
162	mov	r10, r0
163	jmp	@r10
164	 nop
165
166
167	.align	1
168mbr_size:	.word	mbr_end - _C_LABEL(start)
169	.align	1
170stack_offset:	.word	0x1000
171	.align	1
172part_offset:	.word	MBR_PART_OFFSET
173	.align	1
174magic_offset:	.word	MBR_MAGIC_OFFSET
175
176	.align	2
177found_sector:	.long	0
178
179#ifndef	NO_BANNER
180	.align	2
181banner:		.asciz	"\r\nNetBSD MBR boot"
182#endif
183	.align	2
184crlf:		.asciz	"\r\n"
185
186	.align	2
187ERR_INVPART:	.asciz	"No active partition"
188	.align	2
189ERR_READ:	.asciz	"Disk read error"
190	.align	2
191ERR_NOOS:	.asciz	"No operating system"
192
193
194/* space for mbr_dsn */
195	. = _C_LABEL(start) + MBR_DSN_OFFSET
196	.long	0
197
198/* mbr_bootsel_magic */
199	. = _C_LABEL(start) + MBR_BS_MAGIC_OFFSET
200	.word	MBR_BS_MAGIC
201
202/*
203 * MBR partition table
204 */
205	. = _C_LABEL(start) + MBR_PART_OFFSET
206_pbr_part0:
207	.byte	0, 0, 0, 0, 0, 0, 0, 0
208	.byte	0, 0, 0, 0, 0, 0, 0, 0
209_pbr_part1:
210	.byte	0, 0, 0, 0, 0, 0, 0, 0
211	.byte	0, 0, 0, 0, 0, 0, 0, 0
212_pbr_part2:
213	.byte	0, 0, 0, 0, 0, 0, 0, 0
214	.byte	0, 0, 0, 0, 0, 0, 0, 0
215_pbr_part3:
216	.byte	0, 0, 0, 0, 0, 0, 0, 0
217	.byte	0, 0, 0, 0, 0, 0, 0, 0
218
219	. = _C_LABEL(start) + MBR_MAGIC_OFFSET
220magic:
221	.word	MBR_MAGIC
222mbr_end:
223