19c16fa79SAlberto Garcia /* 29c16fa79SAlberto Garcia * QEMU IndustryPack emulation 39c16fa79SAlberto Garcia * 49c16fa79SAlberto Garcia * Copyright (C) 2012 Igalia, S.L. 5b996aed5SAlberto Garcia * Author: Alberto Garcia <berto@igalia.com> 69c16fa79SAlberto Garcia * 79c16fa79SAlberto Garcia * This code is licensed under the GNU GPL v2 or (at your option) any 89c16fa79SAlberto Garcia * later version. 99c16fa79SAlberto Garcia */ 109c16fa79SAlberto Garcia 110430891cSPeter Maydell #include "qemu/osdep.h" 12da34e65cSMarkus Armbruster #include "qapi/error.h" 13*0b8fa32fSMarkus Armbruster #include "qemu/module.h" 141f9c4cfdSAndreas Färber #include "hw/ipack/ipack.h" 159c16fa79SAlberto Garcia 169c16fa79SAlberto Garcia IPackDevice *ipack_device_find(IPackBus *bus, int32_t slot) 179c16fa79SAlberto Garcia { 189c16fa79SAlberto Garcia BusChild *kid; 199c16fa79SAlberto Garcia 209c16fa79SAlberto Garcia QTAILQ_FOREACH(kid, &BUS(bus)->children, sibling) { 219c16fa79SAlberto Garcia DeviceState *qdev = kid->child; 229c16fa79SAlberto Garcia IPackDevice *ip = IPACK_DEVICE(qdev); 239c16fa79SAlberto Garcia if (ip->slot == slot) { 249c16fa79SAlberto Garcia return ip; 259c16fa79SAlberto Garcia } 269c16fa79SAlberto Garcia } 279c16fa79SAlberto Garcia return NULL; 289c16fa79SAlberto Garcia } 299c16fa79SAlberto Garcia 3077cbb28aSAndreas Färber void ipack_bus_new_inplace(IPackBus *bus, size_t bus_size, 3177cbb28aSAndreas Färber DeviceState *parent, 329c16fa79SAlberto Garcia const char *name, uint8_t n_slots, 339c16fa79SAlberto Garcia qemu_irq_handler handler) 349c16fa79SAlberto Garcia { 35fb17dfe0SAndreas Färber qbus_create_inplace(bus, bus_size, TYPE_IPACK_BUS, parent, name); 369c16fa79SAlberto Garcia bus->n_slots = n_slots; 379c16fa79SAlberto Garcia bus->set_irq = handler; 389c16fa79SAlberto Garcia } 399c16fa79SAlberto Garcia 405c570902SAndreas Färber static void ipack_device_realize(DeviceState *dev, Error **errp) 419c16fa79SAlberto Garcia { 425c570902SAndreas Färber IPackDevice *idev = IPACK_DEVICE(dev); 435c570902SAndreas Färber IPackBus *bus = IPACK_BUS(qdev_get_parent_bus(dev)); 449c16fa79SAlberto Garcia IPackDeviceClass *k = IPACK_DEVICE_GET_CLASS(dev); 459c16fa79SAlberto Garcia 465c570902SAndreas Färber if (idev->slot < 0) { 475c570902SAndreas Färber idev->slot = bus->free_slot; 489c16fa79SAlberto Garcia } 495c570902SAndreas Färber if (idev->slot >= bus->n_slots) { 505c570902SAndreas Färber error_setg(errp, "Only %" PRIu8 " slots available.", bus->n_slots); 515c570902SAndreas Färber return; 529c16fa79SAlberto Garcia } 535c570902SAndreas Färber bus->free_slot = idev->slot + 1; 549c16fa79SAlberto Garcia 555c570902SAndreas Färber idev->irq = qemu_allocate_irqs(bus->set_irq, idev, 2); 569c16fa79SAlberto Garcia 575c570902SAndreas Färber k->realize(dev, errp); 589c16fa79SAlberto Garcia } 599c16fa79SAlberto Garcia 605c570902SAndreas Färber static void ipack_device_unrealize(DeviceState *dev, Error **errp) 619c16fa79SAlberto Garcia { 625c570902SAndreas Färber IPackDevice *idev = IPACK_DEVICE(dev); 639c16fa79SAlberto Garcia IPackDeviceClass *k = IPACK_DEVICE_GET_CLASS(dev); 645c570902SAndreas Färber Error *err = NULL; 659c16fa79SAlberto Garcia 665c570902SAndreas Färber if (k->unrealize) { 675c570902SAndreas Färber k->unrealize(dev, &err); 685c570902SAndreas Färber error_propagate(errp, err); 695c570902SAndreas Färber return; 709c16fa79SAlberto Garcia } 719c16fa79SAlberto Garcia 72f173d57aSPeter Crosthwaite qemu_free_irqs(idev->irq, 2); 739c16fa79SAlberto Garcia } 749c16fa79SAlberto Garcia 759c16fa79SAlberto Garcia static Property ipack_device_props[] = { 769c16fa79SAlberto Garcia DEFINE_PROP_INT32("slot", IPackDevice, slot, -1), 779c16fa79SAlberto Garcia DEFINE_PROP_END_OF_LIST() 789c16fa79SAlberto Garcia }; 799c16fa79SAlberto Garcia 809c16fa79SAlberto Garcia static void ipack_device_class_init(ObjectClass *klass, void *data) 819c16fa79SAlberto Garcia { 829c16fa79SAlberto Garcia DeviceClass *k = DEVICE_CLASS(klass); 835c570902SAndreas Färber 84125ee0edSMarcel Apfelbaum set_bit(DEVICE_CATEGORY_INPUT, k->categories); 859c16fa79SAlberto Garcia k->bus_type = TYPE_IPACK_BUS; 865c570902SAndreas Färber k->realize = ipack_device_realize; 875c570902SAndreas Färber k->unrealize = ipack_device_unrealize; 889c16fa79SAlberto Garcia k->props = ipack_device_props; 899c16fa79SAlberto Garcia } 909c16fa79SAlberto Garcia 919c16fa79SAlberto Garcia const VMStateDescription vmstate_ipack_device = { 929c16fa79SAlberto Garcia .name = "ipack_device", 939c16fa79SAlberto Garcia .version_id = 1, 949c16fa79SAlberto Garcia .minimum_version_id = 1, 959c16fa79SAlberto Garcia .fields = (VMStateField[]) { 969c16fa79SAlberto Garcia VMSTATE_INT32(slot, IPackDevice), 979c16fa79SAlberto Garcia VMSTATE_END_OF_LIST() 989c16fa79SAlberto Garcia } 999c16fa79SAlberto Garcia }; 1009c16fa79SAlberto Garcia 1019c16fa79SAlberto Garcia static const TypeInfo ipack_device_info = { 1029c16fa79SAlberto Garcia .name = TYPE_IPACK_DEVICE, 1039c16fa79SAlberto Garcia .parent = TYPE_DEVICE, 1049c16fa79SAlberto Garcia .instance_size = sizeof(IPackDevice), 1059c16fa79SAlberto Garcia .class_size = sizeof(IPackDeviceClass), 1069c16fa79SAlberto Garcia .class_init = ipack_device_class_init, 1079c16fa79SAlberto Garcia .abstract = true, 1089c16fa79SAlberto Garcia }; 1099c16fa79SAlberto Garcia 1109c16fa79SAlberto Garcia static const TypeInfo ipack_bus_info = { 1119c16fa79SAlberto Garcia .name = TYPE_IPACK_BUS, 1129c16fa79SAlberto Garcia .parent = TYPE_BUS, 1139c16fa79SAlberto Garcia .instance_size = sizeof(IPackBus), 1149c16fa79SAlberto Garcia }; 1159c16fa79SAlberto Garcia 1169c16fa79SAlberto Garcia static void ipack_register_types(void) 1179c16fa79SAlberto Garcia { 1189c16fa79SAlberto Garcia type_register_static(&ipack_device_info); 1199c16fa79SAlberto Garcia type_register_static(&ipack_bus_info); 1209c16fa79SAlberto Garcia } 1219c16fa79SAlberto Garcia 1229c16fa79SAlberto Garcia type_init(ipack_register_types) 123