19c16fa79SAlberto Garcia /* 29c16fa79SAlberto Garcia * QEMU IndustryPack emulation 39c16fa79SAlberto Garcia * 49c16fa79SAlberto Garcia * Copyright (C) 2012 Igalia, S.L. 59c16fa79SAlberto Garcia * Author: Alberto Garcia <agarcia@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" 12*da34e65cSMarkus Armbruster #include "qapi/error.h" 131f9c4cfdSAndreas Färber #include "hw/ipack/ipack.h" 149c16fa79SAlberto Garcia 159c16fa79SAlberto Garcia IPackDevice *ipack_device_find(IPackBus *bus, int32_t slot) 169c16fa79SAlberto Garcia { 179c16fa79SAlberto Garcia BusChild *kid; 189c16fa79SAlberto Garcia 199c16fa79SAlberto Garcia QTAILQ_FOREACH(kid, &BUS(bus)->children, sibling) { 209c16fa79SAlberto Garcia DeviceState *qdev = kid->child; 219c16fa79SAlberto Garcia IPackDevice *ip = IPACK_DEVICE(qdev); 229c16fa79SAlberto Garcia if (ip->slot == slot) { 239c16fa79SAlberto Garcia return ip; 249c16fa79SAlberto Garcia } 259c16fa79SAlberto Garcia } 269c16fa79SAlberto Garcia return NULL; 279c16fa79SAlberto Garcia } 289c16fa79SAlberto Garcia 2977cbb28aSAndreas Färber void ipack_bus_new_inplace(IPackBus *bus, size_t bus_size, 3077cbb28aSAndreas Färber DeviceState *parent, 319c16fa79SAlberto Garcia const char *name, uint8_t n_slots, 329c16fa79SAlberto Garcia qemu_irq_handler handler) 339c16fa79SAlberto Garcia { 34fb17dfe0SAndreas Färber qbus_create_inplace(bus, bus_size, TYPE_IPACK_BUS, parent, name); 359c16fa79SAlberto Garcia bus->n_slots = n_slots; 369c16fa79SAlberto Garcia bus->set_irq = handler; 379c16fa79SAlberto Garcia } 389c16fa79SAlberto Garcia 395c570902SAndreas Färber static void ipack_device_realize(DeviceState *dev, Error **errp) 409c16fa79SAlberto Garcia { 415c570902SAndreas Färber IPackDevice *idev = IPACK_DEVICE(dev); 425c570902SAndreas Färber IPackBus *bus = IPACK_BUS(qdev_get_parent_bus(dev)); 439c16fa79SAlberto Garcia IPackDeviceClass *k = IPACK_DEVICE_GET_CLASS(dev); 449c16fa79SAlberto Garcia 455c570902SAndreas Färber if (idev->slot < 0) { 465c570902SAndreas Färber idev->slot = bus->free_slot; 479c16fa79SAlberto Garcia } 485c570902SAndreas Färber if (idev->slot >= bus->n_slots) { 495c570902SAndreas Färber error_setg(errp, "Only %" PRIu8 " slots available.", bus->n_slots); 505c570902SAndreas Färber return; 519c16fa79SAlberto Garcia } 525c570902SAndreas Färber bus->free_slot = idev->slot + 1; 539c16fa79SAlberto Garcia 545c570902SAndreas Färber idev->irq = qemu_allocate_irqs(bus->set_irq, idev, 2); 559c16fa79SAlberto Garcia 565c570902SAndreas Färber k->realize(dev, errp); 579c16fa79SAlberto Garcia } 589c16fa79SAlberto Garcia 595c570902SAndreas Färber static void ipack_device_unrealize(DeviceState *dev, Error **errp) 609c16fa79SAlberto Garcia { 615c570902SAndreas Färber IPackDevice *idev = IPACK_DEVICE(dev); 629c16fa79SAlberto Garcia IPackDeviceClass *k = IPACK_DEVICE_GET_CLASS(dev); 635c570902SAndreas Färber Error *err = NULL; 649c16fa79SAlberto Garcia 655c570902SAndreas Färber if (k->unrealize) { 665c570902SAndreas Färber k->unrealize(dev, &err); 675c570902SAndreas Färber error_propagate(errp, err); 685c570902SAndreas Färber return; 699c16fa79SAlberto Garcia } 709c16fa79SAlberto Garcia 71f173d57aSPeter Crosthwaite qemu_free_irqs(idev->irq, 2); 729c16fa79SAlberto Garcia } 739c16fa79SAlberto Garcia 749c16fa79SAlberto Garcia static Property ipack_device_props[] = { 759c16fa79SAlberto Garcia DEFINE_PROP_INT32("slot", IPackDevice, slot, -1), 769c16fa79SAlberto Garcia DEFINE_PROP_END_OF_LIST() 779c16fa79SAlberto Garcia }; 789c16fa79SAlberto Garcia 799c16fa79SAlberto Garcia static void ipack_device_class_init(ObjectClass *klass, void *data) 809c16fa79SAlberto Garcia { 819c16fa79SAlberto Garcia DeviceClass *k = DEVICE_CLASS(klass); 825c570902SAndreas Färber 83125ee0edSMarcel Apfelbaum set_bit(DEVICE_CATEGORY_INPUT, k->categories); 849c16fa79SAlberto Garcia k->bus_type = TYPE_IPACK_BUS; 855c570902SAndreas Färber k->realize = ipack_device_realize; 865c570902SAndreas Färber k->unrealize = ipack_device_unrealize; 879c16fa79SAlberto Garcia k->props = ipack_device_props; 889c16fa79SAlberto Garcia } 899c16fa79SAlberto Garcia 909c16fa79SAlberto Garcia const VMStateDescription vmstate_ipack_device = { 919c16fa79SAlberto Garcia .name = "ipack_device", 929c16fa79SAlberto Garcia .version_id = 1, 939c16fa79SAlberto Garcia .minimum_version_id = 1, 949c16fa79SAlberto Garcia .fields = (VMStateField[]) { 959c16fa79SAlberto Garcia VMSTATE_INT32(slot, IPackDevice), 969c16fa79SAlberto Garcia VMSTATE_END_OF_LIST() 979c16fa79SAlberto Garcia } 989c16fa79SAlberto Garcia }; 999c16fa79SAlberto Garcia 1009c16fa79SAlberto Garcia static const TypeInfo ipack_device_info = { 1019c16fa79SAlberto Garcia .name = TYPE_IPACK_DEVICE, 1029c16fa79SAlberto Garcia .parent = TYPE_DEVICE, 1039c16fa79SAlberto Garcia .instance_size = sizeof(IPackDevice), 1049c16fa79SAlberto Garcia .class_size = sizeof(IPackDeviceClass), 1059c16fa79SAlberto Garcia .class_init = ipack_device_class_init, 1069c16fa79SAlberto Garcia .abstract = true, 1079c16fa79SAlberto Garcia }; 1089c16fa79SAlberto Garcia 1099c16fa79SAlberto Garcia static const TypeInfo ipack_bus_info = { 1109c16fa79SAlberto Garcia .name = TYPE_IPACK_BUS, 1119c16fa79SAlberto Garcia .parent = TYPE_BUS, 1129c16fa79SAlberto Garcia .instance_size = sizeof(IPackBus), 1139c16fa79SAlberto Garcia }; 1149c16fa79SAlberto Garcia 1159c16fa79SAlberto Garcia static void ipack_register_types(void) 1169c16fa79SAlberto Garcia { 1179c16fa79SAlberto Garcia type_register_static(&ipack_device_info); 1189c16fa79SAlberto Garcia type_register_static(&ipack_bus_info); 1199c16fa79SAlberto Garcia } 1209c16fa79SAlberto Garcia 1219c16fa79SAlberto Garcia type_init(ipack_register_types) 122