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/>.
21 #include "qemu-error.h"
22 #include <hw/ide/internal.h>
26 /* --------------------------------- */
28 static char *idebus_get_fw_dev_path(DeviceState
*dev
);
30 static struct BusInfo ide_bus_info
= {
32 .size
= sizeof(IDEBus
),
33 .get_fw_dev_path
= idebus_get_fw_dev_path
,
34 .props
= (Property
[]) {
35 DEFINE_PROP_UINT32("unit", IDEDevice
, unit
, -1),
36 DEFINE_PROP_END_OF_LIST(),
40 void ide_bus_new(IDEBus
*idebus
, DeviceState
*dev
, int bus_id
)
42 qbus_create_inplace(&idebus
->qbus
, &ide_bus_info
, dev
, NULL
);
43 idebus
->bus_id
= bus_id
;
46 static char *idebus_get_fw_dev_path(DeviceState
*dev
)
50 snprintf(path
, sizeof(path
), "%s@%d", qdev_fw_name(dev
),
51 ((IDEBus
*)dev
->parent_bus
)->bus_id
);
56 static int ide_qdev_init(DeviceState
*qdev
, DeviceInfo
*base
)
58 IDEDevice
*dev
= IDE_DEVICE(qdev
);
59 IDEDeviceClass
*dc
= IDE_DEVICE_GET_CLASS(dev
);
60 IDEBus
*bus
= DO_UPCAST(IDEBus
, qbus
, qdev
->parent_bus
);
63 error_report("No drive specified");
66 if (dev
->unit
== -1) {
67 dev
->unit
= bus
->master
? 1 : 0;
72 error_report("IDE unit %d is in use", dev
->unit
);
79 error_report("IDE unit %d is in use", dev
->unit
);
85 error_report("Invalid IDE unit %d", dev
->unit
);
94 static void ide_qdev_register(DeviceInfo
*info
)
96 info
->init
= ide_qdev_init
;
97 info
->bus_info
= &ide_bus_info
;
98 qdev_register_subclass(info
, TYPE_IDE_DEVICE
);
101 IDEDevice
*ide_create_drive(IDEBus
*bus
, int unit
, DriveInfo
*drive
)
105 dev
= qdev_create(&bus
->qbus
, drive
->media_cd
? "ide-cd" : "ide-hd");
106 qdev_prop_set_uint32(dev
, "unit", unit
);
107 qdev_prop_set_drive_nofail(dev
, "drive", drive
->bdrv
);
108 qdev_init_nofail(dev
);
109 return DO_UPCAST(IDEDevice
, qdev
, dev
);
112 void ide_get_bs(BlockDriverState
*bs
[], BusState
*qbus
)
114 IDEBus
*bus
= DO_UPCAST(IDEBus
, qbus
, qbus
);
115 bs
[0] = bus
->master
? bus
->master
->conf
.bs
: NULL
;
116 bs
[1] = bus
->slave
? bus
->slave
->conf
.bs
: NULL
;
119 /* --------------------------------- */
121 typedef struct IDEDrive
{
125 static int ide_dev_initfn(IDEDevice
*dev
, IDEDriveKind kind
)
127 IDEBus
*bus
= DO_UPCAST(IDEBus
, qbus
, dev
->qdev
.parent_bus
);
128 IDEState
*s
= bus
->ifs
+ dev
->unit
;
132 if (dev
->conf
.discard_granularity
&& dev
->conf
.discard_granularity
!= 512) {
133 error_report("discard_granularity must be 512 for ide");
137 serial
= dev
->serial
;
139 /* try to fall back to value set with legacy -drive serial=... */
140 dinfo
= drive_get_by_blockdev(dev
->conf
.bs
);
141 if (*dinfo
->serial
) {
142 serial
= dinfo
->serial
;
146 if (ide_init_drive(s
, dev
->conf
.bs
, kind
, dev
->version
, serial
) < 0) {
151 dev
->version
= g_strdup(s
->version
);
154 dev
->serial
= g_strdup(s
->drive_serial_str
);
157 add_boot_device_path(dev
->conf
.bootindex
, &dev
->qdev
,
158 dev
->unit
? "/disk@1" : "/disk@0");
163 static int ide_hd_initfn(IDEDevice
*dev
)
165 return ide_dev_initfn(dev
, IDE_HD
);
168 static int ide_cd_initfn(IDEDevice
*dev
)
170 return ide_dev_initfn(dev
, IDE_CD
);
173 static int ide_drive_initfn(IDEDevice
*dev
)
175 DriveInfo
*dinfo
= drive_get_by_blockdev(dev
->conf
.bs
);
177 return ide_dev_initfn(dev
, dinfo
->media_cd
? IDE_CD
: IDE_HD
);
180 #define DEFINE_IDE_DEV_PROPERTIES() \
181 DEFINE_BLOCK_PROPERTIES(IDEDrive, dev.conf), \
182 DEFINE_PROP_STRING("ver", IDEDrive, dev.version), \
183 DEFINE_PROP_STRING("serial", IDEDrive, dev.serial)
185 static void ide_hd_class_init(ObjectClass
*klass
, void *data
)
187 IDEDeviceClass
*k
= IDE_DEVICE_CLASS(klass
);
188 k
->init
= ide_hd_initfn
;
191 static DeviceInfo ide_hd_info
= {
194 .desc
= "virtual IDE disk",
195 .size
= sizeof(IDEDrive
),
196 .class_init
= ide_hd_class_init
,
197 .props
= (Property
[]) {
198 DEFINE_IDE_DEV_PROPERTIES(),
199 DEFINE_PROP_END_OF_LIST(),
203 static void ide_cd_class_init(ObjectClass
*klass
, void *data
)
205 IDEDeviceClass
*k
= IDE_DEVICE_CLASS(klass
);
206 k
->init
= ide_cd_initfn
;
209 static DeviceInfo ide_cd_info
= {
212 .desc
= "virtual IDE CD-ROM",
213 .size
= sizeof(IDEDrive
),
214 .class_init
= ide_cd_class_init
,
215 .props
= (Property
[]) {
216 DEFINE_IDE_DEV_PROPERTIES(),
217 DEFINE_PROP_END_OF_LIST(),
221 static void ide_drive_class_init(ObjectClass
*klass
, void *data
)
223 IDEDeviceClass
*k
= IDE_DEVICE_CLASS(klass
);
224 k
->init
= ide_drive_initfn
;
227 static DeviceInfo ide_drive_info
= {
228 .name
= "ide-drive", /* legacy -device ide-drive */
230 .desc
= "virtual IDE disk or CD-ROM (legacy)",
231 .size
= sizeof(IDEDrive
),
232 .class_init
= ide_drive_class_init
,
233 .props
= (Property
[]) {
234 DEFINE_IDE_DEV_PROPERTIES(),
235 DEFINE_PROP_END_OF_LIST(),
239 static TypeInfo ide_device_type_info
= {
240 .name
= TYPE_IDE_DEVICE
,
241 .parent
= TYPE_DEVICE
,
242 .instance_size
= sizeof(IDEDevice
),
244 .class_size
= sizeof(IDEDeviceClass
),
247 static void ide_dev_register(void)
249 ide_qdev_register(&ide_hd_info
);
250 ide_qdev_register(&ide_cd_info
);
251 ide_qdev_register(&ide_drive_info
);
252 type_register_static(&ide_device_type_info
);
254 device_init(ide_dev_register
);