]> git.saurik.com Git - apple/boot.git/blobdiff - i386/libsaio/disk.c
boot-132.tar.gz
[apple/boot.git] / i386 / libsaio / disk.c
index 3f6157439d8850465a508afdeac9ced4545251e8..afb1057245532ec288d095ce5c5bda97f2fd7f15 100644 (file)
@@ -1,24 +1,23 @@
 /*
- * Copyright (c) 1999 Apple Computer, Inc. All rights reserved.
+ * Copyright (c) 1999-2003 Apple Computer, Inc. All rights reserved.
  *
  * @APPLE_LICENSE_HEADER_START@
  * 
- * Copyright (c) 1999-2003 Apple Computer, Inc.  All Rights Reserved.
- * 
- * This file contains Original Code and/or Modifications of Original Code
- * as defined in and that are subject to the Apple Public Source License
- * Version 2.0 (the 'License'). You may not use this file except in
- * compliance with the License. Please obtain a copy of the License at
- * http://www.opensource.apple.com/apsl/ and read it before using this
- * file.
+ * Portions Copyright (c) 1999-2003 Apple Computer, Inc.  All Rights
+ * Reserved.  This file contains Original Code and/or Modifications of
+ * Original Code as defined in and that are subject to the Apple Public
+ * Source License Version 2.0 (the "License").  You may not use this file
+ * except in compliance with the License.  Please obtain a copy of the
+ * License at http://www.apple.com/publicsource and read it before using
+ * this file.
  * 
  * The Original Code and all software distributed under the License are
- * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
+ * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
- * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
- * Please see the License for the specific language governing rights and
- * limitations under the License.
+ * FITNESS FOR A PARTICULAR PURPOSE OR NON- INFRINGEMENT.  Please see the
+ * License for the specific language governing rights and limitations
+ * under the License.
  * 
  * @APPLE_LICENSE_HEADER_END@
  */
  * All rights reserved.
  */
 
+#define UFS_SUPPORT 1
+
 #include "bootstruct.h"
 #include "libsaio.h"
 #include "fdisk.h"
+#if UFS_SUPPORT
 #include "ufs.h"
+#endif
 #include "hfs.h"
+#include "ntfs.h"
+#include "msdos.h"
 
 #include <limits.h>
 #include <IOKit/storage/IOApplePartitionScheme.h>
@@ -68,8 +73,8 @@
  * biosbuf points to a sector within the track cache, and is
  * updated by Biosread().
  */
