mirror of https://github.com/xqemu/xqemu.git
platform-bus-device: use device plug callback instead of machine_done notifier
platform-bus were using machine_done notifier to get and map (assign irq/mmio resources) dynamically added sysbus devices after all '-device' options had been processed. That however creates non obvious dependencies on ordering of machine_done notifiers and requires carefull line juggling to keep it working. For example see comment above create_platform_bus() and 'straitforward' arm_load_kernel() had to converted to machine_done notifier and that lead to yet another machine_done notifier to keep it working arm_register_platform_bus_fdt_creator(). Instead of hiding resource assignment in platform-bus-device to magically initialize sysbus devices, use device plug callback and assign resources explicitly at board level at the moment each -device option is being processed. That adds a bunch of machine declaration boiler plate to e500plat board, similar to ARM/x86 but gets rid of hidden machine_done notifier and would allow to remove the dependent notifiers in ARM code simplifying it and making code flow easier to follow. Signed-off-by: Igor Mammedov <imammedo@redhat.com> Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org> Acked-by: David Gibson <david@gibson.dropbear.id.au> Message-id: 1525691524-32265-3-git-send-email-imammedo@redhat.com Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
parent
38aefb578d
commit
a3fc839635
|
@ -506,9 +506,6 @@ static void add_all_platform_bus_fdt_nodes(ARMPlatformBusFDTParams *fdt_params)
|
||||||
dev = qdev_find_recursive(sysbus_get_default(), TYPE_PLATFORM_BUS_DEVICE);
|
dev = qdev_find_recursive(sysbus_get_default(), TYPE_PLATFORM_BUS_DEVICE);
|
||||||
pbus = PLATFORM_BUS_DEVICE(dev);
|
pbus = PLATFORM_BUS_DEVICE(dev);
|
||||||
|
|
||||||
/* We can only create dt nodes for dynamic devices when they're ready */
|
|
||||||
assert(pbus->done_gathering);
|
|
||||||
|
|
||||||
PlatformBusFDTData data = {
|
PlatformBusFDTData data = {
|
||||||
.fdt = fdt,
|
.fdt = fdt,
|
||||||
.irq_start = irq_start,
|
.irq_start = irq_start,
|
||||||
|
|
|
@ -1150,6 +1150,7 @@ static void create_platform_bus(VirtMachineState *vms, qemu_irq *pic)
|
||||||
qdev_prop_set_uint32(dev, "mmio_size",
|
qdev_prop_set_uint32(dev, "mmio_size",
|
||||||
platform_bus_params.platform_bus_size);
|
platform_bus_params.platform_bus_size);
|
||||||
qdev_init_nofail(dev);
|
qdev_init_nofail(dev);
|
||||||
|
vms->platform_bus_dev = dev;
|
||||||
s = SYS_BUS_DEVICE(dev);
|
s = SYS_BUS_DEVICE(dev);
|
||||||
|
|
||||||
for (i = 0; i < platform_bus_params.platform_bus_num_irqs; i++) {
|
for (i = 0; i < platform_bus_params.platform_bus_num_irqs; i++) {
|
||||||
|
@ -1627,9 +1628,33 @@ static const CPUArchIdList *virt_possible_cpu_arch_ids(MachineState *ms)
|
||||||
return ms->possible_cpus;
|
return ms->possible_cpus;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void virt_machine_device_plug_cb(HotplugHandler *hotplug_dev,
|
||||||
|
DeviceState *dev, Error **errp)
|
||||||
|
{
|
||||||
|
VirtMachineState *vms = VIRT_MACHINE(hotplug_dev);
|
||||||
|
|
||||||
|
if (vms->platform_bus_dev) {
|
||||||
|
if (object_dynamic_cast(OBJECT(dev), TYPE_SYS_BUS_DEVICE)) {
|
||||||
|
platform_bus_link_device(PLATFORM_BUS_DEVICE(vms->platform_bus_dev),
|
||||||
|
SYS_BUS_DEVICE(dev));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static HotplugHandler *virt_machine_get_hotplug_handler(MachineState *machine,
|
||||||
|
DeviceState *dev)
|
||||||
|
{
|
||||||
|
if (object_dynamic_cast(OBJECT(dev), TYPE_SYS_BUS_DEVICE)) {
|
||||||
|
return HOTPLUG_HANDLER(machine);
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
static void virt_machine_class_init(ObjectClass *oc, void *data)
|
static void virt_machine_class_init(ObjectClass *oc, void *data)
|
||||||
{
|
{
|
||||||
MachineClass *mc = MACHINE_CLASS(oc);
|
MachineClass *mc = MACHINE_CLASS(oc);
|
||||||
|
HotplugHandlerClass *hc = HOTPLUG_HANDLER_CLASS(oc);
|
||||||
|
|
||||||
mc->init = machvirt_init;
|
mc->init = machvirt_init;
|
||||||
/* Start max_cpus at the maximum QEMU supports. We'll further restrict
|
/* Start max_cpus at the maximum QEMU supports. We'll further restrict
|
||||||
|
@ -1648,6 +1673,8 @@ static void virt_machine_class_init(ObjectClass *oc, void *data)
|
||||||
mc->cpu_index_to_instance_props = virt_cpu_index_to_props;
|
mc->cpu_index_to_instance_props = virt_cpu_index_to_props;
|
||||||
mc->default_cpu_type = ARM_CPU_TYPE_NAME("cortex-a15");
|
mc->default_cpu_type = ARM_CPU_TYPE_NAME("cortex-a15");
|
||||||
mc->get_default_cpu_node_id = virt_get_default_cpu_node_id;
|
mc->get_default_cpu_node_id = virt_get_default_cpu_node_id;
|
||||||
|
mc->get_hotplug_handler = virt_machine_get_hotplug_handler;
|
||||||
|
hc->plug = virt_machine_device_plug_cb;
|
||||||
}
|
}
|
||||||
|
|
||||||
static const TypeInfo virt_machine_info = {
|
static const TypeInfo virt_machine_info = {
|
||||||
|
@ -1657,6 +1684,10 @@ static const TypeInfo virt_machine_info = {
|
||||||
.instance_size = sizeof(VirtMachineState),
|
.instance_size = sizeof(VirtMachineState),
|
||||||
.class_size = sizeof(VirtMachineClass),
|
.class_size = sizeof(VirtMachineClass),
|
||||||
.class_init = virt_machine_class_init,
|
.class_init = virt_machine_class_init,
|
||||||
|
.interfaces = (InterfaceInfo[]) {
|
||||||
|
{ TYPE_HOTPLUG_HANDLER },
|
||||||
|
{ }
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
static void machvirt_machine_init(void)
|
static void machvirt_machine_init(void)
|
||||||
|
|
|
@ -103,7 +103,6 @@ static void plaform_bus_refresh_irqs(PlatformBusDevice *pbus)
|
||||||
{
|
{
|
||||||
bitmap_zero(pbus->used_irqs, pbus->num_irqs);
|
bitmap_zero(pbus->used_irqs, pbus->num_irqs);
|
||||||
foreach_dynamic_sysbus_device(platform_bus_count_irqs, pbus);
|
foreach_dynamic_sysbus_device(platform_bus_count_irqs, pbus);
|
||||||
pbus->done_gathering = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void platform_bus_map_irq(PlatformBusDevice *pbus, SysBusDevice *sbdev,
|
static void platform_bus_map_irq(PlatformBusDevice *pbus, SysBusDevice *sbdev,
|
||||||
|
@ -163,12 +162,11 @@ static void platform_bus_map_mmio(PlatformBusDevice *pbus, SysBusDevice *sbdev,
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* For each sysbus device, look for unassigned IRQ lines as well as
|
* Look for unassigned IRQ lines as well as unassociated MMIO regions.
|
||||||
* unassociated MMIO regions. Connect them to the platform bus if available.
|
* Connect them to the platform bus if available.
|
||||||
*/
|
*/
|
||||||
static void link_sysbus_device(SysBusDevice *sbdev, void *opaque)
|
void platform_bus_link_device(PlatformBusDevice *pbus, SysBusDevice *sbdev)
|
||||||
{
|
{
|
||||||
PlatformBusDevice *pbus = opaque;
|
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
for (i = 0; sysbus_has_irq(sbdev, i); i++) {
|
for (i = 0; sysbus_has_irq(sbdev, i); i++) {
|
||||||
|
@ -180,19 +178,6 @@ static void link_sysbus_device(SysBusDevice *sbdev, void *opaque)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void platform_bus_init_notify(Notifier *notifier, void *data)
|
|
||||||
{
|
|
||||||
PlatformBusDevice *pb = container_of(notifier, PlatformBusDevice, notifier);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Generate a bitmap of used IRQ lines, as the user might have specified
|
|
||||||
* them on the command line.
|
|
||||||
*/
|
|
||||||
plaform_bus_refresh_irqs(pb);
|
|
||||||
|
|
||||||
foreach_dynamic_sysbus_device(link_sysbus_device, pb);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void platform_bus_realize(DeviceState *dev, Error **errp)
|
static void platform_bus_realize(DeviceState *dev, Error **errp)
|
||||||
{
|
{
|
||||||
PlatformBusDevice *pbus;
|
PlatformBusDevice *pbus;
|
||||||
|
@ -211,12 +196,8 @@ static void platform_bus_realize(DeviceState *dev, Error **errp)
|
||||||
sysbus_init_irq(d, &pbus->irqs[i]);
|
sysbus_init_irq(d, &pbus->irqs[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/* some devices might be initialized before so update used IRQs map */
|
||||||
* Register notifier that allows us to gather dangling devices once the
|
plaform_bus_refresh_irqs(pbus);
|
||||||
* machine is completely assembled
|
|
||||||
*/
|
|
||||||
pbus->notifier.notify = platform_bus_init_notify;
|
|
||||||
qemu_add_machine_init_done_notifier(&pbus->notifier);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static Property platform_bus_properties[] = {
|
static Property platform_bus_properties[] = {
|
||||||
|
|
|
@ -222,16 +222,15 @@ static void sysbus_device_create_devtree(SysBusDevice *sbdev, void *opaque)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void platform_bus_create_devtree(const PPCE500MachineClass *pmc,
|
static void platform_bus_create_devtree(PPCE500MachineState *pms,
|
||||||
void *fdt, const char *mpic)
|
void *fdt, const char *mpic)
|
||||||
{
|
{
|
||||||
|
const PPCE500MachineClass *pmc = PPCE500_MACHINE_GET_CLASS(pms);
|
||||||
gchar *node = g_strdup_printf("/platform@%"PRIx64, pmc->platform_bus_base);
|
gchar *node = g_strdup_printf("/platform@%"PRIx64, pmc->platform_bus_base);
|
||||||
const char platcomp[] = "qemu,platform\0simple-bus";
|
const char platcomp[] = "qemu,platform\0simple-bus";
|
||||||
uint64_t addr = pmc->platform_bus_base;
|
uint64_t addr = pmc->platform_bus_base;
|
||||||
uint64_t size = pmc->platform_bus_size;
|
uint64_t size = pmc->platform_bus_size;
|
||||||
int irq_start = pmc->platform_bus_first_irq;
|
int irq_start = pmc->platform_bus_first_irq;
|
||||||
PlatformBusDevice *pbus;
|
|
||||||
DeviceState *dev;
|
|
||||||
|
|
||||||
/* Create a /platform node that we can put all devices into */
|
/* Create a /platform node that we can put all devices into */
|
||||||
|
|
||||||
|
@ -246,22 +245,17 @@ static void platform_bus_create_devtree(const PPCE500MachineClass *pmc,
|
||||||
|
|
||||||
qemu_fdt_setprop_phandle(fdt, node, "interrupt-parent", mpic);
|
qemu_fdt_setprop_phandle(fdt, node, "interrupt-parent", mpic);
|
||||||
|
|
||||||
dev = qdev_find_recursive(sysbus_get_default(), TYPE_PLATFORM_BUS_DEVICE);
|
/* Create dt nodes for dynamic devices */
|
||||||
pbus = PLATFORM_BUS_DEVICE(dev);
|
PlatformDevtreeData data = {
|
||||||
|
.fdt = fdt,
|
||||||
|
.mpic = mpic,
|
||||||
|
.irq_start = irq_start,
|
||||||
|
.node = node,
|
||||||
|
.pbus = pms->pbus_dev,
|
||||||
|
};
|
||||||
|
|
||||||
/* We can only create dt nodes for dynamic devices when they're ready */
|
/* Loop through all dynamic sysbus devices and create nodes for them */
|
||||||
if (pbus->done_gathering) {
|
foreach_dynamic_sysbus_device(sysbus_device_create_devtree, &data);
|
||||||
PlatformDevtreeData data = {
|
|
||||||
.fdt = fdt,
|
|
||||||
.mpic = mpic,
|
|
||||||
.irq_start = irq_start,
|
|
||||||
.node = node,
|
|
||||||
.pbus = pbus,
|
|
||||||
};
|
|
||||||
|
|
||||||
/* Loop through all dynamic sysbus devices and create nodes for them */
|
|
||||||
foreach_dynamic_sysbus_device(sysbus_device_create_devtree, &data);
|
|
||||||
}
|
|
||||||
|
|
||||||
g_free(node);
|
g_free(node);
|
||||||
}
|
}
|
||||||
|
@ -533,8 +527,8 @@ static int ppce500_load_device_tree(PPCE500MachineState *pms,
|
||||||
}
|
}
|
||||||
g_free(soc);
|
g_free(soc);
|
||||||
|
|
||||||
if (pmc->has_platform_bus) {
|
if (pms->pbus_dev) {
|
||||||
platform_bus_create_devtree(pmc, fdt, mpic);
|
platform_bus_create_devtree(pms, fdt, mpic);
|
||||||
}
|
}
|
||||||
g_free(mpic);
|
g_free(mpic);
|
||||||
|
|
||||||
|
@ -953,8 +947,9 @@ void ppce500_init(MachineState *machine)
|
||||||
qdev_prop_set_uint32(dev, "num_irqs", pmc->platform_bus_num_irqs);
|
qdev_prop_set_uint32(dev, "num_irqs", pmc->platform_bus_num_irqs);
|
||||||
qdev_prop_set_uint32(dev, "mmio_size", pmc->platform_bus_size);
|
qdev_prop_set_uint32(dev, "mmio_size", pmc->platform_bus_size);
|
||||||
qdev_init_nofail(dev);
|
qdev_init_nofail(dev);
|
||||||
s = SYS_BUS_DEVICE(dev);
|
pms->pbus_dev = PLATFORM_BUS_DEVICE(dev);
|
||||||
|
|
||||||
|
s = SYS_BUS_DEVICE(pms->pbus_dev);
|
||||||
for (i = 0; i < pmc->platform_bus_num_irqs; i++) {
|
for (i = 0; i < pmc->platform_bus_num_irqs; i++) {
|
||||||
int irqn = pmc->platform_bus_first_irq + i;
|
int irqn = pmc->platform_bus_first_irq + i;
|
||||||
sysbus_connect_irq(s, i, qdev_get_gpio_in(mpicdev, irqn));
|
sysbus_connect_irq(s, i, qdev_get_gpio_in(mpicdev, irqn));
|
||||||
|
@ -1097,6 +1092,7 @@ static const TypeInfo ppce500_info = {
|
||||||
.name = TYPE_PPCE500_MACHINE,
|
.name = TYPE_PPCE500_MACHINE,
|
||||||
.parent = TYPE_MACHINE,
|
.parent = TYPE_MACHINE,
|
||||||
.abstract = true,
|
.abstract = true,
|
||||||
|
.instance_size = sizeof(PPCE500MachineState),
|
||||||
.class_size = sizeof(PPCE500MachineClass),
|
.class_size = sizeof(PPCE500MachineClass),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -2,11 +2,16 @@
|
||||||
#define PPCE500_H
|
#define PPCE500_H
|
||||||
|
|
||||||
#include "hw/boards.h"
|
#include "hw/boards.h"
|
||||||
|
#include "hw/platform-bus.h"
|
||||||
|
|
||||||
typedef struct PPCE500MachineState {
|
typedef struct PPCE500MachineState {
|
||||||
/*< private >*/
|
/*< private >*/
|
||||||
MachineState parent_obj;
|
MachineState parent_obj;
|
||||||
|
|
||||||
|
/* points to instance of TYPE_PLATFORM_BUS_DEVICE if
|
||||||
|
* board supports dynamic sysbus devices
|
||||||
|
*/
|
||||||
|
PlatformBusDevice *pbus_dev;
|
||||||
} PPCE500MachineState;
|
} PPCE500MachineState;
|
||||||
|
|
||||||
typedef struct PPCE500MachineClass {
|
typedef struct PPCE500MachineClass {
|
||||||
|
|
|
@ -43,13 +43,40 @@ static void e500plat_init(MachineState *machine)
|
||||||
ppce500_init(machine);
|
ppce500_init(machine);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void e500plat_machine_device_plug_cb(HotplugHandler *hotplug_dev,
|
||||||
|
DeviceState *dev, Error **errp)
|
||||||
|
{
|
||||||
|
PPCE500MachineState *pms = PPCE500_MACHINE(hotplug_dev);
|
||||||
|
|
||||||
|
if (pms->pbus_dev) {
|
||||||
|
if (object_dynamic_cast(OBJECT(dev), TYPE_SYS_BUS_DEVICE)) {
|
||||||
|
platform_bus_link_device(pms->pbus_dev, SYS_BUS_DEVICE(dev));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
HotplugHandler *e500plat_machine_get_hotpug_handler(MachineState *machine,
|
||||||
|
DeviceState *dev)
|
||||||
|
{
|
||||||
|
if (object_dynamic_cast(OBJECT(dev), TYPE_SYS_BUS_DEVICE)) {
|
||||||
|
return HOTPLUG_HANDLER(machine);
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
#define TYPE_E500PLAT_MACHINE MACHINE_TYPE_NAME("ppce500")
|
#define TYPE_E500PLAT_MACHINE MACHINE_TYPE_NAME("ppce500")
|
||||||
|
|
||||||
static void e500plat_machine_class_init(ObjectClass *oc, void *data)
|
static void e500plat_machine_class_init(ObjectClass *oc, void *data)
|
||||||
{
|
{
|
||||||
PPCE500MachineClass *pmc = PPCE500_MACHINE_CLASS(oc);
|
PPCE500MachineClass *pmc = PPCE500_MACHINE_CLASS(oc);
|
||||||
|
HotplugHandlerClass *hc = HOTPLUG_HANDLER_CLASS(oc);
|
||||||
MachineClass *mc = MACHINE_CLASS(oc);
|
MachineClass *mc = MACHINE_CLASS(oc);
|
||||||
|
|
||||||
|
mc->get_hotplug_handler = e500plat_machine_get_hotpug_handler;
|
||||||
|
hc->plug = e500plat_machine_device_plug_cb;
|
||||||
|
|
||||||
pmc->pci_first_slot = 0x1;
|
pmc->pci_first_slot = 0x1;
|
||||||
pmc->pci_nr_slots = PCI_SLOT_MAX - 1;
|
pmc->pci_nr_slots = PCI_SLOT_MAX - 1;
|
||||||
pmc->fixup_devtree = e500plat_fixup_devtree;
|
pmc->fixup_devtree = e500plat_fixup_devtree;
|
||||||
|
@ -77,6 +104,10 @@ static const TypeInfo e500plat_info = {
|
||||||
.name = TYPE_E500PLAT_MACHINE,
|
.name = TYPE_E500PLAT_MACHINE,
|
||||||
.parent = TYPE_PPCE500_MACHINE,
|
.parent = TYPE_PPCE500_MACHINE,
|
||||||
.class_init = e500plat_machine_class_init,
|
.class_init = e500plat_machine_class_init,
|
||||||
|
.interfaces = (InterfaceInfo[]) {
|
||||||
|
{ TYPE_HOTPLUG_HANDLER },
|
||||||
|
{ }
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
static void e500plat_register_types(void)
|
static void e500plat_register_types(void)
|
||||||
|
|
|
@ -99,6 +99,7 @@ typedef struct {
|
||||||
typedef struct {
|
typedef struct {
|
||||||
MachineState parent;
|
MachineState parent;
|
||||||
Notifier machine_done;
|
Notifier machine_done;
|
||||||
|
DeviceState *platform_bus_dev;
|
||||||
FWCfgState *fw_cfg;
|
FWCfgState *fw_cfg;
|
||||||
bool secure;
|
bool secure;
|
||||||
bool highmem;
|
bool highmem;
|
||||||
|
|
|
@ -37,8 +37,6 @@ typedef struct PlatformBusDevice PlatformBusDevice;
|
||||||
struct PlatformBusDevice {
|
struct PlatformBusDevice {
|
||||||
/*< private >*/
|
/*< private >*/
|
||||||
SysBusDevice parent_obj;
|
SysBusDevice parent_obj;
|
||||||
Notifier notifier;
|
|
||||||
bool done_gathering;
|
|
||||||
|
|
||||||
/*< public >*/
|
/*< public >*/
|
||||||
uint32_t mmio_size;
|
uint32_t mmio_size;
|
||||||
|
@ -54,4 +52,6 @@ int platform_bus_get_irqn(PlatformBusDevice *platform_bus, SysBusDevice *sbdev,
|
||||||
hwaddr platform_bus_get_mmio_addr(PlatformBusDevice *pbus, SysBusDevice *sbdev,
|
hwaddr platform_bus_get_mmio_addr(PlatformBusDevice *pbus, SysBusDevice *sbdev,
|
||||||
int n);
|
int n);
|
||||||
|
|
||||||
|
void platform_bus_link_device(PlatformBusDevice *pbus, SysBusDevice *sbdev);
|
||||||
|
|
||||||
#endif /* HW_PLATFORM_BUS_H */
|
#endif /* HW_PLATFORM_BUS_H */
|
||||||
|
|
Loading…
Reference in New Issue