Subject: Re: Adding cd9660 support to the i386/amd64 BIOS bootloader
To: Jason Thorpe <thorpej@shagadelic.org>
From: Bang Jun-Young <junyoung@netbsd.org>
List: port-amd64
Date: 06/22/2005 03:56:52
--Boundary-00=_0LGuCKqbKJ7jqVu
Content-Type: text/plain;
charset="iso-8859-1"
Content-Transfer-Encoding: 7bit
Content-Disposition: inline
Bang Jun-Young wrote:
> I'll look into the sources.
A new patch is available (attached), this time without MI changes. Tested
with QEMU.
Jun-Young
--Boundary-00=_0LGuCKqbKJ7jqVu
Content-Type: text/x-diff;
charset="iso-8859-1";
name="cdboot-20050622a.diff"
Content-Transfer-Encoding: 7bit
Content-Disposition: attachment;
filename="cdboot-20050622a.diff"
Index: sys/arch/i386/stand/boot/Makefile.boot
===================================================================
RCS file: /cvsroot/src/sys/arch/i386/stand/boot/Makefile.boot,v
retrieving revision 1.22
diff -u -r1.22 Makefile.boot
--- sys/arch/i386/stand/boot/Makefile.boot 21 Jun 2005 18:25:14 -0000 1.22
+++ sys/arch/i386/stand/boot/Makefile.boot 21 Jun 2005 18:37:28 -0000
@@ -62,6 +62,7 @@
CPPFLAGS+= -DCONSADDR=boot_params.bp_consaddr
CPPFLAGS+= -DCONSOLE_KEYMAP=boot_params.bp_keymap
+CPPFLAGS+= -DSUPPORT_CD9660
CPPFLAGS+= -DSUPPORT_USTARFS
CPPFLAGS+= -DSUPPORT_DOSFS
CPPFLAGS+= -DPASS_BIOSGEOM
Index: sys/arch/i386/stand/boot/boot2.c
===================================================================
RCS file: /cvsroot/src/sys/arch/i386/stand/boot/boot2.c,v
retrieving revision 1.10
diff -u -r1.10 boot2.c
--- sys/arch/i386/stand/boot/boot2.c 21 Jun 2005 14:20:35 -0000 1.10
+++ sys/arch/i386/stand/boot/boot2.c 21 Jun 2005 18:37:30 -0000
@@ -1,4 +1,4 @@
-/* $NetBSD: boot2.c,v 1.10 2005/06/21 14:20:35 junyoung Exp $ */
+/* $NetBSD: boot2.c,v 1.7 2005/06/15 19:07:36 junyoung Exp $ */
/*
* Copyright (c) 2003
@@ -56,15 +56,16 @@
#include <biosmca.h>
#endif
-int errno;
-extern int boot_biosdev;
-extern int boot_biossector; /* may be wrong... */
-
extern struct x86_boot_params boot_params;
extern const char bootprog_name[], bootprog_rev[], bootprog_date[],
bootprog_maker[];
+int errno;
+
+int boot_biosdev;
+u_int boot_biossector;
+
static const char * const names[][2] = {
{ "netbsd", "netbsd.gz" },
{ "onetbsd", "onetbsd.gz" },
@@ -83,7 +84,7 @@
char *sprint_bootsel(const char *);
void bootit(const char *, int, int);
void print_banner(void);
-void boot2(u_int, u_int);
+void boot2(int, u_int);
void command_help(char *);
void command_ls(char *);
@@ -105,11 +106,12 @@
int
parsebootfile(const char *fname, char **fsname, char **devname,
- u_int *unit, u_int *partition, const char **file)
+ int *unit, int *partition, const char **file)
{
const char *col;
- *fsname = "ufs";
+ if (fsname != NULL)
+ *fsname = "ufs";
*devname = default_devname;
*unit = default_unit;
*partition = default_partition;
@@ -121,7 +123,7 @@
if ((col = strchr(fname, ':')) != NULL) { /* device given */
static char savedevname[MAXDEVNAME+1];
int devlen;
- u_int u = 0, p = 0;
+ int u = 0, p = 0;
int i = 0;
devlen = col - fname;
@@ -172,12 +174,12 @@
char *
sprint_bootsel(const char *filename)
{
- char *fsname, *devname;
+ char *devname;
int unit, partition;
const char *file;
static char buf[80];
- if (parsebootfile(filename, &fsname, &devname, &unit,
+ if (parsebootfile(filename, NULL, &devname, &unit,
&partition, &file) == 0) {
sprintf(buf, "%s%d%c:%s", devname, unit, 'a' + partition, file);
return buf;
@@ -213,8 +215,14 @@
printf(">> Memory: %d/%d k\n", getbasemem(), getextmem());
}
+/*
+ * Called from the initial entry point boot_start in biosboot.S
+ *
+ * biosdev: BIOS drive number the system booted from
+ * biossector: Sector number of the NetBSD partition
+ */
void
-boot2(u_int boot_biosdev, u_int boot_biossector)
+boot2(int biosdev, u_int biossector)
{
int currname;
char c;
@@ -231,9 +239,13 @@
print_banner();
+ /* need to remember these */
+ boot_biosdev = biosdev;
+ boot_biossector = biossector;
+
/* try to set default device to what BIOS tells us */
- bios2dev(boot_biosdev, &default_devname, &default_unit,
- boot_biossector, &default_partition);
+ bios2dev(biosdev, biossector, &default_devname, &default_unit,
+ &default_partition);
/* if the user types "boot" without filename */
default_filename = DEFFILENAME;
@@ -315,7 +327,7 @@
command_dev(char *arg)
{
static char savedevname[MAXDEVNAME + 1];
- char *fsname, *devname;
+ char *devname;
const char *file; /* dummy */
if (*arg == '\0') {
@@ -325,7 +337,7 @@
}
if (strchr(arg, ':') != NULL ||
- parsebootfile(arg, &fsname, &devname, &default_unit,
+ parsebootfile(arg, NULL, &devname, &default_unit,
&default_partition, &file)) {
command_help(NULL);
return;
Index: sys/arch/i386/stand/boot/conf.c
===================================================================
RCS file: /cvsroot/src/sys/arch/i386/stand/boot/conf.c,v
retrieving revision 1.2
diff -u -r1.2 conf.c
--- sys/arch/i386/stand/boot/conf.c 24 Mar 2004 16:54:18 -0000 1.2
+++ sys/arch/i386/stand/boot/conf.c 21 Jun 2005 18:37:30 -0000
@@ -39,6 +39,9 @@
#ifdef SUPPORT_DOSFS
#include <lib/libsa/dosfs.h>
#endif
+#ifdef SUPPORT_CD9660
+#include <lib/libsa/cd9660.h>
+#endif
#include <biosdisk.h>
@@ -48,6 +51,9 @@
int ndevs = sizeof(devsw) / sizeof(struct devsw);
struct fs_ops file_system[] = {
+#ifdef SUPPORT_CD9660
+ FS_OPS(cd9660),
+#endif
#ifdef SUPPORT_USTARFS
FS_OPS(ustarfs),
#endif
Index: sys/arch/i386/stand/boot/devopen.c
===================================================================
RCS file: /cvsroot/src/sys/arch/i386/stand/boot/devopen.c,v
retrieving revision 1.3
diff -u -r1.3 devopen.c
--- sys/arch/i386/stand/boot/devopen.c 15 Jun 2005 19:01:19 -0000 1.3
+++ sys/arch/i386/stand/boot/devopen.c 21 Jun 2005 18:37:30 -0000
@@ -1,5 +1,41 @@
/* $NetBSD: devopen.c,v 1.3 2005/06/15 19:01:19 junyoung Exp $ */
+/*-
+ * Copyright (c) 2005 The NetBSD Foundation, Inc.
+ * All rights reserved.
+ *
+ * This code is derived from software contributed to The NetBSD Foundation
+ * by Bang Jun-Young.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by the NetBSD
+ * Foundation, Inc. and its contributors.
+ * 4. Neither the name of The NetBSD Foundation nor the names of its
+ * contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
+ * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
+ * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
+ * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+
/*
* Copyright (c) 1996, 1997
* Matthias Drochner. All rights reserved.
@@ -41,33 +77,46 @@
#include <biosmca.h>
#endif
-static int dev2bios(char *, u_int, int *);
+static int dev2bios(char *, int, int *);
static int
-dev2bios(char *devname, u_int unit, int *biosdev)
+dev2bios(char *devname, int unit, int *biosdev)
{
+
if (strcmp(devname, "hd") == 0)
*biosdev = 0x80 + unit;
else if (strcmp(devname, "fd") == 0)
*biosdev = 0x00 + unit;
+ else if (strcmp(devname, "cd") == 0)
+ *biosdev = boot_biosdev;
else
return ENXIO;
return 0;
}
-int
-bios2dev(int biosdev, char **devname, u_int *unit, u_int sector, u_int *ptnp)
+void
+bios2dev(int biosdev, u_int sector, char **devname, int *unit, int *partition)
{
- if (biosdev & 0x80)
- *devname = "hd";
- else
- *devname = "fd";
+ /* set default */
*unit = biosdev & 0x7f;
- *ptnp = biosdiskfindptn(biosdev, sector);
- return 0;
+ if (biosdev & 0x80) {
+ /*
+ * There seems to be no standard way of numbering BIOS
+ * CD-ROM drives. The following method is a little tricky
+ * but works nicely.
+ */
+ if (biosdev >= 0x80 + get_harddrives()) {
+ *devname = "cd";
+ *unit = 0; /* override default */
+ } else
+ *devname = "hd";
+ } else
+ *devname = "fd";
+
+ *partition = biosdiskfindptn(biosdev, sector);
}
#ifdef _STANDALONE
@@ -81,7 +130,7 @@
devopen(struct open_file *f, const char *fname, char **file)
{
char *fsname, *devname;
- u_int unit, partition;
+ int unit, partition;
int biosdev;
int error;
Index: sys/arch/i386/stand/boot/devopen.h
===================================================================
RCS file: /cvsroot/src/sys/arch/i386/stand/boot/devopen.h,v
retrieving revision 1.1
diff -u -r1.1 devopen.h
--- sys/arch/i386/stand/boot/devopen.h 16 Apr 2003 22:36:14 -0000 1.1
+++ sys/arch/i386/stand/boot/devopen.h 21 Jun 2005 18:37:30 -0000
@@ -1,3 +1,5 @@
-/* $NetBSD: devopen.h,v 1.1 2003/04/16 22:36:14 dsl Exp $ */
+/* $NetBSD: devopen.h,v 1.1 2003/04/16 22:36:14 dsl Exp $ */
-int bios2dev(int, char **, u_int *, u_int, u_int *);
+extern int boot_biosdev;
+
+void bios2dev(int, u_int, char **, int *, int *);
Index: sys/arch/i386/stand/boot/version
===================================================================
RCS file: /cvsroot/src/sys/arch/i386/stand/boot/version,v
retrieving revision 1.3
diff -u -r1.3 version
--- sys/arch/i386/stand/boot/version 23 Oct 2004 17:20:04 -0000 1.3
+++ sys/arch/i386/stand/boot/version 21 Jun 2005 18:37:30 -0000
@@ -35,3 +35,4 @@
Leave space in bootxx for FAT32 BPB and MBR partition table.
Keep MBR's existing BPB and partition table when installing bootxx.
3.2: Add support for passing boot wedge information to the kernel.
+3.3: Add support for cd9660 file system.
Index: sys/arch/i386/stand/lib/Makefile
===================================================================
RCS file: /cvsroot/src/sys/arch/i386/stand/lib/Makefile,v
retrieving revision 1.22
diff -u -r1.22 Makefile
--- sys/arch/i386/stand/lib/Makefile 16 Apr 2003 22:30:40 -0000 1.22
+++ sys/arch/i386/stand/lib/Makefile 21 Jun 2005 18:37:31 -0000
@@ -18,7 +18,7 @@
SRCS= pcio.c conio.S comio.S comio_direct.c biosvideomode.S
SRCS+= getsecs.c biosgetrtc.S biosdelay.S biosreboot.S gatea20.c
-SRCS+= biosmem.S getextmemx.c biosmemx.S printmemlist.c
+SRCS+= biosmem.S getbasemem.c getextmemx.c biosmemx.S printmemlist.c
SRCS+= pread.c menuutils.c parseutils.c
SRCS+= bootinfo.c bootinfo_biosgeom.c bootinfo_memmap.c
SRCS+= startprog.S panic.c
Index: sys/arch/i386/stand/lib/biosdisk.c
===================================================================
RCS file: /cvsroot/src/sys/arch/i386/stand/lib/biosdisk.c,v
retrieving revision 1.22
diff -u -r1.22 biosdisk.c
--- sys/arch/i386/stand/lib/biosdisk.c 13 Jun 2005 11:23:28 -0000 1.22
+++ sys/arch/i386/stand/lib/biosdisk.c 21 Jun 2005 18:37:31 -0000
@@ -66,6 +66,9 @@
#include <sys/types.h>
#include <sys/disklabel.h>
#include <sys/md5.h>
+#include <sys/param.h>
+
+#include <fs/cd9660/iso.h>
#include <lib/libsa/stand.h>
#include <lib/libsa/saerrno.h>
@@ -78,7 +81,7 @@
#include "bootinfo.h"
#endif
-#define BUFSIZE (1 * BIOSDISK_SECSIZE)
+#define BUFSIZE 2048 /* must be large enough for a CD sector */
struct biosdisk {
struct biosdisk_ll ll;
@@ -93,8 +96,6 @@
#define RF_PROTECTED_SECTORS 64 /* XXX refer to <.../rf_optnames.h> */
-int boot_biossector; /* disk sector partition might have started in */
-
int
biosdiskstrategy(void *devdata, int flag, daddr_t dblk, size_t size,
void *buf, size_t *rsize)
@@ -107,24 +108,27 @@
d = (struct biosdisk *) devdata;
+ if (d->ll.type == BIOSDISK_TYPE_CD)
+ dblk = dblk * DEV_BSIZE / ISO_DEFAULT_BLOCK_SIZE;
+
dblk += d->boff;
- blks = size / BIOSDISK_SECSIZE;
+ blks = size / d->ll.secsize;
if (blks && readsects(&d->ll, dblk, blks, buf, 0)) {
if (rsize)
*rsize = 0;
return EIO;
}
- /* do we really need this? */
- frag = size % BIOSDISK_SECSIZE;
+ /* needed for CD */
+ frag = size % d->ll.secsize;
if (frag) {
if (readsects(&d->ll, dblk + blks, 1, d->buf, 0)) {
if (rsize)
- *rsize = blks * BIOSDISK_SECSIZE;
+ *rsize = blks * d->ll.secsize;
return EIO;
}
- memcpy(buf + blks * BIOSDISK_SECSIZE, d->buf, frag);
+ memcpy(buf + blks * d->ll.secsize, d->buf, frag);
}
if (rsize)
@@ -133,21 +137,21 @@
}
static struct biosdisk *
-alloc_biosdisk(int dev)
+alloc_biosdisk(int biosdev)
{
struct biosdisk *d;
- d = alloc(sizeof *d);
+ d = alloc(sizeof(*d));
if (d == NULL)
return NULL;
- memset(d, 0, sizeof *d);
+ memset(d, 0, sizeof(*d));
- d->ll.dev = dev;
+ d->ll.dev = biosdev;
if (set_geometry(&d->ll, NULL)) {
#ifdef DISK_DEBUG
printf("no geometry information\n");
#endif
- free(d, sizeof *d);
+ free(d, sizeof(*d));
return NULL;
}
return d;
@@ -192,13 +196,13 @@
int sector_386bsd = -1;
#endif
- memset(&dflt_lbl, 0, sizeof dflt_lbl);
+ memset(&dflt_lbl, 0, sizeof(dflt_lbl));
dflt_lbl.d_npartitions = 8;
d->boff = 0;
- if (!(d->ll.dev & 0x80)) /* floppy */
- /* No label on floppy */
+ if (d->ll.type != BIOSDISK_TYPE_HD)
+ /* No label on floppy or CD */
return -1;
/*
@@ -216,7 +220,8 @@
#endif
return EIO;
}
- memcpy(&mbr, ((struct mbr_sector *)d->buf)->mbr_parts, sizeof mbr);
+ memcpy(&mbr, ((struct mbr_sector *)d->buf)->mbr_parts,
+ sizeof(mbr));
/* Look for NetBSD partition ID */
for (i = 0; i < MBR_PART_COUNT; i++) {
typ = mbr[i].mbrp_type;
@@ -279,7 +284,7 @@
*/
/* XXX fill it to make checksum match kernel one */
dflt_lbl.d_checksum = dkcksum(&dflt_lbl);
- memcpy(d->buf, &dflt_lbl, sizeof dflt_lbl);
+ memcpy(d->buf, &dflt_lbl, sizeof(dflt_lbl));
return -1;
}
#endif /* NO_DISKLABEL */
@@ -288,14 +293,14 @@
* partition.
*/
-u_int
+int
biosdiskfindptn(int biosdev, u_int sector)
{
#ifdef NO_DISKLABEL
return 0;
#else
struct biosdisk *d;
- u_int partition = 0;
+ int partition = 0;
struct disklabel *lp;
/* Look for netbsd partition that is the dos boot one */
@@ -313,17 +318,18 @@
}
}
- free(d, sizeof *d);
+ free(d, sizeof(*d));
return partition;
#endif /* NO_DISKLABEL */
}
int
biosdiskopen(struct open_file *f, ...)
-/* file, biosdev, partition */
+/* struct open_file *f, int biosdev, int partition */
{
va_list ap;
struct biosdisk *d;
+ int biosdev;
int partition;
#ifndef NO_DISKLABEL
struct disklabel *lp;
@@ -331,8 +337,9 @@
int error = 0;
va_start(ap, f);
- d = alloc_biosdisk(va_arg(ap, int));
- if (!d) {
+ biosdev = va_arg(ap, int);
+ d = alloc_biosdisk(biosdev);
+ if (d == NULL) {
error = ENXIO;
goto out;
}
@@ -360,7 +367,7 @@
lp = (struct disklabel *) (d->buf + LABELOFFSET);
if (partition >= lp->d_npartitions ||
- lp->d_partitions[partition].p_fstype == FS_UNUSED) {
+ lp->d_partitions[partition].p_fstype == FS_UNUSED) {
#ifdef DISK_DEBUG
printf("illegal partition\n");
#endif
@@ -414,7 +421,8 @@
{
struct biosdisk *d = f->f_devdata;
- if (!(d->ll.dev & 0x80))/* let the floppy drive go off */
+ /* let the floppy drive go off */
+ if (d->ll.type == BIOSDISK_TYPE_FD)
delay(3000000); /* 2s is enough on all PCs I found */
free(d, sizeof(struct biosdisk));
Index: sys/arch/i386/stand/lib/biosdisk.h
===================================================================
RCS file: /cvsroot/src/sys/arch/i386/stand/lib/biosdisk.h,v
retrieving revision 1.4
diff -u -r1.4 biosdisk.h
--- sys/arch/i386/stand/lib/biosdisk.h 24 Mar 2004 16:46:27 -0000 1.4
+++ sys/arch/i386/stand/lib/biosdisk.h 21 Jun 2005 18:37:31 -0000
@@ -29,5 +29,5 @@
int biosdiskopen(struct open_file *, ...);
int biosdiskclose(struct open_file *);
int biosdiskioctl(struct open_file *, u_long, void *);
-u_int biosdiskfindptn(int, u_int);
+int biosdiskfindptn(int, u_int);
Index: sys/arch/i386/stand/lib/biosdisk_ll.c
===================================================================
RCS file: /cvsroot/src/sys/arch/i386/stand/lib/biosdisk_ll.c,v
retrieving revision 1.20
diff -u -r1.20 biosdisk_ll.c
--- sys/arch/i386/stand/lib/biosdisk_ll.c 13 Jun 2005 11:27:40 -0000 1.20
+++ sys/arch/i386/stand/lib/biosdisk_ll.c 21 Jun 2005 18:37:31 -0000
@@ -1,5 +1,41 @@
/* $NetBSD: biosdisk_ll.c,v 1.20 2005/06/13 11:27:40 junyoung Exp $ */
+/*-
+ * Copyright (c) 2005 The NetBSD Foundation, Inc.
+ * All rights reserved.
+ *
+ * This code is derived from software contributed to The NetBSD Foundation
+ * by Bang Jun-Young.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by the NetBSD
+ * Foundation, Inc. and its contributors.
+ * 4. Neither the name of The NetBSD Foundation nor the names of its
+ * contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
+ * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
+ * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
+ * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+
/*
* Copyright (c) 1996
* Matthias Drochner. All rights reserved.
@@ -50,8 +86,8 @@
/*
* we get from get_diskinfo():
- * xxxx %ch %cl %dh (registers after int13/8), ie
- * xxxx cccc Csss hhhh
+ * unused %ch %cl %dh (registers after int13/8), ie
+ * xxxxxxxx cccccccc CCssssss hhhhhhhh
*/
#define SPT(di) (((di)>>8)&0x3f)
#define HEADS(di) (((di)&0xff)+1)
@@ -65,7 +101,6 @@
set_geometry(struct biosdisk_ll *d, struct biosdisk_ext13info *ed)
{
int diskinfo;
- char buf[512];
diskinfo = get_diskinfo(d->dev);
d->sec = SPT(diskinfo);
@@ -73,11 +108,31 @@
d->cyl = CYL(diskinfo);
d->chs_sectors = d->sec * d->head * d->cyl;
+ if (d->dev >= 0x80 + get_harddrives()) {
+ d->secsize = 2048;
+ d->type = BIOSDISK_TYPE_CD;
+ } else {
+ d->secsize = 512;
+ if (d->dev & 0x80)
+ d->type = BIOSDISK_TYPE_HD;
+ else
+ d->type = BIOSDISK_TYPE_FD;
+ }
+
+ /*
+ * Some broken BIOSes such as one found on Soltek SL-75DRV2 report
+ * that they don't support int13 extension for CD-ROM drives while
+ * they actually do. As a workaround, if the boot device is a CD we
+ * assume that the extension is available. Note that only very old
+ * BIOSes don't support the extended mode, and they don't work with
+ * ATAPI CD-ROM drives, either. So there's no problem.
+ */
d->flags = 0;
- if ((d->dev & 0x80) && int13_extension(d->dev)) {
+ if (d->type == BIOSDISK_TYPE_CD ||
+ (d->type == BIOSDISK_TYPE_HD && int13_extension(d->dev))) {
d->flags |= BIOSDISK_EXT13;
if (ed != NULL) {
- ed->size = sizeof *ed;
+ ed->size = sizeof(*ed);
int13_getextinfo(d->dev, ed);
}
}
@@ -86,18 +141,16 @@
* If the drive is 2.88MB floppy drive, check that we can actually
* read sector >= 18. If not, assume 1.44MB floppy disk.
*/
- if (d->dev == 0 && SPT(diskinfo) == 36) {
+ if (d->type == BIOSDISK_TYPE_FD && SPT(diskinfo) == 36) {
+ char buf[512];
+
if (biosread(d->dev, 0, 0, 18, 1, buf)) {
d->sec = 18;
d->chs_sectors /= 2;
}
}
- /*
- * get_diskinfo assumes floppy if BIOS call fails. Check at least
- * "valid" geometry.
- */
- return (!d->sec || !d->head);
+ return 0;
}
/*
@@ -105,7 +158,6 @@
* floppies, the bootstrap has to be loaded on a 64K boundary to ensure that
* this buffer doesn't cross a 64K DMA boundary.
*/
-#define RA_SECTORS (DISKBUFSIZE / BIOSDISK_SECSIZE)
static int ra_dev;
static int ra_end;
static int ra_first;
@@ -126,23 +178,25 @@
(((d)->dev & 0x80) != 0)
#else
#define NEED_INT13EXT(d, dblk, num) \
- (((d)->dev & 0x80) != 0 && ((dblk) + (num)) >= (d)->chs_sectors)
+ (((d)->type == BIOSDISK_TYPE_CD) || \
+ ((d)->type == BIOSDISK_TYPE_HD && \
+ ((dblk) + (num)) >= (d)->chs_sectors))
#endif
static int
do_read(struct biosdisk_ll *d, daddr_t dblk, int num, char *buf)
{
- int cyl, head, sec, nsec, spc, dblk32;
- struct {
- int8_t size;
- int8_t resvd;
- int16_t cnt;
- int16_t off;
- int16_t seg;
- int64_t sec;
- } ext;
if (NEED_INT13EXT(d, dblk, num)) {
+ struct {
+ int8_t size;
+ int8_t resvd;
+ int16_t cnt;
+ int16_t off;
+ int16_t seg;
+ int64_t sec;
+ } ext;
+
if (!(d->flags & BIOSDISK_EXT13))
return -1;
ext.size = sizeof(ext);
@@ -160,6 +214,8 @@
return ext.cnt;
} else {
+ int cyl, head, sec, nsec, spc, dblk32;
+
dblk32 = (int)dblk;
spc = d->head * d->sec;
cyl = dblk32 / spc;
@@ -179,11 +235,10 @@
}
}
-/* NB if 'cold' is set below not all of the program is loaded, so
- * musn't use data segment, bss, call library functions or do
- * read-ahead.
+/*
+ * NB if 'cold' is set below not all of the program is loaded, so
+ * mustn't use data segment, bss, call library functions or do read-ahead.
*/
-
int
readsects(struct biosdisk_ll *d, daddr_t dblk, int num, char *buf, int cold)
{
@@ -209,7 +264,7 @@
} else {
/* fill read-ahead buffer */
trbuf = alloc_diskbuf(0); /* no data yet */
- maxsecs = RA_SECTORS;
+ maxsecs = DISKBUFSIZE / d->secsize;
}
while ((nsec = do_read(d, dblk, maxsecs, trbuf)) < 0) {
@@ -238,13 +293,31 @@
if (nsec > num)
nsec = num;
memcpy(buf,
- diskbufp + (dblk - ra_first) * BIOSDISK_SECSIZE,
- nsec * BIOSDISK_SECSIZE);
+ diskbufp + (dblk - ra_first) * d->secsize,
+ nsec * d->secsize);
}
- buf += nsec * BIOSDISK_SECSIZE;
+ buf += nsec * d->secsize;
num -= nsec;
dblk += nsec;
}
return 0;
}
+
+/*
+ * Return the number of hard disk drives.
+ */
+int
+get_harddrives(void)
+{
+ /*
+ * Some BIOSes are buggy so that they return incorrect number
+ * of hard drives with int13/ah=8. We read a byte at 0040:0075
+ * instead, which is known to be always correct.
+ */
+ int n = 0;
+
+ pvbcopy((void *)0x475, &n, 1);
+
+ return n;
+}
Index: sys/arch/i386/stand/lib/biosdisk_ll.h
===================================================================
RCS file: /cvsroot/src/sys/arch/i386/stand/lib/biosdisk_ll.h,v
retrieving revision 1.10
diff -u -r1.10 biosdisk_ll.h
--- sys/arch/i386/stand/lib/biosdisk_ll.h 13 Jun 2005 11:34:11 -0000 1.10
+++ sys/arch/i386/stand/lib/biosdisk_ll.h 21 Jun 2005 18:37:31 -0000
@@ -46,12 +46,18 @@
*/
struct biosdisk_ll {
int dev; /* BIOS device number */
+ int type; /* device type; see below */
int sec, head, cyl; /* geometry */
int flags; /* see below */
int chs_sectors; /* # of sectors addressable by CHS */
+ int secsize; /* bytes per sector */
};
#define BIOSDISK_EXT13 1 /* BIOS supports int13 extension */
+#define BIOSDISK_TYPE_FD 0
+#define BIOSDISK_TYPE_HD 1
+#define BIOSDISK_TYPE_CD 2
+
/*
* Version 1.x drive parameters from int13 extensions
* - should be supported by every BIOS that supports the extensions.
@@ -114,7 +120,5 @@
#define EXT13_LOCKABLE 0x0020 /* device is lockable */
#define EXT13_MAXGEOM 0x0040 /* geometry set to max; no media */
-#define BIOSDISK_SECSIZE 512
-
int set_geometry(struct biosdisk_ll *, struct biosdisk_ext13info *);
int readsects(struct biosdisk_ll *, daddr_t, int, char *, int);
Index: sys/arch/i386/stand/lib/biosmem.S
===================================================================
RCS file: /cvsroot/src/sys/arch/i386/stand/lib/biosmem.S,v
retrieving revision 1.6
diff -u -r1.6 biosmem.S
--- sys/arch/i386/stand/lib/biosmem.S 1 Feb 2003 14:48:18 -0000 1.6
+++ sys/arch/i386/stand/lib/biosmem.S 21 Jun 2005 18:37:31 -0000
@@ -31,42 +31,12 @@
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
*/
-
-#include <machine/asm.h>
-#define data32 .byte 0x66
+#include <machine/asm.h>
.text
-/* get mem below 1M, in kByte */
-
-ENTRY(getbasemem)
- pushl %ebp
- movl %esp,%ebp
- pushl %ebx
- push %esi
- push %edi
-
- call _C_LABEL(prot_to_real)
- .code16
-
- int $0x12
- # zero-extend 16-bit result to 32 bits.
- movl $0, %ebx
- mov %ax,%bx
-
- calll _C_LABEL(real_to_prot)
- .code32
-
- movl %ebx, %eax
-
- pop %edi
- pop %esi
- popl %ebx
- popl %ebp
- ret
-
/* get mem above 1M, in kByte */
ENTRY(getextmem1)
Index: sys/arch/i386/stand/lib/bootinfo_biosgeom.c
===================================================================
RCS file: /cvsroot/src/sys/arch/i386/stand/lib/bootinfo_biosgeom.c,v
retrieving revision 1.14
diff -u -r1.14 bootinfo_biosgeom.c
--- sys/arch/i386/stand/lib/bootinfo_biosgeom.c 21 Jun 2005 18:34:47 -0000 1.14
+++ sys/arch/i386/stand/lib/bootinfo_biosgeom.c 21 Jun 2005 18:37:31 -0000
@@ -1,4 +1,4 @@
-/* $NetBSD: bootinfo_biosgeom.c,v 1.14 2005/06/21 18:34:47 junyoung Exp $ */
+/* $NetBSD: bootinfo_biosgeom.c,v 1.13 2004/03/24 16:46:28 drochner Exp $ */
/*
* Copyright (c) 1997
@@ -37,6 +37,8 @@
#include "biosdisk_ll.h"
#include "bootinfo.h"
+#define BIOSDISK_SECSIZE 512 /* XXX */
+
#ifdef BIOSDISK_EXT13INFO_V3
static struct {
char *name;
@@ -61,13 +63,13 @@
{
struct btinfo_biosgeom *bibg;
int i, j, nvalid;
- int nhd = 0;
+ int nhd;
unsigned int cksum;
struct biosdisk_ll d;
struct biosdisk_ext13info ed;
char buf[BIOSDISK_SECSIZE];
- pvbcopy((void *)(0x400 + 0x75), &nhd, 1);
+ nhd = get_harddrives();
#ifdef GEOM_DEBUG
printf("nhd %d\n", nhd);
#endif
Index: sys/arch/i386/stand/lib/libi386.h
===================================================================
RCS file: /cvsroot/src/sys/arch/i386/stand/lib/libi386.h,v
retrieving revision 1.19
diff -u -r1.19 libi386.h
--- sys/arch/i386/stand/lib/libi386.h 13 Jun 2005 11:37:41 -0000 1.19
+++ sys/arch/i386/stand/lib/libi386.h 21 Jun 2005 18:37:31 -0000
@@ -68,8 +68,8 @@
char awaitkey(int, int);
/* this is in "user code"! */
-int parsebootfile(const char *, char **, char **, unsigned int *,
- unsigned int *, const char **);
+int parsebootfile(const char *, char **, char **, int *, int *,
+ const char **);
#ifdef XMS
physaddr_t ppbcopy(physaddr_t, physaddr_t, int);
@@ -119,6 +119,8 @@
int int13_extension(int);
struct biosdisk_ext13info;
void int13_getextinfo(int, struct biosdisk_ext13info *);
+int get_harddrives(void);
+
int pcibios_cfgread(unsigned int, int, int *);
int pcibios_cfgwrite(unsigned int, int, int);
int pcibios_finddev(int, int, int, unsigned int *);
--- /dev/null 2005-06-22 03:35:28.000000000 +0900
+++ sys/arch/i386/stand/lib/getbasemem.c 2005-06-20 14:14:27.000000000 +0900
@@ -0,0 +1,28 @@
+/* $NetBSD$ */
+
+/*
+ * Written by Bang Jun-Young, 2005.
+ *
+ * This file is placed in public domain.
+ */
+
+#include <lib/libsa/stand.h>
+
+#include "libi386.h"
+
+/*
+ * Return the size of the base memory in kB.
+ */
+int
+getbasemem(void)
+{
+ int memsize = 0;
+
+ /*
+ * Instead of using int12 which is reported to be buggy on some
+ * BIOSes, read a 16-bit word located in the BIOS data area.
+ */
+ pvbcopy((void *)0x413, &memsize, 2);
+
+ return memsize;
+}
--Boundary-00=_0LGuCKqbKJ7jqVu--