| /* | 
 |  * QEMU IndustryPack emulation | 
 |  * | 
 |  * Copyright (C) 2012 Igalia, S.L. | 
 |  * Author: Alberto Garcia <berto@igalia.com> | 
 |  * | 
 |  * This code is licensed under the GNU GPL v2 or (at your option) any | 
 |  * later version. | 
 |  */ | 
 |  | 
 | #include "qemu/osdep.h" | 
 | #include "qapi/error.h" | 
 | #include "qemu/module.h" | 
 | #include "hw/ipack/ipack.h" | 
 | #include "hw/irq.h" | 
 | #include "hw/qdev-properties.h" | 
 | #include "migration/vmstate.h" | 
 |  | 
 | IPackDevice *ipack_device_find(IPackBus *bus, int32_t slot) | 
 | { | 
 |     BusChild *kid; | 
 |  | 
 |     QTAILQ_FOREACH(kid, &BUS(bus)->children, sibling) { | 
 |         DeviceState *qdev = kid->child; | 
 |         IPackDevice *ip = IPACK_DEVICE(qdev); | 
 |         if (ip->slot == slot) { | 
 |             return ip; | 
 |         } | 
 |     } | 
 |     return NULL; | 
 | } | 
 |  | 
 | void ipack_bus_init(IPackBus *bus, size_t bus_size, | 
 |                     DeviceState *parent, | 
 |                     uint8_t n_slots, | 
 |                     qemu_irq_handler handler) | 
 | { | 
 |     qbus_init(bus, bus_size, TYPE_IPACK_BUS, parent, NULL); | 
 |     bus->n_slots = n_slots; | 
 |     bus->set_irq = handler; | 
 | } | 
 |  | 
 | static void ipack_device_realize(DeviceState *dev, Error **errp) | 
 | { | 
 |     IPackDevice *idev = IPACK_DEVICE(dev); | 
 |     IPackBus *bus = IPACK_BUS(qdev_get_parent_bus(dev)); | 
 |     IPackDeviceClass *k = IPACK_DEVICE_GET_CLASS(dev); | 
 |  | 
 |     if (idev->slot < 0) { | 
 |         idev->slot = bus->free_slot; | 
 |     } | 
 |     if (idev->slot >= bus->n_slots) { | 
 |         error_setg(errp, "Only %" PRIu8 " slots available.", bus->n_slots); | 
 |         return; | 
 |     } | 
 |     bus->free_slot = idev->slot + 1; | 
 |  | 
 |     idev->irq = qemu_allocate_irqs(bus->set_irq, idev, 2); | 
 |  | 
 |     k->realize(dev, errp); | 
 | } | 
 |  | 
 | static void ipack_device_unrealize(DeviceState *dev) | 
 | { | 
 |     IPackDevice *idev = IPACK_DEVICE(dev); | 
 |     IPackDeviceClass *k = IPACK_DEVICE_GET_CLASS(dev); | 
 |  | 
 |     if (k->unrealize) { | 
 |         k->unrealize(dev); | 
 |         return; | 
 |     } | 
 |  | 
 |     qemu_free_irqs(idev->irq, 2); | 
 | } | 
 |  | 
 | static const Property ipack_device_props[] = { | 
 |     DEFINE_PROP_INT32("slot", IPackDevice, slot, -1), | 
 | }; | 
 |  | 
 | static void ipack_device_class_init(ObjectClass *klass, void *data) | 
 | { | 
 |     DeviceClass *k = DEVICE_CLASS(klass); | 
 |  | 
 |     set_bit(DEVICE_CATEGORY_INPUT, k->categories); | 
 |     k->bus_type = TYPE_IPACK_BUS; | 
 |     k->realize = ipack_device_realize; | 
 |     k->unrealize = ipack_device_unrealize; | 
 |     device_class_set_props(k, ipack_device_props); | 
 | } | 
 |  | 
 | const VMStateDescription vmstate_ipack_device = { | 
 |     .name = "ipack_device", | 
 |     .version_id = 1, | 
 |     .minimum_version_id = 1, | 
 |     .fields = (const VMStateField[]) { | 
 |         VMSTATE_INT32(slot, IPackDevice), | 
 |         VMSTATE_END_OF_LIST() | 
 |     } | 
 | }; | 
 |  | 
 | static const TypeInfo ipack_device_info = { | 
 |     .name          = TYPE_IPACK_DEVICE, | 
 |     .parent        = TYPE_DEVICE, | 
 |     .instance_size = sizeof(IPackDevice), | 
 |     .class_size    = sizeof(IPackDeviceClass), | 
 |     .class_init    = ipack_device_class_init, | 
 |     .abstract      = true, | 
 | }; | 
 |  | 
 | static const TypeInfo ipack_bus_info = { | 
 |     .name = TYPE_IPACK_BUS, | 
 |     .parent = TYPE_BUS, | 
 |     .instance_size = sizeof(IPackBus), | 
 | }; | 
 |  | 
 | static void ipack_register_types(void) | 
 | { | 
 |     type_register_static(&ipack_device_info); | 
 |     type_register_static(&ipack_bus_info); | 
 | } | 
 |  | 
 | type_init(ipack_register_types) |