USB floppy sector write timing implemented similar to read timing. It can be

used when asynchronous packet handling is available. Some changes in the
non-async read and write code.
TODO: seek timing (similar to legacy floppy).
This commit is contained in:
Volker Ruppert 2015-09-26 09:19:56 +00:00
parent 8232928096
commit e8d045ad30
2 changed files with 81 additions and 37 deletions

View File

@ -721,14 +721,14 @@ bx_bool usb_cbi_device_c::handle_command(Bit8u *command)
s.sector_count = count; s.sector_count = count;
s.data_len = (count * 512); s.data_len = (count * 512);
s.usb_len = 0; s.usb_len = 0;
if (s.hdimage->lseek(s.sector * 512, SEEK_SET) < 0) {
BX_ERROR(("could not lseek() floppy drive image file"));
ret = 0;
}
if (d.async_mode) { if (d.async_mode) {
start_timer(); start_timer(0);
} else { } else {
bx_gui->statusbar_setitem(s.statusbar_id, 1); // read bx_gui->statusbar_setitem(s.statusbar_id, 1); // read
if (s.hdimage->lseek(s.sector * 512, SEEK_SET) < 0) {
BX_ERROR(("could not lseek() floppy drive image file"));
ret = 0;
}
} }
break; break;
@ -744,10 +744,10 @@ bx_bool usb_cbi_device_c::handle_command(Bit8u *command)
s.asc = 0x3a; s.asc = 0x3a;
break; break;
} }
s.sector = lba;
s.data_len = (count * 512); s.data_len = (count * 512);
s.usb_len = 0; s.usb_len = 0;
bx_gui->statusbar_setitem(s.statusbar_id, 1, 1); // write if (s.hdimage->lseek(s.sector * 512, SEEK_SET) < 0) {
if (s.hdimage->lseek(lba * 512, SEEK_SET) < 0) {
BX_ERROR(("could not lseek() floppy drive image file")); BX_ERROR(("could not lseek() floppy drive image file"));
ret = 0; ret = 0;
} }
@ -896,20 +896,24 @@ int usb_cbi_device_c::handle_data(USBPacket *p)
s.usb_len += len; s.usb_len += len;
s.data_len -= len; s.data_len -= len;
} }
ret = len;
if ((s.data_len == 0) || (s.usb_len >= 512)) { if ((s.data_len == 0) || (s.usb_len >= 512)) {
if (s.hdimage->write((bx_ptr_t) s.usb_buf, 512) < 0) { if (d.async_mode) {
BX_ERROR(("write error")); start_timer(1);
ret = 0; BX_DEBUG(("deferring packet %p", p));
break; usb_defer_packet(p, this);
s.packet = p;
ret = USB_RET_ASYNC;
} else { } else {
s.usb_len = 0; if (!floppy_write_sector()) {
} ret = 0;
if (s.data_len > 0) { break;
bx_gui->statusbar_setitem(s.statusbar_id, 1, 1); // write } else {
bx_gui->statusbar_setitem(s.statusbar_id, 1, 1); // write
}
} }
} }
usb_dump_packet(data, len); if (ret > 0) usb_dump_packet(data, len);
ret = len;
break; break;
case UFI_FORMAT_UNIT: case UFI_FORMAT_UNIT:
@ -957,7 +961,7 @@ int usb_cbi_device_c::handle_data(USBPacket *p)
len = s.data_len; len = s.data_len;
if (d.async_mode) { if (d.async_mode) {
if (len > (int) s.usb_len) { if (len > (int) s.usb_len) {
BX_INFO(("deferring packet %p", p)); BX_DEBUG(("deferring packet %p", p));
usb_defer_packet(p, this); usb_defer_packet(p, this);
s.packet = p; s.packet = p;
ret = USB_RET_ASYNC; ret = USB_RET_ASYNC;
@ -1061,16 +1065,41 @@ fail:
return ret; return ret;
} }
void usb_cbi_device_c::start_timer() void usb_cbi_device_c::start_timer(bx_bool write)
{ {
bx_gui->statusbar_setitem(s.statusbar_id, 1); // read bx_gui->statusbar_setitem(s.statusbar_id, 1, write);
bx_pc_system.activate_timer(s.floppy_timer_index, CBI_SECTOR_TIME, 0); bx_pc_system.activate_timer(s.floppy_timer_index, CBI_SECTOR_TIME, 0);
} }
void usb_cbi_device_c::floppy_timer_handler(void *this_ptr) void usb_cbi_device_c::floppy_timer_handler(void *this_ptr)
{ {
usb_cbi_device_c *class_ptr = (usb_cbi_device_c *) this_ptr; usb_cbi_device_c *class_ptr = (usb_cbi_device_c *) this_ptr;
class_ptr->floppy_read_sector(); class_ptr->floppy_timer();
}
void usb_cbi_device_c::floppy_timer()
{
USBPacket *p = s.packet;
switch (s.cur_command) {
case UFI_READ_10:
case UFI_READ_12:
floppy_read_sector();
break;
case UFI_WRITE_10:
case UFI_WRITE_12:
if (!floppy_write_sector()) {
p->len = 0;
}
if (s.packet != NULL) {
usb_dump_packet(p->data, p->len);
s.packet = NULL;
usb_packet_complete(p);
}
break;
default:
BX_ERROR(("floppy_timer(): unsupported command"));
}
} }
void usb_cbi_device_c::floppy_read_sector() void usb_cbi_device_c::floppy_read_sector()
@ -1079,28 +1108,23 @@ void usb_cbi_device_c::floppy_read_sector()
USBPacket *p = s.packet; USBPacket *p = s.packet;
BX_DEBUG(("floppy_read_sector(): sector = %i", s.sector)); BX_DEBUG(("floppy_read_sector(): sector = %i", s.sector));
if (s.hdimage->lseek(s.sector * 512, SEEK_SET) < 0) { if (((CBI_MAX_SECTORS * 512) - s.usb_len) >= 512) {
BX_ERROR(("could not lseek() floppy drive image file")); ret = s.hdimage->read((bx_ptr_t) s.usb_buf, 512);
s.usb_len = 0; if (ret > 0) {
} else { s.usb_len += ret;
if (((CBI_MAX_SECTORS * 512) - s.usb_len) >= 512) { s.usb_buf += ret;
ret = s.hdimage->read((bx_ptr_t) s.usb_buf, 512);
if (ret > 0) {
s.usb_len += ret;
s.usb_buf += ret;
} else {
BX_ERROR(("read error"));
s.usb_len = 0;
}
} else { } else {
BX_ERROR(("buffer overflow")); BX_ERROR(("read error"));
s.usb_len = 0; s.usb_len = 0;
} }
} else {
BX_ERROR(("buffer overflow"));
s.usb_len = 0;
} }
if (s.usb_len > 0) { if (s.usb_len > 0) {
s.sector++; s.sector++;
if (--s.sector_count > 0) { if (--s.sector_count > 0) {
start_timer(); start_timer(0);
} }
if (s.packet != NULL) { if (s.packet != NULL) {
if (p->len <= (int)s.usb_len) { if (p->len <= (int)s.usb_len) {
@ -1113,6 +1137,24 @@ void usb_cbi_device_c::floppy_read_sector()
} }
} }
bx_bool usb_cbi_device_c::floppy_write_sector()
{
BX_DEBUG(("floppy_write_sector(): sector = %i", s.sector));
if (s.hdimage->write((bx_ptr_t) s.usb_buf, 512) < 0) {
BX_ERROR(("write error"));
return 0;
} else {
s.sector++;
if (s.usb_len > 512) {
s.usb_len -= 512;
memmove(s.usb_buf, s.usb_buf+512, s.usb_len);
} else {
s.usb_len = 0;
}
return 1;
}
}
void usb_cbi_device_c::copy_data(USBPacket *p) void usb_cbi_device_c::copy_data(USBPacket *p)
{ {
int len = p->len; int len = p->len;

View File

@ -94,8 +94,10 @@ private:
} s; } s;
bx_bool handle_command(Bit8u *command); bx_bool handle_command(Bit8u *command);
void start_timer(void); void start_timer(bx_bool write);
void floppy_timer(void);
void floppy_read_sector(void); void floppy_read_sector(void);
bx_bool floppy_write_sector(void);
void copy_data(USBPacket *p); void copy_data(USBPacket *p);
bx_bool set_inserted(bx_bool value); bx_bool set_inserted(bx_bool value);