mirror of
https://github.com/xcat2/xNBA.git
synced 2025-01-11 18:27:56 +00:00
Refactored to reduce code size; now down from 1304 to 1105 bytes.
Tidied up debug messages; the log now contains one line per INT 13 operation, looking like INT 13,08 (80): Get drive parameters INT 13,02 (80): Read: C/H/S 0/47/14 = LBA 0xb9e <-> 1084:0000 (count 106)
This commit is contained in:
parent
af4aacb978
commit
ca2519cea3
@ -64,89 +64,6 @@ extern void int13_exec_fail ( void );
|
||||
/** List of registered emulated drives */
|
||||
static LIST_HEAD ( drives );
|
||||
|
||||
/**
|
||||
* Convert CHS address to linear address
|
||||
*
|
||||
* @v drive Emulated drive
|
||||
* @v ch Low bits of cylinder number
|
||||
* @v cl (bits 7:6) High bits of cylinder number
|
||||
* @v cl (bits 5:0) Sector number
|
||||
* @v dh Head number
|
||||
* @ret lba LBA address
|
||||
*
|
||||
*/
|
||||
static unsigned long chs_to_lba ( struct int13_drive *drive,
|
||||
struct i386_all_regs *ix86 ) {
|
||||
unsigned int cylinder;
|
||||
unsigned int head;
|
||||
unsigned int sector;
|
||||
unsigned long lba;
|
||||
|
||||
cylinder = ( ( ( ix86->regs.cl & 0xc0 ) << 8 ) | ix86->regs.ch );
|
||||
head = ix86->regs.dh;
|
||||
sector = ( ix86->regs.cl & 0x3f );
|
||||
|
||||
assert ( cylinder < drive->cylinders );
|
||||
assert ( head < drive->heads );
|
||||
assert ( ( sector >= 1 ) && ( sector <= drive->sectors_per_track ) );
|
||||
|
||||
lba = ( ( ( ( cylinder * drive->heads ) + head )
|
||||
* drive->sectors_per_track ) + sector - 1 );
|
||||
|
||||
DBG ( "C/H/S address %x/%x/%x -> LBA %lx\n",
|
||||
cylinder, head, sector, lba );
|
||||
|
||||
return lba;
|
||||
}
|
||||
|
||||
/**
|
||||
* Read from drive to real-mode data buffer
|
||||
*
|
||||
* @v drive Emulated drive
|
||||
* @v lba LBA starting sector number
|
||||
* @v data Data buffer
|
||||
* @v count Number of sectors to read
|
||||
* @ret status Status code
|
||||
*/
|
||||
static int int13_read ( struct int13_drive *drive, uint64_t lba,
|
||||
struct segoff data, unsigned long count ) {
|
||||
struct block_device *blockdev = drive->blockdev;
|
||||
userptr_t buffer = real_to_user ( data.segment, data.offset );
|
||||
int rc;
|
||||
|
||||
DBG ( "Read %lx sectors from %llx to %04x:%04x\n", count,
|
||||
( unsigned long long ) lba, data.segment, data.offset );
|
||||
|
||||
if ( ( rc = blockdev->read ( blockdev, lba, count, buffer ) ) != 0 )
|
||||
return INT13_STATUS_READ_ERROR;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Write from real-mode data buffer to drive
|
||||
*
|
||||
* @v drive Emulated drive
|
||||
* @v lba LBA starting sector number
|
||||
* @v data Data buffer
|
||||
* @v count Number of sectors to read
|
||||
* @ret status Status code
|
||||
*/
|
||||
static int int13_write ( struct int13_drive *drive, uint64_t lba,
|
||||
struct segoff data, unsigned long count ) {
|
||||
struct block_device *blockdev = drive->blockdev;
|
||||
userptr_t buffer = real_to_user ( data.segment, data.offset );
|
||||
int rc;
|
||||
|
||||
DBG ( "Write %lx sectors from %04x:%04x to %llx\n", count,
|
||||
data.segment, data.offset, ( unsigned long long ) lba );
|
||||
|
||||
if ( ( rc = blockdev->write ( blockdev, lba, count, buffer ) ) != 0 )
|
||||
return INT13_STATUS_WRITE_ERROR;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* INT 13, 00 - Reset disk system
|
||||
*
|
||||
@ -171,6 +88,61 @@ static int int13_get_last_status ( struct int13_drive *drive,
|
||||
return drive->last_status;
|
||||
}
|
||||
|
||||
/**
|
||||
* Read / write sectors
|
||||
*
|
||||
* @v drive Emulated drive
|
||||
* @v al Number of sectors to read or write (must be nonzero)
|
||||
* @v ch Low bits of cylinder number
|
||||
* @v cl (bits 7:6) High bits of cylinder number
|
||||
* @v cl (bits 5:0) Sector number
|
||||
* @v dh Head number
|
||||
* @v es:bx Data buffer
|
||||
* @v io Read / write method
|
||||
* @ret status Status code
|
||||
* @ret al Number of sectors read or written
|
||||
*/
|
||||
static int int13_rw_sectors ( struct int13_drive *drive,
|
||||
struct i386_all_regs *ix86,
|
||||
int ( * io ) ( struct block_device *blockdev,
|
||||
uint64_t block,
|
||||
unsigned long count,
|
||||
userptr_t buffer ) ) {
|
||||
struct block_device *blockdev = drive->blockdev;
|
||||
unsigned int cylinder, head, sector;
|
||||
unsigned long lba;
|
||||
unsigned int count;
|
||||
userptr_t buffer;
|
||||
|
||||
/* Calculate parameters */
|
||||
cylinder = ( ( ( ix86->regs.cl & 0xc0 ) << 8 ) | ix86->regs.ch );
|
||||
assert ( cylinder < drive->cylinders );
|
||||
head = ix86->regs.dh;
|
||||
assert ( head < drive->heads );
|
||||
sector = ( ix86->regs.cl & 0x3f );
|
||||
assert ( ( sector >= 1 ) && ( sector <= drive->sectors_per_track ) );
|
||||
lba = ( ( ( ( cylinder * drive->heads ) + head )
|
||||
* drive->sectors_per_track ) + sector - 1 );
|
||||
count = ix86->regs.al;
|
||||
buffer = real_to_user ( ix86->segs.es, ix86->regs.bx );
|
||||
|
||||
DBG ( "C/H/S %d/%d/%d = LBA %#lx <-> %04x:%04x (count %d)\n", cylinder,
|
||||
head, sector, lba, ix86->segs.es, ix86->regs.bx, count );
|
||||
|
||||
/* Validate blocksize */
|
||||
if ( blockdev->blksize != INT13_BLKSIZE ) {
|
||||
DBG ( "Invalid blocksize (%zd) for non-extended read/write\n",
|
||||
blockdev->blksize );
|
||||
return INT13_STATUS_INVALID;
|
||||
}
|
||||
|
||||
/* Read from / write to block device */
|
||||
if ( io ( blockdev, lba, count, buffer ) != 0 )
|
||||
return INT13_STATUS_READ_ERROR;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* INT 13, 02 - Read sectors
|
||||
*
|
||||
@ -186,20 +158,8 @@ static int int13_get_last_status ( struct int13_drive *drive,
|
||||
*/
|
||||
static int int13_read_sectors ( struct int13_drive *drive,
|
||||
struct i386_all_regs *ix86 ) {
|
||||
unsigned long lba = chs_to_lba ( drive, ix86 );
|
||||
unsigned int count = ix86->regs.al;
|
||||
struct segoff data = {
|
||||
.segment = ix86->segs.es,
|
||||
.offset = ix86->regs.bx,
|
||||
};
|
||||
|
||||
if ( drive->blockdev->blksize != INT13_BLKSIZE ) {
|
||||
DBG ( "Invalid blocksize (%zd) for non-extended read\n",
|
||||
drive->blockdev->blksize );
|
||||
return INT13_STATUS_INVALID;
|
||||
}
|
||||
|
||||
return int13_read ( drive, lba, data, count );
|
||||
DBG ( "Read: " );
|
||||
return int13_rw_sectors ( drive, ix86, drive->blockdev->read );
|
||||
}
|
||||
|
||||
/**
|
||||
@ -217,20 +177,8 @@ static int int13_read_sectors ( struct int13_drive *drive,
|
||||
*/
|
||||
static int int13_write_sectors ( struct int13_drive *drive,
|
||||
struct i386_all_regs *ix86 ) {
|
||||
unsigned long lba = chs_to_lba ( drive, ix86 );
|
||||
unsigned int count = ix86->regs.al;
|
||||
struct segoff data = {
|
||||
.segment = ix86->segs.es,
|
||||
.offset = ix86->regs.bx,
|
||||
};
|
||||
|
||||
if ( drive->blockdev->blksize != INT13_BLKSIZE ) {
|
||||
DBG ( "Invalid blocksize (%zd) for non-extended write\n",
|
||||
drive->blockdev->blksize );
|
||||
return INT13_STATUS_INVALID;
|
||||
}
|
||||
|
||||
return int13_write ( drive, lba, data, count );
|
||||
DBG ( "Write: " );
|
||||
return int13_rw_sectors ( drive, ix86, drive->blockdev->write );
|
||||
}
|
||||
|
||||
/**
|
||||
@ -280,6 +228,42 @@ static int int13_extension_check ( struct int13_drive *drive __unused,
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Extended read / write
|
||||
*
|
||||
* @v drive Emulated drive
|
||||
* @v ds:si Disk address packet
|
||||
* @v io Read / write method
|
||||
* @ret status Status code
|
||||
*/
|
||||
static int int13_extended_rw ( struct int13_drive *drive,
|
||||
struct i386_all_regs *ix86,
|
||||
int ( * io ) ( struct block_device *blockdev,
|
||||
uint64_t block,
|
||||
unsigned long count,
|
||||
userptr_t buffer ) ) {
|
||||
struct block_device *blockdev = drive->blockdev;
|
||||
struct int13_disk_address addr;
|
||||
uint64_t lba;
|
||||
unsigned long count;
|
||||
userptr_t buffer;
|
||||
|
||||
/* Read parameters from disk address structure */
|
||||
copy_from_real ( &addr, ix86->segs.ds, ix86->regs.si, sizeof ( addr ));
|
||||
lba = addr.lba;
|
||||
count = addr.count;
|
||||
buffer = real_to_user ( addr.buffer.segment, addr.buffer.offset );
|
||||
|
||||
DBG ( "LBA %#llx <-> %04x:%04x (count %ld)\n", (unsigned long long)lba,
|
||||
addr.buffer.segment, addr.buffer.offset, count );
|
||||
|
||||
/* Read from / write to block device */
|
||||
if ( io ( blockdev, lba, count, buffer ) != 0 )
|
||||
return INT13_STATUS_READ_ERROR;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* INT 13, 42 - Extended read
|
||||
*
|
||||
@ -289,11 +273,8 @@ static int int13_extension_check ( struct int13_drive *drive __unused,
|
||||
*/
|
||||
static int int13_extended_read ( struct int13_drive *drive,
|
||||
struct i386_all_regs *ix86 ) {
|
||||
struct int13_disk_address addr;
|
||||
|
||||
copy_from_real ( &addr, ix86->segs.ds, ix86->regs.si,
|
||||
sizeof ( addr ) );
|
||||
return int13_read ( drive, addr.lba, addr.buffer, addr.count );
|
||||
DBG ( "Extended read: " );
|
||||
return int13_extended_rw ( drive, ix86, drive->blockdev->read );
|
||||
}
|
||||
|
||||
/**
|
||||
@ -305,11 +286,8 @@ static int int13_extended_read ( struct int13_drive *drive,
|
||||
*/
|
||||
static int int13_extended_write ( struct int13_drive *drive,
|
||||
struct i386_all_regs *ix86 ) {
|
||||
struct int13_disk_address addr;
|
||||
|
||||
copy_from_real ( &addr, ix86->segs.ds, ix86->regs.si,
|
||||
sizeof ( addr ) );
|
||||
return int13_write ( drive, addr.lba, addr.buffer, addr.count );
|
||||
DBG ( "Extended write: " );
|
||||
return int13_extended_rw ( drive, ix86, drive->blockdev->write );
|
||||
}
|
||||
|
||||
/**
|
||||
@ -351,7 +329,7 @@ static void int13 ( struct i386_all_regs *ix86 ) {
|
||||
if ( drive->drive != ix86->regs.dl )
|
||||
continue;
|
||||
|
||||
DBG ( "INT 13, %02x on drive %02x\n", ix86->regs.ah,
|
||||
DBG ( "INT 13,%02x (%02x): ", ix86->regs.ah,
|
||||
ix86->regs.dl );
|
||||
|
||||
switch ( ix86->regs.ah ) {
|
||||
|
Loading…
x
Reference in New Issue
Block a user