1 /* 2 * isa bus support for qdev. 3 * 4 * Copyright (c) 2009 Gerd Hoffmann <kraxel@redhat.com> 5 * 6 * This library is free software; you can redistribute it and/or 7 * modify it under the terms of the GNU Lesser General Public 8 * License as published by the Free Software Foundation; either 9 * version 2 of the License, or (at your option) any later version. 10 * 11 * This library is distributed in the hope that it will be useful, 12 * but WITHOUT ANY WARRANTY; without even the implied warranty of 13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 14 * Lesser General Public License for more details. 15 * 16 * You should have received a copy of the GNU Lesser General Public 17 * License along with this library; if not, see <http://www.gnu.org/licenses/>. 18 */ 19 #include "hw/hw.h" 20 #include "monitor/monitor.h" 21 #include "hw/sysbus.h" 22 #include "sysemu/sysemu.h" 23 #include "hw/isa/isa.h" 24 #include "hw/i386/pc.h" 25 26 static ISABus *isabus; 27 28 static void isabus_dev_print(Monitor *mon, DeviceState *dev, int indent); 29 static char *isabus_get_fw_dev_path(DeviceState *dev); 30 31 static void isa_bus_class_init(ObjectClass *klass, void *data) 32 { 33 BusClass *k = BUS_CLASS(klass); 34 35 k->print_dev = isabus_dev_print; 36 k->get_fw_dev_path = isabus_get_fw_dev_path; 37 } 38 39 static const TypeInfo isa_bus_info = { 40 .name = TYPE_ISA_BUS, 41 .parent = TYPE_BUS, 42 .instance_size = sizeof(ISABus), 43 .class_init = isa_bus_class_init, 44 }; 45 46 ISABus *isa_bus_new(DeviceState *dev, MemoryRegion* address_space, 47 MemoryRegion *address_space_io, Error **errp) 48 { 49 if (isabus) { 50 error_setg(errp, "Can't create a second ISA bus"); 51 return NULL; 52 } 53 if (!dev) { 54 dev = qdev_create(NULL, "isabus-bridge"); 55 qdev_init_nofail(dev); 56 } 57 58 isabus = ISA_BUS(qbus_create(TYPE_ISA_BUS, dev, NULL)); 59 isabus->address_space = address_space; 60 isabus->address_space_io = address_space_io; 61 return isabus; 62 } 63 64 void isa_bus_irqs(ISABus *bus, qemu_irq *irqs) 65 { 66 bus->irqs = irqs; 67 } 68 69 /* 70 * isa_get_irq() returns the corresponding qemu_irq entry for the i8259. 71 * 72 * This function is only for special cases such as the 'ferr', and 73 * temporary use for normal devices until they are converted to qdev. 74 */ 75 qemu_irq isa_get_irq(ISADevice *dev, int isairq) 76 { 77 assert(!dev || ISA_BUS(qdev_get_parent_bus(DEVICE(dev))) == isabus); 78 if (isairq < 0 || isairq > 15) { 79 hw_error("isa irq %d invalid", isairq); 80 } 81 return isabus->irqs[isairq]; 82 } 83 84 void isa_init_irq(ISADevice *dev, qemu_irq *p, int isairq) 85 { 86 assert(dev->nirqs < ARRAY_SIZE(dev->isairq)); 87 dev->isairq[dev->nirqs] = isairq; 88 *p = isa_get_irq(dev, isairq); 89 dev->nirqs++; 90 } 91 92 static inline void isa_init_ioport(ISADevice *dev, uint16_t ioport) 93 { 94 if (dev && (dev->ioport_id == 0 || ioport < dev->ioport_id)) { 95 dev->ioport_id = ioport; 96 } 97 } 98 99 void isa_register_ioport(ISADevice *dev, MemoryRegion *io, uint16_t start) 100 { 101 memory_region_add_subregion(isabus->address_space_io, start, io); 102 isa_init_ioport(dev, start); 103 } 104 105 void isa_register_portio_list(ISADevice *dev, uint16_t start, 106 const MemoryRegionPortio *pio_start, 107 void *opaque, const char *name) 108 { 109 PortioList piolist; 110 111 /* START is how we should treat DEV, regardless of the actual 112 contents of the portio array. This is how the old code 113 actually handled e.g. the FDC device. */ 114 isa_init_ioport(dev, start); 115 116 /* FIXME: the device should store created PortioList in its state. Note 117 that DEV can be NULL here and that single device can register several 118 portio lists. Current implementation is leaking memory allocated 119 in portio_list_init. The leak is not critical because it happens only 120 at initialization time. */ 121 portio_list_init(&piolist, OBJECT(dev), pio_start, opaque, name); 122 portio_list_add(&piolist, isabus->address_space_io, start); 123 } 124 125 static void isa_device_init(Object *obj) 126 { 127 ISADevice *dev = ISA_DEVICE(obj); 128 129 dev->isairq[0] = -1; 130 dev->isairq[1] = -1; 131 } 132 133 ISADevice *isa_create(ISABus *bus, const char *name) 134 { 135 DeviceState *dev; 136 137 dev = qdev_create(BUS(bus), name); 138 return ISA_DEVICE(dev); 139 } 140 141 ISADevice *isa_try_create(ISABus *bus, const char *name) 142 { 143 DeviceState *dev; 144 145 dev = qdev_try_create(BUS(bus), name); 146 return ISA_DEVICE(dev); 147 } 148 149 ISADevice *isa_create_simple(ISABus *bus, const char *name) 150 { 151 ISADevice *dev; 152 153 dev = isa_create(bus, name); 154 qdev_init_nofail(DEVICE(dev)); 155 return dev; 156 } 157 158 ISADevice *isa_vga_init(ISABus *bus) 159 { 160 switch (vga_interface_type) { 161 case VGA_CIRRUS: 162 return isa_create_simple(bus, "isa-cirrus-vga"); 163 case VGA_QXL: 164 fprintf(stderr, "%s: qxl: no PCI bus\n", __func__); 165 return NULL; 166 case VGA_STD: 167 return isa_create_simple(bus, "isa-vga"); 168 case VGA_VMWARE: 169 fprintf(stderr, "%s: vmware_vga: no PCI bus\n", __func__); 170 return NULL; 171 case VGA_VIRTIO: 172 fprintf(stderr, "%s: virtio-vga: no PCI bus\n", __func__); 173 return NULL; 174 case VGA_NONE: 175 default: 176 return NULL; 177 } 178 } 179 180 static void isabus_dev_print(Monitor *mon, DeviceState *dev, int indent) 181 { 182 ISADevice *d = ISA_DEVICE(dev); 183 184 if (d->isairq[1] != -1) { 185 monitor_printf(mon, "%*sisa irqs %d,%d\n", indent, "", 186 d->isairq[0], d->isairq[1]); 187 } else if (d->isairq[0] != -1) { 188 monitor_printf(mon, "%*sisa irq %d\n", indent, "", 189 d->isairq[0]); 190 } 191 } 192 193 static void isabus_bridge_class_init(ObjectClass *klass, void *data) 194 { 195 DeviceClass *dc = DEVICE_CLASS(klass); 196 197 dc->fw_name = "isa"; 198 } 199 200 static const TypeInfo isabus_bridge_info = { 201 .name = "isabus-bridge", 202 .parent = TYPE_SYS_BUS_DEVICE, 203 .instance_size = sizeof(SysBusDevice), 204 .class_init = isabus_bridge_class_init, 205 }; 206 207 static void isa_device_class_init(ObjectClass *klass, void *data) 208 { 209 DeviceClass *k = DEVICE_CLASS(klass); 210 k->bus_type = TYPE_ISA_BUS; 211 } 212 213 static const TypeInfo isa_device_type_info = { 214 .name = TYPE_ISA_DEVICE, 215 .parent = TYPE_DEVICE, 216 .instance_size = sizeof(ISADevice), 217 .instance_init = isa_device_init, 218 .abstract = true, 219 .class_size = sizeof(ISADeviceClass), 220 .class_init = isa_device_class_init, 221 }; 222 223 static void isabus_register_types(void) 224 { 225 type_register_static(&isa_bus_info); 226 type_register_static(&isabus_bridge_info); 227 type_register_static(&isa_device_type_info); 228 } 229 230 static char *isabus_get_fw_dev_path(DeviceState *dev) 231 { 232 ISADevice *d = ISA_DEVICE(dev); 233 char path[40]; 234 int off; 235 236 off = snprintf(path, sizeof(path), "%s", qdev_fw_name(dev)); 237 if (d->ioport_id) { 238 snprintf(path + off, sizeof(path) - off, "@%04x", d->ioport_id); 239 } 240 241 return g_strdup(path); 242 } 243 244 MemoryRegion *isa_address_space(ISADevice *dev) 245 { 246 if (dev) { 247 return isa_bus_from_device(dev)->address_space; 248 } 249 250 return isabus->address_space; 251 } 252 253 MemoryRegion *isa_address_space_io(ISADevice *dev) 254 { 255 if (dev) { 256 return isa_bus_from_device(dev)->address_space_io; 257 } 258 259 return isabus->address_space_io; 260 } 261 262 type_init(isabus_register_types) 263 264 static void parallel_init(ISABus *bus, int index, CharDriverState *chr) 265 { 266 DeviceState *dev; 267 ISADevice *isadev; 268 269 isadev = isa_create(bus, "isa-parallel"); 270 dev = DEVICE(isadev); 271 qdev_prop_set_uint32(dev, "index", index); 272 qdev_prop_set_chr(dev, "chardev", chr); 273 qdev_init_nofail(dev); 274 } 275 276 void parallel_hds_isa_init(ISABus *bus, int n) 277 { 278 int i; 279 280 assert(n <= MAX_PARALLEL_PORTS); 281 282 for (i = 0; i < n; i++) { 283 if (parallel_hds[i]) { 284 parallel_init(bus, i, parallel_hds[i]); 285 } 286 } 287 } 288