-static const char * const trackbuf = (char *) ptov(BIOS_ADDR);
-static const char * biosbuf;
+static char * const trackbuf = (char *) ptov(BIOS_ADDR);
+static char * biosbuf;
 
 /*
  * Map a disk drive to bootable volumes contained within.
@@ -100,6 +105,7 @@ static int getDriveInfo( int biosdev,  struct driveInfo *dip )
        cc = get_drive_info(biosdev, &cached_di);
         if (cc < 0) {
            cached_di.valid = 0;
+            DEBUG_DISK(("get_drive_info returned error\n"));
            return (-1); // BIOS call error
        }
     }
@@ -161,11 +167,12 @@ static const char * bios_error(int errnum)
 // Return:
 //   0 on success, or an error code from INT13/F2 or INT13/F42 BIOS call.
 
+static BOOL cache_valid = FALSE;
+
 static int Biosread( int biosdev, unsigned int secno )
 {
     static int xbiosdev, xcyl, xhead;
     static unsigned int xsec, xnsecs;
-    static BOOL cache_valid = FALSE;
     struct driveInfo di;
 
     int  rc = -1;
@@ -181,6 +188,9 @@ static int Biosread( int biosdev, unsigned int secno )
        bps = 2048;
     } else {
        bps = di.di.params.phys_nbps;
+        if (bps == 0) {
+            return -1;
+        }
     }
     divisor = bps / BPS;
 
@@ -277,6 +287,7 @@ static int Biosread( int biosdev, unsigned int secno )
 //==========================================================================
 
 static int readBytes( int biosdev, unsigned int blkno,
+                      unsigned int byteoff,
                       unsigned int byteCount, void * buffer )
 {
 
@@ -287,7 +298,7 @@ static int readBytes( int biosdev, unsigned int blkno,
     DEBUG_DISK(("%s: dev %x block %x [%d] -> 0x%x...", __FUNCTION__,
                 biosdev, blkno, byteCount, (unsigned)cbuf));
 
-    for ( ; byteCount; cbuf += BPS, blkno++ )
+    for ( ; byteCount; cbuf += copy_len, blkno++ )
     {
         error = Biosread( biosdev, blkno );
         if ( error )
@@ -296,9 +307,10 @@ static int readBytes( int biosdev, unsigned int blkno,
             return (-1);
         }
 
-        copy_len = (byteCount > BPS) ? BPS : byteCount;
-        bcopy( biosbuf, cbuf, copy_len );
+        copy_len = ((byteCount + byteoff) > BPS) ? (BPS - byteoff) : byteCount;
+        bcopy( biosbuf + byteoff, cbuf, copy_len );
         byteCount -= copy_len;
+        byteoff = 0;
     }
 
     DEBUG_DISK(("done\n"));
@@ -333,6 +345,7 @@ static int getNextFDiskPartition( int biosdev, int * partno,
 {
     static int                 sBiosdev = -1;
     static int                 sNextPartNo;
+    static unsigned int        sFirstBase;
     static unsigned int        sExtBase;
     static unsigned int        sExtDepth;
     static struct fdisk_part * sExtPart;
@@ -345,6 +358,7 @@ static int getNextFDiskPartition( int biosdev, int * partno,
 
         sBiosdev    = biosdev;
         sNextPartNo = 0;
+        sFirstBase  = 0;
         sExtBase    = 0;
         sExtDepth   = 0;
         sExtPart    = NULL;
@@ -360,11 +374,14 @@ static int getNextFDiskPartition( int biosdev, int * partno,
         }
         else if ( sExtPart )
         {
-            unsigned int blkno = sExtPart->relsect + sExtBase;
+            unsigned int blkno = sExtPart->relsect + sFirstBase;
 
             // Save the block offset of the first extended partition.
 
-            if ( sExtDepth == 0 ) sExtBase = sExtPart->relsect;
+            if (sExtDepth == 0) {
+                sFirstBase = blkno;
+            }
+            sExtBase = blkno;
 
             // Load extended partition table.
 
@@ -375,6 +392,7 @@ static int getNextFDiskPartition( int biosdev, int * partno,
                 sExtPart = NULL;
                 continue;
             }
+            // Fall through to part == NULL
         }
 
         if ( part == NULL ) break;  // Reached end of partition chain.
@@ -383,8 +401,6 @@ static int getNextFDiskPartition( int biosdev, int * partno,
 
         sNextPartNo++;
 
-        // Assume at most one extended partition per table.
-
         if ( isExtendedFDiskPartition(part) )
         {
             sExtPart = part;
@@ -399,11 +415,10 @@ static int getNextFDiskPartition( int biosdev, int * partno,
         }
 
         // Change relative offset to an absolute offset.
-
         part->relsect += sExtBase;
 
         *outPart = part;
-        *partno  = sExtDepth ? (int)(sExtDepth + 4) : sNextPartNo;
+        *partno  = sExtDepth ? (int)(sExtDepth + FDISK_NPART) : sNextPartNo;
 
         break;
     }
@@ -416,7 +431,12 @@ static int getNextFDiskPartition( int biosdev, int * partno,
 static BVRef newFDiskBVRef( int biosdev, int partno, unsigned int blkoff,
                             const struct fdisk_part * part,
                             FSInit initFunc, FSLoadFile loadFunc,
-                            FSGetDirEntry getdirFunc, int probe, int type )
+                            FSReadFile readFunc,
+                            FSGetDirEntry getdirFunc,
+                            FSGetFileBlock getBlockFunc,
+                            FSGetUUID getUUIDFunc,
+                            BVGetDescription getDescriptionFunc,
+                            int probe, int type )
 {
     BVRef bvr = (BVRef) malloc( sizeof(*bvr) );
     if ( bvr )
@@ -428,8 +448,12 @@ static BVRef newFDiskBVRef( int biosdev, int partno, unsigned int blkoff,
         bvr->part_boff      = blkoff;
         bvr->part_type      = part->systid;
         bvr->fs_loadfile    = loadFunc;
+        bvr->fs_readfile    = readFunc;
         bvr->fs_getdirentry = getdirFunc;
-        bvr->description    = getVolumeDescription;
+        bvr->fs_getfileblock= getBlockFunc;
+        bvr->fs_getuuid     = getUUIDFunc;
+        bvr->description    = getDescriptionFunc ?
+            getDescriptionFunc : getVolumeDescription;
        bvr->type           = type;
 
         if ( part->bootid & FDISK_ACTIVE )
@@ -470,7 +494,12 @@ static BVRef newFDiskBVRef( int biosdev, int partno, unsigned int blkoff,
 BVRef newAPMBVRef( int biosdev, int partno, unsigned int blkoff,
                    const DPME * part,
                    FSInit initFunc, FSLoadFile loadFunc,
-                   FSGetDirEntry getdirFunc, int probe, int type )
+                   FSReadFile readFunc,
+                   FSGetDirEntry getdirFunc,
+                   FSGetFileBlock getBlockFunc,
+                   FSGetUUID getUUIDFunc,
+                   BVGetDescription getDescriptionFunc,
+                   int probe, int type )
 {
     BVRef bvr = (BVRef) malloc( sizeof(*bvr) );
     if ( bvr )
@@ -481,8 +510,12 @@ BVRef newAPMBVRef( int biosdev, int partno, unsigned int blkoff,
         bvr->part_no        = partno;
         bvr->part_boff      = blkoff;
         bvr->fs_loadfile    = loadFunc;
+        bvr->fs_readfile    = readFunc;
         bvr->fs_getdirentry = getdirFunc;
-        bvr->description    = getVolumeDescription;
+        bvr->fs_getfileblock= getBlockFunc;
+        bvr->fs_getuuid     = getUUIDFunc;
+        bvr->description    = getDescriptionFunc ?
+            getDescriptionFunc : getVolumeDescription;
        bvr->type           = type;
         strlcpy(bvr->name, part->dpme_name, DPISTRLEN);
         strlcpy(bvr->type_name, part->dpme_type, DPISTRLEN);
@@ -526,13 +559,22 @@ BVRef newAPMBVRef( int biosdev, int partno, unsigned int blkoff,
 
 //==========================================================================
 
+/* A note on partition numbers:
+ * IOKit makes the primary partitions numbers 1-4, and then
+ * extended partitions are numbered consecutively 5 and up.
+ * So, for example, if you have two primary partitions and
+ * one extended partition they will be numbered 1, 2, 5.
+ */
+
 static BVRef diskScanFDiskBootVolumes( int biosdev, int * countPtr )
 {
     const struct fdisk_part * part;
     struct DiskBVMap *        map;
     int                       partno  = -1;
     BVRef                     bvr;
+#if UFS_SUPPORT
     BVRef                     booterUFS = NULL;
+#endif
     int                       spc;
     struct driveInfo          di;
     boot_drive_info_t         *dp;
@@ -570,6 +612,7 @@ static BVRef diskScanFDiskBootVolumes( int biosdev, int * countPtr )
 
                 switch ( part->systid )
                 {
+#if UFS_SUPPORT
                     case FDISK_UFS:
                        bvr = newFDiskBVRef(
                                       biosdev, partno,
@@ -577,10 +620,15 @@ static BVRef diskScanFDiskBootVolumes( int biosdev, int * countPtr )
                                       part,
                                       UFSInitPartition,
                                       UFSLoadFile,
+                                      UFSReadFile,
                                       UFSGetDirEntry,
+                                      UFSGetFileBlock,
+                                      UFSGetUUID,
+                                      UFSGetDescription,
                                       0,
                                      kBIOSDevTypeHardDrive);
                         break;
+#endif
 
                     case FDISK_HFS:
                         bvr = newFDiskBVRef(
@@ -589,11 +637,16 @@ static BVRef diskScanFDiskBootVolumes( int biosdev, int * countPtr )
                                       part,
                                       HFSInitPartition,
                                       HFSLoadFile,
+                                      HFSReadFile,
                                       HFSGetDirEntry,
+                                      HFSGetFileBlock,
+                                      HFSGetUUID,
+                                      HFSGetDescription,
                                       0,
                                      kBIOSDevTypeHardDrive);
                         break;
 
+#if UFS_SUPPORT
                     case FDISK_BOOTER:
                         booterUFS = newFDiskBVRef(
                                       biosdev, partno,
@@ -601,17 +654,33 @@ static BVRef diskScanFDiskBootVolumes( int biosdev, int * countPtr )
                                       part,
                                       UFSInitPartition,
                                       UFSLoadFile,
+                                      UFSReadFile,
                                       UFSGetDirEntry,
+                                      UFSGetFileBlock,
+                                      UFSGetUUID,
+                                      UFSGetDescription,
                                       0,
                                      kBIOSDevTypeHardDrive);
                         break;
+#endif
 
+                   case FDISK_NTFS:
+                        bvr = newFDiskBVRef(
+                                      biosdev, partno,
+                                      part->relsect,
+                                      part,
+                                      0, 0, 0, 0, 0, 0,
+                                      NTFSGetDescription,
+                                      0,
+                                     kBIOSDevTypeHardDrive);
+                        break;
+                    
                     default:
                         bvr = newFDiskBVRef(
                                       biosdev, partno,
                                       part->relsect,
                                       part,
-                                      0, 0, 0, 0,
+                                      0, 0, 0, 0, 0, 0, 0, 0,
                                      kBIOSDevTypeHardDrive);
                         break;
                 }
@@ -624,6 +693,7 @@ static BVRef diskScanFDiskBootVolumes( int biosdev, int * countPtr )
                 }
             }
 
