Home | History | Annotate | Download | only in ext2fs
      1 /*
      2  * dosio.c -- Disk I/O module for the ext2fs/DOS library.
      3  *
      4  * Copyright (c) 1997 by Theodore Ts'o.
      5  *
      6  * Copyright (c) 1997 Mark Habersack
      7  *
      8  * %Begin-Header%
      9  * This file may be redistributed under the terms of the GNU Library
     10  * General Public License, version 2.
     11  * %End-Header%
     12  */
     13 
     14 #include "config.h"
     15 #include <stdio.h>
     16 #include <bios.h>
     17 #include <string.h>
     18 #include <ctype.h>
     19 #include <io.h>
     20 #ifdef HAVE_ERRNO_H
     21 #include <errno.h>
     22 #endif
     23 
     24 #include <ext2fs/ext2_types.h>
     25 #include "utils.h"
     26 #include "dosio.h"
     27 #include "et/com_err.h"
     28 #include "ext2_err.h"
     29 #include "ext2fs/io.h"
     30 
     31 /*
     32  * Some helper macros
     33  */
     34 #define LINUX_EXT2FS       0x83
     35 #define LINUX_SWAP         0x82
     36 #define WRITE_ERR(_msg_) write(2, _msg_, strlen(_msg_))
     37 #define WRITE_ERR_S(_msg_) write(2, _msg_, sizeof(_msg_))
     38 
     39 /*
     40  * Exported variables
     41  */
     42 unsigned long        _dio_error;
     43 unsigned long        _dio_hw_error;
     44 
     45 /*
     46  * Array of all opened partitions
     47  */
     48 static PARTITION        **partitions = NULL;
     49 static unsigned short   npart = 0; /* Number of mapped partitions */
     50 static PARTITION        *active = NULL;
     51 
     52 /*
     53  * I/O Manager routine prototypes
     54  */
     55 static errcode_t dos_open(const char *dev, int flags, io_channel *channel);
     56 static errcode_t dos_close(io_channel channel);
     57 static errcode_t dos_set_blksize(io_channel channel, int blksize);
     58 static errcode_t dos_read_blk(io_channel channel, unsigned long block,
     59                                              int count, void *buf);
     60 static errcode_t dos_write_blk(io_channel channel, unsigned long block,
     61                                int count, const void *buf);
     62 static errcode_t dos_flush(io_channel channel);
     63 
     64 static struct struct_io_manager struct_dos_manager = {
     65 	.magic		= EXT2_ET_MAGIC_IO_MANAGER,
     66 	.name		= "DOS I/O Manager",
     67 	.open		= dos_open,
     68 	.close		= dos_close,
     69 	.set_blksize	= dos_set_blksize,
     70 	.read_blk	= dos_read_blk,
     71 	.write_blk	= dos_write_blk,
     72 	.flush		= dos_flush
     73 };
     74 
     75 io_manager dos_io_manager = &struct_dos_manager;
     76 
     77 /*
     78  * Macro taken from unix_io.c
     79  */
     80 /*
     81  * For checking structure magic numbers...
     82  */
     83 
     84 #define EXT2_CHECK_MAGIC(struct, code) \
     85           if ((struct)->magic != (code)) return (code)
     86 
     87 /*
     88  * Calculates a CHS address of a sector from its LBA
     89  * offset for the given partition.
     90  */
     91 static void lba2chs(unsigned long lba_addr, CHS *chs, PARTITION *part)
     92 {
     93   unsigned long      abss;
     94 
     95   chs->offset = lba_addr & 0x000001FF;
     96   abss = (lba_addr >> 9) + part->start;
     97   chs->cyl    = abss / (part->sects * part->heads);
     98   chs->head   = (abss / part->sects) % part->heads;
     99   chs->sector = (abss % part->sects) + 1;
    100 }
    101 
    102 #ifdef __TURBOC__
    103 #pragma argsused
    104 #endif
    105 /*
    106  * Scans the passed partition table looking for *pno partition
    107  * that has LINUX_EXT2FS type.
    108  *
    109  * TODO:
    110  * For partition numbers >5 Linux uses DOS extended partitions -
    111  * dive into them an return an appropriate entry. Also dive into
    112  * extended partitions when scanning for a first Linux/ext2fs.
    113  */
    114 static PTABLE_ENTRY *scan_partition_table(PTABLE_ENTRY *pentry,
    115                                           unsigned short phys,
    116                                           unsigned char *pno)
    117 {
    118   unsigned        i;
    119 
    120   if(*pno != 0xFF && *pno >= 5)
    121      return NULL; /* We don't support extended partitions for now */
    122 
    123   if(*pno != 0xFF)
    124   {
    125     if(pentry[*pno].type == LINUX_EXT2FS)
    126       return &pentry[*pno];
    127     else
    128     {
    129       if(!pentry[*pno].type)
    130         *pno = 0xFE;
    131       else if(pentry[*pno].type == LINUX_SWAP)
    132         *pno = 0xFD;
    133       return NULL;
    134     }
    135   }
    136 
    137   for(i = 0; i < 4; i++)
    138     if(pentry[i].type == LINUX_EXT2FS)
    139     {
    140       *pno = i;
    141       return &pentry[i];
    142     }
    143 
    144   return NULL;
    145 }
    146 
    147 /*
    148  * Allocate libext2fs structures associated with I/O manager
    149  */
    150 static io_channel alloc_io_channel(PARTITION *part)
    151 {
    152   io_channel     ioch;
    153 
    154   ioch = (io_channel)malloc(sizeof(struct struct_io_channel));
    155   if (!ioch)
    156 	  return NULL;
    157   memset(ioch, 0, sizeof(struct struct_io_channel));
    158   ioch->magic = EXT2_ET_MAGIC_IO_CHANNEL;
    159   ioch->manager = dos_io_manager;
    160   ioch->name = (char *)malloc(strlen(part->dev)+1);
    161   if (!ioch->name) {
    162 	  free(ioch);
    163 	  return NULL;
    164   }
    165   strcpy(ioch->name, part->dev);
    166   ioch->private_data = part;
    167   ioch->block_size = 1024; /* The smallest ext2fs block size */
    168   ioch->read_error = 0;
    169   ioch->write_error = 0;
    170 
    171   return ioch;
    172 }
    173 
    174 #ifdef __TURBOC__
    175 #pragma argsused
    176 #endif
    177 /*
    178  * Open the 'name' partition, initialize all information structures
    179  * we need to keep and create libext2fs I/O manager.
    180  */
    181 static errcode_t dos_open(const char *dev, int flags, io_channel *channel)
    182 {
    183   unsigned char  *tmp, sec[512];
    184   PARTITION      *part;
    185   PTABLE_ENTRY   *pent;
    186   PARTITION        **newparts;
    187 
    188   if(!dev)
    189   {
    190     _dio_error = ERR_BADDEV;
    191     return EXT2_ET_BAD_DEVICE_NAME;
    192   }
    193 
    194   /*
    195    * First check whether the dev name is OK
    196    */
    197   tmp = (unsigned char*)strrchr(dev, '/');
    198   if(!tmp)
    199   {
    200     _dio_error = ERR_BADDEV;
    201     return EXT2_ET_BAD_DEVICE_NAME;
    202   }
    203   *tmp = 0;
    204   if(strcmp(dev, "/dev"))
    205   {
    206     _dio_error = ERR_BADDEV;
    207     return EXT2_ET_BAD_DEVICE_NAME;
    208   }
    209   *tmp++ = '/';
    210 
    211   /*
    212    * Check whether the partition data is already in cache
    213    */
    214 
    215   part = (PARTITION*)malloc(sizeof(PARTITION));
    216   if (!part)
    217 	  return ENOMEM;
    218   {
    219     int   i = 0;
    220 
    221     for(;i < npart; i++)
    222       if(!strcmp(partitions[i]->dev, dev))
    223       {
    224         /* Found it! Make it the active one */
    225         active = partitions[i];
    226         *channel = alloc_io_channel(active);
    227 	if (!*channel)
    228 		return ENOMEM;
    229         return 0;
    230       }
    231   }
    232 
    233   /*
    234    * Drive number & optionally partn number
    235    */
    236   switch(tmp[0])
    237   {
    238     case 'h':
    239     case 's':
    240       part->phys = 0x80;
    241       part->phys += toupper(tmp[2]) - 'A';
    242       /*
    243        * Do we have the partition number?
    244        */
    245       if(tmp[3])
    246         part->pno = isdigit((int)tmp[3]) ? tmp[3] - '0' - 1: 0;
    247       else
    248         part->pno = 0xFF;
    249       break;
    250 
    251     case 'f':
    252       if(tmp[2])
    253         part->phys = isdigit((int)tmp[2]) ? tmp[2] - '0' : 0;
    254       else
    255         part->phys = 0x00; /* We'll assume /dev/fd0 */
    256       break;
    257 
    258     default:
    259       _dio_error = ERR_BADDEV;
    260       return ENODEV;
    261   }
    262 
    263   if(part->phys < 0x80)
    264   {
    265      /* We don't support floppies for now */
    266      _dio_error = ERR_NOTSUPP;
    267      return EINVAL;
    268   }
    269 
    270   part->dev = strdup(dev);
    271 
    272   /*
    273    * Get drive's geometry
    274    */
    275   _dio_hw_error = biosdisk(DISK_GET_GEOMETRY,
    276                            part->phys,
    277                            0, /* head */
    278                            0, /* cylinder */
    279                            1, /* sector */
    280                            1, /* just one sector */
    281                            sec);
    282 
    283   if(!HW_OK())
    284   {
    285     _dio_error = ERR_HARDWARE;
    286     free(part->dev);
    287     free(part);
    288     return EFAULT;
    289   }
    290 
    291   /*
    292    * Calculate the geometry
    293    */
    294   part->cyls  = (unsigned short)(((sec[0] >> 6) << 8) + sec[1] + 1);
    295   part->heads = sec[3] + 1;
    296   part->sects = sec[0] & 0x3F;
    297 
    298   /*
    299    * Now that we know all we need, let's look for the partition
    300    */
    301   _dio_hw_error = biosdisk(DISK_READ, part->phys, 0, 0, 1, 1, sec);
    302 
    303   if(!HW_OK())
    304   {
    305     _dio_error = ERR_HARDWARE;
    306     free(part->dev);
    307     free(part);
    308     return EFAULT;
    309   }
    310 
    311   pent = (PTABLE_ENTRY*)&sec[0x1BE];
    312   pent = scan_partition_table(pent, part->phys, &part->pno);
    313 
    314   if(!pent)
    315   {
    316     _dio_error = part->pno == 0xFE ? ERR_EMPTYPART :
    317                  part->pno == 0xFD ? ERR_LINUXSWAP : ERR_NOTEXT2FS;
    318     free(part->dev);
    319     free(part);
    320     return ENODEV;
    321   }
    322 
    323   /*
    324    * Calculate the remaining figures
    325    */
    326   {
    327     unsigned long    fsec, fhead, fcyl;
    328 
    329     fsec = (unsigned long)(pent->start_sec & 0x3F);
    330     fhead = (unsigned long)pent->start_head;
    331     fcyl = ((pent->start_sec >> 6) << 8) + pent->start_cyl;
    332     part->start = fsec + fhead * part->sects + fcyl *
    333                   (part->heads * part->sects) - 1;
    334     part->len = pent->size;
    335   }
    336 
    337   /*
    338    * Add the partition to the table
    339    */
    340   newparts = (PARTITION**)realloc(partitions, sizeof(PARTITION) * npart);
    341   if (!newparts) {
    342 	  free(part);
    343 	  return ENOMEM;
    344   }
    345   partitions = newparts;
    346   partitions[npart++] = active = part;
    347 
    348   /*
    349    * Now alloc all libe2fs structures
    350    */
    351   *channel = alloc_io_channel(active);
    352   if (!*channel)
    353 	  return ENOMEM;
    354 
    355   return 0;
    356 }
    357 
    358 static errcode_t dos_close(io_channel channel)
    359 {
    360 	free(channel->name);
    361 	free(channel);
    362 
    363 	return 0;
    364 }
    365 
    366 static errcode_t dos_set_blksize(io_channel channel, int blksize)
    367 {
    368   channel->block_size = blksize;
    369 
    370   return 0;
    371 }
    372 
    373 static errcode_t dos_read_blk(io_channel channel, unsigned long block,
    374                                              int count, void *buf)
    375 {
    376   PARTITION     *part;
    377   size_t        size;
    378   ext2_loff_t   loc;
    379   CHS           chs;
    380 
    381   EXT2_CHECK_MAGIC(channel, EXT2_ET_MAGIC_IO_CHANNEL);
    382   part = (PARTITION*)channel->private_data;
    383 
    384   size = (size_t)((count < 0) ? -count : count * channel->block_size);
    385   loc = (ext2_loff_t) block * channel->block_size;
    386 
    387   lba2chs(loc, &chs, part);
    388   /*
    389    * Potential bug here:
    390    *   If DJGPP is used then reads of >18 sectors will fail!
    391    *   Have to rewrite biosdisk.
    392    */
    393   _dio_hw_error = biosdisk(DISK_READ,
    394                            part->phys,
    395                            chs.head,
    396                            chs.cyl,
    397                            chs.sector,
    398                            size < 512 ? 1 : size/512,
    399                            buf);
    400 
    401   if(!HW_OK())
    402   {
    403     _dio_error = ERR_HARDWARE;
    404     return EFAULT;
    405   }
    406 
    407   return 0;
    408 }
    409 
    410 static errcode_t dos_write_blk(io_channel channel, unsigned long block,
    411                                int count, const void *buf)
    412 {
    413   PARTITION     *part;
    414   size_t        size;
    415   ext2_loff_t   loc;
    416   CHS           chs;
    417 
    418   EXT2_CHECK_MAGIC(channel, EXT2_ET_MAGIC_IO_CHANNEL);
    419   part = (PARTITION*)channel->private_data;
    420 
    421   if(count == 1)
    422     size = (size_t)channel->block_size;
    423   else
    424   {
    425     if (count < 0)
    426       size = (size_t)-count;
    427     else
    428       size = (size_t)(count * channel->block_size);
    429   }
    430 
    431   loc = (ext2_loff_t)block * channel->block_size;
    432   lba2chs(loc, &chs, part);
    433   _dio_hw_error = biosdisk(DISK_WRITE,
    434                            part->phys,
    435                            chs.head,
    436                            chs.cyl,
    437                            chs.sector,
    438                            size < 512 ? 1 : size/512,
    439                            (void*)buf);
    440 
    441   if(!HW_OK())
    442   {
    443     _dio_error = ERR_HARDWARE;
    444     return EFAULT;
    445   }
    446 
    447   return 0;
    448 }
    449 
    450 #ifdef __TURBOC__
    451 #pragma argsused
    452 #endif
    453 static errcode_t dos_flush(io_channel channel)
    454 {
    455   /*
    456    * No buffers, no flush...
    457    */
    458   return 0;
    459 }
    460