Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 1 | /* |
| 2 | * QEMU IndustryPack emulation |
| 3 | * |
| 4 | * Copyright (C) 2012 Igalia, S.L. |
Alberto Garcia | b996aed | 2016-02-23 10:44:25 +0200 | [diff] [blame] | 5 | * Author: Alberto Garcia <berto@igalia.com> |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 6 | * |
| 7 | * This code is licensed under the GNU GPL v2 or (at your option) any |
| 8 | * later version. |
| 9 | */ |
| 10 | |
Peter Maydell | 0430891 | 2016-01-26 18:17:30 +0000 | [diff] [blame] | 11 | #include "qemu/osdep.h" |
Markus Armbruster | da34e65 | 2016-03-14 09:01:28 +0100 | [diff] [blame] | 12 | #include "qapi/error.h" |
Markus Armbruster | 0b8fa32 | 2019-05-23 16:35:07 +0200 | [diff] [blame] | 13 | #include "qemu/module.h" |
Andreas Färber | 1f9c4cf | 2013-08-02 00:48:40 +0200 | [diff] [blame] | 14 | #include "hw/ipack/ipack.h" |
Markus Armbruster | 64552b6 | 2019-08-12 07:23:42 +0200 | [diff] [blame] | 15 | #include "hw/irq.h" |
Markus Armbruster | a27bd6c | 2019-08-12 07:23:51 +0200 | [diff] [blame] | 16 | #include "hw/qdev-properties.h" |
Markus Armbruster | d645427 | 2019-08-12 07:23:45 +0200 | [diff] [blame] | 17 | #include "migration/vmstate.h" |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 18 | |
| 19 | IPackDevice *ipack_device_find(IPackBus *bus, int32_t slot) |
| 20 | { |
| 21 | BusChild *kid; |
| 22 | |
| 23 | QTAILQ_FOREACH(kid, &BUS(bus)->children, sibling) { |
| 24 | DeviceState *qdev = kid->child; |
| 25 | IPackDevice *ip = IPACK_DEVICE(qdev); |
| 26 | if (ip->slot == slot) { |
| 27 | return ip; |
| 28 | } |
| 29 | } |
| 30 | return NULL; |
| 31 | } |
| 32 | |
Peter Maydell | 43417c0 | 2021-09-23 13:11:49 +0100 | [diff] [blame] | 33 | void ipack_bus_init(IPackBus *bus, size_t bus_size, |
| 34 | DeviceState *parent, |
| 35 | uint8_t n_slots, |
| 36 | qemu_irq_handler handler) |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 37 | { |
Peter Maydell | d637e1d | 2021-09-23 13:11:51 +0100 | [diff] [blame] | 38 | qbus_init(bus, bus_size, TYPE_IPACK_BUS, parent, NULL); |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 39 | bus->n_slots = n_slots; |
| 40 | bus->set_irq = handler; |
| 41 | } |
| 42 | |
Andreas Färber | 5c57090 | 2013-08-01 18:45:02 +0200 | [diff] [blame] | 43 | static void ipack_device_realize(DeviceState *dev, Error **errp) |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 44 | { |
Andreas Färber | 5c57090 | 2013-08-01 18:45:02 +0200 | [diff] [blame] | 45 | IPackDevice *idev = IPACK_DEVICE(dev); |
| 46 | IPackBus *bus = IPACK_BUS(qdev_get_parent_bus(dev)); |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 47 | IPackDeviceClass *k = IPACK_DEVICE_GET_CLASS(dev); |
| 48 | |
Andreas Färber | 5c57090 | 2013-08-01 18:45:02 +0200 | [diff] [blame] | 49 | if (idev->slot < 0) { |
| 50 | idev->slot = bus->free_slot; |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 51 | } |
Andreas Färber | 5c57090 | 2013-08-01 18:45:02 +0200 | [diff] [blame] | 52 | if (idev->slot >= bus->n_slots) { |
| 53 | error_setg(errp, "Only %" PRIu8 " slots available.", bus->n_slots); |
| 54 | return; |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 55 | } |
Andreas Färber | 5c57090 | 2013-08-01 18:45:02 +0200 | [diff] [blame] | 56 | bus->free_slot = idev->slot + 1; |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 57 | |
Andreas Färber | 5c57090 | 2013-08-01 18:45:02 +0200 | [diff] [blame] | 58 | idev->irq = qemu_allocate_irqs(bus->set_irq, idev, 2); |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 59 | |
Andreas Färber | 5c57090 | 2013-08-01 18:45:02 +0200 | [diff] [blame] | 60 | k->realize(dev, errp); |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 61 | } |
| 62 | |
Markus Armbruster | b69c3c2 | 2020-05-05 17:29:24 +0200 | [diff] [blame] | 63 | static void ipack_device_unrealize(DeviceState *dev) |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 64 | { |
Andreas Färber | 5c57090 | 2013-08-01 18:45:02 +0200 | [diff] [blame] | 65 | IPackDevice *idev = IPACK_DEVICE(dev); |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 66 | IPackDeviceClass *k = IPACK_DEVICE_GET_CLASS(dev); |
| 67 | |
Andreas Färber | 5c57090 | 2013-08-01 18:45:02 +0200 | [diff] [blame] | 68 | if (k->unrealize) { |
Markus Armbruster | b69c3c2 | 2020-05-05 17:29:24 +0200 | [diff] [blame] | 69 | k->unrealize(dev); |
Andreas Färber | 5c57090 | 2013-08-01 18:45:02 +0200 | [diff] [blame] | 70 | return; |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 71 | } |
| 72 | |
Peter Crosthwaite | f173d57 | 2014-06-18 00:56:31 -0700 | [diff] [blame] | 73 | qemu_free_irqs(idev->irq, 2); |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 74 | } |
| 75 | |
| 76 | static Property ipack_device_props[] = { |
| 77 | DEFINE_PROP_INT32("slot", IPackDevice, slot, -1), |
| 78 | DEFINE_PROP_END_OF_LIST() |
| 79 | }; |
| 80 | |
| 81 | static void ipack_device_class_init(ObjectClass *klass, void *data) |
| 82 | { |
| 83 | DeviceClass *k = DEVICE_CLASS(klass); |
Andreas Färber | 5c57090 | 2013-08-01 18:45:02 +0200 | [diff] [blame] | 84 | |
Marcel Apfelbaum | 125ee0e | 2013-07-29 17:17:45 +0300 | [diff] [blame] | 85 | set_bit(DEVICE_CATEGORY_INPUT, k->categories); |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 86 | k->bus_type = TYPE_IPACK_BUS; |
Andreas Färber | 5c57090 | 2013-08-01 18:45:02 +0200 | [diff] [blame] | 87 | k->realize = ipack_device_realize; |
| 88 | k->unrealize = ipack_device_unrealize; |
Marc-André Lureau | 4f67d30 | 2020-01-10 19:30:32 +0400 | [diff] [blame] | 89 | device_class_set_props(k, ipack_device_props); |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 90 | } |
| 91 | |
| 92 | const VMStateDescription vmstate_ipack_device = { |
| 93 | .name = "ipack_device", |
| 94 | .version_id = 1, |
| 95 | .minimum_version_id = 1, |
Richard Henderson | 8913d05 | 2023-12-21 14:16:16 +1100 | [diff] [blame] | 96 | .fields = (const VMStateField[]) { |
Alberto Garcia | 9c16fa7 | 2013-01-11 18:25:29 +0100 | [diff] [blame] | 97 | VMSTATE_INT32(slot, IPackDevice), |
| 98 | VMSTATE_END_OF_LIST() |
| 99 | } |
| 100 | }; |
| 101 | |
| 102 | static const TypeInfo ipack_device_info = { |
| 103 | .name = TYPE_IPACK_DEVICE, |
| 104 | .parent = TYPE_DEVICE, |
| 105 | .instance_size = sizeof(IPackDevice), |
| 106 | .class_size = sizeof(IPackDeviceClass), |
| 107 | .class_init = ipack_device_class_init, |
| 108 | .abstract = true, |
| 109 | }; |
| 110 | |
| 111 | static const TypeInfo ipack_bus_info = { |
| 112 | .name = TYPE_IPACK_BUS, |
| 113 | .parent = TYPE_BUS, |
| 114 | .instance_size = sizeof(IPackBus), |
| 115 | }; |
| 116 | |
| 117 | static void ipack_register_types(void) |
| 118 | { |
| 119 | type_register_static(&ipack_device_info); |
| 120 | type_register_static(&ipack_bus_info); |
| 121 | } |
| 122 | |
| 123 | type_init(ipack_register_types) |