xref: /openbmc/qemu/hw/isa/isa-bus.c (revision d341d9f3)
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