/* * 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 registers */ #define IWM_PH0L 0 #define IWM_PH0H 1 #define IWM_PH1L 2 #define IWM_PH1H 3 #define IWM_PH2L 4 #define IWM_PH2H 5 #define IWM_PH3L 6 #define IWM_PH3H 7 #define IWM_MTROFF 8 #define IWM_MTRON 9 #define IWM_INTDRIVE 10 #define IWM_EXTDRIVE 11 #define IWM_Q6L 12 #define IWM_Q6H 13 #define IWM_Q7L 14 #define IWM_Q7H 15 /* 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_IWM 0 #define SWIM_MODE_SWIM 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 *swim_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 reg, uint64_t value, unsigned size) { SWIMCtrl *swimctrl = opaque; reg >>= REG_SHIFT; swimctrl->regs[reg >> 1] = reg & 1; trace_swim_iwmctrl_write((reg >> 1), size, (reg & 1)); if (swimctrl->regs[IWM_Q6] && swimctrl->regs[IWM_Q7]) { if (swimctrl->regs[IWM_MTR]) { /* data register */ swimctrl->iwm_data = value; } else { /* mode register */ swimctrl->iwm_mode = value; /* detect sequence to switch from IWM mode to SWIM mode */ switch (swimctrl->iwm_switch) { case 0: if (value == 0x57) { swimctrl->iwm_switch++; } break; case 1: if (value == 0x17) { swimctrl->iwm_switch++; } break; case 2: if (value == 0x57) { swimctrl->iwm_switch++; } break; case 3: if (value == 0x57) { swimctrl->mode = SWIM_MODE_SWIM; swimctrl->iwm_switch = 0; trace_swim_iwm_switch(); } break; } } } } static uint64_t iwmctrl_read(void *opaque, hwaddr reg, unsigned size) { SWIMCtrl *swimctrl = opaque; reg >>= REG_SHIFT; swimctrl->regs[reg >> 1] = reg & 1; trace_swim_iwmctrl_read((reg >> 1), size, (reg & 1)); return 0; } static void swimctrl_write(void *opaque, hwaddr reg, uint64_t value, unsigned size) { SWIMCtrl *swimctrl = opaque; if (swimctrl->mode == SWIM_MODE_IWM) { iwmctrl_write(opaque, reg, value, size); return; } reg >>= REG_SHIFT; trace_swim_swimctrl_write(reg, swim_reg_names[reg], size, value); switch (reg) { case SWIM_WRITE_PHASE: swimctrl->swim_phase = value; break; case SWIM_WRITE_MODE0: swimctrl->swim_mode &= ~value; break; case SWIM_WRITE_MODE1: swimctrl->swim_mode |= value; break; case SWIM_WRITE_DATA: case SWIM_WRITE_MARK: case SWIM_WRITE_CRC: case SWIM_WRITE_PARAMETER: case SWIM_WRITE_SETUP: break; } } static uint64_t swimctrl_read(void *opaque, hwaddr reg, unsigned size) { SWIMCtrl *swimctrl = opaque; uint32_t value = 0; if (swimctrl->mode == SWIM_MODE_IWM) { return iwmctrl_read(opaque, reg, size); } 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_DATA: case SWIM_READ_MARK: case SWIM_READ_ERROR: case SWIM_READ_PARAMETER: case SWIM_READ_SETUP: case SWIM_READ_STATUS: break; } trace_swim_swimctrl_read(reg, swim_reg_names[reg], size, value); return value; } static const MemoryRegionOps swimctrl_mem_ops = { .write = swimctrl_write, .read = swimctrl_read, .endianness = DEVICE_NATIVE_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; for (i = 0; i < 8; i++) { ctrl->regs[i] = 0; } ctrl->iwm_data = 0; ctrl->iwm_mode = 0; ctrl->swim_phase = 0; ctrl->swim_mode = 0; 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_io(&swimctrl->iomem, obj, &swimctrl_mem_ops, swimctrl, "swim", 0x2000); sysbus_init_mmio(sbd, &swimctrl->iomem); } 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; } static const VMStateDescription vmstate_fdrive = { .name = "fdrive", .version_id = 1, .minimum_version_id = 1, .fields = (VMStateField[]) { VMSTATE_END_OF_LIST() }, }; static const VMStateDescription vmstate_swim = { .name = "swim", .version_id = 1, .minimum_version_id = 1, .fields = (VMStateField[]) { VMSTATE_INT32(mode, SWIMCtrl), /* IWM mode */ VMSTATE_INT32(iwm_switch, SWIMCtrl), VMSTATE_UINT16_ARRAY(regs, SWIMCtrl, 8), VMSTATE_UINT8(iwm_data, SWIMCtrl), VMSTATE_UINT8(iwm_mode, SWIMCtrl), /* SWIM mode */ 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 = (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; dc->reset = 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)