+#if UFS_SUPPORT
             // Booting from a CD with an UFS filesystem embedded
             // in a booter partition.
 
@@ -636,6 +706,7 @@ static BVRef diskScanFDiskBootVolumes( int biosdev, int * countPtr )
                 }
                 else free( booterUFS );
             }
+#endif
         }
     } while (0);
 
@@ -643,7 +714,7 @@ static BVRef diskScanFDiskBootVolumes( int biosdev, int * countPtr )
      * If no FDisk partition, then we will check for
      * an Apple partition map elsewhere.
      */
-#if 0
+#if UNUSED
     if (map->bvrcnt == 0) {
        static struct fdisk_part cdpart;
        cdpart.systid = 0xCD;
@@ -655,7 +726,10 @@ static BVRef diskScanFDiskBootVolumes( int biosdev, int * countPtr )
                            &cdpart,
                            HFSInitPartition,
                            HFSLoadFile,
+                            HFSReadFile,
                            HFSGetDirEntry,
+                            HFSGetFileBlock,
+                            HFSGetUUID,
                            0,
                            kBIOSDevTypeHardDrive);
        bvr->next = map->bvr;
@@ -680,12 +754,12 @@ static BVRef diskScanAPMBootVolumes( int biosdev, int * countPtr )
     void *buffer = malloc(BPS);
 
     /* Check for alternate block size */
