mirror of https://gitee.com/openkylin/qemu.git
qdev: refactor device creation to allow bus_info to be set only in class
As we use class_init to set class members, DeviceInfo no longer holds this information. Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
This commit is contained in:
parent
d253e09619
commit
18b6dade8c
42
hw/qdev.c
42
hw/qdev.c
|
@ -119,21 +119,29 @@ const char *qdev_fw_name(DeviceState *dev)
|
||||||
return object_get_typename(OBJECT(dev));
|
return object_get_typename(OBJECT(dev));
|
||||||
}
|
}
|
||||||
|
|
||||||
void qdev_register_subclass(DeviceInfo *info, const char *parent)
|
static void qdev_do_register_subclass(DeviceInfo *info, const char *parent,
|
||||||
|
const char *name)
|
||||||
{
|
{
|
||||||
TypeInfo type_info = {};
|
TypeInfo type_info = {};
|
||||||
|
|
||||||
assert(info->size >= sizeof(DeviceState));
|
assert(info->size >= sizeof(DeviceState));
|
||||||
assert(!info->next);
|
assert(!info->next);
|
||||||
|
|
||||||
type_info.name = info->name;
|
type_info.name = name;
|
||||||
type_info.parent = parent;
|
type_info.parent = parent;
|
||||||
type_info.instance_size = info->size;
|
type_info.instance_size = info->size;
|
||||||
type_info.class_init = qdev_subclass_init;
|
type_info.class_init = qdev_subclass_init;
|
||||||
type_info.class_data = info;
|
type_info.class_data = info;
|
||||||
|
|
||||||
type_register_static(&type_info);
|
type_register_static(&type_info);
|
||||||
|
}
|
||||||
|
|
||||||
|
void qdev_register_subclass(DeviceInfo *info, const char *parent)
|
||||||
|
{
|
||||||
|
qdev_do_register_subclass(info, parent, info->name);
|
||||||
|
if (info->alias) {
|
||||||
|
qdev_do_register_subclass(info, parent, info->alias);
|
||||||
|
}
|
||||||
info->next = device_info_list;
|
info->next = device_info_list;
|
||||||
device_info_list = info;
|
device_info_list = info;
|
||||||
}
|
}
|
||||||
|
@ -173,12 +181,12 @@ bool qdev_exists(const char *name)
|
||||||
static void qdev_property_add_legacy(DeviceState *dev, Property *prop,
|
static void qdev_property_add_legacy(DeviceState *dev, Property *prop,
|
||||||
Error **errp);
|
Error **errp);
|
||||||
|
|
||||||
static DeviceState *qdev_create_from_info(BusState *bus, DeviceInfo *info)
|
static DeviceState *qdev_create_from_info(BusState *bus, const char *typename)
|
||||||
{
|
{
|
||||||
DeviceState *dev;
|
DeviceState *dev;
|
||||||
Property *prop;
|
Property *prop;
|
||||||
|
|
||||||
dev = DEVICE(object_new(info->name));
|
dev = DEVICE(object_new(typename));
|
||||||
dev->parent_bus = bus;
|
dev->parent_bus = bus;
|
||||||
qdev_prop_set_defaults(dev, qdev_get_props(dev));
|
qdev_prop_set_defaults(dev, qdev_get_props(dev));
|
||||||
qdev_prop_set_defaults(dev, dev->parent_bus->info->props);
|
qdev_prop_set_defaults(dev, dev->parent_bus->info->props);
|
||||||
|
@ -230,18 +238,11 @@ DeviceState *qdev_create(BusState *bus, const char *name)
|
||||||
|
|
||||||
DeviceState *qdev_try_create(BusState *bus, const char *name)
|
DeviceState *qdev_try_create(BusState *bus, const char *name)
|
||||||
{
|
{
|
||||||
DeviceInfo *info;
|
|
||||||
|
|
||||||
if (!bus) {
|
if (!bus) {
|
||||||
bus = sysbus_get_default();
|
bus = sysbus_get_default();
|
||||||
}
|
}
|
||||||
|
|
||||||
info = qdev_find_info(bus->info, name);
|
return qdev_create_from_info(bus, name);
|
||||||
if (!info) {
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
return qdev_create_from_info(bus, info);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void qdev_print_devinfo(DeviceInfo *info)
|
static void qdev_print_devinfo(DeviceInfo *info)
|
||||||
|
@ -352,8 +353,8 @@ static DeviceState *qdev_get_peripheral_anon(void)
|
||||||
|
|
||||||
DeviceState *qdev_device_add(QemuOpts *opts)
|
DeviceState *qdev_device_add(QemuOpts *opts)
|
||||||
{
|
{
|
||||||
|
DeviceClass *k;
|
||||||
const char *driver, *path, *id;
|
const char *driver, *path, *id;
|
||||||
DeviceInfo *info;
|
|
||||||
DeviceState *qdev;
|
DeviceState *qdev;
|
||||||
BusState *bus;
|
BusState *bus;
|
||||||
|
|
||||||
|
@ -364,12 +365,7 @@ DeviceState *qdev_device_add(QemuOpts *opts)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* find driver */
|
/* find driver */
|
||||||
info = qdev_find_info(NULL, driver);
|
k = DEVICE_CLASS(object_class_by_name(driver));
|
||||||
if (!info || info->no_user) {
|
|
||||||
qerror_report(QERR_INVALID_PARAMETER_VALUE, "driver", "a driver name");
|
|
||||||
error_printf_unless_qmp("Try with argument '?' for a list.\n");
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* find bus */
|
/* find bus */
|
||||||
path = qemu_opt_get(opts, "bus");
|
path = qemu_opt_get(opts, "bus");
|
||||||
|
@ -378,16 +374,16 @@ DeviceState *qdev_device_add(QemuOpts *opts)
|
||||||
if (!bus) {
|
if (!bus) {
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
if (bus->info != info->bus_info) {
|
if (bus->info != k->bus_info) {
|
||||||
qerror_report(QERR_BAD_BUS_FOR_DEVICE,
|
qerror_report(QERR_BAD_BUS_FOR_DEVICE,
|
||||||
driver, bus->info->name);
|
driver, bus->info->name);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
bus = qbus_find_recursive(main_system_bus, NULL, info->bus_info);
|
bus = qbus_find_recursive(main_system_bus, NULL, k->bus_info);
|
||||||
if (!bus) {
|
if (!bus) {
|
||||||
qerror_report(QERR_NO_BUS_FOR_DEVICE,
|
qerror_report(QERR_NO_BUS_FOR_DEVICE,
|
||||||
info->name, info->bus_info->name);
|
driver, k->bus_info->name);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -397,7 +393,7 @@ DeviceState *qdev_device_add(QemuOpts *opts)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* create device, set properties */
|
/* create device, set properties */
|
||||||
qdev = qdev_create_from_info(bus, info);
|
qdev = qdev_create_from_info(bus, driver);
|
||||||
id = qemu_opts_id(opts);
|
id = qemu_opts_id(opts);
|
||||||
if (id) {
|
if (id) {
|
||||||
qdev->id = id;
|
qdev->id = id;
|
||||||
|
|
Loading…
Reference in New Issue