- multi sector read/write commands implemented

- ATA command 'set multiple mode' fixed (missing interrupt and value check)
- hard disk sector read/write rewritten and simplified
- hard disk now reporting ATA-3 supported
- updated drive buffer size for save/restore
This commit is contained in:
Volker Ruppert 2006-07-17 18:40:26 +00:00
parent 93d8cc2b8a
commit 92412a3fee
2 changed files with 154 additions and 203 deletions

View File

@ -1,5 +1,5 @@
///////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////
// $Id: harddrv.cc,v 1.175 2006-07-14 17:23:58 vruppert Exp $ // $Id: harddrv.cc,v 1.176 2006-07-17 18:40:26 vruppert Exp $
///////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////
// //
// Copyright (C) 2002 MandrakeSoft S.A. // Copyright (C) 2002 MandrakeSoft S.A.
@ -140,7 +140,7 @@ void bx_hard_drive_c::init(void)
char ata_name[20]; char ata_name[20];
bx_list_c *base; bx_list_c *base;
BX_DEBUG(("Init $Id: harddrv.cc,v 1.175 2006-07-14 17:23:58 vruppert Exp $")); BX_DEBUG(("Init $Id: harddrv.cc,v 1.176 2006-07-17 18:40:26 vruppert Exp $"));
for (channel=0; channel<BX_MAX_ATA_CHANNEL; channel++) { for (channel=0; channel<BX_MAX_ATA_CHANNEL; channel++) {
sprintf(ata_name, "ata.%d.resources", channel); sprintf(ata_name, "ata.%d.resources", channel);
@ -631,7 +631,7 @@ void bx_hard_drive_c::register_state(void)
if (BX_DRIVE_IS_PRESENT(i, j)) { if (BX_DRIVE_IS_PRESENT(i, j)) {
sprintf(dname, "drive%d", i); sprintf(dname, "drive%d", i);
drive = new bx_list_c(chan, dname, 20); drive = new bx_list_c(chan, dname, 20);
new bx_shadow_data_c(drive, "buffer", BX_CONTROLLER(i, j).buffer, 2352); new bx_shadow_data_c(drive, "buffer", BX_CONTROLLER(i, j).buffer, MAX_MULTIPLE_SECTORS * 512 + 4);
status = new bx_list_c(drive, "status", 9); status = new bx_list_c(drive, "status", 9);
new bx_shadow_bool_c(status, "busy", &BX_CONTROLLER(i, j).status.busy); new bx_shadow_bool_c(status, "busy", &BX_CONTROLLER(i, j).status.busy);
new bx_shadow_bool_c(status, "drive_ready", &BX_CONTROLLER(i, j).status.drive_ready); new bx_shadow_bool_c(status, "drive_ready", &BX_CONTROLLER(i, j).status.drive_ready);
@ -717,7 +717,6 @@ Bit32u bx_hard_drive_c::read(Bit32u address, unsigned io_len)
Bit8u value8; Bit8u value8;
Bit16u value16; Bit16u value16;
Bit32u value32; Bit32u value32;
Bit32u counter;
Bit8u channel = BX_MAX_ATA_CHANNEL; Bit8u channel = BX_MAX_ATA_CHANNEL;
Bit32u port = 0xff; // undefined Bit32u port = 0xff; // undefined
@ -799,16 +798,7 @@ Bit32u bx_hard_drive_c::read(Bit32u address, unsigned io_len)
// drive, head, status // drive, head, status
// if there are more sectors, read next one in... // if there are more sectors, read next one in...
// //
BX_SELECTED_CONTROLLER(channel).buffer_index = 0; if (BX_SELECTED_CONTROLLER(channel).current_command == 0xC4) {
if (BX_SELECTED_CONTROLLER(channel).current_command != 0xC4) {
increment_address(channel);
} else {
counter = BX_SELECTED_CONTROLLER(channel).buffer_size;
do {
increment_address(channel);
counter -= 512;
} while (counter > 0);
if (BX_SELECTED_CONTROLLER(channel).sector_count > BX_SELECTED_CONTROLLER(channel).multiple_sectors) { if (BX_SELECTED_CONTROLLER(channel).sector_count > BX_SELECTED_CONTROLLER(channel).multiple_sectors) {
BX_SELECTED_CONTROLLER(channel).buffer_size = BX_SELECTED_CONTROLLER(channel).multiple_sectors * 512; BX_SELECTED_CONTROLLER(channel).buffer_size = BX_SELECTED_CONTROLLER(channel).multiple_sectors * 512;
} else { } else {
@ -825,44 +815,18 @@ Bit32u bx_hard_drive_c::read(Bit32u address, unsigned io_len)
if (BX_SELECTED_CONTROLLER(channel).sector_count==0) { if (BX_SELECTED_CONTROLLER(channel).sector_count==0) {
BX_SELECTED_CONTROLLER(channel).status.drq = 0; BX_SELECTED_CONTROLLER(channel).status.drq = 0;
} } else { /* read next one into controller buffer */
else { /* read next one into controller buffer */
off_t logical_sector;
off_t ret;
BX_SELECTED_CONTROLLER(channel).status.drq = 1; BX_SELECTED_CONTROLLER(channel).status.drq = 1;
BX_SELECTED_CONTROLLER(channel).status.seek_complete = 1; BX_SELECTED_CONTROLLER(channel).status.seek_complete = 1;
if (!calculate_logical_address(channel, &logical_sector)) { if (ide_read_sector(channel, BX_SELECTED_CONTROLLER(channel).buffer,
BX_ERROR(("multi-sector read reached invalid sector %lu, aborting", (unsigned long)logical_sector)); BX_SELECTED_CONTROLLER(channel).buffer_size)) {
command_aborted (channel, BX_SELECTED_CONTROLLER(channel).current_command); BX_SELECTED_CONTROLLER(channel).buffer_index = 0;
GOTO_RETURN_VALUE; raise_interrupt(channel);
} }
ret = BX_SELECTED_DRIVE(channel).hard_drive->lseek(logical_sector * 512, SEEK_SET); }
if (ret < 0) { }
BX_ERROR(("could not lseek() hard drive image file")); GOTO_RETURN_VALUE;
command_aborted (channel, BX_SELECTED_CONTROLLER(channel).current_command);
GOTO_RETURN_VALUE;
}
/* set status bar conditions for device */
if (!BX_SELECTED_DRIVE(channel).iolight_counter)
bx_gui->statusbar_setitem(BX_SELECTED_DRIVE(channel).statusbar_id, 1);
BX_SELECTED_DRIVE(channel).iolight_counter = 5;
bx_pc_system.activate_timer( BX_HD_THIS iolight_timer_index, 100000, 0 );
ret = BX_SELECTED_DRIVE(channel).hard_drive->read((bx_ptr_t) BX_SELECTED_CONTROLLER(channel).buffer,
BX_SELECTED_CONTROLLER(channel).buffer_size);
if (ret < BX_SELECTED_CONTROLLER(channel).buffer_size) {
BX_ERROR(("logical sector was %lu", (unsigned long)logical_sector));
BX_ERROR(("could not read() hard drive image file at byte %lu", (unsigned long)logical_sector*512));
command_aborted (channel, BX_SELECTED_CONTROLLER(channel).current_command);
GOTO_RETURN_VALUE;
}
BX_SELECTED_CONTROLLER(channel).buffer_index = 0;
raise_interrupt(channel);
}
}
GOTO_RETURN_VALUE;
break; break;
case 0xec: // IDENTIFY DEVICE case 0xec: // IDENTIFY DEVICE
@ -1206,7 +1170,6 @@ void bx_hard_drive_c::write(Bit32u address, Bit32u value, unsigned io_len)
UNUSED(this_ptr); UNUSED(this_ptr);
#endif // !BX_USE_HD_SMF #endif // !BX_USE_HD_SMF
off_t logical_sector; off_t logical_sector;
off_t ret;
bx_bool prev_control_reset; bx_bool prev_control_reset;
Bit8u channel = BX_MAX_ATA_CHANNEL; Bit8u channel = BX_MAX_ATA_CHANNEL;
@ -1262,15 +1225,16 @@ void bx_hard_drive_c::write(Bit32u address, Bit32u value, unsigned io_len)
case 0x00: // 0x1f0 case 0x00: // 0x1f0
switch (BX_SELECTED_CONTROLLER(channel).current_command) { switch (BX_SELECTED_CONTROLLER(channel).current_command) {
case 0x30: // WRITE SECTORS case 0x30: // WRITE SECTORS
if (BX_SELECTED_CONTROLLER(channel).buffer_index >= 512) case 0xC5: // WRITE MULTIPLE SECTORS
BX_PANIC(("IO write(0x%04x): buffer_index >= 512", address)); if (BX_SELECTED_CONTROLLER(channel).buffer_index >= BX_SELECTED_CONTROLLER(channel).buffer_size)
BX_PANIC(("IO write(0x%04x): buffer_index >= %d", address, BX_SELECTED_CONTROLLER(channel).buffer_size));
#if BX_SupportRepeatSpeedups #if BX_SupportRepeatSpeedups
if (DEV_bulk_io_quantum_requested()) { if (DEV_bulk_io_quantum_requested()) {
unsigned transferLen, quantumsMax; unsigned transferLen, quantumsMax;
quantumsMax = quantumsMax =
(512 - BX_SELECTED_CONTROLLER(channel).buffer_index) / io_len; (BX_SELECTED_CONTROLLER(channel).buffer_size - BX_SELECTED_CONTROLLER(channel).buffer_index) / io_len;
if ( quantumsMax == 0) if ( quantumsMax == 0)
BX_PANIC(("IO write(0x%04x): not enough space for write", address)); BX_PANIC(("IO write(0x%04x): not enough space for write", address));
DEV_bulk_io_quantum_transferred() = DEV_bulk_io_quantum_transferred() =
@ -1300,62 +1264,39 @@ void bx_hard_drive_c::write(Bit32u address, Bit32u value, unsigned io_len)
} }
/* if buffer completely writtten */ /* if buffer completely writtten */
if (BX_SELECTED_CONTROLLER(channel).buffer_index >= 512) { if (BX_SELECTED_CONTROLLER(channel).buffer_index >= BX_SELECTED_CONTROLLER(channel).buffer_size) {
off_t logical_sector; if (ide_write_sector(channel, BX_SELECTED_CONTROLLER(channel).buffer,
off_t ret; BX_SELECTED_CONTROLLER(channel).buffer_size)) {
if (BX_SELECTED_CONTROLLER(channel).current_command == 0xC5) {
if (BX_SELECTED_CONTROLLER(channel).sector_count > BX_SELECTED_CONTROLLER(channel).multiple_sectors) {
BX_SELECTED_CONTROLLER(channel).buffer_size = BX_SELECTED_CONTROLLER(channel).multiple_sectors * 512;
} else {
BX_SELECTED_CONTROLLER(channel).buffer_size = BX_SELECTED_CONTROLLER(channel).sector_count * 512;
}
}
BX_SELECTED_CONTROLLER(channel).buffer_index = 0;
if (!calculate_logical_address(channel, &logical_sector)) { /* When the write is complete, controller clears the DRQ bit and
BX_ERROR(("write reached invalid sector %lu, aborting", (unsigned long)logical_sector)); * sets the BSY bit.
command_aborted (channel, BX_SELECTED_CONTROLLER(channel).current_command); * If at least one more sector is to be written, controller sets DRQ bit,
return; * clears BSY bit, and issues IRQ
*/
if (BX_SELECTED_CONTROLLER(channel).sector_count!=0) {
BX_SELECTED_CONTROLLER(channel).status.busy = 0;
BX_SELECTED_CONTROLLER(channel).status.drive_ready = 1;
BX_SELECTED_CONTROLLER(channel).status.drq = 1;
BX_SELECTED_CONTROLLER(channel).status.corrected_data = 0;
BX_SELECTED_CONTROLLER(channel).status.err = 0;
} else { /* no more sectors to write */
BX_SELECTED_CONTROLLER(channel).status.busy = 0;
BX_SELECTED_CONTROLLER(channel).status.drive_ready = 1;
BX_SELECTED_CONTROLLER(channel).status.drq = 0;
BX_SELECTED_CONTROLLER(channel).status.err = 0;
BX_SELECTED_CONTROLLER(channel).status.corrected_data = 0;
}
raise_interrupt(channel);
} }
ret = BX_SELECTED_DRIVE(channel).hard_drive->lseek(logical_sector * 512, SEEK_SET);
if (ret < 0) {
BX_ERROR(("could not lseek() hard drive image file at byte %lu", (unsigned long)logical_sector * 512));
command_aborted (channel, BX_SELECTED_CONTROLLER(channel).current_command);
return;
}
/* set status bar conditions for device */
if (!BX_SELECTED_DRIVE(channel).iolight_counter)
bx_gui->statusbar_setitem(BX_SELECTED_DRIVE(channel).statusbar_id, 1);
BX_SELECTED_DRIVE(channel).iolight_counter = 5;
bx_pc_system.activate_timer( BX_HD_THIS iolight_timer_index, 100000, 0 );
ret = BX_SELECTED_DRIVE(channel).hard_drive->write((bx_ptr_t) BX_SELECTED_CONTROLLER(channel).buffer, 512);
if (ret < 512) {
BX_ERROR(("could not write() hard drive image file at byte %lu", (unsigned long)logical_sector*512));
command_aborted (channel, BX_SELECTED_CONTROLLER(channel).current_command);
return;
}
BX_SELECTED_CONTROLLER(channel).buffer_index = 0;
/* update sector count, sector number, cylinder,
* drive, head, status
* if there are more sectors, read next one in...
*/
increment_address(channel);
/* When the write is complete, controller clears the DRQ bit and
* sets the BSY bit.
* If at least one more sector is to be written, controller sets DRQ bit,
* clears BSY bit, and issues IRQ
*/
if (BX_SELECTED_CONTROLLER(channel).sector_count!=0) {
BX_SELECTED_CONTROLLER(channel).status.busy = 0;
BX_SELECTED_CONTROLLER(channel).status.drive_ready = 1;
BX_SELECTED_CONTROLLER(channel).status.drq = 1;
BX_SELECTED_CONTROLLER(channel).status.corrected_data = 0;
BX_SELECTED_CONTROLLER(channel).status.err = 0;
} else { /* no more sectors to write */
BX_SELECTED_CONTROLLER(channel).status.busy = 0;
BX_SELECTED_CONTROLLER(channel).status.drive_ready = 1;
BX_SELECTED_CONTROLLER(channel).status.drq = 0;
BX_SELECTED_CONTROLLER(channel).status.err = 0;
BX_SELECTED_CONTROLLER(channel).status.corrected_data = 0;
}
raise_interrupt(channel);
} }
break; break;
@ -2093,14 +2034,9 @@ void bx_hard_drive_c::write(Bit32u address, Bit32u value, unsigned io_len)
raise_interrupt(channel); raise_interrupt(channel);
break; break;
case 0xC4: // READ MULTIPLE SECTORS (present below, but not yet complete)
BX_ERROR(("write cmd 0xC4 (READ MULTIPLE) not supported yet"));
command_aborted(channel, 0xC4);
break;
case 0x20: // READ SECTORS, with retries case 0x20: // READ SECTORS, with retries
case 0x21: // READ SECTORS, without retries case 0x21: // READ SECTORS, without retries
// case 0xC4: // READ MULTIPLE SECTORS case 0xC4: // READ MULTIPLE SECTORS
/* update sector_no, always points to current sector /* update sector_no, always points to current sector
* after each sector is read to buffer, DRQ bit set and issue IRQ * after each sector is read to buffer, DRQ bit set and issue IRQ
* if interrupt handler transfers all data words into main memory, * if interrupt handler transfers all data words into main memory,
@ -2129,23 +2065,6 @@ void bx_hard_drive_c::write(Bit32u address, Bit32u value, unsigned io_len)
break; break;
} }
if (!calculate_logical_address(channel, &logical_sector)) {
BX_ERROR(("initial read from sector %lu out of bounds, aborting", (unsigned long)logical_sector));
command_aborted(channel, value);
break;
}
ret=BX_SELECTED_DRIVE(channel).hard_drive->lseek(logical_sector * 512, SEEK_SET);
if (ret < 0) {
BX_ERROR (("could not lseek() hard drive image file, aborting"));
command_aborted(channel, value);
break;
}
/* set status bar conditions for device */
if (!BX_SELECTED_DRIVE(channel).iolight_counter)
bx_gui->statusbar_setitem(BX_SELECTED_DRIVE(channel).statusbar_id, 1);
BX_SELECTED_DRIVE(channel).iolight_counter = 5;
bx_pc_system.activate_timer( BX_HD_THIS iolight_timer_index, 100000, 0 );
if (value == 0xC4) { if (value == 0xC4) {
if (BX_SELECTED_CONTROLLER(channel).sector_count > BX_SELECTED_CONTROLLER(channel).multiple_sectors) { if (BX_SELECTED_CONTROLLER(channel).sector_count > BX_SELECTED_CONTROLLER(channel).multiple_sectors) {
BX_SELECTED_CONTROLLER(channel).buffer_size = BX_SELECTED_CONTROLLER(channel).multiple_sectors * 512; BX_SELECTED_CONTROLLER(channel).buffer_size = BX_SELECTED_CONTROLLER(channel).multiple_sectors * 512;
@ -2155,27 +2074,22 @@ void bx_hard_drive_c::write(Bit32u address, Bit32u value, unsigned io_len)
} else { } else {
BX_SELECTED_CONTROLLER(channel).buffer_size = 512; BX_SELECTED_CONTROLLER(channel).buffer_size = 512;
} }
ret = BX_SELECTED_DRIVE(channel).hard_drive->read((bx_ptr_t) BX_SELECTED_CONTROLLER(channel).buffer, if (ide_read_sector(channel, BX_SELECTED_CONTROLLER(channel).buffer,
BX_SELECTED_CONTROLLER(channel).buffer_size); BX_SELECTED_CONTROLLER(channel).buffer_size)) {
if (ret < BX_SELECTED_CONTROLLER(channel).buffer_size) { BX_SELECTED_CONTROLLER(channel).error_register = 0;
BX_ERROR(("logical sector was %lu", (unsigned long)logical_sector)); BX_SELECTED_CONTROLLER(channel).status.busy = 0;
BX_ERROR(("could not read() hard drive image file at byte %lu", (unsigned long)logical_sector*512)); BX_SELECTED_CONTROLLER(channel).status.drive_ready = 1;
command_aborted(channel, value); BX_SELECTED_CONTROLLER(channel).status.seek_complete = 1;
break; BX_SELECTED_CONTROLLER(channel).status.drq = 1;
} BX_SELECTED_CONTROLLER(channel).status.corrected_data = 0;
BX_SELECTED_CONTROLLER(channel).status.err = 0;
BX_SELECTED_CONTROLLER(channel).error_register = 0; BX_SELECTED_CONTROLLER(channel).buffer_index = 0;
BX_SELECTED_CONTROLLER(channel).status.busy = 0; raise_interrupt(channel);
BX_SELECTED_CONTROLLER(channel).status.drive_ready = 1; }
BX_SELECTED_CONTROLLER(channel).status.seek_complete = 1;
BX_SELECTED_CONTROLLER(channel).status.drq = 1;
BX_SELECTED_CONTROLLER(channel).status.corrected_data = 0;
BX_SELECTED_CONTROLLER(channel).status.err = 0;
BX_SELECTED_CONTROLLER(channel).buffer_index = 0;
raise_interrupt(channel);
break; break;
case 0x30: /* WRITE SECTORS, with retries */ case 0x30: // WRITE SECTORS, with retries
case 0xC5: // WRITE MULTIPLE SECTORS
/* update sector_no, always points to current sector /* update sector_no, always points to current sector
* after each sector is read to buffer, DRQ bit set and issue IRQ * after each sector is read to buffer, DRQ bit set and issue IRQ
* if interrupt handler transfers all data words into main memory, * if interrupt handler transfers all data words into main memory,
@ -2185,7 +2099,7 @@ void bx_hard_drive_c::write(Bit32u address, Bit32u value, unsigned io_len)
*/ */
if (!BX_SELECTED_IS_HD(channel)) { if (!BX_SELECTED_IS_HD(channel)) {
BX_PANIC(("ata%d-%d: write multiple issued to non-disk", BX_PANIC(("ata%d-%d: write sectors issued to non-disk",
channel, BX_SLAVE_SELECTED(channel))); channel, BX_SLAVE_SELECTED(channel)));
command_aborted(channel, value); command_aborted(channel, value);
break; break;
@ -2198,6 +2112,15 @@ void bx_hard_drive_c::write(Bit32u address, Bit32u value, unsigned io_len)
} }
BX_SELECTED_CONTROLLER(channel).current_command = value; BX_SELECTED_CONTROLLER(channel).current_command = value;
if (value == 0xC5) {
if (BX_SELECTED_CONTROLLER(channel).sector_count > BX_SELECTED_CONTROLLER(channel).multiple_sectors) {
BX_SELECTED_CONTROLLER(channel).buffer_size = BX_SELECTED_CONTROLLER(channel).multiple_sectors * 512;
} else {
BX_SELECTED_CONTROLLER(channel).buffer_size = BX_SELECTED_CONTROLLER(channel).sector_count * 512;
}
} else {
BX_SELECTED_CONTROLLER(channel).buffer_size = 512;
}
// implicit seek done :^) // implicit seek done :^)
BX_SELECTED_CONTROLLER(channel).error_register = 0; BX_SELECTED_CONTROLLER(channel).error_register = 0;
BX_SELECTED_CONTROLLER(channel).status.busy = 0; BX_SELECTED_CONTROLLER(channel).status.busy = 0;
@ -2348,6 +2271,7 @@ void bx_hard_drive_c::write(Bit32u address, Bit32u value, unsigned io_len)
case 0xc6: // SET MULTIPLE MODE case 0xc6: // SET MULTIPLE MODE
if ((BX_SELECTED_CONTROLLER(channel).sector_count > MAX_MULTIPLE_SECTORS) || if ((BX_SELECTED_CONTROLLER(channel).sector_count > MAX_MULTIPLE_SECTORS) ||
(BX_SELECTED_CONTROLLER(channel).sector_count & (BX_SELECTED_CONTROLLER(channel).sector_count - 1) != 0) ||
(BX_SELECTED_CONTROLLER(channel).sector_count == 0)) { (BX_SELECTED_CONTROLLER(channel).sector_count == 0)) {
command_aborted(channel, value); command_aborted(channel, value);
} else { } else {
@ -2361,6 +2285,7 @@ void bx_hard_drive_c::write(Bit32u address, Bit32u value, unsigned io_len)
BX_SELECTED_CONTROLLER(channel).status.write_fault = 0; BX_SELECTED_CONTROLLER(channel).status.write_fault = 0;
BX_SELECTED_CONTROLLER(channel).status.drq = 0; BX_SELECTED_CONTROLLER(channel).status.drq = 0;
BX_SELECTED_CONTROLLER(channel).status.err = 0; BX_SELECTED_CONTROLLER(channel).status.err = 0;
raise_interrupt(channel);
} }
break; break;
@ -2553,7 +2478,6 @@ void bx_hard_drive_c::write(Bit32u address, Bit32u value, unsigned io_len)
case 0xB0: BX_ERROR(("write cmd 0xB0 (SMART commands) not supported"));command_aborted(channel, 0xB0); break; case 0xB0: BX_ERROR(("write cmd 0xB0 (SMART commands) not supported"));command_aborted(channel, 0xB0); break;
case 0xB1: BX_ERROR(("write cmd 0xB1 (DEVICE CONFIGURATION commands) not supported"));command_aborted(channel, 0xB1); break; case 0xB1: BX_ERROR(("write cmd 0xB1 (DEVICE CONFIGURATION commands) not supported"));command_aborted(channel, 0xB1); break;
case 0xC0: BX_ERROR(("write cmd 0xC0 (CFA ERASE SECTORS) not supported"));command_aborted(channel, 0xC0); break; case 0xC0: BX_ERROR(("write cmd 0xC0 (CFA ERASE SECTORS) not supported"));command_aborted(channel, 0xC0); break;
case 0xC5: BX_ERROR(("write cmd 0xC5 (WRITE MULTIPLE) not supported"));command_aborted(channel, 0xC5); break;
case 0xC7: BX_ERROR(("write cmd 0xC7 (READ DMA QUEUED) not supported"));command_aborted(channel, 0xC7); break; case 0xC7: BX_ERROR(("write cmd 0xC7 (READ DMA QUEUED) not supported"));command_aborted(channel, 0xC7); break;
case 0xC9: BX_ERROR(("write cmd 0xC9 (READ DMA NO RETRY) not supported")); command_aborted(channel, 0xC9); break; case 0xC9: BX_ERROR(("write cmd 0xC9 (READ DMA NO RETRY) not supported")); command_aborted(channel, 0xC9); break;
case 0xCC: BX_ERROR(("write cmd 0xCC (WRITE DMA QUEUED) not supported"));command_aborted(channel, 0xCC); break; case 0xCC: BX_ERROR(("write cmd 0xCC (WRITE DMA QUEUED) not supported"));command_aborted(channel, 0xCC); break;
@ -3127,7 +3051,7 @@ void bx_hard_drive_c::identify_drive(Bit8u channel)
// 2 supports ATA-2 // 2 supports ATA-2
// 1 supports ATA-1 // 1 supports ATA-1
// 0 reserved // 0 reserved
BX_SELECTED_DRIVE(channel).id_drive[80] = (1 << 2) | (1 << 1); BX_SELECTED_DRIVE(channel).id_drive[80] = (1 << 3) | (1 << 2) | (1 << 1);
// Word 81: Minor version number // Word 81: Minor version number
BX_SELECTED_DRIVE(channel).id_drive[81] = 0; BX_SELECTED_DRIVE(channel).id_drive[81] = 0;
@ -3426,35 +3350,11 @@ bx_bool bx_hard_drive_c::bmdma_present(void)
#if BX_SUPPORT_PCI #if BX_SUPPORT_PCI
bx_bool bx_hard_drive_c::bmdma_read_sector(Bit8u channel, Bit8u *buffer, Bit32u *sector_size) bx_bool bx_hard_drive_c::bmdma_read_sector(Bit8u channel, Bit8u *buffer, Bit32u *sector_size)
{ {
off_t logical_sector;
off_t ret;
if (BX_SELECTED_CONTROLLER(channel).current_command == 0xC8) { if (BX_SELECTED_CONTROLLER(channel).current_command == 0xC8) {
*sector_size = 512; *sector_size = 512;
if (!calculate_logical_address(channel, &logical_sector)) { if (!ide_read_sector(channel, buffer, 512)) {
BX_ERROR(("BM-DMA read sector reached invalid sector %lu, aborting", (unsigned long)logical_sector));
command_aborted (channel, BX_SELECTED_CONTROLLER(channel).current_command);
return 0; return 0;
} }
ret = BX_SELECTED_DRIVE(channel).hard_drive->lseek(logical_sector * 512, SEEK_SET);
if (ret < 0) {
BX_ERROR(("could not lseek() hard drive image file"));
command_aborted (channel, BX_SELECTED_CONTROLLER(channel).current_command);
return 0;
}
/* set status bar conditions for device */
if (!BX_SELECTED_DRIVE(channel).iolight_counter)
bx_gui->statusbar_setitem(BX_SELECTED_DRIVE(channel).statusbar_id, 1);
BX_SELECTED_DRIVE(channel).iolight_counter = 5;
bx_pc_system.activate_timer( BX_HD_THIS iolight_timer_index, 100000, 0 );
ret = BX_SELECTED_DRIVE(channel).hard_drive->read((bx_ptr_t) buffer, 512);
if (ret < 512) {
BX_ERROR(("logical sector was %lu", (unsigned long)logical_sector));
BX_ERROR(("could not read() hard drive image file at byte %lu", (unsigned long)logical_sector*512));
command_aborted (channel, BX_SELECTED_CONTROLLER(channel).current_command);
return 0;
}
increment_address(channel);
} else if (BX_SELECTED_CONTROLLER(channel).current_command == 0xA0) { } else if (BX_SELECTED_CONTROLLER(channel).current_command == 0xA0) {
if (BX_SELECTED_CONTROLLER(channel).packet_dma) { if (BX_SELECTED_CONTROLLER(channel).packet_dma) {
*sector_size = BX_SELECTED_CONTROLLER(channel).buffer_size; *sector_size = BX_SELECTED_CONTROLLER(channel).buffer_size;
@ -3490,37 +3390,14 @@ bx_bool bx_hard_drive_c::bmdma_read_sector(Bit8u channel, Bit8u *buffer, Bit32u
bx_bool bx_hard_drive_c::bmdma_write_sector(Bit8u channel, Bit8u *buffer) bx_bool bx_hard_drive_c::bmdma_write_sector(Bit8u channel, Bit8u *buffer)
{ {
off_t logical_sector;
off_t ret;
if (BX_SELECTED_CONTROLLER(channel).current_command != 0xCA) { if (BX_SELECTED_CONTROLLER(channel).current_command != 0xCA) {
BX_ERROR(("command 0xCA (WRITE DMA) not active")); BX_ERROR(("command 0xCA (WRITE DMA) not active"));
command_aborted (channel, BX_SELECTED_CONTROLLER(channel).current_command); command_aborted (channel, BX_SELECTED_CONTROLLER(channel).current_command);
return 0; return 0;
} }
if (!calculate_logical_address(channel, &logical_sector)) { if (!ide_write_sector(channel, buffer, 512)) {
BX_ERROR(("BM-DMA read sector reached invalid sector %lu, aborting", (unsigned long)logical_sector));
command_aborted (channel, BX_SELECTED_CONTROLLER(channel).current_command);
return 0; return 0;
} }
ret = BX_SELECTED_DRIVE(channel).hard_drive->lseek(logical_sector * 512, SEEK_SET);
if (ret < 0) {
BX_ERROR(("could not lseek() hard drive image file"));
command_aborted (channel, BX_SELECTED_CONTROLLER(channel).current_command);
return 0;
}
/* set status bar conditions for device */
if (!BX_SELECTED_DRIVE(channel).iolight_counter)
bx_gui->statusbar_setitem(BX_SELECTED_DRIVE(channel).statusbar_id, 1);
BX_SELECTED_DRIVE(channel).iolight_counter = 5;
bx_pc_system.activate_timer( BX_HD_THIS iolight_timer_index, 100000, 0 );
ret = BX_SELECTED_DRIVE(channel).hard_drive->write((bx_ptr_t) buffer, 512);
if (ret < 512) {
BX_ERROR(("could not write() hard drive image file at byte %lu", (unsigned long)logical_sector*512));
command_aborted (channel, BX_SELECTED_CONTROLLER(channel).current_command);
return 0;
}
increment_address(channel);
return 1; return 1;
} }
@ -3558,7 +3435,79 @@ void bx_hard_drive_c::set_signature(Bit8u channel, Bit8u id)
} }
} }
error_recovery_t::error_recovery_t () bx_bool bx_hard_drive_c::ide_read_sector(Bit8u channel, Bit8u *buffer, Bit32u buffer_size)
{
off_t logical_sector;
off_t ret;
int sector_count = (buffer_size / 512);
Bit8u *bufptr = buffer;
do {
if (!calculate_logical_address(channel, &logical_sector)) {
BX_ERROR(("ide_read_sector() reached invalid sector %lu, aborting", (unsigned long)logical_sector));
command_aborted(channel, BX_SELECTED_CONTROLLER(channel).current_command);
return 0;
}
ret = BX_SELECTED_DRIVE(channel).hard_drive->lseek(logical_sector * 512, SEEK_SET);
if (ret < 0) {
BX_ERROR(("could not lseek() hard drive image file"));
command_aborted(channel, BX_SELECTED_CONTROLLER(channel).current_command);
return 0;
}
/* set status bar conditions for device */
if (!BX_SELECTED_DRIVE(channel).iolight_counter)
bx_gui->statusbar_setitem(BX_SELECTED_DRIVE(channel).statusbar_id, 1);
BX_SELECTED_DRIVE(channel).iolight_counter = 5;
bx_pc_system.activate_timer(BX_HD_THIS iolight_timer_index, 100000, 0);
ret = BX_SELECTED_DRIVE(channel).hard_drive->read((bx_ptr_t)bufptr, 512);
if (ret < 512) {
BX_ERROR(("could not read() hard drive image file at byte %lu", (unsigned long)logical_sector*512));
command_aborted(channel, BX_SELECTED_CONTROLLER(channel).current_command);
return 0;
}
increment_address(channel);
bufptr += 512;
} while (--sector_count > 0);
return 1;
}
bx_bool bx_hard_drive_c::ide_write_sector(Bit8u channel, Bit8u *buffer, Bit32u buffer_size)
{
off_t logical_sector;
off_t ret;
int sector_count = (buffer_size / 512);
Bit8u *bufptr = buffer;
do {
if (!calculate_logical_address(channel, &logical_sector)) {
BX_ERROR(("ide_write_sector() reached invalid sector %lu, aborting", (unsigned long)logical_sector));
command_aborted(channel, BX_SELECTED_CONTROLLER(channel).current_command);
return 0;
}
ret = BX_SELECTED_DRIVE(channel).hard_drive->lseek(logical_sector * 512, SEEK_SET);
if (ret < 0) {
BX_ERROR(("could not lseek() hard drive image file at byte %lu", (unsigned long)logical_sector * 512));
command_aborted(channel, BX_SELECTED_CONTROLLER(channel).current_command);
return 0;
}
/* set status bar conditions for device */
if (!BX_SELECTED_DRIVE(channel).iolight_counter)
bx_gui->statusbar_setitem(BX_SELECTED_DRIVE(channel).statusbar_id, 1);
BX_SELECTED_DRIVE(channel).iolight_counter = 5;
bx_pc_system.activate_timer(BX_HD_THIS iolight_timer_index, 100000, 0);
ret = BX_SELECTED_DRIVE(channel).hard_drive->write((bx_ptr_t)bufptr, 512);
if (ret < 512) {
BX_ERROR(("could not write() hard drive image file at byte %lu", (unsigned long)logical_sector*512));
command_aborted(channel, BX_SELECTED_CONTROLLER(channel).current_command);
return 0;
}
increment_address(channel);
bufptr += 512;
} while (--sector_count > 0);
return 1;
}
error_recovery_t::error_recovery_t()
{ {
if (sizeof(error_recovery_t) != 8) { if (sizeof(error_recovery_t) != 8) {
BX_PANIC(("error_recovery_t has size != 8")); BX_PANIC(("error_recovery_t has size != 8"));

View File

@ -1,5 +1,5 @@
///////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////
// $Id: harddrv.h,v 1.41 2006-07-13 17:34:44 vruppert Exp $ // $Id: harddrv.h,v 1.42 2006-07-17 18:40:26 vruppert Exp $
///////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////
// //
// Copyright (C) 2002 MandrakeSoft S.A. // Copyright (C) 2002 MandrakeSoft S.A.
@ -211,6 +211,8 @@ private:
BX_HD_SMF void atapi_cmd_nop(Bit8u channel) BX_CPP_AttrRegparmN(1); BX_HD_SMF void atapi_cmd_nop(Bit8u channel) BX_CPP_AttrRegparmN(1);
BX_HD_SMF bx_bool bmdma_present(void); BX_HD_SMF bx_bool bmdma_present(void);
BX_HD_SMF void set_signature(Bit8u channel, Bit8u id); BX_HD_SMF void set_signature(Bit8u channel, Bit8u id);
BX_HD_SMF bx_bool ide_read_sector(Bit8u channel, Bit8u *buffer, Bit32u buffer_size);
BX_HD_SMF bx_bool ide_write_sector(Bit8u channel, Bit8u *buffer, Bit32u buffer_size);
// FIXME: // FIXME:
// For each ATA channel we should have one controller struct // For each ATA channel we should have one controller struct