123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116 |
- /*
- * 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);
- }
|