mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-02 07:13:54 -06:00

- Ignore writes to CNTP_CTL_EL0 on HVF ARM (Alexander) - Add '-d invalid_mem' logging option (Zoltan) - Create QOM containers explicitly (Peter) - Rename sysemu/ -> system/ (Philippe) - Re-orderning of include/exec/ headers (Philippe) Move a lot of declarations from these legacy mixed bag headers: . "exec/cpu-all.h" . "exec/cpu-common.h" . "exec/cpu-defs.h" . "exec/exec-all.h" . "exec/translate-all" to these more specific ones: . "exec/page-protection.h" . "exec/translation-block.h" . "user/cpu_loop.h" . "user/guest-host.h" . "user/page-protection.h" -----BEGIN PGP SIGNATURE----- iQIzBAABCAAdFiEE+qvnXhKRciHc/Wuy4+MsLN6twN4FAmdlnyAACgkQ4+MsLN6t wN6mBw//QFWi7CrU+bb8KMM53kOU9C507tjn99LLGFb5or73/umDsw6eo/b8DHBt KIwGLgATel42oojKfNKavtAzLK5rOrywpboPDpa3SNeF1onW+99NGJ52LQUqIX6K A6bS0fPdGG9ZzEuPpbjDXlp++0yhDcdSgZsS42fEsT7Dyj5gzJYlqpqhiXGqpsn8 4Y0UMxSL21K3HEexlzw2hsoOBFA3tUm2ujNDhNkt8QASr85yQVLCypABJnuoe/// 5Ojl5wTBeDwhANET0rhwHK8eIYaNboiM9fHopJYhvyw1bz6yAu9jQwzF/MrL3s/r xa4OBHBy5mq2hQV9Shcl3UfCQdk/vDaYaWpgzJGX8stgMGYfnfej1SIl8haJIfcl VMX8/jEFdYbjhO4AeGRYcBzWjEJymkDJZoiSWp2NuEDi6jqIW+7yW1q0Rnlg9lay ShAqLK5Pv4zUw3t0Jy3qv9KSW8sbs6PQxtzXjk8p97rTf76BJ2pF8sv1tVzmsidP 9L92Hv5O34IqzBu2oATOUZYJk89YGmTIUSLkpT7asJZpBLwNM2qLp5jO00WVU0Sd +kAn324guYPkko/TVnjC/AY7CMu55EOtD9NU35k3mUAnxXT9oDUeL4NlYtfgrJx6 x1Nzr2FkS68+wlPAFKNSSU5lTjsjNaFM0bIJ4LCNtenJVP+SnRo= =cjz8 -----END PGP SIGNATURE----- Merge tag 'exec-20241220' of https://github.com/philmd/qemu into staging Accel & Exec patch queue - Ignore writes to CNTP_CTL_EL0 on HVF ARM (Alexander) - Add '-d invalid_mem' logging option (Zoltan) - Create QOM containers explicitly (Peter) - Rename sysemu/ -> system/ (Philippe) - Re-orderning of include/exec/ headers (Philippe) Move a lot of declarations from these legacy mixed bag headers: . "exec/cpu-all.h" . "exec/cpu-common.h" . "exec/cpu-defs.h" . "exec/exec-all.h" . "exec/translate-all" to these more specific ones: . "exec/page-protection.h" . "exec/translation-block.h" . "user/cpu_loop.h" . "user/guest-host.h" . "user/page-protection.h" # -----BEGIN PGP SIGNATURE----- # # iQIzBAABCAAdFiEE+qvnXhKRciHc/Wuy4+MsLN6twN4FAmdlnyAACgkQ4+MsLN6t # wN6mBw//QFWi7CrU+bb8KMM53kOU9C507tjn99LLGFb5or73/umDsw6eo/b8DHBt # KIwGLgATel42oojKfNKavtAzLK5rOrywpboPDpa3SNeF1onW+99NGJ52LQUqIX6K # A6bS0fPdGG9ZzEuPpbjDXlp++0yhDcdSgZsS42fEsT7Dyj5gzJYlqpqhiXGqpsn8 # 4Y0UMxSL21K3HEexlzw2hsoOBFA3tUm2ujNDhNkt8QASr85yQVLCypABJnuoe/// # 5Ojl5wTBeDwhANET0rhwHK8eIYaNboiM9fHopJYhvyw1bz6yAu9jQwzF/MrL3s/r # xa4OBHBy5mq2hQV9Shcl3UfCQdk/vDaYaWpgzJGX8stgMGYfnfej1SIl8haJIfcl # VMX8/jEFdYbjhO4AeGRYcBzWjEJymkDJZoiSWp2NuEDi6jqIW+7yW1q0Rnlg9lay # ShAqLK5Pv4zUw3t0Jy3qv9KSW8sbs6PQxtzXjk8p97rTf76BJ2pF8sv1tVzmsidP # 9L92Hv5O34IqzBu2oATOUZYJk89YGmTIUSLkpT7asJZpBLwNM2qLp5jO00WVU0Sd # +kAn324guYPkko/TVnjC/AY7CMu55EOtD9NU35k3mUAnxXT9oDUeL4NlYtfgrJx6 # x1Nzr2FkS68+wlPAFKNSSU5lTjsjNaFM0bIJ4LCNtenJVP+SnRo= # =cjz8 # -----END PGP SIGNATURE----- # gpg: Signature made Fri 20 Dec 2024 11:45:20 EST # gpg: using RSA key FAABE75E12917221DCFD6BB2E3E32C2CDEADC0DE # gpg: Good signature from "Philippe Mathieu-Daudé (F4BUG) <f4bug@amsat.org>" [unknown] # gpg: WARNING: This key is not certified with a trusted signature! # gpg: There is no indication that the signature belongs to the owner. # Primary key fingerprint: FAAB E75E 1291 7221 DCFD 6BB2 E3E3 2C2C DEAD C0DE * tag 'exec-20241220' of https://github.com/philmd/qemu: (59 commits) util/qemu-timer: fix indentation meson: Do not define CONFIG_DEVICES on user emulation system/accel-ops: Remove unnecessary 'exec/cpu-common.h' header system/numa: Remove unnecessary 'exec/cpu-common.h' header hw/xen: Remove unnecessary 'exec/cpu-common.h' header target/mips: Drop left-over comment about Jazz machine target/mips: Remove tswap() calls in semihosting uhi_fstat_cb() target/xtensa: Remove tswap() calls in semihosting simcall() helper accel/tcg: Un-inline translator_is_same_page() accel/tcg: Include missing 'exec/translation-block.h' header accel/tcg: Move tcg_cflags_has/set() to 'exec/translation-block.h' accel/tcg: Restrict curr_cflags() declaration to 'internal-common.h' qemu/coroutine: Include missing 'qemu/atomic.h' header exec/translation-block: Include missing 'qemu/atomic.h' header accel/tcg: Declare cpu_loop_exit_requested() in 'exec/cpu-common.h' exec/cpu-all: Include 'cpu.h' earlier so MMU_USER_IDX is always defined target/sparc: Move sparc_restore_state_to_opc() to cpu.c target/sparc: Uninline cpu_get_tb_cpu_state() target/loongarch: Declare loongarch_cpu_dump_state() locally user: Move various declarations out of 'exec/exec-all.h' ... Conflicts: hw/char/riscv_htif.c hw/intc/riscv_aplic.c target/s390x/cpu.c Apply sysemu header path changes to not in the pull request. Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
577 lines
16 KiB
C
577 lines
16 KiB
C
/*
|
|
* 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 "system/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 const Property swim_drive_properties[] = {
|
|
DEFINE_PROP_INT32("unit", SWIMDrive, unit, -1),
|
|
DEFINE_BLOCK_PROPERTIES(SWIMDrive, conf),
|
|
};
|
|
|
|
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)
|