-    if (readBytes( biosdev, 0, BPS, buffer ) != 0) {
+    if (readBytes( biosdev, 0, 0, BPS, buffer ) != 0) {
         return NULL;
     }
     block0_p = buffer;
-    if (NXSwapBigShortToHost(block0_p->sbSig) == BLOCK0_SIGNATURE) {
-        blksize = NXSwapBigShortToHost(block0_p->sbBlkSize);
+    if (OSSwapBigToHostInt16(block0_p->sbSig) == BLOCK0_SIGNATURE) {
+        blksize = OSSwapBigToHostInt16(block0_p->sbBlkSize);
         if (blksize != BPS) {
             free(buffer);
             buffer = malloc(blksize);
@@ -714,14 +788,14 @@ static BVRef diskScanAPMBootVolumes( int biosdev, int * countPtr )
             gDiskBVMap   = map;
 
             for (i=0; i<npart; i++) {
-                error = readBytes( biosdev, (kAPMSector + i) * factor, blksize, buffer );
+                error = readBytes( biosdev, (kAPMSector + i) * factor, 0, blksize, buffer );
 
-                if (error || NXSwapBigShortToHost(dpme_p->dpme_signature) != DPME_SIGNATURE) {
+                if (error || OSSwapBigToHostInt16(dpme_p->dpme_signature) != DPME_SIGNATURE) {
                     break;
                 }
 
                 if (i==0) {
-                    npart = NXSwapBigLongToHost(dpme_p->dpme_map_entries);
+                    npart = OSSwapBigToHostInt32(dpme_p->dpme_map_entries);
                 }
                 /*
                 printf("name = %s, %s%s  %d -> %d [%d -> %d] {%d}\n",
@@ -734,11 +808,15 @@ static BVRef diskScanAPMBootVolumes( int biosdev, int * countPtr )
                 if (strcmp(dpme_p->dpme_type, "Apple_HFS") == 0) {
                     bvr = newAPMBVRef(biosdev,
                                       i,
-                                      NXSwapBigLongToHost(dpme_p->dpme_pblock_start) * factor,
+                                      OSSwapBigToHostInt32(dpme_p->dpme_pblock_start) * factor,
                                       dpme_p,
                                       HFSInitPartition,
                                       HFSLoadFile,
+                                      HFSReadFile,
                                       HFSGetDirEntry,
+                                      HFSGetFileBlock,
+                                      HFSGetUUID,
+                                      HFSGetDescription,
                                       0,
                                       kBIOSDevTypeHardDrive);
                     bvr->next = map->bvr;
@@ -790,8 +868,8 @@ BVRef diskScanBootVolumes( int biosdev, int * countPtr )
 
 static const struct NamedValue fdiskTypes[] =
 {
-    { 0x07,         "Windows NTFS"   },
-    { 0x0c,         "Windows FAT32"  },
+    { FDISK_NTFS,   "Windows NTFS"   },
+    { FDISK_FAT32,  "Windows FAT32"  },
     { 0x83,         "Linux"          },
     { FDISK_UFS,    "Apple UFS"      },
     { FDISK_HFS,    "Apple HFS"      },
@@ -800,20 +878,110 @@ static const struct NamedValue fdiskTypes[] =
     { 0x00,         0                }  /* must be last */
 };
 
-static void getVolumeDescription( BVRef bvr, char * str, long strMaxLen )
+//==========================================================================
+
+void getBootVolumeDescription( BVRef bvr, char * str, long strMaxLen, BOOL verbose )
 {
     unsigned char type = (unsigned char) bvr->part_type;
     const char * name = getNameForValue( fdiskTypes, type );
+    char *p;
+
+    if (name == NULL)
+        name = bvr->type_name;
+
+    p = str;
+    if ( name && verbose ) {
+        sprintf( str, "hd(%d,%d) ",
+                 BIOS_DEV_UNIT(bvr), bvr->part_no);
+        for (; strMaxLen > 0 && *p != '\0'; p++, strMaxLen--);
+    } else {
+        *p = '\0';
+    }
+    bvr->description(bvr, p, strMaxLen);
+    if (*p == '\0') {
+        const char * name = getNameForValue( fdiskTypes, type );
+        if (name == NULL) {
+            name = bvr->type_name;
+        }
+        if (name == NULL) {
+            sprintf(p, "TYPE %02x", type);
+        } else {
+            strncpy(p, name, strMaxLen);
+        }
+    }
+}
+
+#if UNUSED
+//==========================================================================
+
+static int
+getFAT32VolumeDescription( BVRef bvr, char *str, long strMaxLen)
+{
+    struct fat32_header {
+        unsigned char code[3];
+        unsigned char oem_id[8];
+        unsigned char data[56];
+        unsigned long serial;
+        unsigned char label[11];
+        unsigned char fsid[8];
+        unsigned char reserved[420];
+        unsigned short signature;
+    } __attribute__((packed));
+
+    char *buf, *name;
+    struct fat32_header *fat32_p;
+    int label_len = sizeof(fat32_p->label);
+    int error;
+
+    buf = (char *)malloc(BPS);
+    name = (char *)malloc(label_len + 1);
+    fat32_p = (struct fat32_header *)buf;
+
+    diskSeek(bvr, 0ULL);
+    error = diskRead(bvr, (long)buf, BPS);
+    if ( error ) return 0;
+
+    if (fat32_p->signature != 0xaa55) return 0;
+
+    if (strMaxLen < label_len) label_len = strMaxLen;
+    strncpy(str, fat32_p->label, label_len);
+    str[label_len] = '\0';
+    return 1;
+}
+#endif
+
+//==========================================================================
+
+static void getVolumeDescription( BVRef bvr, char * str, long strMaxLen )
+{
+    unsigned char type = (unsigned char) bvr->part_type;
+    const char * name = NULL;
+
+    /* First try a few types that we can figure out the
+     * volume description.
+     */
+    switch(type) {
+    case FDISK_FAT32:
+        str[0] = '\0';
+        MSDOSGetDescription(bvr, str, strMaxLen);
+        if (str[0] != '\0')
+            return;
+        break;
+
+    default: // Not one of our known types
+        break;
+    }
+
+    if (name == NULL)
+        name = getNameForValue( fdiskTypes, type );
 
     if (name == NULL)
         name = bvr->type_name;
 
     if ( name )
-        sprintf( str, "hd(%d,%d) %s",
-                 BIOS_DEV_UNIT(bvr), bvr->part_no, name );
+        strncpy( str, name, strMaxLen);
     else
-        sprintf( str, "hd(%d,%d) TYPE %02x",
-                 BIOS_DEV_UNIT(bvr), bvr->part_no, type );
+        sprintf( str, "TYPE %02x", type);
 }
 
 
@@ -834,7 +1002,7 @@ int readBootSector( int biosdev, unsigned int secno, void * buffer )
         bootSector = gBootSector;
     }
 
-    error = readBytes( biosdev, secno, BPS, bootSector );
+    error = readBytes( biosdev, secno, 0, BPS, bootSector );
     if ( error || bootSector->signature != DISK_SIGNATURE )
         return -1;
 
@@ -847,6 +1015,7 @@ int readBootSector( int biosdev, unsigned int secno, void * buffer )
 void diskSeek( BVRef bvr, long long position )
 {
     bvr->fs_boff = position / BPS;
+    bvr->fs_byteoff = position % BPS;
 }
 
 //==========================================================================
@@ -856,17 +1025,93 @@ int diskRead( BVRef bvr, long addr, long length )
 {
     return readBytes( bvr->biosdev,
                       bvr->fs_boff + bvr->part_boff,
+                      bvr->fs_byteoff,
                       length,
                       (void *) addr );
 }
 
-void turnOffFloppy(void)
+int rawDiskRead( BVRef bvr, unsigned int secno, void *buffer, unsigned int len )
 {
-    /*
-     * Disable floppy:
-     * Hold controller in reset,
-     * disable DMA and IRQ,
-     * turn off floppy motors.
-     */
-    outb(0x3F2, 0x00);
+    int secs;
+    unsigned char *cbuf = (unsigned char *)buffer;
+    unsigned int copy_len;
+    int rc;
+
+    if ((len & (BPS-1)) != 0) {
+        error("raw disk read not sector aligned");
+        return -1;
+    }
+    secno += bvr->part_boff;
+
+    cache_valid = FALSE;
+
+    while (len > 0) {
+        secs = len / BPS;
+        if (secs > N_CACHE_SECS) secs = N_CACHE_SECS;
+        copy_len = secs * BPS;
+
+        //printf("rdr: ebiosread(%d, %d, %d)\n", bvr->biosdev, secno, secs);
+        if ((rc = ebiosread(bvr->biosdev, secno, secs)) != 0) {
+            /* Ignore corrected ECC errors */
+            if (rc != ECC_CORRECTED_ERR) {
+                error("  EBIOS read error: %s\n", bios_error(rc), rc);
+                error("    Block %d Sectors %d\n", secno, secs);
+                return rc;
+            }
+        }
+        bcopy( trackbuf, cbuf, copy_len );
+        len -= copy_len;
+        cbuf += copy_len;
+        secno += secs;
+        spinActivityIndicator();
+    }
+
+    return 0;
+}
+
+int rawDiskWrite( BVRef bvr, unsigned int secno, void *buffer, unsigned int len )
+{
+    int secs;
+    unsigned char *cbuf = (unsigned char *)buffer;
+    unsigned int copy_len;
+    int rc;
+
+    if ((len & (BPS-1)) != 0) {
+        error("raw disk write not sector aligned");
+        return -1;
+    }
+    secno += bvr->part_boff;
+
+    cache_valid = FALSE;
+
+    while (len > 0) {
+        secs = len / BPS;
+        if (secs > N_CACHE_SECS) secs = N_CACHE_SECS;
+        copy_len = secs * BPS;
+
+        bcopy( cbuf, trackbuf, copy_len );
+        //printf("rdr: ebioswrite(%d, %d, %d)\n", bvr->biosdev, secno, secs);
+        if ((rc = ebioswrite(bvr->biosdev, secno, secs)) != 0) {
+            error("  EBIOS write error: %s\n", bios_error(rc), rc);
+            error("    Block %d Sectors %d\n", secno, secs);
+            return rc;
+        }
+        len -= copy_len;
+        cbuf += copy_len;
+        secno += secs;
+        spinActivityIndicator();
+    }
+
+    return 0;
+}
+
+
+int diskIsCDROM(BVRef bvr)
+{
+    struct driveInfo          di;
+
+    if (getDriveInfo(bvr->biosdev, &di) == 0 && di.no_emulation) {
+       return 1;
+    }
+    return 0;
 }