aboutsummaryrefslogtreecommitdiff
path: root/e2fslib/dosio.c
diff options
context:
space:
mode:
Diffstat (limited to 'e2fslib/dosio.c')
-rw-r--r--e2fslib/dosio.c456
1 files changed, 0 insertions, 456 deletions
diff --git a/e2fslib/dosio.c b/e2fslib/dosio.c
deleted file mode 100644
index d695b18..0000000
--- a/e2fslib/dosio.c
+++ /dev/null
@@ -1,456 +0,0 @@
-/*
- * dosio.c -- Disk I/O module for the ext2fs/DOS library.
- *
- * Copyright (c) 1997 by Theodore Ts'o.
- *
- * Copyright (c) 1997 Mark Habersack
- * This file may be distributed under the terms of the GNU Public License.
- *
- */
-
-#include <stdio.h>
-#include <bios.h>
-#include <string.h>
-#include <ctype.h>
-#include <io.h>
-#ifdef HAVE_ERRNO_H
-#include <errno.h>
-#endif
-
-#include <ext2fs/ext2_types.h>
-#include "utils.h"
-#include "dosio.h"
-#include "et/com_err.h"
-#include "ext2_err.h"
-#include "ext2fs/io.h"
-
-/*
- * Some helper macros
- */
-#define LINUX_EXT2FS 0x83
-#define LINUX_SWAP 0x82
-#define WRITE_ERR(_msg_) write(2, _msg_, strlen(_msg_))
-#define WRITE_ERR_S(_msg_) write(2, _msg_, sizeof(_msg_))
-
-/*
- * Exported variables
- */
-unsigned long _dio_error;
-unsigned long _dio_hw_error;
-
-/*
- * Array of all opened partitions
- */
-static PARTITION **partitions = NULL;
-static unsigned short npart = 0; /* Number of mapped partitions */
-static PARTITION *active = NULL;
-
-/*
- * I/O Manager routine prototypes
- */
-static errcode_t dos_open(const char *dev, int flags, io_channel *channel);
-static errcode_t dos_close(io_channel channel);
-static errcode_t dos_set_blksize(io_channel channel, int blksize);
-static errcode_t dos_read_blk(io_channel channel, unsigned long block,
- int count, void *buf);
-static errcode_t dos_write_blk(io_channel channel, unsigned long block,
- int count, const void *buf);
-static errcode_t dos_flush(io_channel channel);
-
-static struct struct_io_manager struct_dos_manager = {
- EXT2_ET_MAGIC_IO_MANAGER,
- "DOS I/O Manager",
- dos_open,
- dos_close,
- dos_set_blksize,
- dos_read_blk,
- dos_write_blk,
- dos_flush
-};
-io_manager dos_io_manager = &struct_dos_manager;
-
-/*
- * Macro taken from unix_io.c
- */
-/*
- * For checking structure magic numbers...
- */
-
-#define EXT2_CHECK_MAGIC(struct, code) \
- if ((struct)->magic != (code)) return (code)
-
-/*
- * Calculates a CHS address of a sector from its LBA
- * offset for the given partition.
- */
-static void lba2chs(unsigned long lba_addr, CHS *chs, PARTITION *part)
-{
- unsigned long abss;
-
- chs->offset = lba_addr & 0x000001FF;
- abss = (lba_addr >> 9) + part->start;
- chs->cyl = abss / (part->sects * part->heads);
- chs->head = (abss / part->sects) % part->heads;
- chs->sector = (abss % part->sects) + 1;
-}
-
-#ifdef __TURBOC__
-#pragma argsused
-#endif
-/*
- * Scans the passed partition table looking for *pno partition
- * that has LINUX_EXT2FS type.
- *
- * TODO:
- * For partition numbers >5 Linux uses DOS extended partitions -
- * dive into them an return an appropriate entry. Also dive into
- * extended partitions when scanning for a first Linux/ext2fs.
- */
-static PTABLE_ENTRY *scan_partition_table(PTABLE_ENTRY *pentry,
- unsigned short phys,
- unsigned char *pno)
-{
- unsigned i;
-
- if(*pno != 0xFF && *pno >= 5)
- return NULL; /* We don't support extended partitions for now */
-
- if(*pno != 0xFF)
- {
- if(pentry[*pno].type == LINUX_EXT2FS)
- return &pentry[*pno];
- else
- {
- if(!pentry[*pno].type)
- *pno = 0xFE;
- else if(pentry[*pno].type == LINUX_SWAP)
- *pno = 0xFD;
- return NULL;
- }
- }
-
- for(i = 0; i < 4; i++)
- if(pentry[i].type == LINUX_EXT2FS)
- {
- *pno = i;
- return &pentry[i];
- }
-
- return NULL;
-}
-
-/*
- * Allocate libext2fs structures associated with I/O manager
- */
-static io_channel alloc_io_channel(PARTITION *part)
-{
- io_channel ioch;
-
- ioch = (io_channel)malloc(sizeof(struct struct_io_channel));
- if (!ioch)
- return NULL;
- memset(ioch, 0, sizeof(struct struct_io_channel));
- ioch->magic = EXT2_ET_MAGIC_IO_CHANNEL;
- ioch->manager = dos_io_manager;
- ioch->name = (char *)malloc(strlen(part->dev)+1);
- if (!ioch->name) {
- free(ioch);
- return NULL;
- }
- strcpy(ioch->name, part->dev);
- ioch->private_data = part;
- ioch->block_size = 1024; /* The smallest ext2fs block size */
- ioch->read_error = 0;
- ioch->write_error = 0;
-
- return ioch;
-}
-
-#ifdef __TURBOC__
-#pragma argsused
-#endif
-/*
- * Open the 'name' partition, initialize all information structures
- * we need to keep and create libext2fs I/O manager.
- */
-static errcode_t dos_open(const char *dev, int flags, io_channel *channel)
-{
- unsigned char *tmp, sec[512];
- PARTITION *part;
- PTABLE_ENTRY *pent;
- PARTITION **newparts;
-
- if(!dev)
- {
- _dio_error = ERR_BADDEV;
- return EXT2_ET_BAD_DEVICE_NAME;
- }
-
- /*
- * First check whether the dev name is OK
- */
- tmp = (unsigned char*)strrchr(dev, '/');
- if(!tmp)
- {
- _dio_error = ERR_BADDEV;
- return EXT2_ET_BAD_DEVICE_NAME;
- }
- *tmp = 0;
- if(strcmp(dev, "/dev"))
- {
- _dio_error = ERR_BADDEV;
- return EXT2_ET_BAD_DEVICE_NAME;
- }
- *tmp++ = '/';
-
- /*
- * Check whether the partition data is already in cache
- */
-
- part = (PARTITION*)malloc(sizeof(PARTITION));
- if (!part)
- return ENOMEM;
- {
- int i = 0;
-
- for(;i < npart; i++)
- if(!strcmp(partitions[i]->dev, dev))
- {
- /* Found it! Make it the active one */
- active = partitions[i];
- *channel = alloc_io_channel(active);
- if (!*channel)
- return ENOMEM;
- return 0;
- }
- }
-
- /*
- * Drive number & optionally partn number
- */
- switch(tmp[0])
- {
- case 'h':
- case 's':
- part->phys = 0x80;
- part->phys += toupper(tmp[2]) - 'A';
- /*
- * Do we have the partition number?
- */
- if(tmp[3])
- part->pno = isdigit((int)tmp[3]) ? tmp[3] - '0' - 1: 0;
- else
- part->pno = 0xFF;
- break;
-
- case 'f':
- if(tmp[2])
- part->phys = isdigit((int)tmp[2]) ? tmp[2] - '0' : 0;
- else
- part->phys = 0x00; /* We'll assume /dev/fd0 */
- break;
-
- default:
- _dio_error = ERR_BADDEV;
- return ENODEV;
- }
-
- if(part->phys < 0x80)
- {
- /* We don't support floppies for now */
- _dio_error = ERR_NOTSUPP;
- return EINVAL;
- }
-
- part->dev = strdup(dev);
-
- /*
- * Get drive's geometry
- */
- _dio_hw_error = biosdisk(DISK_GET_GEOMETRY,
- part->phys,
- 0, /* head */
- 0, /* cylinder */
- 1, /* sector */
- 1, /* just one sector */
- sec);
-
- if(!HW_OK())
- {
- _dio_error = ERR_HARDWARE;
- if (part)
- free(part);
- return EFAULT;
- }
-
- /*
- * Calculate the geometry
- */
- part->cyls = (unsigned short)(((sec[0] >> 6) << 8) + sec[1] + 1);
- part->heads = sec[3] + 1;
- part->sects = sec[0] & 0x3F;
-
- /*
- * Now that we know all we need, let's look for the partition
- */
- _dio_hw_error = biosdisk(DISK_READ, part->phys, 0, 0, 1, 1, sec);
-
- if(!HW_OK())
- {
- _dio_error = ERR_HARDWARE;
- if (part)
- free(part);
- return EFAULT;
- }
-
- pent = (PTABLE_ENTRY*)&sec[0x1BE];
- pent = scan_partition_table(pent, part->phys, &part->pno);
-
- if(!pent)
- {
- _dio_error = part->pno == 0xFE ? ERR_EMPTYPART :
- part->pno == 0xFD ? ERR_LINUXSWAP : ERR_NOTEXT2FS;
- if (part)
- free(part);
- return ENODEV;
- }
-
- /*
- * Calculate the remaining figures
- */
- {
- unsigned long fsec, fhead, fcyl;
-
- fsec = (unsigned long)(pent->start_sec & 0x3F);
- fhead = (unsigned long)pent->start_head;
- fcyl = ((pent->start_sec >> 6) << 8) + pent->start_cyl;
- part->start = fsec + fhead * part->sects + fcyl *
- (part->heads * part->sects) - 1;
- part->len = pent->size;
- }
-
- /*
- * Add the partition to the table
- */
- newparts = (PARTITION**)realloc(partitions, sizeof(PARTITION) * npart);
- if (!newparts) {
- free(part);
- return ENOMEM;
- }
- partitions = newparts;
- partitions[npart++] = active = part;
-
- /*
- * Now alloc all libe2fs structures
- */
- *channel = alloc_io_channel(active);
- if (!*channel)
- return ENOMEM;
-
- return 0;
-}
-
-static errcode_t dos_close(io_channel channel)
-{
- if (channel->name)
- free(channel->name);
- if (channel)
- free(channel);
-
- return 0;
-}
-
-static errcode_t dos_set_blksize(io_channel channel, int blksize)
-{
- channel->block_size = blksize;
-
- return 0;
-}
-
-static errcode_t dos_read_blk(io_channel channel, unsigned long block,
- int count, void *buf)
-{
- PARTITION *part;
- size_t size;
- ext2_loff_t loc;
- CHS chs;
-
- EXT2_CHECK_MAGIC(channel, EXT2_ET_MAGIC_IO_CHANNEL);
- part = (PARTITION*)channel->private_data;
-
- size = (size_t)((count < 0) ? -count : count * channel->block_size);
- loc = (ext2_loff_t) block * channel->block_size;
-
- lba2chs(loc, &chs, part);
- /*
- * Potential bug here:
- * If DJGPP is used then reads of >18 sectors will fail!
- * Have to rewrite biosdisk.
- */
- _dio_hw_error = biosdisk(DISK_READ,
- part->phys,
- chs.head,
- chs.cyl,
- chs.sector,
- size < 512 ? 1 : size/512,
- buf);
-
- if(!HW_OK())
- {
- _dio_error = ERR_HARDWARE;
- return EFAULT;
- }
-
- return 0;
-}
-
-static errcode_t dos_write_blk(io_channel channel, unsigned long block,
- int count, const void *buf)
-{
- PARTITION *part;
- size_t size;
- ext2_loff_t loc;
- CHS chs;
-
- EXT2_CHECK_MAGIC(channel, EXT2_ET_MAGIC_IO_CHANNEL);
- part = (PARTITION*)channel->private_data;
-
- if(count == 1)
- size = (size_t)channel->block_size;
- else
- {
- if (count < 0)
- size = (size_t)-count;
- else
- size = (size_t)(count * channel->block_size);
- }
-
- loc = (ext2_loff_t)block * channel->block_size;
- lba2chs(loc, &chs, part);
- _dio_hw_error = biosdisk(DISK_WRITE,
- part->phys,
- chs.head,
- chs.cyl,
- chs.sector,
- size < 512 ? 1 : size/512,
- (void*)buf);
-
- if(!HW_OK())
- {
- _dio_error = ERR_HARDWARE;
- return EFAULT;
- }
-
- return 0;
-}
-
-#ifdef __TURBOC__
-#pragma argsused
-#endif
-static errcode_t dos_flush(io_channel channel)
-{
- /*
- * No buffers, no flush...
- */
- return 0;
-}