summaryrefslogtreecommitdiff
path: root/sys/arch/i386/stand/libsa
diff options
context:
space:
mode:
authorTobias Weingartner <weingart@cvs.openbsd.org>2003-05-31 00:15:30 +0000
committerTobias Weingartner <weingart@cvs.openbsd.org>2003-05-31 00:15:30 +0000
commitcf6ecfdc46a2e16ba53c638c6eaddeda169e1307 (patch)
tree056694a54102e3247af5777b73693bd322025686 /sys/arch/i386/stand/libsa
parente99ca4bfa2a21d1f4a6cf106d422aa7807b702ed (diff)
First lba support in /boot. Not complete lba boot support, but it's the
first step on the way there. Ok deraadt@, espie@, todd@, and others.
Diffstat (limited to 'sys/arch/i386/stand/libsa')
-rw-r--r--sys/arch/i386/stand/libsa/biosdev.c191
-rw-r--r--sys/arch/i386/stand/libsa/biosdev.h4
-rw-r--r--sys/arch/i386/stand/libsa/cmd_i386.c6
-rw-r--r--sys/arch/i386/stand/libsa/diskprobe.c11
4 files changed, 109 insertions, 103 deletions
diff --git a/sys/arch/i386/stand/libsa/biosdev.c b/sys/arch/i386/stand/libsa/biosdev.c
index 937368404d0..34521d72bc3 100644
--- a/sys/arch/i386/stand/libsa/biosdev.c
+++ b/sys/arch/i386/stand/libsa/biosdev.c
@@ -1,7 +1,8 @@
-/* $OpenBSD: biosdev.c,v 1.55 2003/04/17 03:43:18 drahn Exp $ */
+/* $OpenBSD: biosdev.c,v 1.56 2003/05/31 00:15:29 weingart Exp $ */
/*
* Copyright (c) 1996 Michael Shalayeff
+ * Copyright (c) 2003 Tobias Weingartner
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
@@ -46,6 +47,9 @@
static const char *biosdisk_err(u_int);
static int biosdisk_errno(u_int);
+static int CHS_rw __P((int, int, int, int, int, int, void *));
+static int EDD_rw __P((int, int, u_int64_t, u_int32_t, void *));
+
extern int debug;
#if 0
@@ -58,9 +62,11 @@ struct biosdisk {
struct EDD_CB {
u_int8_t edd_len; /* size of packet */
- u_int8_t edd_res; /* reserved */
- u_int16_t edd_nblk; /* # of blocks to transfer */
- u_int32_t edd_buf; /* address of buffer */
+ u_int8_t edd_res1; /* reserved */
+ u_int8_t edd_nblk; /* # of blocks to transfer */
+ u_int8_t edd_res2; /* reserved */
+ u_int16_t edd_off; /* address of buffer (offset) */
+ u_int16_t edd_seg; /* address of buffer (segment) */
u_int64_t edd_daddr; /* starting block */
};
@@ -87,10 +93,10 @@ bios_getdiskinfo(dev, pdi)
int dev;
bios_diskinfo_t *pdi;
{
- u_int rv;
+ u_int rv, secl, sech;
/* Just reset, don't check return code */
- biosdreset(dev);
+ rv = biosdreset(dev);
#ifdef BIOS_DEBUG
if (debug)
@@ -113,7 +119,7 @@ bios_getdiskinfo(dev, pdi)
}
#endif
if (rv & 0xff)
- return(1);
+ return (1);
/* Fix up info */
pdi->bios_number = dev;
@@ -121,46 +127,46 @@ bios_getdiskinfo(dev, pdi)
pdi->bios_cylinders &= 0x3ff;
pdi->bios_cylinders++;
-#if 0
+ /*
+ * NOTE: This seems to hang on certain machines. Use function #8
+ * first, and verify with #21 IFF #8 succeeds first.
+ * Do not try this for floppy 0 (to support CD-ROM boot).
+ */
+ if (dev) {
+ __asm __volatile (DOINT(0x13) ";setc %b0"
+ : "=a" (rv), "=d" (secl), "=c" (sech)
+ : "0" (0x15FF), "1" (dev), "2" (0xFFFF)
+ : "cc");
+ if (!(rv & 0xff00))
+ return (1);
+ if (rv & 0xff)
+ return (1);
+ }
+
/* NOTE:
* This currently hangs/reboots some machines
* The IBM Thinkpad 750ED for one.
*
* Funny that an IBM/MS extension would not be
* implemented by an IBM system...
+ *
+ * Future hangs (when reported) can be "fixed"
+ * with getSYSCONFaddr() and an exceptions list.
*/
if (dev & 0x80) {
int bm;
/* EDD support check */
- __asm __volatile("int $2;" DOINT(0x13) "; setc %b0"
+ __asm __volatile(DOINT(0x13) "; setc %b0"
: "=a" (rv), "=c" (bm)
- : "0" (0x4100), "2" (0x55aa), "d" (dev) : "cc");
- DUMP_REGS;
+ : "0" (0x4100), "b" (0x55aa), "d" (dev) : "cc");
if (!(rv & 0xff) && (BIOS_regs.biosr_bx & 0xffff) == 0xaa55)
- pdi->bios_edd = bm & 0xffff;
+ pdi->bios_edd = (bm & 0xffff) | ((rv & 0xff) << 16);
else
pdi->bios_edd = -1;
} else
pdi->bios_edd = -1;
-#else
- pdi->bios_edd = -1;
-#endif
- /*
- * NOTE: This seems to hang on certain machines. Use function #8
- * first, and verify with #21 IFF #8 succeeds first.
- * Do not try this for floppy 0 (to support CD-ROM boot).
- */
- if (dev) {
- __asm __volatile (DOINT(0x13) "; setc %b0"
- : "=a" (rv) : "0" (0x1500), "d" (dev) : "%ecx", "cc");
- if(!(rv & 0xff00))
- return(1);
- if(rv & 0xff)
- return(1);
- }
-
- /* XXX - Sanity check */
+ /* Sanity check */
if (!pdi->bios_cylinders || !pdi->bios_heads || !pdi->bios_sectors)
return(1);
@@ -175,7 +181,7 @@ bios_getdiskinfo(dev, pdi)
* Read/Write a block from given place using the BIOS.
*/
static __inline int
-biosd_rw(rw, dev, cyl, head, sect, nsect, buf)
+CHS_rw(rw, dev, cyl, head, sect, nsect, buf)
int rw, dev, cyl, head;
int sect, nsect;
void * buf;
@@ -201,7 +207,7 @@ biosd_rw(rw, dev, cyl, head, sect, nsect, buf)
return (rv & 0xff)? rv >> 8 : 0;
}
-int
+static __inline int
EDD_rw(rw, dev, daddr, nblk, buf)
int rw, dev;
u_int64_t daddr;
@@ -209,16 +215,28 @@ EDD_rw(rw, dev, daddr, nblk, buf)
void *buf;
{
int rv;
- struct EDD_CB cb;
+ volatile static struct EDD_CB cb;
+ /* Zero out reserved stuff */
+ cb.edd_res1 = 0;
+ cb.edd_res2 = 0;
+
+ /* Fill in parameters */
cb.edd_len = sizeof(cb);
cb.edd_nblk = nblk;
- cb.edd_buf = (u_int32_t)buf;
+ cb.edd_seg = ((u_int32_t)buf >> 4) & 0xffff;
+ cb.edd_off = (u_int32_t)buf & 0xf;
cb.edd_daddr = daddr;
+ /* if offset/segment are zero, punt */
+ if (!cb.edd_seg && !cb.edd_off)
+ return (1);
+
+ /* Call extended read/write (with disk packet) */
+ BIOS_regs.biosr_ds = (u_int32_t)&cb >> 4;
__asm __volatile (DOINT(0x13) "; setc %b0" : "=a" (rv)
: "0" ((rw == F_READ)? 0x4200: 0x4300),
- "d" (dev), "S" (&cb) : "%ecx", "cc");
+ "d" (dev), "S" ((int) (&cb) & 0xf) : "%ecx", "cc");
return (rv & 0xff)? rv >> 8 : 0;
}
@@ -226,21 +244,17 @@ EDD_rw(rw, dev, daddr, nblk, buf)
* Read given sector, handling retry/errors/etc.
*/
int
-biosd_io(rw, dev, cyl, head, sect, nsect, buf)
- int rw, dev, cyl, head;
- int sect, nsect;
- void * buf;
+biosd_io(rw, bd, off, nsect, buf)
+ int rw;
+ bios_diskinfo_t *bd;
+ daddr_t off;
+ int nsect;
+ void* buf;
{
+ int dev = bd->bios_number;
int j, error;
void *bb;
-#ifdef BIOS_DEBUG
- if (debug)
- printf("biosd_io(%s,%x,%u,%u,%u,%u,%p)\n",
- (rw==F_READ?"reading":"writing"), dev,
- cyl, head, sect, nsect, buf);
-#endif
-
/* use a bounce buffer to not cross 64k DMA boundary */
if ((((u_int32_t)buf) & ~0xffff) !=
(((u_int32_t)buf + nsect * DEV_BSIZE) & ~0xffff)) {
@@ -253,13 +267,35 @@ biosd_io(rw, dev, cyl, head, sect, nsect, buf)
bcopy (buf, bb, nsect * DEV_BSIZE);
} else
bb = buf;
-#ifdef BIOS_DEBUG
- if (debug)
- printf(" (%d,%d,%d,%d)@%p", cyl, head, sect, nsect, bb);
-#endif
+
/* Try to do operation up to 5 times */
- for (error = 1, j = 5; j-- && error;)
- switch (error = biosd_rw(rw, dev, cyl, head, sect, nsect, bb)) {
+ for (error = 1, j = 5; j-- && error;) {
+ /* CHS or LBA access? */
+ if (bd->bios_edd != -1) {
+ error = EDD_rw(rw, dev, off, nsect, bb);
+ } else {
+ int cyl, head, sect;
+ size_t i, n;
+ char *p = bb;
+
+ /* Handle track boundaries */
+ for (error = i = 0; error == 0 && i < nsect;
+ i += n, off += n, p += n * DEV_BSIZE) {
+
+ btochs(off, cyl, head, sect, bd->bios_heads, bd->bios_sectors);
+ if ((sect + (nsect - i)) >= bd->bios_sectors)
+ n = bd->bios_sectors - sect;
+ else
+ n = nsect - i;
+
+ error = CHS_rw(rw, dev, cyl, head, sect, n, p);
+
+ /* ECC corrected */
+ if (error == 0x11)
+ error = 0;
+ }
+ }
+ switch (error) {
case 0x00: /* No errors */
case 0x11: /* ECC corrected */
error = 0;
@@ -274,6 +310,7 @@ biosd_io(rw, dev, cyl, head, sect, nsect, buf)
biosdreset(dev);
break;
}
+ }
if (bb != buf && rw == F_READ)
bcopy (bb, buf, nsect * DEV_BSIZE);
@@ -286,7 +323,7 @@ biosd_io(rw, dev, cyl, head, sect, nsect, buf)
}
#endif
- return biosdisk_errno(error);
+ return (error);
}
/*
@@ -300,22 +337,17 @@ bios_getdisklabel(bd, label)
daddr_t off = LABELSECTOR;
char *buf;
struct dos_mbr mbr;
- int cyl, head, sect;
int error, i;
- /* XXX - Sanity check */
+ /* Sanity check */
if(bd->bios_heads == 0 || bd->bios_sectors == 0)
return("failed to read disklabel");
- /* MBR is a hard thing */
+ /* MBR is a harddisk thing */
if (bd->bios_number & 0x80) {
/* Read MBR */
- btochs(DOSBBSECTOR, cyl, head, sect,
- bd->bios_heads, bd->bios_sectors);
-
- error = biosd_io(F_READ, bd->bios_number,
- cyl, head, sect, 1, &mbr);
- if(error)
+ error = biosd_io(F_READ, bd, DOSBBSECTOR, 1, &mbr);
+ if (error)
return(biosdisk_err(error));
/* check mbr signature */
@@ -345,8 +377,7 @@ bios_getdisklabel(bd, label)
printf("loading disklabel @ %u\n", off);
#endif
/* read disklabel */
- btochs(off, cyl, head, sect, bd->bios_heads, bd->bios_sectors);
- error = biosd_io(F_READ, bd->bios_number, cyl, head, sect, 1, buf);
+ error = biosd_io(F_READ, bd, off, 1, buf);
if(error)
return("failed to read disklabel");
@@ -555,38 +586,18 @@ biosstrategy(devdata, rw, blk, size, buf, rsize)
void *buf;
size_t *rsize;
{
- u_int8_t error = 0;
struct diskinfo *dip = (struct diskinfo *)devdata;
- register size_t i, nsect, n, spt, tpc;
- int dev;
+ bios_diskinfo_t *bd = &dip->bios_info;
+ u_int8_t error = 0;
+ size_t nsect;
nsect = (size + DEV_BSIZE-1) / DEV_BSIZE;
if (rsize != NULL)
blk += dip->disklabel.
d_partitions[B_PARTITION(dip->bsddev)].p_offset;
- /* handle floppies w/ different from drive geometry */
- if (!(dip->bios_info.bios_number & 0x80) &&
- dip->disklabel.d_nsectors != 0)
- spt = dip->disklabel.d_nsectors;
- else
- spt = dip->bios_info.bios_sectors;
-
- tpc = dip->bios_info.bios_heads;
- dev = dip->bios_info.bios_number;
-
- for (i = 0; error == 0 && i < nsect;
- i += n, blk += n, buf += n * DEV_BSIZE) {
- register int cyl, hd, sect;
-
- btochs(blk, cyl, hd, sect, tpc, spt);
- if ((sect + (nsect - i)) >= spt)
- n = spt - sect;
- else
- n = nsect - i;
-
- error = biosd_io(rw, dev, cyl, hd, sect, n, buf);
- }
+ /* Read all, sub-functions handle track boundaries */
+ error = biosd_io(rw, bd, blk, nsect, buf);
#ifdef BIOS_DEBUG
if (debug) {
@@ -597,7 +608,7 @@ biosstrategy(devdata, rw, blk, size, buf, rsize)
#endif
if (rsize != NULL)
- *rsize = i * DEV_BSIZE;
+ *rsize = nsect * DEV_BSIZE;
return biosdisk_errno(error);
}
diff --git a/sys/arch/i386/stand/libsa/biosdev.h b/sys/arch/i386/stand/libsa/biosdev.h
index 156519f2bb7..51b59e0d9a9 100644
--- a/sys/arch/i386/stand/libsa/biosdev.h
+++ b/sys/arch/i386/stand/libsa/biosdev.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: biosdev.h,v 1.27 2002/03/14 03:15:54 millert Exp $ */
+/* $OpenBSD: biosdev.h,v 1.28 2003/05/31 00:15:29 weingart Exp $ */
/*
* Copyright (c) 1996 Michael Shalayeff
@@ -42,7 +42,7 @@ int biosopen(struct open_file *, ...);
int biosclose(struct open_file *);
int biosioctl(struct open_file *, u_long, void *);
int bios_getdiskinfo(int, bios_diskinfo_t *);
-int biosd_io(int, int, int, int, int, int, void *);
+int biosd_io(int, bios_diskinfo_t *, daddr_t, int, void *);
const char * bios_getdisklabel(bios_diskinfo_t *, struct disklabel *);
/* diskprobe.c */
diff --git a/sys/arch/i386/stand/libsa/cmd_i386.c b/sys/arch/i386/stand/libsa/cmd_i386.c
index f0de0e8c9ce..4a1c9c70ff8 100644
--- a/sys/arch/i386/stand/libsa/cmd_i386.c
+++ b/sys/arch/i386/stand/libsa/cmd_i386.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: cmd_i386.c,v 1.24 2002/03/14 03:15:54 millert Exp $ */
+/* $OpenBSD: cmd_i386.c,v 1.25 2003/05/31 00:15:29 weingart Exp $ */
/*
* Copyright (c) 1997-1999 Michael Shalayeff
@@ -87,6 +87,7 @@ Xboot()
{
#ifndef _TEST
int dev, part, st;
+ bios_diskinfo_t *bd = NULL;
char buf[DEV_BSIZE], *dest = (void *)BOOTBIOS_ADDR;
if(cmd.argc != 2) {
@@ -118,7 +119,8 @@ Xboot()
printf("[%x]\n", dev);
/* Read boot sector from device */
- st = biosd_io(F_READ, dev, 0, 0, 0, 1, buf);
+ bd = bios_dklookup(dev);
+ st = biosd_io(F_READ, bd, 0, 1, buf);
if(st) goto bad;
/* Frob boot flag in buffer from HD */
diff --git a/sys/arch/i386/stand/libsa/diskprobe.c b/sys/arch/i386/stand/libsa/diskprobe.c
index 1b19b2ec1c5..a5efd53a913 100644
--- a/sys/arch/i386/stand/libsa/diskprobe.c
+++ b/sys/arch/i386/stand/libsa/diskprobe.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: diskprobe.c,v 1.18 2002/03/14 01:26:34 millert Exp $ */
+/* $OpenBSD: diskprobe.c,v 1.19 2003/05/31 00:15:29 weingart Exp $ */
/*
* Copyright (c) 1997 Tobias Weingartner
@@ -267,9 +267,7 @@ disksum(blk)
{
struct diskinfo *dip, *dip2;
int st, reprobe = 0;
- int hpc, spt, dev;
char *buf;
- int cyl, head, sect;
buf = alloca(DEV_BSIZE);
for(dip = TAILQ_FIRST(&disklist); dip; dip = TAILQ_NEXT(dip, list)){
@@ -279,13 +277,8 @@ disksum(blk)
if (!(bdi->bios_number & 0x80) || bdi->flags & BDI_INVALID)
continue;
- dev = bdi->bios_number;
- hpc = bdi->bios_heads;
- spt = bdi->bios_sectors;
-
/* Adler32 checksum */
- btochs(blk, cyl, head, sect, hpc, spt);
- st = biosd_io(F_READ, dev, cyl, head, sect, 1, buf);
+ st = biosd_io(F_READ, bdi, blk, 1, buf);
if (st) {
bdi->flags |= BDI_INVALID;
continue;