116 lines
3 KiB
C
116 lines
3 KiB
C
/*
|
|
* QDev Hotplug handlers
|
|
*
|
|
* Copyright (c) Red Hat
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0-or-later
|
|
*
|
|
* This work is licensed under the terms of the GNU GPL, version 2 or later.
|
|
* See the COPYING file in the top-level directory.
|
|
*/
|
|
|
|
#include "qemu/osdep.h"
|
|
#include "hw/qdev-core.h"
|
|
#include "hw/boards.h"
|
|
#include "qapi/error.h"
|
|
|
|
HotplugHandler *qdev_get_machine_hotplug_handler(DeviceState *dev)
|
|
{
|
|
MachineState *machine;
|
|
MachineClass *mc;
|
|
Object *m_obj = qdev_get_machine();
|
|
|
|
if (object_dynamic_cast(m_obj, TYPE_MACHINE)) {
|
|
machine = MACHINE(m_obj);
|
|
mc = MACHINE_GET_CLASS(machine);
|
|
if (mc->get_hotplug_handler) {
|
|
return mc->get_hotplug_handler(machine, dev);
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static bool qdev_hotplug_unplug_allowed_common(DeviceState *dev, BusState *bus,
|
|
Error **errp)
|
|
{
|
|
DeviceClass *dc = DEVICE_GET_CLASS(dev);
|
|
|
|
if (!dc->hotpluggable) {
|
|
error_setg(errp, "Device '%s' does not support hotplugging",
|
|
object_get_typename(OBJECT(dev)));
|
|
return false;
|
|
}
|
|
|
|
if (bus) {
|
|
if (!qbus_is_hotpluggable(bus)) {
|
|
error_setg(errp, "Bus '%s' does not support hotplugging",
|
|
bus->name);
|
|
return false;
|
|
}
|
|
} else {
|
|
if (!qdev_get_machine_hotplug_handler(dev)) {
|
|
/*
|
|
* No bus, no machine hotplug handler --> device is not hotpluggable
|
|
*/
|
|
error_setg(errp,
|
|
"Device '%s' can not be hotplugged on this machine",
|
|
object_get_typename(OBJECT(dev)));
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool qdev_hotplug_allowed(DeviceState *dev, BusState *bus, Error **errp)
|
|
{
|
|
MachineState *machine;
|
|
MachineClass *mc;
|
|
Object *m_obj = qdev_get_machine();
|
|
|
|
if (!qdev_hotplug_unplug_allowed_common(dev, bus, errp)) {
|
|
return false;
|
|
}
|
|
|
|
if (object_dynamic_cast(m_obj, TYPE_MACHINE)) {
|
|
machine = MACHINE(m_obj);
|
|
mc = MACHINE_GET_CLASS(machine);
|
|
if (mc->hotplug_allowed) {
|
|
return mc->hotplug_allowed(machine, dev, errp);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool qdev_hotunplug_allowed(DeviceState *dev, Error **errp)
|
|
{
|
|
return !qdev_unplug_blocked(dev, errp) &&
|
|
qdev_hotplug_unplug_allowed_common(dev, dev->parent_bus, errp);
|
|
}
|
|
|
|
HotplugHandler *qdev_get_bus_hotplug_handler(DeviceState *dev)
|
|
{
|
|
if (dev->parent_bus) {
|
|
return dev->parent_bus->hotplug_handler;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
HotplugHandler *qdev_get_hotplug_handler(DeviceState *dev)
|
|
{
|
|
HotplugHandler *hotplug_ctrl = qdev_get_machine_hotplug_handler(dev);
|
|
|
|
if (hotplug_ctrl == NULL && dev->parent_bus) {
|
|
hotplug_ctrl = qdev_get_bus_hotplug_handler(dev);
|
|
}
|
|
return hotplug_ctrl;
|
|
}
|
|
|
|
/* can be used as ->unplug() callback for the simple cases */
|
|
void qdev_simple_device_unplug_cb(HotplugHandler *hotplug_dev,
|
|
DeviceState *dev, Error **errp)
|
|
{
|
|
qdev_unrealize(dev);
|
|
}
|