biosdisk.c revision 1.36 1 1.36 jakllsch /* $NetBSD: biosdisk.c,v 1.36 2011/01/05 22:06:59 jakllsch Exp $ */
2 1.1 perry
3 1.1 perry /*
4 1.8 drochner * Copyright (c) 1996, 1998
5 1.1 perry * Matthias Drochner. All rights reserved.
6 1.1 perry *
7 1.1 perry * Redistribution and use in source and binary forms, with or without
8 1.1 perry * modification, are permitted provided that the following conditions
9 1.1 perry * are met:
10 1.1 perry * 1. Redistributions of source code must retain the above copyright
11 1.1 perry * notice, this list of conditions and the following disclaimer.
12 1.1 perry * 2. Redistributions in binary form must reproduce the above copyright
13 1.1 perry * notice, this list of conditions and the following disclaimer in the
14 1.1 perry * documentation and/or other materials provided with the distribution.
15 1.1 perry *
16 1.1 perry * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17 1.1 perry * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18 1.1 perry * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19 1.1 perry * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20 1.1 perry * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21 1.1 perry * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22 1.1 perry * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23 1.1 perry * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24 1.1 perry * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25 1.1 perry * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 1.1 perry *
27 1.1 perry */
28 1.1 perry
29 1.3 thorpej /*
30 1.4 drochner * raw BIOS disk device for libsa.
31 1.4 drochner * needs lowlevel parts from bios_disk.S and biosdisk_ll.c
32 1.4 drochner * partly from netbsd:sys/arch/i386/boot/disk.c
33 1.4 drochner * no bad144 handling!
34 1.17 dsl *
35 1.17 dsl * A lot of this must match sys/kern/subr_disk_mbr.c
36 1.1 perry */
37 1.1 perry
38 1.1 perry /*
39 1.1 perry * Ported to boot 386BSD by Julian Elischer (julian (at) tfs.com) Sept 1992
40 1.1 perry *
41 1.1 perry * Mach Operating System
42 1.1 perry * Copyright (c) 1992, 1991 Carnegie Mellon University
43 1.1 perry * All Rights Reserved.
44 1.3 thorpej *
45 1.1 perry * Permission to use, copy, modify and distribute this software and its
46 1.1 perry * documentation is hereby granted, provided that both the copyright
47 1.1 perry * notice and this permission notice appear in all copies of the
48 1.1 perry * software, derivative works or modified versions, and any portions
49 1.1 perry * thereof, and that both notices appear in supporting documentation.
50 1.3 thorpej *
51 1.1 perry * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
52 1.1 perry * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
53 1.1 perry * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
54 1.3 thorpej *
55 1.1 perry * Carnegie Mellon requests users of this software to return to
56 1.3 thorpej *
57 1.1 perry * Software Distribution Coordinator or Software.Distribution (at) CS.CMU.EDU
58 1.1 perry * School of Computer Science
59 1.1 perry * Carnegie Mellon University
60 1.1 perry * Pittsburgh PA 15213-3890
61 1.3 thorpej *
62 1.1 perry * any improvements or extensions that they make and grant Carnegie Mellon
63 1.1 perry * the rights to redistribute these changes.
64 1.1 perry */
65 1.1 perry
66 1.36 jakllsch #if !defined(NO_DISKLABEL) || !defined(NO_GPT)
67 1.29 jmcneill #define FSTYPENAMES
68 1.29 jmcneill #endif
69 1.29 jmcneill
70 1.36 jakllsch #include <lib/libkern/libkern.h>
71 1.36 jakllsch #include <lib/libsa/stand.h>
72 1.36 jakllsch
73 1.1 perry #include <sys/types.h>
74 1.21 thorpej #include <sys/md5.h>
75 1.24 junyoung #include <sys/param.h>
76 1.29 jmcneill #include <sys/disklabel.h>
77 1.36 jakllsch #include <sys/disklabel_gpt.h>
78 1.36 jakllsch #include <sys/uuid.h>
79 1.24 junyoung
80 1.24 junyoung #include <fs/cd9660/iso.h>
81 1.1 perry
82 1.1 perry #include <lib/libsa/saerrno.h>
83 1.5 drochner #include <machine/stdarg.h>
84 1.29 jmcneill #include <machine/cpu.h>
85 1.1 perry
86 1.1 perry #include "libi386.h"
87 1.1 perry #include "biosdisk_ll.h"
88 1.5 drochner #include "biosdisk.h"
89 1.9 drochner #ifdef _STANDALONE
90 1.5 drochner #include "bootinfo.h"
91 1.9 drochner #endif
92 1.16 dsl
93 1.24 junyoung #define BUFSIZE 2048 /* must be large enough for a CD sector */
94 1.1 perry
95 1.36 jakllsch #define BIOSDISKNPART 26
96 1.36 jakllsch
97 1.3 thorpej struct biosdisk {
98 1.3 thorpej struct biosdisk_ll ll;
99 1.32 jakllsch daddr_t boff;
100 1.3 thorpej char buf[BUFSIZE];
101 1.36 jakllsch #if !defined(NO_DISKLABEL) || !defined(NO_GPT)
102 1.36 jakllsch struct {
103 1.36 jakllsch daddr_t offset;
104 1.36 jakllsch daddr_t size;
105 1.36 jakllsch int fstype;
106 1.36 jakllsch } part[BIOSDISKNPART];
107 1.36 jakllsch #endif
108 1.1 perry };
109 1.1 perry
110 1.36 jakllsch #ifndef NO_GPT
111 1.36 jakllsch const struct uuid GET_nbsd_raid = GPT_ENT_TYPE_NETBSD_RAIDFRAME;
112 1.36 jakllsch const struct uuid GET_nbsd_ffs = GPT_ENT_TYPE_NETBSD_FFS;
113 1.36 jakllsch const struct uuid GET_nbsd_lfs = GPT_ENT_TYPE_NETBSD_LFS;
114 1.36 jakllsch const struct uuid GET_nbsd_swap = GPT_ENT_TYPE_NETBSD_SWAP;
115 1.36 jakllsch #endif /* NO_GPT */
116 1.36 jakllsch
117 1.9 drochner #ifdef _STANDALONE
118 1.5 drochner static struct btinfo_bootdisk bi_disk;
119 1.21 thorpej static struct btinfo_bootwedge bi_wedge;
120 1.9 drochner #endif
121 1.5 drochner
122 1.13 lukem #define RF_PROTECTED_SECTORS 64 /* XXX refer to <.../rf_optnames.h> */
123 1.13 lukem
124 1.22 junyoung int
125 1.24 junyoung biosdisk_strategy(void *devdata, int flag, daddr_t dblk, size_t size,
126 1.24 junyoung void *buf, size_t *rsize)
127 1.1 perry {
128 1.3 thorpej struct biosdisk *d;
129 1.22 junyoung int blks, frag;
130 1.1 perry
131 1.3 thorpej if (flag != F_READ)
132 1.22 junyoung return EROFS;
133 1.1 perry
134 1.3 thorpej d = (struct biosdisk *) devdata;
135 1.1 perry
136 1.24 junyoung if (d->ll.type == BIOSDISK_TYPE_CD)
137 1.24 junyoung dblk = dblk * DEV_BSIZE / ISO_DEFAULT_BLOCK_SIZE;
138 1.24 junyoung
139 1.3 thorpej dblk += d->boff;
140 1.3 thorpej
141 1.24 junyoung blks = size / d->ll.secsize;
142 1.3 thorpej if (blks && readsects(&d->ll, dblk, blks, buf, 0)) {
143 1.3 thorpej if (rsize)
144 1.3 thorpej *rsize = 0;
145 1.22 junyoung return EIO;
146 1.3 thorpej }
147 1.22 junyoung
148 1.24 junyoung /* needed for CD */
149 1.24 junyoung frag = size % d->ll.secsize;
150 1.3 thorpej if (frag) {
151 1.3 thorpej if (readsects(&d->ll, dblk + blks, 1, d->buf, 0)) {
152 1.3 thorpej if (rsize)
153 1.24 junyoung *rsize = blks * d->ll.secsize;
154 1.22 junyoung return EIO;
155 1.3 thorpej }
156 1.24 junyoung memcpy(buf + blks * d->ll.secsize, d->buf, frag);
157 1.3 thorpej }
158 1.22 junyoung
159 1.3 thorpej if (rsize)
160 1.3 thorpej *rsize = size;
161 1.22 junyoung return 0;
162 1.1 perry }
163 1.1 perry
164 1.16 dsl static struct biosdisk *
165 1.23 junyoung alloc_biosdisk(int biosdev)
166 1.1 perry {
167 1.3 thorpej struct biosdisk *d;
168 1.1 perry
169 1.23 junyoung d = alloc(sizeof(*d));
170 1.22 junyoung if (d == NULL)
171 1.16 dsl return NULL;
172 1.23 junyoung memset(d, 0, sizeof(*d));
173 1.16 dsl
174 1.23 junyoung d->ll.dev = biosdev;
175 1.11 fvdl if (set_geometry(&d->ll, NULL)) {
176 1.1 perry #ifdef DISK_DEBUG
177 1.3 thorpej printf("no geometry information\n");
178 1.1 perry #endif
179 1.26 christos dealloc(d, sizeof(*d));
180 1.16 dsl return NULL;
181 1.3 thorpej }
182 1.16 dsl return d;
183 1.16 dsl }
184 1.16 dsl
185 1.36 jakllsch #if !defined(NO_DISKLABEL) || !defined(NO_GPT)
186 1.36 jakllsch static void
187 1.36 jakllsch md5(void *hash, const void *data, size_t len)
188 1.36 jakllsch {
189 1.36 jakllsch MD5_CTX ctx;
190 1.36 jakllsch
191 1.36 jakllsch MD5Init(&ctx);
192 1.36 jakllsch MD5Update(&ctx, data, len);
193 1.36 jakllsch MD5Final(hash, &ctx);
194 1.36 jakllsch
195 1.36 jakllsch return;
196 1.36 jakllsch }
197 1.36 jakllsch #endif
198 1.36 jakllsch
199 1.36 jakllsch #ifndef NO_GPT
200 1.36 jakllsch static bool
201 1.36 jakllsch guid_is_nil(const struct uuid *u)
202 1.36 jakllsch {
203 1.36 jakllsch static const struct uuid nil = { .time_low = 0 };
204 1.36 jakllsch return (memcmp(u, &nil, sizeof(*u)) == 0 ? true : false);
205 1.36 jakllsch }
206 1.36 jakllsch
207 1.36 jakllsch static bool
208 1.36 jakllsch guid_is_equal(const struct uuid *a, const struct uuid *b)
209 1.36 jakllsch {
210 1.36 jakllsch return (memcmp(a, b, sizeof(*a)) == 0 ? true : false);
211 1.36 jakllsch }
212 1.36 jakllsch
213 1.36 jakllsch static int
214 1.36 jakllsch check_gpt(struct biosdisk *d, daddr_t sector)
215 1.36 jakllsch {
216 1.36 jakllsch struct gpt_hdr gpth;
217 1.36 jakllsch const struct gpt_ent *ep;
218 1.36 jakllsch const struct uuid *u;
219 1.36 jakllsch daddr_t entblk;
220 1.36 jakllsch size_t size;
221 1.36 jakllsch uint32_t crc;
222 1.36 jakllsch int sectors;
223 1.36 jakllsch int entries;
224 1.36 jakllsch int entry;
225 1.36 jakllsch int i, j;
226 1.36 jakllsch
227 1.36 jakllsch /* read in gpt_hdr sector */
228 1.36 jakllsch if (readsects(&d->ll, sector, 1, d->buf, 1)) {
229 1.36 jakllsch #ifdef DISK_DEBUG
230 1.36 jakllsch printf("Error reading GPT header at %"PRId64"\n", sector);
231 1.36 jakllsch #endif
232 1.36 jakllsch return EIO;
233 1.36 jakllsch }
234 1.36 jakllsch
235 1.36 jakllsch gpth = *(const struct gpt_hdr *)d->buf;
236 1.36 jakllsch
237 1.36 jakllsch if (memcmp(GPT_HDR_SIG, gpth.hdr_sig, sizeof(gpth.hdr_sig)))
238 1.36 jakllsch return -1;
239 1.36 jakllsch
240 1.36 jakllsch crc = gpth.hdr_crc_self;
241 1.36 jakllsch gpth.hdr_crc_self = 0;
242 1.36 jakllsch gpth.hdr_crc_self = crc32(0, (const void *)&gpth, GPT_HDR_SIZE);
243 1.36 jakllsch if (gpth.hdr_crc_self != crc) {
244 1.36 jakllsch return -1;
245 1.36 jakllsch }
246 1.36 jakllsch
247 1.36 jakllsch if (gpth.hdr_lba_self != sector)
248 1.36 jakllsch return -1;
249 1.36 jakllsch
250 1.36 jakllsch #ifdef _STANDALONE
251 1.36 jakllsch bi_wedge.matchblk = sector;
252 1.36 jakllsch bi_wedge.matchnblks = 1;
253 1.36 jakllsch
254 1.36 jakllsch md5(bi_wedge.matchhash, d->buf, d->ll.secsize);
255 1.36 jakllsch #endif
256 1.36 jakllsch
257 1.36 jakllsch sectors = sizeof(d->buf)/d->ll.secsize; /* sectors per buffer */
258 1.36 jakllsch entries = sizeof(d->buf)/gpth.hdr_entsz; /* entries per buffer */
259 1.36 jakllsch entblk = gpth.hdr_lba_table;
260 1.36 jakllsch crc = crc32(0, NULL, 0);
261 1.36 jakllsch
262 1.36 jakllsch j = 0;
263 1.36 jakllsch ep = (const struct gpt_ent *)d->buf;
264 1.36 jakllsch
265 1.36 jakllsch for (entry = 0; entry < gpth.hdr_entries; entry += entries) {
266 1.36 jakllsch size = MIN(sizeof(d->buf),
267 1.36 jakllsch (gpth.hdr_entries - entry) * gpth.hdr_entsz);
268 1.36 jakllsch entries = size / gpth.hdr_entsz;
269 1.36 jakllsch sectors = roundup(size, d->ll.secsize) / d->ll.secsize;
270 1.36 jakllsch if (readsects(&d->ll, entblk, sectors, d->buf, 1))
271 1.36 jakllsch return -1;
272 1.36 jakllsch entblk += sectors;
273 1.36 jakllsch crc = crc32(crc, (const void *)d->buf, size);
274 1.36 jakllsch
275 1.36 jakllsch for (i = 0; j < BIOSDISKNPART && i < entries; i++, j++) {
276 1.36 jakllsch u = (const struct uuid *)ep[i].ent_type;
277 1.36 jakllsch if (!guid_is_nil(u)) {
278 1.36 jakllsch d->part[j].offset = ep[i].ent_lba_start;
279 1.36 jakllsch d->part[j].size = ep[i].ent_lba_end -
280 1.36 jakllsch ep[i].ent_lba_start + 1;
281 1.36 jakllsch if (guid_is_equal(u, &GET_nbsd_ffs))
282 1.36 jakllsch d->part[j].fstype = FS_BSDFFS;
283 1.36 jakllsch else if (guid_is_equal(u, &GET_nbsd_lfs))
284 1.36 jakllsch d->part[j].fstype = FS_BSDLFS;
285 1.36 jakllsch else if (guid_is_equal(u, &GET_nbsd_raid))
286 1.36 jakllsch d->part[j].fstype = FS_RAID;
287 1.36 jakllsch else if (guid_is_equal(u, &GET_nbsd_swap))
288 1.36 jakllsch d->part[j].fstype = FS_SWAP;
289 1.36 jakllsch else
290 1.36 jakllsch d->part[j].fstype = FS_OTHER;
291 1.36 jakllsch }
292 1.36 jakllsch }
293 1.36 jakllsch
294 1.36 jakllsch }
295 1.36 jakllsch
296 1.36 jakllsch if (crc != gpth.hdr_crc_table) {
297 1.36 jakllsch #ifdef DISK_DEBUG
298 1.36 jakllsch printf("GPT table CRC invalid\n");
299 1.36 jakllsch #endif
300 1.36 jakllsch return -1;
301 1.36 jakllsch }
302 1.36 jakllsch
303 1.36 jakllsch return 0;
304 1.36 jakllsch }
305 1.36 jakllsch
306 1.36 jakllsch static int
307 1.36 jakllsch read_gpt(struct biosdisk *d)
308 1.36 jakllsch {
309 1.36 jakllsch struct biosdisk_extinfo ed;
310 1.36 jakllsch daddr_t gptsector[2];
311 1.36 jakllsch int i, error;
312 1.36 jakllsch
313 1.36 jakllsch gptsector[0] = GPT_HDR_BLKNO;
314 1.36 jakllsch if (set_geometry(&d->ll, &ed) == 0 && d->ll.flags & BIOSDISK_INT13EXT) {
315 1.36 jakllsch gptsector[1] = ed.totsec - 1;
316 1.36 jakllsch d->ll.secsize = ed.sbytes;
317 1.36 jakllsch } else {
318 1.36 jakllsch #ifdef DISK_DEBUG
319 1.36 jakllsch printf("Unable to determine extended disk geometry - "
320 1.36 jakllsch "using CHS\n");
321 1.36 jakllsch #endif
322 1.36 jakllsch /* at least try some other reasonable values then */
323 1.36 jakllsch gptsector[1] = d->ll.chs_sectors - 1;
324 1.36 jakllsch }
325 1.36 jakllsch
326 1.36 jakllsch /*
327 1.36 jakllsch * Use any valid GPT available, do not require both GPTs to be valid
328 1.36 jakllsch */
329 1.36 jakllsch for (i = 0; i < __arraycount(gptsector); i++) {
330 1.36 jakllsch error = check_gpt(d, gptsector[i]);
331 1.36 jakllsch if (error == 0)
332 1.36 jakllsch break;
333 1.36 jakllsch }
334 1.36 jakllsch
335 1.36 jakllsch if (i >= __arraycount(gptsector)) {
336 1.36 jakllsch memset(d->part, 0, sizeof(d->part));
337 1.36 jakllsch return -1;
338 1.36 jakllsch }
339 1.36 jakllsch
340 1.36 jakllsch #ifdef DISK_DEBUG
341 1.36 jakllsch printf("using %s GPT\n", (i == 0) ? "primary" : "secondary");
342 1.36 jakllsch #endif
343 1.36 jakllsch return 0;
344 1.36 jakllsch }
345 1.36 jakllsch #endif /* !NO_GPT */
346 1.36 jakllsch
347 1.16 dsl #ifndef NO_DISKLABEL
348 1.16 dsl static int
349 1.32 jakllsch check_label(struct biosdisk *d, daddr_t sector)
350 1.17 dsl {
351 1.17 dsl struct disklabel *lp;
352 1.36 jakllsch int part;
353 1.17 dsl
354 1.17 dsl /* find partition in NetBSD disklabel */
355 1.17 dsl if (readsects(&d->ll, sector + LABELSECTOR, 1, d->buf, 0)) {
356 1.17 dsl #ifdef DISK_DEBUG
357 1.17 dsl printf("Error reading disklabel\n");
358 1.17 dsl #endif
359 1.17 dsl return EIO;
360 1.17 dsl }
361 1.17 dsl lp = (struct disklabel *) (d->buf + LABELOFFSET);
362 1.17 dsl if (lp->d_magic != DISKMAGIC || dkcksum(lp)) {
363 1.17 dsl #ifdef DISK_DEBUG
364 1.33 jakllsch printf("warning: no disklabel in sector %"PRId64"\n", sector);
365 1.17 dsl #endif
366 1.17 dsl return -1;
367 1.17 dsl }
368 1.17 dsl
369 1.36 jakllsch memset(d->part, 0, sizeof(d->part));
370 1.36 jakllsch for (part = 0; part < lp->d_npartitions; part++) {
371 1.36 jakllsch if (lp->d_partitions[part].p_size == 0)
372 1.36 jakllsch continue;
373 1.36 jakllsch if (lp->d_partitions[part].p_fstype == FS_UNUSED)
374 1.36 jakllsch continue;
375 1.36 jakllsch d->part[part].fstype = lp->d_partitions[part].p_fstype;
376 1.36 jakllsch d->part[part].offset = lp->d_partitions[part].p_offset;
377 1.36 jakllsch d->part[part].size = lp->d_partitions[part].p_size;
378 1.36 jakllsch }
379 1.36 jakllsch
380 1.17 dsl d->boff = sector;
381 1.36 jakllsch
382 1.36 jakllsch #ifdef _STANDALONE
383 1.36 jakllsch bi_disk.labelsector = d->boff + LABELSECTOR;
384 1.36 jakllsch bi_disk.label.type = lp->d_type;
385 1.36 jakllsch memcpy(bi_disk.label.packname, lp->d_packname, 16);
386 1.36 jakllsch bi_disk.label.checksum = lp->d_checksum;
387 1.36 jakllsch
388 1.36 jakllsch bi_wedge.matchblk = d->boff + LABELSECTOR;
389 1.36 jakllsch bi_wedge.matchnblks = 1;
390 1.36 jakllsch
391 1.36 jakllsch md5(bi_wedge.matchhash, d->buf, d->ll.secsize);
392 1.36 jakllsch #endif
393 1.36 jakllsch
394 1.17 dsl return 0;
395 1.17 dsl }
396 1.17 dsl
397 1.17 dsl static int
398 1.16 dsl read_label(struct biosdisk *d)
399 1.16 dsl {
400 1.17 dsl struct disklabel dflt_lbl;
401 1.18 lukem struct mbr_partition mbr[MBR_PART_COUNT];
402 1.17 dsl struct partition *p;
403 1.16 dsl int sector, i;
404 1.17 dsl int error;
405 1.17 dsl int typ;
406 1.17 dsl int ext_base, this_ext, next_ext;
407 1.17 dsl #ifdef COMPAT_386BSD_MBRPART
408 1.17 dsl int sector_386bsd = -1;
409 1.17 dsl #endif
410 1.17 dsl
411 1.23 junyoung memset(&dflt_lbl, 0, sizeof(dflt_lbl));
412 1.17 dsl dflt_lbl.d_npartitions = 8;
413 1.5 drochner
414 1.7 drochner d->boff = 0;
415 1.7 drochner
416 1.24 junyoung if (d->ll.type != BIOSDISK_TYPE_HD)
417 1.24 junyoung /* No label on floppy and CD */
418 1.16 dsl return -1;
419 1.7 drochner
420 1.3 thorpej /*
421 1.7 drochner * find NetBSD Partition in DOS partition table
422 1.7 drochner * XXX check magic???
423 1.3 thorpej */
424 1.17 dsl ext_base = 0;
425 1.17 dsl next_ext = 0;
426 1.17 dsl for (;;) {
427 1.17 dsl this_ext = ext_base + next_ext;
428 1.17 dsl next_ext = 0;
429 1.17 dsl if (readsects(&d->ll, this_ext, 1, d->buf, 0)) {
430 1.1 perry #ifdef DISK_DEBUG
431 1.17 dsl printf("error reading MBR sector %d\n", this_ext);
432 1.17 dsl #endif
433 1.17 dsl return EIO;
434 1.17 dsl }
435 1.23 junyoung memcpy(&mbr, ((struct mbr_sector *)d->buf)->mbr_parts,
436 1.23 junyoung sizeof(mbr));
437 1.17 dsl /* Look for NetBSD partition ID */
438 1.18 lukem for (i = 0; i < MBR_PART_COUNT; i++) {
439 1.18 lukem typ = mbr[i].mbrp_type;
440 1.17 dsl if (typ == 0)
441 1.17 dsl continue;
442 1.17 dsl sector = this_ext + mbr[i].mbrp_start;
443 1.27 dsl #ifdef DISK_DEBUG
444 1.34 jakllsch printf("ptn type %d in sector %d\n", typ, sector);
445 1.27 dsl #endif
446 1.17 dsl if (typ == MBR_PTYPE_NETBSD) {
447 1.17 dsl error = check_label(d, sector);
448 1.17 dsl if (error >= 0)
449 1.17 dsl return error;
450 1.17 dsl }
451 1.17 dsl if (MBR_IS_EXTENDED(typ)) {
452 1.17 dsl next_ext = mbr[i].mbrp_start;
453 1.17 dsl continue;
454 1.17 dsl }
455 1.17 dsl #ifdef COMPAT_386BSD_MBRPART
456 1.17 dsl if (this_ext == 0 && typ == MBR_PTYPE_386BSD)
457 1.17 dsl sector_386bsd = sector;
458 1.1 perry #endif
459 1.17 dsl if (this_ext != 0) {
460 1.17 dsl if (dflt_lbl.d_npartitions >= MAXPARTITIONS)
461 1.17 dsl continue;
462 1.17 dsl p = &dflt_lbl.d_partitions[dflt_lbl.d_npartitions++];
463 1.17 dsl } else
464 1.17 dsl p = &dflt_lbl.d_partitions[i];
465 1.17 dsl p->p_offset = sector;
466 1.17 dsl p->p_size = mbr[i].mbrp_size;
467 1.17 dsl p->p_fstype = xlat_mbr_fstype(typ);
468 1.17 dsl }
469 1.17 dsl if (next_ext == 0)
470 1.3 thorpej break;
471 1.17 dsl if (ext_base == 0) {
472 1.17 dsl ext_base = next_ext;
473 1.17 dsl next_ext = 0;
474 1.3 thorpej }
475 1.17 dsl }
476 1.17 dsl
477 1.17 dsl sector = 0;
478 1.8 drochner #ifdef COMPAT_386BSD_MBRPART
479 1.17 dsl if (sector_386bsd != -1) {
480 1.17 dsl printf("old BSD partition ID!\n");
481 1.17 dsl sector = sector_386bsd;
482 1.8 drochner }
483 1.8 drochner #endif
484 1.7 drochner
485 1.17 dsl /*
486 1.17 dsl * One of two things:
487 1.17 dsl * 1. no MBR
488 1.17 dsl * 2. no NetBSD partition in MBR
489 1.17 dsl *
490 1.17 dsl * We simply default to "start of disk" in this case and
491 1.17 dsl * press on.
492 1.17 dsl */
493 1.17 dsl error = check_label(d, sector);
494 1.17 dsl if (error >= 0)
495 1.17 dsl return error;
496 1.16 dsl
497 1.17 dsl /*
498 1.17 dsl * Nothing at start of disk, return info from mbr partitions.
499 1.17 dsl */
500 1.17 dsl /* XXX fill it to make checksum match kernel one */
501 1.17 dsl dflt_lbl.d_checksum = dkcksum(&dflt_lbl);
502 1.23 junyoung memcpy(d->buf, &dflt_lbl, sizeof(dflt_lbl));
503 1.27 dsl return 0;
504 1.16 dsl }
505 1.16 dsl #endif /* NO_DISKLABEL */
506 1.16 dsl
507 1.36 jakllsch #if !defined(NO_DISKLABEL) || !defined(NO_GPT)
508 1.36 jakllsch static int
509 1.36 jakllsch read_partitions(struct biosdisk *d)
510 1.36 jakllsch {
511 1.36 jakllsch int error;
512 1.36 jakllsch
513 1.36 jakllsch error = -1;
514 1.36 jakllsch
515 1.36 jakllsch #ifndef NO_GPT
516 1.36 jakllsch error = read_gpt(d);
517 1.36 jakllsch if (error == 0)
518 1.36 jakllsch return 0;
519 1.36 jakllsch
520 1.36 jakllsch #endif
521 1.36 jakllsch #ifndef NO_DISKLABEL
522 1.36 jakllsch error = read_label(d);
523 1.36 jakllsch
524 1.36 jakllsch #endif
525 1.36 jakllsch return error;
526 1.36 jakllsch }
527 1.36 jakllsch #endif
528 1.36 jakllsch
529 1.29 jmcneill void
530 1.29 jmcneill biosdisk_probe(void)
531 1.29 jmcneill {
532 1.29 jmcneill struct biosdisk d;
533 1.29 jmcneill struct biosdisk_extinfo ed;
534 1.29 jmcneill uint64_t size;
535 1.36 jakllsch int first;
536 1.29 jmcneill int i;
537 1.36 jakllsch #if !defined(NO_DISKLABEL) || !defined(NO_GPT)
538 1.36 jakllsch int part;
539 1.36 jakllsch #endif
540 1.29 jmcneill
541 1.29 jmcneill for (i = 0; i < MAX_BIOSDISKS + 2; i++) {
542 1.29 jmcneill first = 1;
543 1.29 jmcneill memset(&d, 0, sizeof(d));
544 1.29 jmcneill memset(&ed, 0, sizeof(ed));
545 1.29 jmcneill if (i >= MAX_BIOSDISKS)
546 1.29 jmcneill d.ll.dev = 0x00 + i - MAX_BIOSDISKS; /* fd */
547 1.29 jmcneill else
548 1.29 jmcneill d.ll.dev = 0x80 + i; /* hd/cd */
549 1.29 jmcneill if (set_geometry(&d.ll, &ed))
550 1.29 jmcneill continue;
551 1.30 jmcneill printf("disk ");
552 1.29 jmcneill switch (d.ll.type) {
553 1.29 jmcneill case BIOSDISK_TYPE_CD:
554 1.30 jmcneill printf("cd0\n cd0a\n");
555 1.29 jmcneill break;
556 1.29 jmcneill case BIOSDISK_TYPE_FD:
557 1.30 jmcneill printf("fd%d\n", d.ll.dev & 0x7f);
558 1.30 jmcneill printf(" fd%da\n", d.ll.dev & 0x7f);
559 1.29 jmcneill break;
560 1.29 jmcneill case BIOSDISK_TYPE_HD:
561 1.30 jmcneill printf("hd%d", d.ll.dev & 0x7f);
562 1.29 jmcneill if (d.ll.flags & BIOSDISK_INT13EXT) {
563 1.29 jmcneill printf(" size ");
564 1.29 jmcneill size = ed.totsec * ed.sbytes;
565 1.29 jmcneill if (size >= (10ULL * 1024 * 1024 * 1024))
566 1.33 jakllsch printf("%"PRIu64" GB",
567 1.29 jmcneill size / (1024 * 1024 * 1024));
568 1.29 jmcneill else
569 1.33 jakllsch printf("%"PRIu64" MB",
570 1.29 jmcneill size / (1024 * 1024));
571 1.29 jmcneill }
572 1.29 jmcneill printf("\n");
573 1.29 jmcneill break;
574 1.29 jmcneill }
575 1.36 jakllsch #if !defined(NO_DISKLABEL) || !defined(NO_GPT)
576 1.30 jmcneill if (d.ll.type != BIOSDISK_TYPE_HD)
577 1.30 jmcneill continue;
578 1.36 jakllsch
579 1.36 jakllsch if (read_partitions(&d) != 0)
580 1.36 jakllsch continue;
581 1.36 jakllsch
582 1.36 jakllsch for (part = 0; part < BIOSDISKNPART; part++) {
583 1.36 jakllsch if (d.part[part].size == 0)
584 1.29 jmcneill continue;
585 1.36 jakllsch if (d.part[part].fstype == FS_UNUSED)
586 1.29 jmcneill continue;
587 1.29 jmcneill if (first) {
588 1.29 jmcneill printf(" ");
589 1.29 jmcneill first = 0;
590 1.29 jmcneill }
591 1.29 jmcneill printf(" hd%d%c(", d.ll.dev & 0x7f, part + 'a');
592 1.36 jakllsch if (d.part[part].fstype < FSMAXTYPES)
593 1.29 jmcneill printf("%s",
594 1.36 jakllsch fstypenames[d.part[part].fstype]);
595 1.29 jmcneill else
596 1.36 jakllsch printf("%d", d.part[part].fstype);
597 1.29 jmcneill printf(")");
598 1.29 jmcneill }
599 1.36 jakllsch #endif
600 1.29 jmcneill if (first == 0)
601 1.29 jmcneill printf("\n");
602 1.29 jmcneill }
603 1.29 jmcneill }
604 1.29 jmcneill
605 1.16 dsl /* Determine likely partition for possible sector number of dos
606 1.17 dsl * partition.
607 1.17 dsl */
608 1.16 dsl
609 1.24 junyoung int
610 1.32 jakllsch biosdisk_findpartition(int biosdev, daddr_t sector)
611 1.16 dsl {
612 1.36 jakllsch #if defined(NO_DISKLABEL) && defined(NO_GPT)
613 1.16 dsl return 0;
614 1.16 dsl #else
615 1.16 dsl struct biosdisk *d;
616 1.24 junyoung int partition = 0;
617 1.27 dsl #ifdef DISK_DEBUG
618 1.33 jakllsch printf("looking for partition device %x, sector %"PRId64"\n", biosdev, sector);
619 1.27 dsl #endif
620 1.16 dsl
621 1.16 dsl /* Look for netbsd partition that is the dos boot one */
622 1.16 dsl d = alloc_biosdisk(biosdev);
623 1.17 dsl if (d == NULL)
624 1.17 dsl return 0;
625 1.17 dsl
626 1.36 jakllsch if (read_partitions(d) == 0) {
627 1.36 jakllsch for (partition = (BIOSDISKNPART-1); --partition;) {
628 1.36 jakllsch if (d->part[partition].fstype == FS_UNUSED)
629 1.16 dsl continue;
630 1.36 jakllsch if (d->part[partition].offset == sector)
631 1.16 dsl break;
632 1.16 dsl }
633 1.16 dsl }
634 1.16 dsl
635 1.26 christos dealloc(d, sizeof(*d));
636 1.16 dsl return partition;
637 1.36 jakllsch #endif /* NO_DISKLABEL && NO_GPT */
638 1.36 jakllsch }
639 1.36 jakllsch
640 1.36 jakllsch #ifdef _STANDALONE
641 1.36 jakllsch static void
642 1.36 jakllsch add_biosdisk_bootinfo(void)
643 1.36 jakllsch {
644 1.36 jakllsch static bool done;
645 1.36 jakllsch
646 1.36 jakllsch if (bootinfo == NULL) {
647 1.36 jakllsch done = false;
648 1.36 jakllsch return;
649 1.36 jakllsch }
650 1.36 jakllsch
651 1.36 jakllsch if (done)
652 1.36 jakllsch return;
653 1.36 jakllsch
654 1.36 jakllsch BI_ADD(&bi_disk, BTINFO_BOOTDISK, sizeof(bi_disk));
655 1.36 jakllsch BI_ADD(&bi_wedge, BTINFO_BOOTWEDGE, sizeof(bi_wedge));
656 1.36 jakllsch
657 1.36 jakllsch done = true;
658 1.36 jakllsch
659 1.36 jakllsch return;
660 1.16 dsl }
661 1.16 dsl
662 1.36 jakllsch #endif
663 1.36 jakllsch
664 1.22 junyoung int
665 1.24 junyoung biosdisk_open(struct open_file *f, ...)
666 1.23 junyoung /* struct open_file *f, int biosdev, int partition */
667 1.16 dsl {
668 1.16 dsl va_list ap;
669 1.16 dsl struct biosdisk *d;
670 1.23 junyoung int biosdev;
671 1.16 dsl int partition;
672 1.16 dsl int error = 0;
673 1.16 dsl
674 1.16 dsl va_start(ap, f);
675 1.23 junyoung biosdev = va_arg(ap, int);
676 1.23 junyoung d = alloc_biosdisk(biosdev);
677 1.23 junyoung if (d == NULL) {
678 1.16 dsl error = ENXIO;
679 1.16 dsl goto out;
680 1.16 dsl }
681 1.16 dsl
682 1.16 dsl partition = va_arg(ap, int);
683 1.16 dsl #ifdef _STANDALONE
684 1.16 dsl bi_disk.biosdev = d->ll.dev;
685 1.16 dsl bi_disk.partition = partition;
686 1.16 dsl bi_disk.labelsector = -1;
687 1.21 thorpej
688 1.21 thorpej bi_wedge.biosdev = d->ll.dev;
689 1.21 thorpej bi_wedge.matchblk = -1;
690 1.16 dsl #endif
691 1.16 dsl
692 1.36 jakllsch #if !defined(NO_DISKLABEL) || !defined(NO_GPT)
693 1.36 jakllsch error = read_partitions(d);
694 1.16 dsl if (error == -1) {
695 1.16 dsl error = 0;
696 1.16 dsl goto nolabel;
697 1.16 dsl }
698 1.16 dsl if (error)
699 1.16 dsl goto out;
700 1.16 dsl
701 1.36 jakllsch if (partition >= BIOSDISKNPART ||
702 1.36 jakllsch d->part[partition].fstype == FS_UNUSED) {
703 1.1 perry #ifdef DISK_DEBUG
704 1.3 thorpej printf("illegal partition\n");
705 1.1 perry #endif
706 1.3 thorpej error = EPART;
707 1.3 thorpej goto out;
708 1.16 dsl }
709 1.21 thorpej
710 1.36 jakllsch d->boff = d->part[partition].offset;
711 1.36 jakllsch
712 1.36 jakllsch if (d->part[partition].fstype == FS_RAID)
713 1.36 jakllsch d->boff += RF_PROTECTED_SECTORS;
714 1.21 thorpej
715 1.36 jakllsch #ifdef _STANDALONE
716 1.36 jakllsch bi_wedge.startblk = d->part[partition].offset;
717 1.36 jakllsch bi_wedge.nblks = d->part[partition].size;
718 1.16 dsl #endif
719 1.36 jakllsch
720 1.7 drochner nolabel:
721 1.36 jakllsch #endif
722 1.1 perry #ifdef DISK_DEBUG
723 1.34 jakllsch printf("partition @%"PRId64"\n", d->boff);
724 1.1 perry #endif
725 1.1 perry
726 1.9 drochner #ifdef _STANDALONE
727 1.36 jakllsch add_biosdisk_bootinfo();
728 1.9 drochner #endif
729 1.5 drochner
730 1.3 thorpej f->f_devdata = d;
731 1.1 perry out:
732 1.5 drochner va_end(ap);
733 1.3 thorpej if (error)
734 1.35 jakllsch dealloc(d, sizeof(*d));
735 1.22 junyoung return error;
736 1.1 perry }
737 1.1 perry
738 1.12 drochner #ifndef LIBSA_NO_FS_CLOSE
739 1.22 junyoung int
740 1.24 junyoung biosdisk_close(struct open_file *f)
741 1.1 perry {
742 1.3 thorpej struct biosdisk *d = f->f_devdata;
743 1.1 perry
744 1.24 junyoung /* let the floppy drive go off */
745 1.24 junyoung if (d->ll.type == BIOSDISK_TYPE_FD)
746 1.31 tsutsui wait_sec(3); /* 2s is enough on all PCs I found */
747 1.1 perry
748 1.35 jakllsch dealloc(d, sizeof(*d));
749 1.3 thorpej f->f_devdata = NULL;
750 1.22 junyoung return 0;
751 1.1 perry }
752 1.12 drochner #endif
753 1.1 perry
754 1.22 junyoung int
755 1.24 junyoung biosdisk_ioctl(struct open_file *f, u_long cmd, void *arg)
756 1.1 perry {
757 1.3 thorpej return EIO;
758 1.1 perry }
759