/* * QEMU Macintosh floppy disk controller emulator (SWIM) * * Copyright (c) 2014-2018 Laurent Vivier * * This work is licensed under the terms of the GNU GPL, version 2. See * the COPYING file in the top-level directory. * * Only the basic support: it allows to switch from IWM (Integrated WOZ * Machine) mode to the SWIM mode and makes the linux driver happy. */ #include "qemu/osdep.h" #include "qemu/main-loop.h" #include "qapi/error.h" #include "sysemu/block-backend.h" #include "hw/sysbus.h" #include "migration/vmstate.h" #include "hw/block/block.h" #include "hw/block/swim.h" #include "hw/qdev-properties.h" #include "trace.h" /* IWM latch bits */ #define IWMLB_PHASE0 0 #define IWMLB_PHASE1 1 #define IWMLB_PHASE2 2 #define IWMLB_PHASE3 3 #define IWMLB_MOTORON 4 #define IWMLB_DRIVESEL 5 #define IWMLB_L6 6 #define IWMLB_L7 7 /* IWM registers */ #define IWM_READALLONES 0 #define IWM_READDATA 1 #define IWM_READSTATUS0 2 #define IWM_READSTATUS1 3 #define IWM_READWHANDSHAKE0 4 #define IWM_READWHANDSHAKE1 5 #define IWM_WRITESETMODE 6 #define IWM_WRITEDATA 7 /* SWIM registers */ #define SWIM_WRITE_DATA 0 #define SWIM_WRITE_MARK 1 #define SWIM_WRITE_CRC 2 #define SWIM_WRITE_PARAMETER 3 #define SWIM_WRITE_PHASE 4 #define SWIM_WRITE_SETUP 5 #define SWIM_WRITE_MODE0 6 #define SWIM_WRITE_MODE1 7 #define SWIM_READ_DATA 8 #define SWIM_READ_MARK 9 #define SWIM_READ_ERROR 10 #define SWIM_READ_PARAMETER 11 #define SWIM_READ_PHASE 12 #define SWIM_READ_SETUP 13 #define SWIM_READ_STATUS 14 #define SWIM_READ_HANDSHAKE 15 #define REG_SHIFT 9 #define SWIM_MODE_STATUS_BIT 6 #define SWIM_MODE_IWM 0 #define SWIM_MODE_ISM 1 /* bits in phase register */ #define SWIM_SEEK_NEGATIVE 0x074 #define SWIM_STEP 0x071 #define SWIM_MOTOR_ON 0x072 #define SWIM_MOTOR_OFF 0x076 #define SWIM_INDEX 0x073 #define SWIM_EJECT 0x077 #define SWIM_SETMFM 0x171 #define SWIM_SETGCR 0x175 #define SWIM_RELAX 0x033 #define SWIM_LSTRB 0x008 #define SWIM_CA_MASK 0x077 /* Select values for swim_select and swim_readbit */ #define SWIM_READ_DATA_0 0x074 #define SWIM_TWOMEG_DRIVE 0x075 #define SWIM_SINGLE_SIDED 0x076 #define SWIM_DRIVE_PRESENT 0x077 #define SWIM_DISK_IN 0x170 #define SWIM_WRITE_PROT 0x171 #define SWIM_TRACK_ZERO 0x172 #define SWIM_TACHO 0x173 #define SWIM_READ_DATA_1 0x174 #define SWIM_MFM_MODE 0x175 #define SWIM_SEEK_COMPLETE 0x176 #define SWIM_ONEMEG_MEDIA 0x177 /* Bits in handshake register */ #define SWIM_MARK_BYTE 0x01 #define SWIM_CRC_ZERO 0x02 #define SWIM_RDDATA 0x04 #define SWIM_SENSE 0x08 #define SWIM_MOTEN 0x10 #define SWIM_ERROR 0x20 #define SWIM_DAT2BYTE 0x40 #define SWIM_DAT1BYTE 0x80 /* bits in setup register */ #define SWIM_S_INV_WDATA 0x01 #define SWIM_S_3_5_SELECT 0x02 #define SWIM_S_GCR 0x04 #define SWIM_S_FCLK_DIV2 0x08 #define SWIM_S_ERROR_CORR 0x10 #define SWIM_S_IBM_DRIVE 0x20 #define SWIM_S_GCR_WRITE 0x40 #define SWIM_S_TIMEOUT 0x80 /* bits in mode register */ #define SWIM_CLFIFO 0x01 #define SWIM_ENBL1 0x02 #define SWIM_ENBL2 0x04 #define SWIM_ACTION 0x08 #define SWIM_WRITE_MODE 0x10 #define SWIM_HEDSEL 0x20 #define SWIM_MOTON 0x80 static const char *iwm_reg_names[] = { "READALLONES", "READDATA", "READSTATUS0", "READSTATUS1", "READWHANDSHAKE0", "READWHANDSHAKE1", "WRITESETMODE", "WRITEDATA" }; static const char *ism_reg_names[] = { "WRITE_DATA", "WRITE_MARK", "WRITE_CRC", "WRITE_PARAMETER", "WRITE_PHASE", "WRITE_SETUP", "WRITE_MODE0", "WRITE_MODE1", "READ_DATA", "READ_MARK", "READ_ERROR", "READ_PARAMETER", "READ_PHASE", "READ_SETUP", "READ_STATUS", "READ_HANDSHAKE" }; static void fd_recalibrate(FDrive *drive) { } static void swim_change_cb(void *opaque, bool load, Error **errp) { FDrive *drive = opaque; if (!load) { blk_set_perm(drive->blk, 0, BLK_PERM_ALL, &error_abort); } else { if (!blkconf_apply_backend_options(drive->conf, !blk_supports_write_perm(drive->blk), false, errp)) { return; } } } static const BlockDevOps swim_block_ops = { .change_media_cb = swim_change_cb, }; static Property swim_drive_properties[] = { DEFINE_PROP_INT32("unit", SWIMDrive, unit, -1), DEFINE_BLOCK_PROPERTIES(SWIMDrive, conf), DEFINE_PROP_END_OF_LIST(), }; static void swim_drive_realize(DeviceState *qdev, Error **errp) { SWIMDrive *dev = SWIM_DRIVE(qdev); SWIMBus *bus = SWIM_BUS(qdev->parent_bus); FDrive *drive; int ret; if (dev->unit == -1) { for (dev->unit = 0; dev->unit < SWIM_MAX_FD; dev->unit++) { drive = &bus->ctrl->drives[dev->unit]; if (!drive->blk) { break; } } } if (dev->unit >= SWIM_MAX_FD) { error_setg(errp, "Can't create floppy unit %d, bus supports " "only %d units", dev->unit, SWIM_MAX_FD); return; } drive = &bus->ctrl->drives[dev->unit]; if (drive->blk) { error_setg(errp, "Floppy unit %d is in use", dev->unit); return; } if (!dev->conf.blk) { /* Anonymous BlockBackend for an empty drive */ dev->conf.blk = blk_new(qemu_get_aio_context(), 0, BLK_PERM_ALL); ret = blk_attach_dev(dev->conf.blk, qdev); assert(ret == 0); } if (!blkconf_blocksizes(&dev->conf, errp)) { return; } if (dev->conf.logical_block_size != 512 || dev->conf.physical_block_size != 512) { error_setg(errp, "Physical and logical block size must " "be 512 for floppy"); return; } /* * rerror/werror aren't supported by fdc and therefore not even registered * with qdev. So set the defaults manually before they are used in * blkconf_apply_backend_options(). */ dev->conf.rerror = BLOCKDEV_ON_ERROR_AUTO; dev->conf.werror = BLOCKDEV_ON_ERROR_AUTO; if (!blkconf_apply_backend_options(&dev->conf, !blk_supports_write_perm(dev->conf.blk), false, errp)) { return; } /* * 'enospc' is the default for -drive, 'report' is what blk_new() gives us * for empty drives. */ if (blk_get_on_error(dev->conf.blk, 0) != BLOCKDEV_ON_ERROR_ENOSPC && blk_get_on_error(dev->conf.blk, 0) != BLOCKDEV_ON_ERROR_REPORT) { error_setg(errp, "fdc doesn't support drive option werror"); return; } if (blk_get_on_error(dev->conf.blk, 1) != BLOCKDEV_ON_ERROR_REPORT) { error_setg(errp, "fdc doesn't support drive option rerror"); return; } drive->conf = &dev->conf; drive->blk = dev->conf.blk; drive->swimctrl = bus->ctrl; blk_set_dev_ops(drive->blk, &swim_block_ops, drive); } static void swim_drive_class_init(ObjectClass *klass, void *data) { DeviceClass *k = DEVICE_CLASS(klass); k->realize = swim_drive_realize; set_bit(DEVICE_CATEGORY_STORAGE, k->categories); k->bus_type = TYPE_SWIM_BUS; device_class_set_props(k, swim_drive_properties); k->desc = "virtual SWIM drive"; } static const TypeInfo swim_drive_info = { .name = TYPE_SWIM_DRIVE, .parent = TYPE_DEVICE, .instance_size = sizeof(SWIMDrive), .class_init = swim_drive_class_init, }; static const TypeInfo swim_bus_info = { .name = TYPE_SWIM_BUS, .parent = TYPE_BUS, .instance_size = sizeof(SWIMBus), }; static void iwmctrl_write(void *opaque, hwaddr addr, uint64_t value, unsigned size) { SWIMCtrl *swimctrl = opaque; uint8_t latch, reg, ism_bit; addr >>= REG_SHIFT; /* A3-A1 select a latch, A0 specifies the value */ latch = (addr >> 1) & 7; if (addr & 1) { swimctrl->iwm_latches |= (1 << latch); } else { swimctrl->iwm_latches &= ~(1 << latch); } reg = (swimctrl->iwm_latches & 0xc0) >> 5 | (swimctrl->iwm_latches & 0x10) >> 4; swimctrl->iwmregs[reg] = value; trace_swim_iwmctrl_write(reg, iwm_reg_names[reg], size, value); switch (reg) { case IWM_WRITESETMODE: /* detect sequence to switch from IWM mode to SWIM mode */ ism_bit = (value & (1 << SWIM_MODE_STATUS_BIT)); switch (swimctrl->iwm_switch) { case 0: if (ism_bit) { /* 1 */ swimctrl->iwm_switch++; } break; case 1: if (!ism_bit) { /* 0 */ swimctrl->iwm_switch++; } break; case 2: if (ism_bit) { /* 1 */ swimctrl->iwm_switch++; } break; case 3: if (ism_bit) { /* 1 */ swimctrl->iwm_switch++; swimctrl->mode = SWIM_MODE_ISM; swimctrl->swim_mode |= (1 << SWIM_MODE_STATUS_BIT); swimctrl->iwm_switch = 0; trace_swim_switch_to_ism(); /* Switch to ISM registers */ memory_region_del_subregion(&swimctrl->swim, &swimctrl->iwm); memory_region_add_subregion(&swimctrl->swim, 0x0, &swimctrl->ism); } break; } break; default: break; } } static uint64_t iwmctrl_read(void *opaque, hwaddr addr, unsigned size) { SWIMCtrl *swimctrl = opaque; uint8_t latch, reg, value; addr >>= REG_SHIFT; /* A3-A1 select a latch, A0 specifies the value */ latch = (addr >> 1) & 7; if (addr & 1) { swimctrl->iwm_latches |= (1 << latch); } else { swimctrl->iwm_latches &= ~(1 << latch); } reg = (swimctrl->iwm_latches & 0xc0) >> 5 | (swimctrl->iwm_latches & 0x10) >> 4; switch (reg) { case IWM_READALLONES: value = 0xff; break; default: value = 0; break; } trace_swim_iwmctrl_read(reg, iwm_reg_names[reg], size, value); return value; } static const MemoryRegionOps swimctrl_iwm_ops = { .write = iwmctrl_write, .read = iwmctrl_read, .endianness = DEVICE_BIG_ENDIAN, }; static void ismctrl_write(void *opaque, hwaddr reg, uint64_t value, unsigned size) { SWIMCtrl *swimctrl = opaque; reg >>= REG_SHIFT; trace_swim_ismctrl_write(reg, ism_reg_names[reg], size, value); switch (reg) { case SWIM_WRITE_PHASE: swimctrl->swim_phase = value; break; case SWIM_WRITE_MODE0: swimctrl->swim_mode &= ~value; /* Any access to MODE0 register resets PRAM index */ swimctrl->pram_idx = 0; if (!(swimctrl->swim_mode & (1 << SWIM_MODE_STATUS_BIT))) { /* Clearing the mode bit switches to IWM mode */ swimctrl->mode = SWIM_MODE_IWM; swimctrl->iwm_latches = 0; trace_swim_switch_to_iwm(); /* Switch to IWM registers */ memory_region_del_subregion(&swimctrl->swim, &swimctrl->ism); memory_region_add_subregion(&swimctrl->swim, 0x0, &swimctrl->iwm); } break; case SWIM_WRITE_MODE1: swimctrl->swim_mode |= value; break; case SWIM_WRITE_PARAMETER: swimctrl->pram[swimctrl->pram_idx++] = value; swimctrl->pram_idx &= 0xf; break; case SWIM_WRITE_DATA: case SWIM_WRITE_MARK: case SWIM_WRITE_CRC: case SWIM_WRITE_SETUP: break; } } static uint64_t ismctrl_read(void *opaque, hwaddr reg, unsigned size) { SWIMCtrl *swimctrl = opaque; uint32_t value = 0; reg >>= REG_SHIFT; switch (reg) { case SWIM_READ_PHASE: value = swimctrl->swim_phase; break; case SWIM_READ_HANDSHAKE: if (swimctrl->swim_phase == SWIM_DRIVE_PRESENT) { /* always answer "no drive present" */ value = SWIM_SENSE; } break; case SWIM_READ_PARAMETER: value = swimctrl->pram[swimctrl->pram_idx++]; swimctrl->pram_idx &= 0xf; break; case SWIM_READ_STATUS: value = swimctrl->swim_status & ~(1 << SWIM_MODE_STATUS_BIT); if (swimctrl->swim_mode == SWIM_MODE_ISM) { value |= (1 << SWIM_MODE_STATUS_BIT); } break; case SWIM_READ_DATA: case SWIM_READ_MARK: case SWIM_READ_ERROR: case SWIM_READ_SETUP: break; } trace_swim_ismctrl_read(reg, ism_reg_names[reg], size, value); return value; } static const MemoryRegionOps swimctrl_ism_ops = { .write = ismctrl_write, .read = ismctrl_read, .endianness = DEVICE_BIG_ENDIAN, }; static void sysbus_swim_reset(DeviceState *d) { Swim *sys = SWIM(d); SWIMCtrl *ctrl = &sys->ctrl; int i; ctrl->mode = 0; ctrl->iwm_switch = 0; memset(ctrl->iwmregs, 0, sizeof(ctrl->iwmregs)); ctrl->swim_phase = 0; ctrl->swim_mode = 0; memset(ctrl->ismregs, 0, sizeof(ctrl->ismregs)); for (i = 0; i < SWIM_MAX_FD; i++) { fd_recalibrate(&ctrl->drives[i]); } } static void sysbus_swim_init(Object *obj) { SysBusDevice *sbd = SYS_BUS_DEVICE(obj); Swim *sbs = SWIM(obj); SWIMCtrl *swimctrl = &sbs->ctrl; memory_region_init(&swimctrl->swim, obj, "swim", 0x2000); memory_region_init_io(&swimctrl->iwm, obj, &swimctrl_iwm_ops, swimctrl, "iwm", 0x2000); memory_region_init_io(&swimctrl->ism, obj, &swimctrl_ism_ops, swimctrl, "ism", 0x2000); sysbus_init_mmio(sbd, &swimctrl->swim); } static void sysbus_swim_realize(DeviceState *dev, Error **errp) { Swim *sys = SWIM(dev); SWIMCtrl *swimctrl = &sys->ctrl; qbus_init(&swimctrl->bus, sizeof(SWIMBus), TYPE_SWIM_BUS, dev, NULL); swimctrl->bus.ctrl = swimctrl; /* Default register set is IWM */ memory_region_add_subregion(&swimctrl->swim, 0x0, &swimctrl->iwm); } static const VMStateDescription vmstate_fdrive = { .name = "fdrive", .version_id = 1, .minimum_version_id = 1, .fields = (const VMStateField[]) { VMSTATE_END_OF_LIST() }, }; static const VMStateDescription vmstate_swim = { .name = "swim", .version_id = 1, .minimum_version_id = 1, .fields = (const VMStateField[]) { VMSTATE_INT32(mode, SWIMCtrl), /* IWM mode */ VMSTATE_INT32(iwm_switch, SWIMCtrl), VMSTATE_UINT8(iwm_latches, SWIMCtrl), VMSTATE_UINT8_ARRAY(iwmregs, SWIMCtrl, 8), /* SWIM mode */ VMSTATE_UINT8_ARRAY(ismregs, SWIMCtrl, 16), VMSTATE_UINT8(swim_phase, SWIMCtrl), VMSTATE_UINT8(swim_mode, SWIMCtrl), /* Drives */ VMSTATE_STRUCT_ARRAY(drives, SWIMCtrl, SWIM_MAX_FD, 1, vmstate_fdrive, FDrive), VMSTATE_END_OF_LIST() }, }; static const VMStateDescription vmstate_sysbus_swim = { .name = "SWIM", .version_id = 1, .fields = (const VMStateField[]) { VMSTATE_STRUCT(ctrl, Swim, 0, vmstate_swim, SWIMCtrl), VMSTATE_END_OF_LIST() } }; static void sysbus_swim_class_init(ObjectClass *oc, void *data) { DeviceClass *dc = DEVICE_CLASS(oc); dc->realize = sysbus_swim_realize; device_class_set_legacy_reset(dc, sysbus_swim_reset); dc->vmsd = &vmstate_sysbus_swim; } static const TypeInfo sysbus_swim_info = { .name = TYPE_SWIM, .parent = TYPE_SYS_BUS_DEVICE, .instance_size = sizeof(Swim), .instance_init = sysbus_swim_init, .class_init = sysbus_swim_class_init, }; static void swim_register_types(void) { type_register_static(&sysbus_swim_info); type_register_static(&swim_bus_info); type_register_static(&swim_drive_info); } type_init(swim_register_types)