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