///////////////////////////////////////////////////////////////////////// // $Id$ ///////////////////////////////////////////////////////////////////////// // // UFI/CBI floppy disk storage device support // // Copyright (c) 2015 Benjamin David Lunt // Copyright (C) 2015-2016 The Bochs Project // // This library is free software; you can redistribute it and/or // modify it under the terms of the GNU Lesser General Public // License as published by the Free Software Foundation; either // version 2 of the License, or (at your option) any later version. // // This library is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU // Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public // License along with this library; if not, write to the Free Software // Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA ///////////////////////////////////////////////////////////////////////// /* Notes by Ben Lunt: - My purpose of coding this emulation was/is to test some other code. - This emulation will emulate an external USB Floppy Disk Drive with an inserted 1.44 meg *only* disk. - This emulation shows different formats allowed, but does not support changing to them. Use with caution if you change formats. - If you have any questions, send me a note at fys [at] fysnet [dot] net */ // Define BX_PLUGGABLE in files that can be compiled into plugins. For // platforms that require a special tag on exported symbols, BX_PLUGGABLE // is used to know when we are exporting symbols and when we are importing. #define BX_PLUGGABLE #include "iodev.h" #if BX_SUPPORT_PCI && BX_SUPPORT_PCIUSB #include "usb_common.h" #include "hdimage/hdimage.h" #include "usb_cbi.h" #define LOG_THIS // maximum size of the read buffer in sectors #define CBI_MAX_SECTORS 18 // useconds per sector at 300 RPM #define CBI_SECTOR_TIME 11111 // Set this to zero if you only support CB interface. Set to 1 if you support CBI. #define USB_CBI_USE_INTERRUPT 1 /* A well known, somewhat older, but still widely used Operating System * must have the Vendor ID as the TEAC external drive emulated below. If * the VendorID is not the TEAC id, this operating system doesn't know what * to do with the drive. Go figure. Therefore, use the optionsX="model=teac" * options in the bochsrc.txt file to set this model. If you do not, a * default model will be used. */ // USB requests #define GetMaxLun 0xfe // Full-speed only static Bit8u bx_cbi_dev_descriptor[] = { 0x12, /* u8 bLength; */ 0x01, /* u8 bDescriptorType; Device */ 0x00, 0x02, /* u16 bcdUSB; v2.0 */ 0x00, /* u8 bDeviceClass; */ 0x00, /* u8 bDeviceSubClass; */ 0x00, /* u8 bDeviceProtocol; [ low/full speeds only ] */ 0x08, /* u8 bMaxPacketSize0; 8 Bytes */ /* Vendor and product id are arbitrary. */ 0x00, 0x00, /* u16 idVendor; */ 0x00, 0x00, /* u16 idProduct; */ 0x00, 0x00, /* u16 bcdDevice */ 0x01, /* u8 iManufacturer; */ 0x02, /* u8 iProduct; */ 0x03, /* u8 iSerialNumber; */ 0x01 /* u8 bNumConfigurations; */ }; // Full-speed only static const Bit8u bx_cbi_config_descriptor[] = { /* one configuration */ 0x09, /* u8 bLength; */ 0x02, /* u8 bDescriptorType; Configuration */ 0x27, 0x00, /* u16 wTotalLength; */ 0x01, /* u8 bNumInterfaces; (1) */ 0x01, /* u8 bConfigurationValue; */ 0x00, /* u8 iConfiguration; */ 0x80, /* u8 bmAttributes; Bit 7: must be set, 6: Self-powered, 5: Remote wakeup, 4..0: resvd */ 0xFA, /* u8 MaxPower; */ /* one interface */ 0x09, /* u8 if_bLength; */ 0x04, /* u8 if_bDescriptorType; Interface */ 0x00, /* u8 if_bInterfaceNumber; */ 0x00, /* u8 if_bAlternateSetting; */ #if USB_CBI_USE_INTERRUPT 0x03, /* u8 if_bNumEndpoints; (CBI has 3)*/ #else 0x02, /* u8 if_bNumEndpoints; (CB has 2) */ #endif 0x08, /* u8 if_bInterfaceClass; MASS STORAGE */ 0x04, /* u8 if_bInterfaceSubClass; CB(I) */ #if USB_CBI_USE_INTERRUPT 0x00, /* u8 if_bInterfaceProtocol; CB with Interrupt */ #else 0x01, /* u8 if_bInterfaceProtocol; CB with out Interrupt */ #endif 0x00, /* u8 if_iInterface; */ /* Bulk-In endpoint */ 0x07, /* u8 ep_bLength; */ 0x05, /* u8 ep_bDescriptorType; Endpoint */ 0x81, /* u8 ep_bEndpointAddress; IN Endpoint 1 */ 0x02, /* u8 ep_bmAttributes; Bulk */ 0x40, 0x00, /* u16 ep_wMaxPacketSize; */ 0x00, /* u8 ep_bInterval; */ /* Bulk-Out endpoint */ 0x07, /* u8 ep_bLength; */ 0x05, /* u8 ep_bDescriptorType; Endpoint */ 0x02, /* u8 ep_bEndpointAddress; OUT Endpoint 2 */ 0x02, /* u8 ep_bmAttributes; Bulk */ 0x40, 0x00, /* u16 ep_wMaxPacketSize; */ 0x00, /* u8 ep_bInterval; */ #if USB_CBI_USE_INTERRUPT /* Interrupt endpoint */ 0x07, /* u8 ep_bLength; */ 0x05, /* u8 ep_bDescriptorType; Endpoint */ 0x83, /* u8 ep_bEndpointAddress; IN Endpoint 3 */ 0x03, /* u8 ep_bmAttributes; Interrupt */ 0x02, 0x00, /* u16 ep_wMaxPacketSize; */ 0x20 /* u8 ep_bInterval; */ #endif }; static const Bit8u bx_cbi_dev_inquiry_teac[] = { 0x00, /* perifpheral device type */ 0x80, /* RMB = 1, reserved = 0 */ 0x00, /* ISO version, ecma version , ansi version */ 0x01, /* response data format */ 0x1F, /* additional length */ 0x00, 0x00, 0x00, /* reserved */ 0x54, 0x45, 0x41, 0x43, 0x20, 0x20, 0x20, 0x20, /* vendor information */ 0x46, 0x44, 0x2D, 0x30, 0x35, 0x50, 0x55, 0x57, /* product information */ 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, /* (cont.) */ 0x33, 0x30, 0x30, 0x30 /* revision level */ }; static const Bit8u bx_cbi_dev_inquiry_bochs[] = { 0x00, /* perifpheral device type */ 0x80, /* RMB = 1, reserved = 0 */ 0x00, /* ISO version, ecma version , ansi version */ 0x01, /* response data format */ 0x1F, /* additional length */ 0x00, 0x00, 0x00, /* reserved */ 0x42, 0x4F, 0x43, 0x48, 0x53, 0x20, 0x20, 0x20, /* vendor information */ 0x55, 0x53, 0x42, 0x20, 0x43, 0x42, 0x49, 0x20, /* product information */ 0x46, 0x4C, 0x4F, 0x50, 0x50, 0x59, 0x20, 0x20, /* (cont.) */ 0x30, 0x30, 0x31, 0x30 /* revision level */ }; static Bit8u bx_cbi_dev_frmt_capacity[] = { 0x00, 0x00, 0x00, // reserved 32, // remaining list in bytes // current: 1.44meg 0x00, 0x00, 0x0B, 0x40, // lba's 0x02, // descriptor code 0x00, 0x02, 0x00, // 512-byte sectors // allowed #1: 1.44meg 0x00, 0x00, 0x0B, 0x40, // lba's 0x00, // descriptor code 0x00, 0x02, 0x00, // 512-byte sectors // allowed #2: 1.25meg 0x00, 0x00, 0x04, 0xD0, // lba's 0x02, // descriptor code 0x00, 0x04, 0x00, // 1024-byte sectors // allowed #3: 1.28meg 0x00, 0x00, 0x09, 0x60, // lba's (2400) 0x02, // descriptor code 0x00, 0x02, 0x00 // 512-byte sectors }; static const Bit8u bx_cbi_dev_capacity[] = { 0x00, 0x00, 0x0B, 0x3f, // lba's 0x00, 0x00, 0x02, 0x00 // 512-byte sectors }; static Bit8u bx_cbi_dev_req_sense[] = { 0x70, // valid and error code 0x00, // reserved 0x00, // sense key 0x00, 0x00, 0x00, 0x00, // (lba ?) information 0x0A, // additional sense length 0x00, 0x00, 0x00, 0x00, // reserved 0x00, // additional sense code 0x00, // additional sense code qualifier 0x00, 0x00, 0x00, 0x00 // reserved }; #define PAGE_CODE_01_OFF 8 #define PAGE_CODE_05_OFF 20 #define PAGE_CODE_1B_OFF 52 #define PAGE_CODE_1C_OFF 64 static Bit8u bx_cbi_dev_mode_sense_cur[] = { ///////////////////////////////////////// // header (8 bytes) 0x00, 0x46, // length of remaining data (72 - 2) // medium type code (See page 25 of ufi v1.0 specification) // 0x1E, // (720k) // 0x93, // (1.25meg) 0x94, // (1.44meg) // WP (bit 7), reserved 0x00, // ben // reserved 0x00, 0x00, 0x00, 0x00, ///////////////////////////////////////// // Page Code 01 starts here // (12 bytes) 0x01, // page code 01 0x0A, // 10 more bytes 0x05, // error recovery parameters (5 = PER | DCR) (PER = Post Error, DCR = Disable Correction) 0x03, // read re-try count 0x00, 0x00, 0x00, 0x00, // reserved 0x03, // write re-try count 0x00, 0x00, 0x00, // reserved ///////////////////////////////////////// // Page Code 05 starts here // (32 bytes) 0x05, // page code 5 0x1E, // 30 more bytes 0x03, 0xE8, // transfer rate (0x03E8 = 1000d = 1,000 Kbits per second) 0x02, // head count (2) 0x12, // spt (18) 0x02, 0x00, // bytes per sector (512) 0x00, 0x50, // cylinder count (80) 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // reserved 0x08, // motor on delay (10ths of a second) 0x1E, // motor off delay (10ths of a second) 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // reserved 0x02, 0x58, // medium rotation rate (0x0258) 0x00, 0x00, // reserved ///////////////////////////////////////// // Page Code 1B starts here // (12 bytes) 0x1B, // page code 1B 0x0A, // 10 more bytes 0x80, // System Floppy = 1, Format Progress = 0 0x01, // 1 LUN supported 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // reserved ///////////////////////////////////////// // Page Code 1C starts here (Timer and Protect Page) // (8 bytes) 0x1C, // page code 1C 0x06, // 6 more bytes 0x00, // reserved 0x05, // inactivity multiplier 0x00, // reserved in UFI 0x00, 0x00, 0x00 // reserved }; void usb_cbi_restore_handler(void *dev, bx_list_c *conf); const char *fdimage_mode_names[] = { "flat", "vvfat", NULL }; static int usb_floppy_count = 0; usb_cbi_device_c::usb_cbi_device_c(const char *filename) { char pname[10]; char label[32]; char tmpfname[BX_PATHNAME_LEN]; char *ptr1, *ptr2; bx_param_string_c *path; bx_param_bool_c *readonly; bx_param_enum_c *status, *mode; d.type = USB_DEV_TYPE_FLOPPY; d.speed = d.minspeed = d.maxspeed = USB_SPEED_FULL; memset((void*)&s, 0, sizeof(s)); strcpy(d.devname, "BOCHS USB CBI FLOPPY"); d.dev_descriptor = bx_cbi_dev_descriptor; d.config_descriptor = bx_cbi_config_descriptor; d.device_desc_size = sizeof(bx_cbi_dev_descriptor); d.config_desc_size = sizeof(bx_cbi_config_descriptor); // set the model information // s.model == 1 if use teac model, else use bochs model if (s.model) { bx_cbi_dev_descriptor[8] = 0x44; bx_cbi_dev_descriptor[9] = 0x06; d.vendor_desc = "TEAC "; d.product_desc = "TEAC FD-05PUW "; d.serial_num = "3000"; } else { bx_cbi_dev_descriptor[8] = 0x00; bx_cbi_dev_descriptor[9] = 0x00; d.vendor_desc = "BOCHS "; d.product_desc = d.devname; d.serial_num = "00.10"; } s.inserted = 0; strcpy(tmpfname, filename); ptr1 = strtok(tmpfname, ":"); ptr2 = strtok(NULL, ":"); if ((ptr2 == NULL) || (strlen(ptr1) < 2)) { s.image_mode = BX_HDIMAGE_MODE_FLAT; s.fname = filename; } else { s.image_mode = SIM->hdimage_get_mode(ptr1); s.fname = filename+strlen(ptr1)+1; if ((s.image_mode != BX_HDIMAGE_MODE_FLAT) && (s.image_mode != BX_HDIMAGE_MODE_VVFAT)) { BX_PANIC(("USB floppy only supports image modes 'flat' and 'vvfat'")); } } s.dev_buffer = new Bit8u[CBI_MAX_SECTORS * 512]; s.statusbar_id = bx_gui->register_statusitem("USB-FD", 1); s.floppy_timer_index = DEV_register_timer(this, floppy_timer_handler, CBI_SECTOR_TIME, 0, 0, "USB FD timer"); // config options bx_list_c *usb_rt = (bx_list_c*)SIM->get_param(BXPN_MENU_RUNTIME_USB); sprintf(pname, "floppy%d", ++usb_floppy_count); sprintf(label, "USB floppy #%d Configuration", usb_floppy_count); s.config = new bx_list_c(usb_rt, pname, label); s.config->set_options(bx_list_c::SERIES_ASK | bx_list_c::USE_BOX_TITLE); s.config->set_device_param(this); path = new bx_param_string_c(s.config, "path", "Path", "", "", BX_PATHNAME_LEN); path->set(s.fname); path->set_handler(floppy_path_handler); mode = new bx_param_enum_c(s.config, "mode", "Image mode", "Mode of the floppy image", fdimage_mode_names, 0, 0); if (s.image_mode == BX_HDIMAGE_MODE_VVFAT) { mode->set(1); } mode->set_handler(floppy_param_handler); mode->set_ask_format("Enter mode of floppy image, (flat or vvfat): [%s] "); readonly = new bx_param_bool_c(s.config, "readonly", "Write Protection", "Floppy media write protection", 0); readonly->set_handler(floppy_param_handler); readonly->set_ask_format("Is media write protected? [%s] "); status = new bx_param_enum_c(s.config, "status", "Status", "Floppy media status (inserted / ejected)", media_status_names, BX_INSERTED, BX_EJECTED); status->set_handler(floppy_param_handler); status->set_ask_format("Is the device inserted or ejected? [%s] "); if (SIM->is_wx_selected()) { bx_list_c *usb = (bx_list_c*)SIM->get_param("ports.usb"); usb->add(s.config); } put("usb_cbi", "USBCBI"); } usb_cbi_device_c::~usb_cbi_device_c(void) { d.sr->clear(); bx_gui->unregister_statusitem(s.statusbar_id); set_inserted(0); if (s.dev_buffer != NULL) delete [] s.dev_buffer; if (SIM->is_wx_selected()) { bx_list_c *usb = (bx_list_c*)SIM->get_param("ports.usb"); usb->remove(s.config->get_name()); } bx_list_c *usb_rt = (bx_list_c*)SIM->get_param(BXPN_MENU_RUNTIME_USB); usb_rt->remove(s.config->get_name()); bx_pc_system.deactivate_timer(s.floppy_timer_index); bx_pc_system.unregisterTimer(s.floppy_timer_index); } bx_bool usb_cbi_device_c::set_option(const char *option) { if (!strncmp(option, "write_protected:", 16)) { SIM->get_param_bool("readonly", s.config)->set(atol(&option[16])); return 1; } else if (!strncmp(option, "model:", 6)) { if (!strcmp(option+6, "teac")) { s.model = 1; } else { s.model = 0; } return 1; } return 0; } bx_bool usb_cbi_device_c::init() { if (set_inserted(1)) { sprintf(s.info_txt, "USB CBI: path='%s', mode='%s'", s.fname, hdimage_mode_names[s.image_mode]); } else { sprintf(s.info_txt, "USB CBI: media not present"); } d.connected = 1; s.did_inquiry_fail = 0; s.fail_count = 0; s.status_changed = 0; return 1; } const char* usb_cbi_device_c::get_info() { // set the write protected bit given by parameter in bochsrc.txt file bx_cbi_dev_mode_sense_cur[3] &= ~0x80; bx_cbi_dev_mode_sense_cur[3] |= ((s.wp > 0) << 7); return s.info_txt; } void usb_cbi_device_c::register_state_specific(bx_list_c *parent) { bx_list_c *list = new bx_list_c(parent, "s", "UFI/CBI Floppy Disk State"); bx_list_c *rt_config = new bx_list_c(list, "rt_config"); rt_config->add(s.config->get_by_name("path")); rt_config->add(s.config->get_by_name("readonly")); rt_config->add(s.config->get_by_name("status")); rt_config->set_restore_handler(this, usb_cbi_restore_handler); BXRS_DEC_PARAM_FIELD(list, usb_len, s.usb_len); BXRS_DEC_PARAM_FIELD(list, data_len, s.data_len); BXRS_DEC_PARAM_FIELD(list, sector, s.sector); BXRS_DEC_PARAM_FIELD(list, sector_count, s.sector_count); BXRS_DEC_PARAM_FIELD(list, cur_command, s.cur_command); BXRS_DEC_PARAM_FIELD(list, cur_track, s.cur_track); BXRS_DEC_PARAM_FIELD(list, sense, s.sense); BXRS_DEC_PARAM_FIELD(list, asc, s.asc); BXRS_DEC_PARAM_FIELD(list, fail_count, s.fail_count); BXRS_PARAM_BOOL(list, did_inquiry_fail, s.did_inquiry_fail); BXRS_PARAM_BOOL(list, seek_pending, s.seek_pending); BXRS_PARAM_SPECIAL32(list, usb_buf, param_save_handler, param_restore_handler); new bx_shadow_data_c(list, "dev_buffer", s.dev_buffer, CBI_MAX_SECTORS * 512); // TODO: async usb packet } void usb_cbi_device_c::handle_reset() { BX_DEBUG(("Reset")); } int usb_cbi_device_c::handle_control(int request, int value, int index, int length, Bit8u *data) { int ret; ret = handle_control_common(request, value, index, length, data); if (ret >= 0) { return ret; } ret = 0; switch (request) { case DeviceOutRequest | USB_REQ_CLEAR_FEATURE: BX_INFO(("USB_REQ_CLEAR_FEATURE: Not handled: %i %i %i %i", request, value, index, length )); // It's okay that we don't handle this. Most likely the guest is just // clearing the toggle bit. Since we don't worry about the toggle bit (yet?), // we can just ignore and continue. ret = 0; break; case DeviceOutRequest | USB_REQ_SET_FEATURE: BX_DEBUG(("USB_REQ_SET_FEATURE:")); switch (value) { case USB_DEVICE_REMOTE_WAKEUP: case USB_DEVICE_U1_ENABLE: case USB_DEVICE_U2_ENABLE: break; default: BX_DEBUG(("USB_REQ_SET_FEATURE: Not handled: %i %i %i %i", request, value, index, length )); goto fail; } ret = 0; break; case DeviceRequest | USB_REQ_GET_DESCRIPTOR: switch (value >> 8) { case USB_DT_STRING: BX_DEBUG(("USB_REQ_GET_DESCRIPTOR: String")); switch(value & 0xff) { case 0xEE: // Microsoft OS Descriptor check // We don't support this check, so fail goto fail; default: BX_ERROR(("USB CBI handle_control: unknown string descriptor 0x%02x", value & 0xff)); goto fail; } break; case USB_DT_DEVICE_QUALIFIER: BX_DEBUG(("USB_REQ_GET_DESCRIPTOR: Device Qualifier")); // device qualifier // a low- or full-speed only device (i.e.: a non high-speed device) must return // request error on this function BX_ERROR(("USB CBI handle_control: full-speed only device returning stall on Device Qualifier.")); goto fail; default: BX_ERROR(("USB CBI handle_control: unknown descriptor type 0x%02x", value >> 8)); goto fail; } break; case EndpointOutRequest | USB_REQ_CLEAR_FEATURE: BX_DEBUG(("USB_REQ_CLEAR_FEATURE:")); // It's okay that we don't handle this. Most likely the guest is just // clearing the toggle bit. Since we don't worry about the toggle bit (yet?), // we can just ignore and continue. //if (value == 0 && index != 0x81) { /* clear ep halt */ // goto fail; //} ret = 0; break; case DeviceOutRequest | USB_REQ_SET_SEL: // Set U1 and U2 System Exit Latency BX_DEBUG(("SET_SEL (U1 and U2):")); ret = 0; break; // Class specific requests case InterfaceInClassRequest | GetMaxLun: case GetMaxLun: BX_DEBUG(("MASS STORAGE: GET MAX LUN")); data[0] = 0; ret = 1; break; // this is the request where we should receive 12 bytes // of command data. case InterfaceOutClassRequest: if (!handle_command(data)) goto fail; break; default: BX_ERROR(("USB CBI handle_control: unknown request 0x%04X", request)); fail: BX_ERROR(("USB CBI handle_control: stalled on request: 0x%04X", request)); d.stall = 1; ret = USB_RET_STALL; } return ret; } bx_bool usb_cbi_device_c::handle_command(Bit8u *command) { Bit32u lba, count; int pc, pagecode; bx_bool ret = 1; // assume valid return s.cur_command = command[0]; s.usb_buf = s.dev_buffer; s.usb_len = 0; s.data_len = 0; // most commands that use the LBA field, use the same place for this field lba = ((command[2] << 24) | (command[3] << 16) | (command[4] << 8) | (command[5] << 0)); // then assume the 16-bit count field (32-bit commands will update it first) count = ((command[7] << 8) | (command[8] << 0)); // get the LUN from the command if (((command[1] >> 5) & 0x07) != 0) { BX_ERROR(("Command sent a lun value != 0")); return 0; } // to be consistant with real hardware, we need to fail with // a STALL twice for any command after the first Inquiry except // for the inquiry and request_sense commands. // (I don't know why, and will document further when I know more) if ((s.fail_count > 0) && (s.cur_command != UFI_INQUIRY) && (s.cur_command != UFI_REQUEST_SENSE)) { s.fail_count--; return 0; } if (s.cur_command != UFI_REQUEST_SENSE) { s.sense = 0; s.asc = 0; } switch (s.cur_command) { case UFI_INQUIRY: BX_DEBUG(("UFI INQUIRY COMMAND")); if (s.model) { memcpy(s.usb_buf, bx_cbi_dev_inquiry_teac, sizeof(bx_cbi_dev_inquiry_teac)); s.usb_len = sizeof(bx_cbi_dev_inquiry_teac); } else { memcpy(s.usb_buf, bx_cbi_dev_inquiry_bochs, sizeof(bx_cbi_dev_inquiry_bochs)); s.usb_len = sizeof(bx_cbi_dev_inquiry_bochs); } s.data_len = command[4]; if (s.data_len > s.usb_len) s.data_len = s.usb_len; if (s.did_inquiry_fail == 0) { s.fail_count = 2; s.did_inquiry_fail = 1; } break; case UFI_READ_FORMAT_CAPACITIES: BX_DEBUG(("UFI_READ_FORMAT_CAPACITIES COMMAND")); if (s.inserted) { bx_cbi_dev_frmt_capacity[3] = 32; bx_cbi_dev_frmt_capacity[8] = 0x02; memcpy(s.usb_buf, bx_cbi_dev_frmt_capacity, sizeof(bx_cbi_dev_frmt_capacity)); s.usb_len = sizeof(bx_cbi_dev_frmt_capacity); } else { bx_cbi_dev_frmt_capacity[3] = 8; bx_cbi_dev_frmt_capacity[8] = 0x03; memcpy(s.usb_buf, bx_cbi_dev_frmt_capacity, 12); s.usb_len = 12; } s.data_len = (unsigned) ((command[7] << 8) | command[8]); if (s.data_len > s.usb_len) s.data_len = s.usb_len; break; case UFI_REQUEST_SENSE: BX_DEBUG(("UFI_REQUEST_SENSE COMMAND")); bx_cbi_dev_req_sense[2] = s.sense; bx_cbi_dev_req_sense[12] = s.asc; if (s.sense == 6) { s.sense = 0; } memcpy(s.usb_buf, bx_cbi_dev_req_sense, sizeof(bx_cbi_dev_req_sense)); s.usb_len = sizeof(bx_cbi_dev_req_sense); s.data_len = command[4]; if (s.data_len > s.usb_len) s.data_len = s.usb_len; break; case UFI_READ_12: count = ((command[6] << 24) | (command[7] << 16) | (command[8] << 8) | (command[9] << 0)); case UFI_READ_10: BX_DEBUG(("UFI_READ_%i COMMAND (lba = %i, count = %i)", (s.cur_command == UFI_READ_12) ? 12 : 10, lba, count)); if (!s.inserted) { s.sense = 2; s.asc = 0x3a; break; } s.sector = lba; s.sector_count = count; s.data_len = (count * 512); 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) { s.seek_pending = 1; start_timer(0); } else { bx_gui->statusbar_setitem(s.statusbar_id, 1); // read } break; case UFI_WRITE_12: count = ((command[6] << 24) | (command[7] << 16) | (command[8] << 8) | (command[9] << 0)); case UFI_WRITE_10: BX_DEBUG(("UFI_WRITE_%i COMMAND (lba = %i, count = %i)", (s.cur_command == UFI_WRITE_12) ? 12 : 10, lba, count)); if (!s.inserted) { s.sense = 2; s.asc = 0x3a; break; } s.sector = lba; s.data_len = (count * 512); 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) { s.seek_pending = 1; } break; case UFI_READ_CAPACITY: BX_DEBUG(("UFI_READ_CAPACITY COMMAND")); if (!s.inserted) { s.sense = 2; s.asc = 0x3a; break; } memcpy(s.usb_buf, bx_cbi_dev_capacity, sizeof(bx_cbi_dev_capacity)); s.usb_len = sizeof(bx_cbi_dev_capacity); s.data_len = sizeof(bx_cbi_dev_capacity); break; case UFI_TEST_UNIT_READY: // This is a zero data command. It simply sets the status bytes for // the interrupt in part of the CBI BX_DEBUG(("UFI_TEST_UNIT_READY COMMAND")); if (!s.inserted) { s.sense = 2; s.asc = 0x3a; break; } break; case UFI_PREVENT_ALLOW_REMOVAL: BX_DEBUG(("UFI_PREVENT_ALLOW_REMOVAL COMMAND (prevent = %i)", (command[4] & 1) > 0)); if (command[4] & 1) { s.sense = 5; s.asc = 0x24; break; } break; case UFI_MODE_SENSE: pc = command[2] >> 6; pagecode = command[2] & 0x3F; BX_DEBUG(("UFI_MODE_SENSE COMMAND. PC = %i, PageCode = %02X", pc, pagecode)); switch (pc) { case 0: // current values switch (pagecode) { case 0x01: memcpy(s.usb_buf, bx_cbi_dev_mode_sense_cur, 8); // header first memcpy(s.usb_buf + 8, bx_cbi_dev_mode_sense_cur + PAGE_CODE_01_OFF, 12); s.usb_len = 8 + 12; break; case 0x05: memcpy(s.usb_buf, bx_cbi_dev_mode_sense_cur, 8); // header first memcpy(s.usb_buf + 8, bx_cbi_dev_mode_sense_cur + PAGE_CODE_05_OFF, 32); s.usb_len = 8 + 32; break; case 0x1B: memcpy(s.usb_buf, bx_cbi_dev_mode_sense_cur, 8); // header first memcpy(s.usb_buf + 8, bx_cbi_dev_mode_sense_cur + PAGE_CODE_1B_OFF, 12); s.usb_len = 8 + 12; break; case 0x1C: memcpy(s.usb_buf, bx_cbi_dev_mode_sense_cur, 8); // header first memcpy(s.usb_buf + 8, bx_cbi_dev_mode_sense_cur + PAGE_CODE_1C_OFF, 8); s.usb_len = 8 + 8; break; case 0x3F: memcpy(s.usb_buf, bx_cbi_dev_mode_sense_cur, 8); // header first memcpy(s.usb_buf + 8, bx_cbi_dev_mode_sense_cur + PAGE_CODE_01_OFF, 64); s.usb_len = 8 + 64; break; default: ret = 0; } break; case 1: // changable values case 2: // default values case 3: // saved values ret = 0; break; } s.data_len = (unsigned) ((command[7] << 8) | command[8]); if (s.data_len > s.usb_len) s.data_len = s.usb_len; // set the length of the data returned s.usb_buf[0] = 0; s.usb_buf[1] = (s.usb_len & 0xFF); break; case UFI_START_STOP_UNIT: BX_DEBUG(("UFI_START_STOP_UNIT COMMAND (start = %i)", command[4] & 1)); // The UFI specs say that access to the media is allowed // even if the start/stop command is used to stop the device. // However, we'll allow the command to return valid return. if ((command[4] & 2) != 0) { s.sense = 5; s.asc = 0x24; break; } break; case UFI_FORMAT_UNIT: BX_DEBUG(("UFI_FORMAT_UNIT COMMAND (track = %i)", command[2])); if (!s.inserted) { s.sense = 2; s.asc = 0x3a; break; } s.sector = command[2] * 36; s.data_len = (unsigned) ((command[7] << 8) | command[8]); if (d.async_mode) { s.seek_pending = 1; } break; case UFI_REZERO: case UFI_SEND_DIAGNOSTIC: case UFI_SEEK_10: case UFI_WRITE_VERIFY: case UFI_VERIFY: case UFI_MODE_SELECT: default: BX_ERROR(("Unknown UFI/CBI Command: 0x%02X", s.cur_command)); usb_dump_packet(command, 12); ret = 0; } return ret; } int usb_cbi_device_c::handle_data(USBPacket *p) { int ret = 0; Bit8u devep = p->devep; Bit8u *data = p->data, *tmpbuf; int len = p->len, len1; Bit32u count, max_sectors; switch (p->pid) { case USB_TOKEN_OUT: if (devep != 2) goto fail; BX_DEBUG(("Bulk OUT: %d/%d", len, s.data_len)); switch (s.cur_command) { case UFI_WRITE_10: case UFI_WRITE_12: if (s.wp) goto fail; if (len > (int) s.data_len) goto fail; if (len > 0) { memcpy(s.usb_buf+s.usb_len, data, len); s.usb_len += len; s.data_len -= len; } ret = len; if ((s.data_len == 0) || (s.usb_len >= 512)) { if (d.async_mode) { start_timer(1); BX_DEBUG(("deferring packet %p", p)); usb_defer_packet(p, this); s.packet = p; ret = USB_RET_ASYNC; } else { if (floppy_write_sector() < 0) { ret = 0; break; } else { bx_gui->statusbar_setitem(s.statusbar_id, 1, 1); // write } } } if (ret > 0) usb_dump_packet(data, len); break; case UFI_FORMAT_UNIT: if (s.wp) goto fail; if (len > (int) s.data_len) goto fail; BX_DEBUG(("FORMAT UNIT: single track = %i, side = %i", (data[1] >> 4) & 1, data[1] & 1)); if ((data[1] >> 4) & 1) { if (data[1] & 1) { s.sector += 18; } if (s.hdimage->lseek(s.sector * 512, SEEK_SET) < 0) { BX_ERROR(("could not lseek() floppy drive image file")); break; } if (d.async_mode) { start_timer(2); BX_DEBUG(("deferring packet %p", p)); usb_defer_packet(p, this); s.packet = p; ret = USB_RET_ASYNC; } else { bx_gui->statusbar_setitem(s.statusbar_id, 1, 1); // write memset(s.dev_buffer, 0xff, 18 * 512); if (s.hdimage->write((bx_ptr_t) s.dev_buffer, 18 * 512) < 0) { BX_ERROR(("write error")); break; } ret = len; } } else { BX_ERROR(("FORMAT UNIT with no SINGLE TRACK bit set not yet supported")); } if (ret > 0) usb_dump_packet(data, len); break; default: goto fail; } break; case USB_TOKEN_IN: if (devep == 1) { // Bulk In EP BX_DEBUG(("Bulk IN: %d/%d", len, s.data_len)); switch (s.cur_command) { case UFI_READ_10: case UFI_READ_12: if (len > (int) s.data_len) len = s.data_len; if (d.async_mode) { if (len > (int) s.usb_len) { BX_DEBUG(("deferring packet %p", p)); usb_defer_packet(p, this); s.packet = p; ret = USB_RET_ASYNC; } else { copy_data(p); ret = len; } } else { tmpbuf = data; len1 = len; while (len1 > 0) { if ((int)s.usb_len < len1) { count = s.sector_count; max_sectors = CBI_MAX_SECTORS - (s.usb_len + 511) / 512; if (count > max_sectors) { count = max_sectors; } s.sector_count -= count; ret = (int)s.hdimage->read((bx_ptr_t) s.usb_buf, count * 512); if (ret > 0) { s.usb_len += ret; s.usb_buf += ret; } else { BX_ERROR(("read error")); ret = 0; break; } } if (len1 <= (int)s.usb_len) { memcpy(tmpbuf, s.dev_buffer, len1); s.data_len -= len1; if (s.data_len > 0) { if ((int)s.usb_len > len1) { s.usb_len -= len1; memmove(s.dev_buffer, s.dev_buffer+len1, s.usb_len); s.usb_buf -= len1; } else { s.usb_len = 0; s.usb_buf = s.dev_buffer; } } len1 = 0; } } if (s.data_len > 0) { bx_gui->statusbar_setitem(s.statusbar_id, 1); // read } ret = len; } if (ret > 0) usb_dump_packet(data, ret); break; case UFI_READ_CAPACITY: case UFI_MODE_SENSE: case UFI_READ_FORMAT_CAPACITIES: case UFI_INQUIRY: case UFI_REQUEST_SENSE: if (len > (int) s.data_len) len = s.data_len; memcpy(data, s.usb_buf, len); s.usb_buf += len; s.data_len -= len; usb_dump_packet(data, len); ret = len; break; case UFI_START_STOP_UNIT: case UFI_TEST_UNIT_READY: case UFI_PREVENT_ALLOW_REMOVAL: case UFI_REZERO: case UFI_FORMAT_UNIT: case UFI_SEND_DIAGNOSTIC: case UFI_SEEK_10: case UFI_VERIFY: case UFI_MODE_SELECT: default: goto fail; } #if USB_CBI_USE_INTERRUPT } else if (devep == 3) { // Interrupt In EP BX_DEBUG(("Interrupt IN: 2 bytes")); // We currently do not support error reporting. // We currently assume all transfers are successful memset(data, 0, 2); data[0] = s.asc; ret = 2; #endif } else goto fail; break; default: BX_ERROR(("USB CBI handle_data: bad token")); fail: d.stall = 1; ret = USB_RET_STALL; BX_ERROR(("USB CBI handle_data: stalled")); break; } return ret; } void usb_cbi_device_c::start_timer(Bit8u mode) { Bit32u delay = CBI_SECTOR_TIME; Bit8u new_track, steps; if (mode == 2) { delay *= 18; } bx_gui->statusbar_setitem(s.statusbar_id, 1, (mode > 0)); if (s.seek_pending) { new_track = (s.sector / 36); steps = abs(new_track - s.cur_track); if (steps == 0) steps = 1; // FIXME: this is okay for selected data rate 1000 kbps delay += (steps * 4000); s.cur_track = new_track; s.seek_pending = 0; } bx_pc_system.activate_timer(s.floppy_timer_index, delay, 0); } void usb_cbi_device_c::floppy_timer_handler(void *this_ptr) { usb_cbi_device_c *class_ptr = (usb_cbi_device_c *) this_ptr; class_ptr->floppy_timer(); } void usb_cbi_device_c::floppy_timer() { USBPacket *p = s.packet; int ret = 1; switch (s.cur_command) { case UFI_READ_10: case UFI_READ_12: ret = floppy_read_sector(); break; case UFI_WRITE_10: case UFI_WRITE_12: ret = floppy_write_sector(); break; case UFI_FORMAT_UNIT: memset(s.dev_buffer, 0xff, 18 * 512); if (s.hdimage->write((bx_ptr_t) s.dev_buffer, 18 * 512) < 0) { BX_ERROR(("write error")); ret = -1; } break; default: BX_ERROR(("floppy_timer(): unsupported command")); ret = -1; } if (ret < 0) { p->len = 0; } // ret: 0 = not complete / 1 = complete / -1 = error if ((s.packet != NULL) && (ret != 0)) { usb_dump_packet(p->data, p->len); s.packet = NULL; usb_packet_complete(p); } } int usb_cbi_device_c::floppy_read_sector() { ssize_t ret; USBPacket *p = s.packet; BX_DEBUG(("floppy_read_sector(): sector = %i", s.sector)); if (((CBI_MAX_SECTORS * 512) - s.usb_len) >= 512) { ret = s.hdimage->read((bx_ptr_t) s.usb_buf, 512); if (ret > 0) { s.usb_len += (Bit32u)ret; s.usb_buf += ret; } else { BX_ERROR(("read error")); s.usb_len = 0; } } else { BX_ERROR(("buffer overflow")); s.usb_len = 0; } if (s.usb_len > 0) { s.sector++; s.cur_track = (s.sector / 36); if (--s.sector_count > 0) { start_timer(0); } if (s.packet != NULL) { if (p->len <= (int)s.usb_len) { copy_data(p); } else { return 0; } } return 1; } else { return -1; } } int 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 -1; } else { s.sector++; s.cur_track = (s.sector / 36); 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) { int len = p->len; memcpy(p->data, s.dev_buffer, len); s.data_len -= len; if (s.data_len > 0) { if ((int)s.usb_len > len) { s.usb_len -= len; memmove(s.dev_buffer, s.dev_buffer+len, s.usb_len); s.usb_buf -= len; } else { s.usb_len = 0; s.usb_buf = s.dev_buffer; } } } void usb_cbi_device_c::cancel_packet(USBPacket *p) { bx_pc_system.deactivate_timer(s.floppy_timer_index); s.packet = NULL; } bx_bool usb_cbi_device_c::set_inserted(bx_bool value) { Bit8u mode; s.inserted = value; if (value) { s.fname = SIM->get_param_string("path", s.config)->getptr(); if ((strlen(s.fname) > 0) && (strcmp(s.fname, "none"))) { mode = SIM->get_param_enum("mode", s.config)->get(); s.image_mode = (mode == 1) ? BX_HDIMAGE_MODE_VVFAT : BX_HDIMAGE_MODE_FLAT; s.hdimage = DEV_hdimage_init_image(s.image_mode, 1474560, ""); if ((s.hdimage->open(s.fname)) < 0) { BX_ERROR(("could not open floppy image file '%s'", s.fname)); set_inserted(0); SIM->get_param_enum("status", s.config)->set(BX_EJECTED); } else { s.wp = SIM->get_param_bool("readonly", s.config)->get(); s.sense = 6; s.asc = 0x28; } } else { set_inserted(0); SIM->get_param_enum("status", s.config)->set(BX_EJECTED); } } else { if (s.hdimage != NULL) { s.hdimage->close(); delete s.hdimage; s.hdimage = NULL; } } return s.inserted; } void usb_cbi_device_c::runtime_config(void) { if (s.status_changed) { set_inserted(0); if (SIM->get_param_enum("status", s.config)->get() == BX_INSERTED) { set_inserted(1); } s.status_changed = 0; } } #undef LOG_THIS #define LOG_THIS floppy-> // USB floppy runtime parameter handlers const char *usb_cbi_device_c::floppy_path_handler(bx_param_string_c *param, int set, const char *oldval, const char *val, int maxlen) { usb_cbi_device_c *floppy; if (set) { if (strlen(val) < 1) { val = "none"; } floppy = (usb_cbi_device_c*) param->get_parent()->get_device_param(); if (floppy != NULL) { floppy->s.status_changed = 1; } else { BX_PANIC(("floppy_path_handler: floppy not found")); } } return val; } Bit64s usb_cbi_device_c::floppy_param_handler(bx_param_c *param, int set, Bit64s val) { usb_cbi_device_c *floppy; if (set) { floppy = (usb_cbi_device_c*) param->get_parent()->get_device_param(); if (floppy != NULL) { floppy->s.status_changed = 1; } else { BX_PANIC(("floppy_status_handler: floppy not found")); } } return val; } Bit64s usb_cbi_device_c::param_save_handler(void *devptr, bx_param_c *param) { Bit64s val = 0; usb_cbi_device_c *floppy = (usb_cbi_device_c*) devptr; if (!strcmp(param->get_name(), "usb_buf")) { if (floppy->s.usb_buf != NULL) { val = (Bit32u)(floppy->s.usb_buf - floppy->s.dev_buffer); } else { val = 0; } } else { val = 0; } return val; } void usb_cbi_device_c::param_restore_handler(void *devptr, bx_param_c *param, Bit64s val) { usb_cbi_device_c *floppy = (usb_cbi_device_c*) devptr; if (!strcmp(param->get_name(), "usb_buf")) { floppy->s.usb_buf = floppy->s.dev_buffer + val; } } void usb_cbi_restore_handler(void *dev, bx_list_c *conf) { ((usb_cbi_device_c*)dev)->restore_handler(conf); } void usb_cbi_device_c::restore_handler(bx_list_c *conf) { runtime_config(); } #endif // BX_SUPPORT_PCI && BX_SUPPORT_PCIUSB