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 111f9c4cfdSAndreas Färber #include "hw/ipack/ipack.h" 129c16fa79SAlberto Garcia 139c16fa79SAlberto Garcia IPackDevice *ipack_device_find(IPackBus *bus, int32_t slot) 149c16fa79SAlberto Garcia { 159c16fa79SAlberto Garcia BusChild *kid; 169c16fa79SAlberto Garcia 179c16fa79SAlberto Garcia QTAILQ_FOREACH(kid, &BUS(bus)->children, sibling) { 189c16fa79SAlberto Garcia DeviceState *qdev = kid->child; 199c16fa79SAlberto Garcia IPackDevice *ip = IPACK_DEVICE(qdev); 209c16fa79SAlberto Garcia if (ip->slot == slot) { 219c16fa79SAlberto Garcia return ip; 229c16fa79SAlberto Garcia } 239c16fa79SAlberto Garcia } 249c16fa79SAlberto Garcia return NULL; 259c16fa79SAlberto Garcia } 269c16fa79SAlberto Garcia 2777cbb28aSAndreas Färber void ipack_bus_new_inplace(IPackBus *bus, size_t bus_size, 2877cbb28aSAndreas Färber DeviceState *parent, 299c16fa79SAlberto Garcia const char *name, uint8_t n_slots, 309c16fa79SAlberto Garcia qemu_irq_handler handler) 319c16fa79SAlberto Garcia { 32fb17dfe0SAndreas Färber qbus_create_inplace(bus, bus_size, TYPE_IPACK_BUS, parent, name); 339c16fa79SAlberto Garcia bus->n_slots = n_slots; 349c16fa79SAlberto Garcia bus->set_irq = handler; 359c16fa79SAlberto Garcia } 369c16fa79SAlberto Garcia 375c570902SAndreas Färber static void ipack_device_realize(DeviceState *dev, Error **errp) 389c16fa79SAlberto Garcia { 395c570902SAndreas Färber IPackDevice *idev = IPACK_DEVICE(dev); 405c570902SAndreas Färber IPackBus *bus = IPACK_BUS(qdev_get_parent_bus(dev)); 419c16fa79SAlberto Garcia IPackDeviceClass *k = IPACK_DEVICE_GET_CLASS(dev); 429c16fa79SAlberto Garcia 435c570902SAndreas Färber if (idev->slot < 0) { 445c570902SAndreas Färber idev->slot = bus->free_slot; 459c16fa79SAlberto Garcia } 465c570902SAndreas Färber if (idev->slot >= bus->n_slots) { 475c570902SAndreas Färber error_setg(errp, "Only %" PRIu8 " slots available.", bus->n_slots); 485c570902SAndreas Färber return; 499c16fa79SAlberto Garcia } 505c570902SAndreas Färber bus->free_slot = idev->slot + 1; 519c16fa79SAlberto Garcia 525c570902SAndreas Färber idev->irq = qemu_allocate_irqs(bus->set_irq, idev, 2); 539c16fa79SAlberto Garcia 545c570902SAndreas Färber k->realize(dev, errp); 559c16fa79SAlberto Garcia } 569c16fa79SAlberto Garcia 575c570902SAndreas Färber static void ipack_device_unrealize(DeviceState *dev, Error **errp) 589c16fa79SAlberto Garcia { 595c570902SAndreas Färber IPackDevice *idev = IPACK_DEVICE(dev); 609c16fa79SAlberto Garcia IPackDeviceClass *k = IPACK_DEVICE_GET_CLASS(dev); 615c570902SAndreas Färber Error *err = NULL; 629c16fa79SAlberto Garcia 635c570902SAndreas Färber if (k->unrealize) { 645c570902SAndreas Färber k->unrealize(dev, &err); 655c570902SAndreas Färber error_propagate(errp, err); 665c570902SAndreas Färber return; 679c16fa79SAlberto Garcia } 689c16fa79SAlberto Garcia 69*f173d57aSPeter Crosthwaite qemu_free_irqs(idev->irq, 2); 709c16fa79SAlberto Garcia } 719c16fa79SAlberto Garcia 729c16fa79SAlberto Garcia static Property ipack_device_props[] = { 739c16fa79SAlberto Garcia DEFINE_PROP_INT32("slot", IPackDevice, slot, -1), 749c16fa79SAlberto Garcia DEFINE_PROP_END_OF_LIST() 759c16fa79SAlberto Garcia }; 769c16fa79SAlberto Garcia 779c16fa79SAlberto Garcia static void ipack_device_class_init(ObjectClass *klass, void *data) 789c16fa79SAlberto Garcia { 799c16fa79SAlberto Garcia DeviceClass *k = DEVICE_CLASS(klass); 805c570902SAndreas Färber 81125ee0edSMarcel Apfelbaum set_bit(DEVICE_CATEGORY_INPUT, k->categories); 829c16fa79SAlberto Garcia k->bus_type = TYPE_IPACK_BUS; 835c570902SAndreas Färber k->realize = ipack_device_realize; 845c570902SAndreas Färber k->unrealize = ipack_device_unrealize; 859c16fa79SAlberto Garcia k->props = ipack_device_props; 869c16fa79SAlberto Garcia } 879c16fa79SAlberto Garcia 889c16fa79SAlberto Garcia const VMStateDescription vmstate_ipack_device = { 899c16fa79SAlberto Garcia .name = "ipack_device", 909c16fa79SAlberto Garcia .version_id = 1, 919c16fa79SAlberto Garcia .minimum_version_id = 1, 929c16fa79SAlberto Garcia .fields = (VMStateField[]) { 939c16fa79SAlberto Garcia VMSTATE_INT32(slot, IPackDevice), 949c16fa79SAlberto Garcia VMSTATE_END_OF_LIST() 959c16fa79SAlberto Garcia } 969c16fa79SAlberto Garcia }; 979c16fa79SAlberto Garcia 989c16fa79SAlberto Garcia static const TypeInfo ipack_device_info = { 999c16fa79SAlberto Garcia .name = TYPE_IPACK_DEVICE, 1009c16fa79SAlberto Garcia .parent = TYPE_DEVICE, 1019c16fa79SAlberto Garcia .instance_size = sizeof(IPackDevice), 1029c16fa79SAlberto Garcia .class_size = sizeof(IPackDeviceClass), 1039c16fa79SAlberto Garcia .class_init = ipack_device_class_init, 1049c16fa79SAlberto Garcia .abstract = true, 1059c16fa79SAlberto Garcia }; 1069c16fa79SAlberto Garcia 1079c16fa79SAlberto Garcia static const TypeInfo ipack_bus_info = { 1089c16fa79SAlberto Garcia .name = TYPE_IPACK_BUS, 1099c16fa79SAlberto Garcia .parent = TYPE_BUS, 1109c16fa79SAlberto Garcia .instance_size = sizeof(IPackBus), 1119c16fa79SAlberto Garcia }; 1129c16fa79SAlberto Garcia 1139c16fa79SAlberto Garcia static void ipack_register_types(void) 1149c16fa79SAlberto Garcia { 1159c16fa79SAlberto Garcia type_register_static(&ipack_device_info); 1169c16fa79SAlberto Garcia type_register_static(&ipack_bus_info); 1179c16fa79SAlberto Garcia } 1189c16fa79SAlberto Garcia 1199c16fa79SAlberto Garcia type_init(ipack_register_types) 120