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>
24 #include "hw/block-common.h"
27 /* --------------------------------- */
29 static char *idebus_get_fw_dev_path(DeviceState
*dev
);
31 static Property ide_props
[] = {
32 DEFINE_PROP_UINT32("unit", IDEDevice
, unit
, -1),
33 DEFINE_PROP_END_OF_LIST(),
36 static void ide_bus_class_init(ObjectClass
*klass
, void *data
)
38 BusClass
*k
= BUS_CLASS(klass
);
40 k
->get_fw_dev_path
= idebus_get_fw_dev_path
;
43 static const TypeInfo ide_bus_info
= {
46 .instance_size
= sizeof(IDEBus
),
47 .class_init
= ide_bus_class_init
,
50 void ide_bus_new(IDEBus
*idebus
, DeviceState
*dev
, int bus_id
)
52 qbus_create_inplace(&idebus
->qbus
, TYPE_IDE_BUS
, dev
, NULL
);
53 idebus
->bus_id
= bus_id
;
56 static char *idebus_get_fw_dev_path(DeviceState
*dev
)
60 snprintf(path
, sizeof(path
), "%s@%d", qdev_fw_name(dev
),
61 ((IDEBus
*)dev
->parent_bus
)->bus_id
);
63 return g_strdup(path
);
66 static int ide_qdev_init(DeviceState
*qdev
)
68 IDEDevice
*dev
= IDE_DEVICE(qdev
);
69 IDEDeviceClass
*dc
= IDE_DEVICE_GET_CLASS(dev
);
70 IDEBus
*bus
= DO_UPCAST(IDEBus
, qbus
, qdev
->parent_bus
);
73 error_report("No drive specified");
76 if (dev
->unit
== -1) {
77 dev
->unit
= bus
->master
? 1 : 0;
82 error_report("IDE unit %d is in use", dev
->unit
);
89 error_report("IDE unit %d is in use", dev
->unit
);
95 error_report("Invalid IDE unit %d", dev
->unit
);
104 IDEDevice
*ide_create_drive(IDEBus
*bus
, int unit
, DriveInfo
*drive
)
108 dev
= qdev_create(&bus
->qbus
, drive
->media_cd
? "ide-cd" : "ide-hd");
109 qdev_prop_set_uint32(dev
, "unit", unit
);
110 qdev_prop_set_drive_nofail(dev
, "drive", drive
->bdrv
);
111 qdev_init_nofail(dev
);
112 return DO_UPCAST(IDEDevice
, qdev
, dev
);
115 int ide_get_geometry(BusState
*bus
, int unit
,
116 int16_t *cyls
, int8_t *heads
, int8_t *secs
)
118 IDEState
*s
= &DO_UPCAST(IDEBus
, qbus
, bus
)->ifs
[unit
];
120 if (s
->drive_kind
!= IDE_HD
|| !s
->bs
) {
124 *cyls
= s
->cylinders
;
130 int ide_get_bios_chs_trans(BusState
*bus
, int unit
)
132 return DO_UPCAST(IDEBus
, qbus
, bus
)->ifs
[unit
].chs_trans
;
135 /* --------------------------------- */
137 typedef struct IDEDrive
{
141 static int ide_dev_initfn(IDEDevice
*dev
, IDEDriveKind kind
)
143 IDEBus
*bus
= DO_UPCAST(IDEBus
, qbus
, dev
->qdev
.parent_bus
);
144 IDEState
*s
= bus
->ifs
+ dev
->unit
;
146 if (dev
->conf
.discard_granularity
&& dev
->conf
.discard_granularity
!= 512) {
147 error_report("discard_granularity must be 512 for ide");
151 blkconf_serial(&dev
->conf
, &dev
->serial
);
153 && blkconf_geometry(&dev
->conf
, &dev
->chs_trans
, 65536, 16, 255) < 0) {
157 if (ide_init_drive(s
, dev
->conf
.bs
, kind
,
158 dev
->version
, dev
->serial
, dev
->model
, dev
->wwn
,
159 dev
->conf
.cyls
, dev
->conf
.heads
, dev
->conf
.secs
,
160 dev
->chs_trans
) < 0) {
165 dev
->version
= g_strdup(s
->version
);
168 dev
->serial
= g_strdup(s
->drive_serial_str
);
171 add_boot_device_path(dev
->conf
.bootindex
, &dev
->qdev
,
172 dev
->unit
? "/disk@1" : "/disk@0");
177 static int ide_hd_initfn(IDEDevice
*dev
)
179 return ide_dev_initfn(dev
, IDE_HD
);
182 static int ide_cd_initfn(IDEDevice
*dev
)
184 return ide_dev_initfn(dev
, IDE_CD
);
187 static int ide_drive_initfn(IDEDevice
*dev
)
189 DriveInfo
*dinfo
= drive_get_by_blockdev(dev
->conf
.bs
);
191 return ide_dev_initfn(dev
, dinfo
->media_cd
? IDE_CD
: IDE_HD
);
194 #define DEFINE_IDE_DEV_PROPERTIES() \
195 DEFINE_BLOCK_PROPERTIES(IDEDrive, dev.conf), \
196 DEFINE_PROP_STRING("ver", IDEDrive, dev.version), \
197 DEFINE_PROP_HEX64("wwn", IDEDrive, dev.wwn, 0), \
198 DEFINE_PROP_STRING("serial", IDEDrive, dev.serial),\
199 DEFINE_PROP_STRING("model", IDEDrive, dev.model)
201 static Property ide_hd_properties
[] = {
202 DEFINE_IDE_DEV_PROPERTIES(),
203 DEFINE_BLOCK_CHS_PROPERTIES(IDEDrive
, dev
.conf
),
204 DEFINE_PROP_BIOS_CHS_TRANS("bios-chs-trans",
205 IDEDrive
, dev
.chs_trans
, BIOS_ATA_TRANSLATION_AUTO
),
206 DEFINE_PROP_END_OF_LIST(),
209 static void ide_hd_class_init(ObjectClass
*klass
, void *data
)
211 DeviceClass
*dc
= DEVICE_CLASS(klass
);
212 IDEDeviceClass
*k
= IDE_DEVICE_CLASS(klass
);
213 k
->init
= ide_hd_initfn
;
214 dc
->fw_name
= "drive";
215 dc
->desc
= "virtual IDE disk";
216 dc
->props
= ide_hd_properties
;
219 static TypeInfo ide_hd_info
= {
221 .parent
= TYPE_IDE_DEVICE
,
222 .instance_size
= sizeof(IDEDrive
),
223 .class_init
= ide_hd_class_init
,
226 static Property ide_cd_properties
[] = {
227 DEFINE_IDE_DEV_PROPERTIES(),
228 DEFINE_PROP_END_OF_LIST(),
231 static void ide_cd_class_init(ObjectClass
*klass
, void *data
)
233 DeviceClass
*dc
= DEVICE_CLASS(klass
);
234 IDEDeviceClass
*k
= IDE_DEVICE_CLASS(klass
);
235 k
->init
= ide_cd_initfn
;
236 dc
->fw_name
= "drive";
237 dc
->desc
= "virtual IDE CD-ROM";
238 dc
->props
= ide_cd_properties
;
241 static TypeInfo ide_cd_info
= {
243 .parent
= TYPE_IDE_DEVICE
,
244 .instance_size
= sizeof(IDEDrive
),
245 .class_init
= ide_cd_class_init
,
248 static Property ide_drive_properties
[] = {
249 DEFINE_IDE_DEV_PROPERTIES(),
250 DEFINE_PROP_END_OF_LIST(),
253 static void ide_drive_class_init(ObjectClass
*klass
, void *data
)
255 DeviceClass
*dc
= DEVICE_CLASS(klass
);
256 IDEDeviceClass
*k
= IDE_DEVICE_CLASS(klass
);
257 k
->init
= ide_drive_initfn
;
258 dc
->fw_name
= "drive";
259 dc
->desc
= "virtual IDE disk or CD-ROM (legacy)";
260 dc
->props
= ide_drive_properties
;
263 static TypeInfo ide_drive_info
= {
265 .parent
= TYPE_IDE_DEVICE
,
266 .instance_size
= sizeof(IDEDrive
),
267 .class_init
= ide_drive_class_init
,
270 static void ide_device_class_init(ObjectClass
*klass
, void *data
)
272 DeviceClass
*k
= DEVICE_CLASS(klass
);
273 k
->init
= ide_qdev_init
;
274 k
->bus_type
= TYPE_IDE_BUS
;
275 k
->props
= ide_props
;
278 static TypeInfo ide_device_type_info
= {
279 .name
= TYPE_IDE_DEVICE
,
280 .parent
= TYPE_DEVICE
,
281 .instance_size
= sizeof(IDEDevice
),
283 .class_size
= sizeof(IDEDeviceClass
),
284 .class_init
= ide_device_class_init
,
287 static void ide_register_types(void)
289 type_register_static(&ide_bus_info
);
290 type_register_static(&ide_hd_info
);
291 type_register_static(&ide_cd_info
);
292 type_register_static(&ide_drive_info
);
293 type_register_static(&ide_device_type_info
);
296 type_init(ide_register_types
)