1 /* $NetBSD: biosdisk.c,v 1.43 2013/10/31 20:31:04 christos Exp $ */
4 * Copyright (c) 1996, 1998
5 * Matthias Drochner. All rights reserved.
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
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.
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30 * raw BIOS disk device for libsa.
31 * needs lowlevel parts from bios_disk.S and biosdisk_ll.c
32 * partly from netbsd:sys/arch/i386/boot/disk.c
35 * A lot of this must match sys/kern/subr_disk_mbr.c
39 * Ported to boot 386BSD by Julian Elischer (julian@tfs.com) Sept 1992
41 * Mach Operating System
42 * Copyright (c) 1992, 1991 Carnegie Mellon University
43 * All Rights Reserved.
45 * Permission to use, copy, modify and distribute this software and its
46 * documentation is hereby granted, provided that both the copyright
47 * notice and this permission notice appear in all copies of the
48 * software, derivative works or modified versions, and any portions
49 * thereof, and that both notices appear in supporting documentation.
51 * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
52 * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
53 * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
55 * Carnegie Mellon requests users of this software to return to
57 * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
58 * School of Computer Science
59 * Carnegie Mellon University
60 * Pittsburgh PA 15213-3890
62 * any improvements or extensions that they make and grant Carnegie Mellon
63 * the rights to redistribute these changes.
66 #if !defined(NO_DISKLABEL) || !defined(NO_GPT)
70 #include <lib/libkern/libkern.h>
71 #include <lib/libsa/stand.h>
73 #include <sys/types.h>
75 #include <sys/param.h>
76 #include <sys/disklabel.h>
77 #include <sys/disklabel_gpt.h>
80 #include <fs/cd9660/iso.h>
82 #include <lib/libsa/saerrno.h>
83 #include <machine/cpu.h>
86 #include "biosdisk_ll.h"
92 #define BUFSIZE 2048 /* must be large enough for a CD sector */
94 #define BIOSDISKNPART 26
97 struct biosdisk_ll ll
;
100 #if !defined(NO_DISKLABEL) || !defined(NO_GPT)
105 } part
[BIOSDISKNPART
];
110 const struct uuid GET_nbsd_raid
= GPT_ENT_TYPE_NETBSD_RAIDFRAME
;
111 const struct uuid GET_nbsd_ffs
= GPT_ENT_TYPE_NETBSD_FFS
;
112 const struct uuid GET_nbsd_lfs
= GPT_ENT_TYPE_NETBSD_LFS
;
113 const struct uuid GET_nbsd_swap
= GPT_ENT_TYPE_NETBSD_SWAP
;
117 static struct btinfo_bootdisk bi_disk
;
118 static struct btinfo_bootwedge bi_wedge
;
121 #define MBR_PARTS(buf) ((char *)(buf) + offsetof(struct mbr_sector, mbr_parts))
123 #define RF_PROTECTED_SECTORS 64 /* XXX refer to <.../rf_optnames.h> */
126 biosdisk_strategy(void *devdata
, int flag
, daddr_t dblk
, size_t size
,
127 void *buf
, size_t *rsize
)
135 d
= (struct biosdisk
*) devdata
;
137 if (d
->ll
.type
== BIOSDISK_TYPE_CD
)
138 dblk
= dblk
* DEV_BSIZE
/ ISO_DEFAULT_BLOCK_SIZE
;
142 blks
= size
/ d
->ll
.secsize
;
143 if (blks
&& readsects(&d
->ll
, dblk
, blks
, buf
, 0)) {
150 frag
= size
% d
->ll
.secsize
;
152 if (readsects(&d
->ll
, dblk
+ blks
, 1, d
->buf
, 0)) {
154 *rsize
= blks
* d
->ll
.secsize
;
157 memcpy(buf
+ blks
* d
->ll
.secsize
, d
->buf
, frag
);
165 static struct biosdisk
*
166 alloc_biosdisk(int biosdev
)
170 d
= alloc(sizeof(*d
));
173 memset(d
, 0, sizeof(*d
));
176 if (set_geometry(&d
->ll
, NULL
)) {
178 printf("no geometry information\n");
180 dealloc(d
, sizeof(*d
));
186 #if !defined(NO_DISKLABEL) || !defined(NO_GPT)
188 md5(void *hash
, const void *data
, size_t len
)
193 MD5Update(&ctx
, data
, len
);
194 MD5Final(hash
, &ctx
);
202 guid_is_nil(const struct uuid
*u
)
204 static const struct uuid nil
= { .time_low
= 0 };
205 return (memcmp(u
, &nil
, sizeof(*u
)) == 0 ? true : false);
209 guid_is_equal(const struct uuid
*a
, const struct uuid
*b
)
211 return (memcmp(a
, b
, sizeof(*a
)) == 0 ? true : false);
215 check_gpt(struct biosdisk
*d
, daddr_t sector
)
218 const struct gpt_ent
*ep
;
219 const struct uuid
*u
;
228 /* read in gpt_hdr sector */
229 if (readsects(&d
->ll
, sector
, 1, d
->buf
, 1)) {
231 printf("Error reading GPT header at %"PRId64
"\n", sector
);
236 memcpy(&gpth
, d
->buf
, sizeof(gpth
));
238 if (memcmp(GPT_HDR_SIG
, gpth
.hdr_sig
, sizeof(gpth
.hdr_sig
)))
241 crc
= gpth
.hdr_crc_self
;
242 gpth
.hdr_crc_self
= 0;
243 gpth
.hdr_crc_self
= crc32(0, (const void *)&gpth
, GPT_HDR_SIZE
);
244 if (gpth
.hdr_crc_self
!= crc
) {
248 if (gpth
.hdr_lba_self
!= sector
)
252 bi_wedge
.matchblk
= sector
;
253 bi_wedge
.matchnblks
= 1;
255 md5(bi_wedge
.matchhash
, d
->buf
, d
->ll
.secsize
);
258 sectors
= sizeof(d
->buf
)/d
->ll
.secsize
; /* sectors per buffer */
259 entries
= sizeof(d
->buf
)/gpth
.hdr_entsz
; /* entries per buffer */
260 entblk
= gpth
.hdr_lba_table
;
261 crc
= crc32(0, NULL
, 0);
264 ep
= (const struct gpt_ent
*)d
->buf
;
266 for (entry
= 0; entry
< gpth
.hdr_entries
; entry
+= entries
) {
267 size
= MIN(sizeof(d
->buf
),
268 (gpth
.hdr_entries
- entry
) * gpth
.hdr_entsz
);
269 entries
= size
/ gpth
.hdr_entsz
;
270 sectors
= roundup(size
, d
->ll
.secsize
) / d
->ll
.secsize
;
271 if (readsects(&d
->ll
, entblk
, sectors
, d
->buf
, 1))
274 crc
= crc32(crc
, (const void *)d
->buf
, size
);
276 for (i
= 0; j
< BIOSDISKNPART
&& i
< entries
; i
++, j
++) {
277 u
= (const struct uuid
*)ep
[i
].ent_type
;
278 if (!guid_is_nil(u
)) {
279 d
->part
[j
].offset
= ep
[i
].ent_lba_start
;
280 d
->part
[j
].size
= ep
[i
].ent_lba_end
-
281 ep
[i
].ent_lba_start
+ 1;
282 if (guid_is_equal(u
, &GET_nbsd_ffs
))
283 d
->part
[j
].fstype
= FS_BSDFFS
;
284 else if (guid_is_equal(u
, &GET_nbsd_lfs
))
285 d
->part
[j
].fstype
= FS_BSDLFS
;
286 else if (guid_is_equal(u
, &GET_nbsd_raid
))
287 d
->part
[j
].fstype
= FS_RAID
;
288 else if (guid_is_equal(u
, &GET_nbsd_swap
))
289 d
->part
[j
].fstype
= FS_SWAP
;
291 d
->part
[j
].fstype
= FS_OTHER
;
297 if (crc
!= gpth
.hdr_crc_table
) {
299 printf("GPT table CRC invalid\n");
308 read_gpt(struct biosdisk
*d
)
310 struct biosdisk_extinfo ed
;
311 daddr_t gptsector
[2];
314 if (d
->ll
.type
!= BIOSDISK_TYPE_HD
)
315 /* No GPT on floppy and CD */
318 gptsector
[0] = GPT_HDR_BLKNO
;
319 if (set_geometry(&d
->ll
, &ed
) == 0 && d
->ll
.flags
& BIOSDISK_INT13EXT
) {
320 gptsector
[1] = ed
.totsec
- 1;
321 /* Sanity check values returned from BIOS */
322 if (ed
.sbytes
>= 512 && (ed
.sbytes
& (ed
.sbytes
- 1)) == 0)
323 d
->ll
.secsize
= ed
.sbytes
;
326 printf("Unable to determine extended disk geometry - "
329 /* at least try some other reasonable values then */
330 gptsector
[1] = d
->ll
.chs_sectors
- 1;
334 * Use any valid GPT available, do not require both GPTs to be valid
336 for (i
= 0; i
< __arraycount(gptsector
); i
++) {
337 error
= check_gpt(d
, gptsector
[i
]);
342 if (i
>= __arraycount(gptsector
)) {
343 memset(d
->part
, 0, sizeof(d
->part
));
348 printf("using %s GPT\n", (i
== 0) ? "primary" : "secondary");
356 ingest_label(struct biosdisk
*d
, struct disklabel
*lp
)
360 memset(d
->part
, 0, sizeof(d
->part
));
362 for (part
= 0; part
< lp
->d_npartitions
; part
++) {
363 if (lp
->d_partitions
[part
].p_size
== 0)
365 if (lp
->d_partitions
[part
].p_fstype
== FS_UNUSED
)
367 d
->part
[part
].fstype
= lp
->d_partitions
[part
].p_fstype
;
368 d
->part
[part
].offset
= lp
->d_partitions
[part
].p_offset
;
369 d
->part
[part
].size
= lp
->d_partitions
[part
].p_size
;
374 check_label(struct biosdisk
*d
, daddr_t sector
)
376 struct disklabel
*lp
;
378 /* find partition in NetBSD disklabel */
379 if (readsects(&d
->ll
, sector
+ LABELSECTOR
, 1, d
->buf
, 0)) {
381 printf("Error reading disklabel\n");
385 lp
= (struct disklabel
*) (d
->buf
+ LABELOFFSET
);
386 if (lp
->d_magic
!= DISKMAGIC
|| dkcksum(lp
)) {
388 printf("warning: no disklabel in sector %"PRId64
"\n", sector
);
396 bi_disk
.labelsector
= sector
+ LABELSECTOR
;
397 bi_disk
.label
.type
= lp
->d_type
;
398 memcpy(bi_disk
.label
.packname
, lp
->d_packname
, 16);
399 bi_disk
.label
.checksum
= lp
->d_checksum
;
401 bi_wedge
.matchblk
= sector
+ LABELSECTOR
;
402 bi_wedge
.matchnblks
= 1;
404 md5(bi_wedge
.matchhash
, d
->buf
, d
->ll
.secsize
);
411 read_minix_subp(struct biosdisk
*d
, struct disklabel
* dflt_lbl
,
412 int this_ext
, daddr_t sector
)
414 struct mbr_partition mbr
[MBR_PART_COUNT
];
419 if (readsects(&d
->ll
, sector
, 1, d
->buf
, 0)) {
421 printf("Error reading MFS sector %"PRId64
"\n", sector
);
425 if ((uint8_t)d
->buf
[510] != 0x55 || (uint8_t)d
->buf
[511] != 0xAA) {
428 memcpy(&mbr
, MBR_PARTS(d
->buf
), sizeof(mbr
));
429 for (i
= 0; i
< MBR_PART_COUNT
; i
++) {
430 typ
= mbr
[i
].mbrp_type
;
433 sector
= this_ext
+ mbr
[i
].mbrp_start
;
434 if (dflt_lbl
->d_npartitions
>= MAXPARTITIONS
)
436 p
= &dflt_lbl
->d_partitions
[dflt_lbl
->d_npartitions
++];
437 p
->p_offset
= sector
;
438 p
->p_size
= mbr
[i
].mbrp_size
;
439 p
->p_fstype
= xlat_mbr_fstype(typ
);
445 read_label(struct biosdisk
*d
)
447 struct disklabel dflt_lbl
;
448 struct mbr_partition mbr
[MBR_PART_COUNT
];
454 uint32_t ext_base
, this_ext
, next_ext
;
455 #ifdef COMPAT_386BSD_MBRPART
456 int sector_386bsd
= -1;
459 memset(&dflt_lbl
, 0, sizeof(dflt_lbl
));
460 dflt_lbl
.d_npartitions
= 8;
464 if (d
->ll
.type
!= BIOSDISK_TYPE_HD
)
465 /* No label on floppy and CD */
469 * find NetBSD Partition in DOS partition table
475 this_ext
= ext_base
+ next_ext
;
477 if (readsects(&d
->ll
, this_ext
, 1, d
->buf
, 0)) {
479 printf("error reading MBR sector %u\n", this_ext
);
483 memcpy(&mbr
, MBR_PARTS(d
->buf
), sizeof(mbr
));
484 /* Look for NetBSD partition ID */
485 for (i
= 0; i
< MBR_PART_COUNT
; i
++) {
486 typ
= mbr
[i
].mbrp_type
;
489 sector
= this_ext
+ mbr
[i
].mbrp_start
;
491 printf("ptn type %d in sector %u\n", typ
, sector
);
493 if (typ
== MBR_PTYPE_MINIX_14B
) {
494 if (!read_minix_subp(d
, &dflt_lbl
,
496 /* Don't add "container" partition */
500 if (typ
== MBR_PTYPE_NETBSD
) {
501 error
= check_label(d
, sector
);
505 if (MBR_IS_EXTENDED(typ
)) {
506 next_ext
= mbr
[i
].mbrp_start
;
509 #ifdef COMPAT_386BSD_MBRPART
510 if (this_ext
== 0 && typ
== MBR_PTYPE_386BSD
)
511 sector_386bsd
= sector
;
514 if (dflt_lbl
.d_npartitions
>= MAXPARTITIONS
)
516 p
= &dflt_lbl
.d_partitions
[dflt_lbl
.d_npartitions
++];
518 p
= &dflt_lbl
.d_partitions
[i
];
519 p
->p_offset
= sector
;
520 p
->p_size
= mbr
[i
].mbrp_size
;
521 p
->p_fstype
= xlat_mbr_fstype(typ
);
532 #ifdef COMPAT_386BSD_MBRPART
533 if (sector_386bsd
!= -1) {
534 printf("old BSD partition ID!\n");
535 sector
= sector_386bsd
;
542 * 2. no NetBSD partition in MBR
544 * We simply default to "start of disk" in this case and
547 error
= check_label(d
, sector
);
552 * Nothing at start of disk, return info from mbr partitions.
554 /* XXX fill it to make checksum match kernel one */
555 dflt_lbl
.d_checksum
= dkcksum(&dflt_lbl
);
556 ingest_label(d
, &dflt_lbl
);
559 #endif /* NO_DISKLABEL */
561 #if !defined(NO_DISKLABEL) || !defined(NO_GPT)
563 read_partitions(struct biosdisk
*d
)
576 error
= read_label(d
);
587 struct biosdisk_extinfo ed
;
591 #if !defined(NO_DISKLABEL) || !defined(NO_GPT)
595 for (i
= 0; i
< MAX_BIOSDISKS
+ 2; i
++) {
597 memset(&d
, 0, sizeof(d
));
598 memset(&ed
, 0, sizeof(ed
));
599 if (i
>= MAX_BIOSDISKS
)
600 d
.ll
.dev
= 0x00 + i
- MAX_BIOSDISKS
; /* fd */
602 d
.ll
.dev
= 0x80 + i
; /* hd/cd */
603 if (set_geometry(&d
.ll
, &ed
))
607 case BIOSDISK_TYPE_CD
:
608 printf("cd0\n cd0a\n");
610 case BIOSDISK_TYPE_FD
:
611 printf("fd%d\n", d
.ll
.dev
& 0x7f);
612 printf(" fd%da\n", d
.ll
.dev
& 0x7f);
614 case BIOSDISK_TYPE_HD
:
615 printf("hd%d", d
.ll
.dev
& 0x7f);
616 if (d
.ll
.flags
& BIOSDISK_INT13EXT
) {
618 size
= ed
.totsec
* ed
.sbytes
;
619 if (size
>= (10ULL * 1024 * 1024 * 1024))
620 printf("%"PRIu64
" GB",
621 size
/ (1024 * 1024 * 1024));
623 printf("%"PRIu64
" MB",
624 size
/ (1024 * 1024));
629 #if !defined(NO_DISKLABEL) || !defined(NO_GPT)
630 if (d
.ll
.type
!= BIOSDISK_TYPE_HD
)
633 if (read_partitions(&d
) != 0)
636 for (part
= 0; part
< BIOSDISKNPART
; part
++) {
637 if (d
.part
[part
].size
== 0)
639 if (d
.part
[part
].fstype
== FS_UNUSED
)
645 printf(" hd%d%c(", d
.ll
.dev
& 0x7f, part
+ 'a');
646 if (d
.part
[part
].fstype
< FSMAXTYPES
)
648 fstypenames
[d
.part
[part
].fstype
]);
650 printf("%d", d
.part
[part
].fstype
);
659 /* Determine likely partition for possible sector number of dos
664 biosdisk_findpartition(int biosdev
, daddr_t sector
)
666 #if defined(NO_DISKLABEL) && defined(NO_GPT)
672 printf("looking for partition device %x, sector %"PRId64
"\n", biosdev
, sector
);
675 /* Look for netbsd partition that is the dos boot one */
676 d
= alloc_biosdisk(biosdev
);
680 if (read_partitions(d
) == 0) {
681 for (partition
= (BIOSDISKNPART
-1); --partition
;) {
682 if (d
->part
[partition
].fstype
== FS_UNUSED
)
684 if (d
->part
[partition
].offset
== sector
)
689 dealloc(d
, sizeof(*d
));
691 #endif /* NO_DISKLABEL && NO_GPT */
696 add_biosdisk_bootinfo(void)
700 if (bootinfo
== NULL
) {
708 BI_ADD(&bi_disk
, BTINFO_BOOTDISK
, sizeof(bi_disk
));
709 BI_ADD(&bi_wedge
, BTINFO_BOOTWEDGE
, sizeof(bi_wedge
));
719 biosdisk_open(struct open_file
*f
, ...)
720 /* struct open_file *f, int biosdev, int partition */
729 biosdev
= va_arg(ap
, int);
730 d
= alloc_biosdisk(biosdev
);
736 partition
= va_arg(ap
, int);
738 bi_disk
.biosdev
= d
->ll
.dev
;
739 bi_disk
.partition
= partition
;
740 bi_disk
.labelsector
= -1;
742 bi_wedge
.biosdev
= d
->ll
.dev
;
743 bi_wedge
.matchblk
= -1;
746 #if !defined(NO_DISKLABEL) || !defined(NO_GPT)
747 error
= read_partitions(d
);
755 if (partition
>= BIOSDISKNPART
||
756 d
->part
[partition
].fstype
== FS_UNUSED
) {
758 printf("illegal partition\n");
764 d
->boff
= d
->part
[partition
].offset
;
766 if (d
->part
[partition
].fstype
== FS_RAID
)
767 d
->boff
+= RF_PROTECTED_SECTORS
;
770 bi_wedge
.startblk
= d
->part
[partition
].offset
;
771 bi_wedge
.nblks
= d
->part
[partition
].size
;
777 printf("partition @%"PRId64
"\n", d
->boff
);
781 add_biosdisk_bootinfo();
788 dealloc(d
, sizeof(*d
));
792 #ifndef LIBSA_NO_FS_CLOSE
794 biosdisk_close(struct open_file
*f
)
796 struct biosdisk
*d
= f
->f_devdata
;
798 /* let the floppy drive go off */
799 if (d
->ll
.type
== BIOSDISK_TYPE_FD
)
800 wait_sec(3); /* 2s is enough on all PCs I found */
802 dealloc(d
, sizeof(*d
));
809 biosdisk_ioctl(struct open_file
*f
, u_long cmd
, void *arg
)