2 * ide bus support for qdev.
4 * Copyright (c) 2009 Gerd Hoffmann <kraxel@redhat.com>
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, see <http://www.gnu.org/licenses/>.
22 #include <hw/ide/internal.h>
24 /* --------------------------------- */
26 static struct BusInfo ide_bus_info
= {
28 .size
= sizeof(IDEBus
),
31 void ide_bus_new(IDEBus
*idebus
, DeviceState
*dev
)
33 qbus_create_inplace(&idebus
->qbus
, &ide_bus_info
, dev
, NULL
);
36 static int ide_qdev_init(DeviceState
*qdev
, DeviceInfo
*base
)
38 IDEDevice
*dev
= DO_UPCAST(IDEDevice
, qdev
, qdev
);
39 IDEDeviceInfo
*info
= DO_UPCAST(IDEDeviceInfo
, qdev
, base
);
40 IDEBus
*bus
= DO_UPCAST(IDEBus
, qbus
, qdev
->parent_bus
);
43 fprintf(stderr
, "%s: no drive specified\n", qdev
->info
->name
);
46 if (dev
->unit
== -1) {
47 dev
->unit
= bus
->master
? 1 : 0;
52 fprintf(stderr
, "ide: tried to assign master twice\n");
59 fprintf(stderr
, "ide: tried to assign slave twice\n");
67 return info
->init(dev
);
73 static void ide_qdev_register(IDEDeviceInfo
*info
)
75 info
->qdev
.init
= ide_qdev_init
;
76 info
->qdev
.bus_info
= &ide_bus_info
;
77 qdev_register(&info
->qdev
);
80 IDEDevice
*ide_create_drive(IDEBus
*bus
, int unit
, DriveInfo
*drive
)
84 dev
= qdev_create(&bus
->qbus
, "ide-drive");
85 qdev_prop_set_uint32(dev
, "unit", unit
);
86 qdev_prop_set_drive_nofail(dev
, "drive", drive
->bdrv
);
87 qdev_init_nofail(dev
);
88 return DO_UPCAST(IDEDevice
, qdev
, dev
);
91 void ide_get_bs(BlockDriverState
*bs
[], BusState
*qbus
)
93 IDEBus
*bus
= DO_UPCAST(IDEBus
, qbus
, qbus
);
94 bs
[0] = bus
->master
? bus
->master
->conf
.bs
: NULL
;
95 bs
[1] = bus
->slave
? bus
->slave
->conf
.bs
: NULL
;
98 /* --------------------------------- */
100 typedef struct IDEDrive
{
104 static int ide_drive_initfn(IDEDevice
*dev
)
106 IDEBus
*bus
= DO_UPCAST(IDEBus
, qbus
, dev
->qdev
.parent_bus
);
107 IDEState
*s
= bus
->ifs
+ dev
->unit
;
111 serial
= dev
->serial
;
113 /* try to fall back to value set with legacy -drive serial=... */
114 dinfo
= drive_get_by_blockdev(dev
->conf
.bs
);
115 if (*dinfo
->serial
) {
116 serial
= dinfo
->serial
;
120 ide_init_drive(s
, dev
->conf
.bs
, dev
->version
, serial
);
123 dev
->version
= qemu_strdup(s
->version
);
126 dev
->serial
= qemu_strdup(s
->drive_serial_str
);
131 static IDEDeviceInfo ide_drive_info
= {
132 .qdev
.name
= "ide-drive",
133 .qdev
.size
= sizeof(IDEDrive
),
134 .init
= ide_drive_initfn
,
135 .qdev
.props
= (Property
[]) {
136 DEFINE_PROP_UINT32("unit", IDEDrive
, dev
.unit
, -1),
137 DEFINE_BLOCK_PROPERTIES(IDEDrive
, dev
.conf
),
138 DEFINE_PROP_STRING("ver", IDEDrive
, dev
.version
),
139 DEFINE_PROP_STRING("serial", IDEDrive
, dev
.serial
),
140 DEFINE_PROP_END_OF_LIST(),
144 static void ide_drive_register(void)
146 ide_qdev_register(&ide_drive_info
);
148 device_init(ide_drive_register
);