| /* |
| * QEMU Macintosh floppy disk controller emulator (SWIM) |
| * |
| * Copyright (c) 2014-2018 Laurent Vivier <laurent@vivier.eu> |
| * |
| * 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; |
| 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) |