Re: [Qemu-devel] [PATCH 2/3] fdc: Add a floppy drive qdev
On 09/05/2016 11:43 AM, Kevin Wolf wrote: Floppy controllers automatically create two floppy drive devices in qdev now. (They always created two drives, but managed them only internally.) This appears to *actually* create and expose two drives by default, is this intentional? Previously, we created "two drives", but only exposed ones for which drive->blk was true -- which was in practice only the first drive by default. The CMOS magic we'd expose would be either 0x40 or 0x50 (1.44MB and no drive or 2.88MB and no drive) -- now it's 0x55 (two 2.88MB drives.) As it stands right now, both drives get made as type "AUTO". If drive->blk is present, we'll evaluate this to either 128, 144 or 288. We'll use a fallback type of 288 if no medium is present or we couldn't figure out what was in there. Otherwise, we'll fall through to fd_revalidate's "No Drive Connected" segment which will more or less delete the drive. This changes fdc-test and acpi-test which will need to be changed if this is really what you want. --js Signed-off-by: Kevin Wolf --- hw/block/fdc.c | 152 +++-- 1 file changed, 115 insertions(+), 37 deletions(-) diff --git a/hw/block/fdc.c b/hw/block/fdc.c index 4164b31..d1e2339 100644 --- a/hw/block/fdc.c +++ b/hw/block/fdc.c @@ -60,6 +60,8 @@ #define FLOPPY_BUS(obj) OBJECT_CHECK(FloppyBus, (obj), TYPE_FLOPPY_BUS) typedef struct FDCtrl FDCtrl; +typedef struct FDrive FDrive; +static FDrive *get_drv(FDCtrl *fdctrl, int unit); typedef struct FloppyBus { BusState bus; @@ -180,7 +182,7 @@ typedef enum FDiskFlags { FDISK_DBL_SIDES = 0x01, } FDiskFlags; -typedef struct FDrive { +struct FDrive { FDCtrl *fdctrl; BlockBackend *blk; /* Drive status */ @@ -201,7 +203,7 @@ typedef struct FDrive { uint8_t media_rate; /* Data rate of medium*/ bool media_validated; /* Have we validated the media? */ -} FDrive; +}; static FloppyDriveType get_fallback_drive_type(FDrive *drv); @@ -466,6 +468,101 @@ static void fd_revalidate(FDrive *drv) } } +static void fd_change_cb(void *opaque, bool load) +{ +FDrive *drive = opaque; + +drive->media_changed = 1; +drive->media_validated = false; +fd_revalidate(drive); +} + +static const BlockDevOps fd_block_ops = { +.change_media_cb = fd_change_cb, +}; + + +#define TYPE_FLOPPY_DRIVE "floppy" +#define FLOPPY_DRIVE(obj) \ + OBJECT_CHECK(FloppyDrive, (obj), TYPE_FLOPPY_DRIVE) + +typedef struct FloppyDrive { +DeviceState qdev; +uint32_tunit; +} FloppyDrive; + +static Property floppy_drive_properties[] = { +DEFINE_PROP_UINT32("unit", FloppyDrive, unit, -1), +DEFINE_PROP_END_OF_LIST(), +}; + +static int floppy_drive_init(DeviceState *qdev) +{ +FloppyDrive *dev = FLOPPY_DRIVE(qdev); +FloppyBus *bus = DO_UPCAST(FloppyBus, bus, dev->qdev.parent_bus); +FDrive *drive; + +if (dev->unit == -1) { +for (dev->unit = 0; dev->unit < MAX_FD; dev->unit++) { +drive = get_drv(bus->fdc, dev->unit); +if (!drive->blk) { +break; +} +} +} + +if (dev->unit >= MAX_FD) { +error_report("Can't create floppy unit %d, bus supports only %d units", + dev->unit, MAX_FD); +return -1; +} + +/* TODO Check whether unit is in use */ + +drive = get_drv(bus->fdc, dev->unit); +drive->fdctrl = bus->fdc; + +if (drive->blk) { +if (blk_get_on_error(drive->blk, 0) != BLOCKDEV_ON_ERROR_ENOSPC) { +error_report("fdc doesn't support drive option werror"); +return -1; +} +if (blk_get_on_error(drive->blk, 1) != BLOCKDEV_ON_ERROR_REPORT) { +error_report("fdc doesn't support drive option rerror"); +return -1; +} +} else { +/* Anonymous BlockBackend for an empty drive */ +drive->blk = blk_new(); +} + +fd_init(drive); +if (drive->blk) { +blk_set_dev_ops(drive->blk, &fd_block_ops, drive); +pick_drive_type(drive); +} +fd_revalidate(drive); + +return 0; +} + +static void floppy_drive_class_init(ObjectClass *klass, void *data) +{ +DeviceClass *k = DEVICE_CLASS(klass); +k->init = floppy_drive_init; +set_bit(DEVICE_CATEGORY_STORAGE, k->categories); +k->bus_type = TYPE_FLOPPY_BUS; +k->props = floppy_drive_properties; +k->desc = "virtual floppy drive"; +} + +static const TypeInfo floppy_drive_info = { +.name = TYPE_FLOPPY_DRIVE, +.parent = TYPE_DEVICE, +.instance_size = sizeof(FloppyDrive), +.class_init = floppy_drive_class_init, +}; + // /* Intel 82078 floppy disk controller emulation */ @@ -1184,9 +1281,9 @@ static inline FDrive *drv3(FDCtrl *fdctrl) } #endif -static FDrive *get_cur_drv(FDCtrl *fdctrl) +static FDrive *get_drv(FDCtrl *fdctrl, int unit) { -switch (fdc
[Qemu-devel] [PATCH 2/3] fdc: Add a floppy drive qdev
Floppy controllers automatically create two floppy drive devices in qdev now. (They always created two drives, but managed them only internally.) Signed-off-by: Kevin Wolf --- hw/block/fdc.c | 152 +++-- 1 file changed, 115 insertions(+), 37 deletions(-) diff --git a/hw/block/fdc.c b/hw/block/fdc.c index 4164b31..d1e2339 100644 --- a/hw/block/fdc.c +++ b/hw/block/fdc.c @@ -60,6 +60,8 @@ #define FLOPPY_BUS(obj) OBJECT_CHECK(FloppyBus, (obj), TYPE_FLOPPY_BUS) typedef struct FDCtrl FDCtrl; +typedef struct FDrive FDrive; +static FDrive *get_drv(FDCtrl *fdctrl, int unit); typedef struct FloppyBus { BusState bus; @@ -180,7 +182,7 @@ typedef enum FDiskFlags { FDISK_DBL_SIDES = 0x01, } FDiskFlags; -typedef struct FDrive { +struct FDrive { FDCtrl *fdctrl; BlockBackend *blk; /* Drive status */ @@ -201,7 +203,7 @@ typedef struct FDrive { uint8_t media_rate; /* Data rate of medium*/ bool media_validated; /* Have we validated the media? */ -} FDrive; +}; static FloppyDriveType get_fallback_drive_type(FDrive *drv); @@ -466,6 +468,101 @@ static void fd_revalidate(FDrive *drv) } } +static void fd_change_cb(void *opaque, bool load) +{ +FDrive *drive = opaque; + +drive->media_changed = 1; +drive->media_validated = false; +fd_revalidate(drive); +} + +static const BlockDevOps fd_block_ops = { +.change_media_cb = fd_change_cb, +}; + + +#define TYPE_FLOPPY_DRIVE "floppy" +#define FLOPPY_DRIVE(obj) \ + OBJECT_CHECK(FloppyDrive, (obj), TYPE_FLOPPY_DRIVE) + +typedef struct FloppyDrive { +DeviceState qdev; +uint32_tunit; +} FloppyDrive; + +static Property floppy_drive_properties[] = { +DEFINE_PROP_UINT32("unit", FloppyDrive, unit, -1), +DEFINE_PROP_END_OF_LIST(), +}; + +static int floppy_drive_init(DeviceState *qdev) +{ +FloppyDrive *dev = FLOPPY_DRIVE(qdev); +FloppyBus *bus = DO_UPCAST(FloppyBus, bus, dev->qdev.parent_bus); +FDrive *drive; + +if (dev->unit == -1) { +for (dev->unit = 0; dev->unit < MAX_FD; dev->unit++) { +drive = get_drv(bus->fdc, dev->unit); +if (!drive->blk) { +break; +} +} +} + +if (dev->unit >= MAX_FD) { +error_report("Can't create floppy unit %d, bus supports only %d units", + dev->unit, MAX_FD); +return -1; +} + +/* TODO Check whether unit is in use */ + +drive = get_drv(bus->fdc, dev->unit); +drive->fdctrl = bus->fdc; + +if (drive->blk) { +if (blk_get_on_error(drive->blk, 0) != BLOCKDEV_ON_ERROR_ENOSPC) { +error_report("fdc doesn't support drive option werror"); +return -1; +} +if (blk_get_on_error(drive->blk, 1) != BLOCKDEV_ON_ERROR_REPORT) { +error_report("fdc doesn't support drive option rerror"); +return -1; +} +} else { +/* Anonymous BlockBackend for an empty drive */ +drive->blk = blk_new(); +} + +fd_init(drive); +if (drive->blk) { +blk_set_dev_ops(drive->blk, &fd_block_ops, drive); +pick_drive_type(drive); +} +fd_revalidate(drive); + +return 0; +} + +static void floppy_drive_class_init(ObjectClass *klass, void *data) +{ +DeviceClass *k = DEVICE_CLASS(klass); +k->init = floppy_drive_init; +set_bit(DEVICE_CATEGORY_STORAGE, k->categories); +k->bus_type = TYPE_FLOPPY_BUS; +k->props = floppy_drive_properties; +k->desc = "virtual floppy drive"; +} + +static const TypeInfo floppy_drive_info = { +.name = TYPE_FLOPPY_DRIVE, +.parent = TYPE_DEVICE, +.instance_size = sizeof(FloppyDrive), +.class_init = floppy_drive_class_init, +}; + // /* Intel 82078 floppy disk controller emulation */ @@ -1184,9 +1281,9 @@ static inline FDrive *drv3(FDCtrl *fdctrl) } #endif -static FDrive *get_cur_drv(FDCtrl *fdctrl) +static FDrive *get_drv(FDCtrl *fdctrl, int unit) { -switch (fdctrl->cur_drv) { +switch (unit) { case 0: return drv0(fdctrl); case 1: return drv1(fdctrl); #if MAX_FD == 4 @@ -1197,6 +1294,11 @@ static FDrive *get_cur_drv(FDCtrl *fdctrl) } } +static FDrive *get_cur_drv(FDCtrl *fdctrl) +{ +return get_drv(fdctrl, fdctrl->cur_drv); +} + /* Status A register : 0x00 (read-only) */ static uint32_t fdctrl_read_statusA(FDCtrl *fdctrl) { @@ -2356,46 +2458,21 @@ static void fdctrl_result_timer(void *opaque) } } -static void fdctrl_change_cb(void *opaque, bool load) -{ -FDrive *drive = opaque; - -drive->media_changed = 1; -drive->media_validated = false; -fd_revalidate(drive); -} - -static const BlockDevOps fdctrl_block_ops = { -.change_media_cb = fdctrl_change_cb, -}; - /* Init functions */ static void fdctrl_connect_drives(FDCtrl *fdctrl, Error **errp) { unsigned int i; -FDrive