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