2 * QEMU Intel i82378 emulation (PCI to ISA bridge)
4 * Copyright (c) 2010-2011 Hervé Poussineau
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.
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.
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/>.
23 //#define DEBUG_I82378
26 #define DPRINTF(fmt, ...) \
27 do { fprintf(stderr, "i82378: " fmt , ## __VA_ARGS__); } while (0)
29 #define DPRINTF(fmt, ...) \
33 #define BADF(fmt, ...) \
34 do { fprintf(stderr, "i82378 ERROR: " fmt , ## __VA_ARGS__); } while (0)
36 typedef struct I82378State
{
43 typedef struct PCIi82378State
{
46 uint32_t isa_mem_base
;
50 static const VMStateDescription vmstate_pci_i82378
= {
53 .minimum_version_id
= 0,
54 .fields
= (VMStateField
[]) {
55 VMSTATE_PCI_DEVICE(pci_dev
, PCIi82378State
),
60 static void i82378_io_write(void *opaque
, target_phys_addr_t addr
,
61 uint64_t value
, unsigned int size
)
65 DPRINTF("%s: " TARGET_FMT_plx
"=%02" PRIx64
"\n", __func__
,
67 cpu_outb(addr
, value
);
70 DPRINTF("%s: " TARGET_FMT_plx
"=%04" PRIx64
"\n", __func__
,
72 cpu_outw(addr
, value
);
75 DPRINTF("%s: " TARGET_FMT_plx
"=%08" PRIx64
"\n", __func__
,
77 cpu_outl(addr
, value
);
84 static uint64_t i82378_io_read(void *opaque
, target_phys_addr_t addr
,
87 DPRINTF("%s: " TARGET_FMT_plx
"\n", __func__
, addr
);
100 static const MemoryRegionOps i82378_io_ops
= {
101 .read
= i82378_io_read
,
102 .write
= i82378_io_write
,
103 .endianness
= DEVICE_LITTLE_ENDIAN
,
106 static void i82378_mem_write(void *opaque
, target_phys_addr_t addr
,
107 uint64_t value
, unsigned int size
)
111 DPRINTF("%s: " TARGET_FMT_plx
"=%02" PRIx64
"\n", __func__
,
113 cpu_outb(addr
, value
);
116 DPRINTF("%s: " TARGET_FMT_plx
"=%04" PRIx64
"\n", __func__
,
118 cpu_outw(addr
, value
);
121 DPRINTF("%s: " TARGET_FMT_plx
"=%08" PRIx64
"\n", __func__
,
123 cpu_outl(addr
, value
);
130 static uint64_t i82378_mem_read(void *opaque
, target_phys_addr_t addr
,
133 DPRINTF("%s: " TARGET_FMT_plx
"\n", __func__
, addr
);
136 return cpu_inb(addr
);
138 return cpu_inw(addr
);
140 return cpu_inl(addr
);
146 static const MemoryRegionOps i82378_mem_ops
= {
147 .read
= i82378_mem_read
,
148 .write
= i82378_mem_write
,
149 .endianness
= DEVICE_LITTLE_ENDIAN
,
152 static void i82378_request_out0_irq(void *opaque
, int irq
, int level
)
154 I82378State
*s
= opaque
;
155 qemu_set_irq(s
->out
[0], level
);
158 static void i82378_request_pic_irq(void *opaque
, int irq
, int level
)
160 DeviceState
*dev
= opaque
;
161 PCIDevice
*pci
= DO_UPCAST(PCIDevice
, qdev
, dev
);
162 PCIi82378State
*s
= DO_UPCAST(PCIi82378State
, pci_dev
, pci
);
164 qemu_set_irq(s
->state
.i8259
[irq
], level
);
167 static void i82378_init(DeviceState
*dev
, I82378State
*s
)
169 ISABus
*isabus
= DO_UPCAST(ISABus
, qbus
, qdev_get_child_bus(dev
, "isa.0"));
178 Utility Bus Support Registers
180 All devices accept byte access only, except timer
183 qdev_init_gpio_out(dev
, s
->out
, 2);
184 qdev_init_gpio_in(dev
, i82378_request_pic_irq
, 16);
186 /* Workaround the fact that i8259 is not qdev'ified... */
187 out0_irq
= qemu_allocate_irqs(i82378_request_out0_irq
, s
, 1);
190 s
->i8259
= i8259_init(isabus
, *out0_irq
);
191 isa_bus_irqs(isabus
, s
->i8259
);
194 pit
= pit_init(isabus
, 0x40, 0);
200 DMA_init(1, &s
->out
[1]);
201 isa_create_simple(isabus
, "i82374");
204 isa_create_simple(isabus
, "mc146818rtc");
207 static int pci_i82378_init(PCIDevice
*dev
)
209 PCIi82378State
*pci
= DO_UPCAST(PCIi82378State
, pci_dev
, dev
);
210 I82378State
*s
= &pci
->state
;
213 pci_conf
= dev
->config
;
214 pci_set_word(pci_conf
+ PCI_COMMAND
,
215 PCI_COMMAND_IO
| PCI_COMMAND_MEMORY
| PCI_COMMAND_MASTER
);
216 pci_set_word(pci_conf
+ PCI_STATUS
,
217 PCI_STATUS_DEVSEL_MEDIUM
);
219 pci_conf
[PCI_INTERRUPT_PIN
] = 1; /* interrupt pin 0 */
221 memory_region_init_io(&s
->io
, &i82378_io_ops
, s
, "i82378-io", 0x00010000);
222 pci_register_bar(dev
, 0, PCI_BASE_ADDRESS_SPACE_MEMORY
, &s
->io
);
224 memory_region_init_io(&s
->mem
, &i82378_mem_ops
, s
, "i82378-mem", 0x01000000);
225 memory_region_set_coalescing(&s
->mem
);
226 pci_register_bar(dev
, 1, PCI_BASE_ADDRESS_SPACE_MEMORY
, &s
->mem
);
228 /* Make I/O address read only */
229 pci_set_word(dev
->wmask
+ PCI_COMMAND
, PCI_COMMAND_SPECIAL
);
230 pci_set_long(dev
->wmask
+ PCI_BASE_ADDRESS_0
, 0);
231 pci_set_long(pci_conf
+ PCI_BASE_ADDRESS_0
, pci
->isa_io_base
);
233 isa_mem_base
= pci
->isa_mem_base
;
234 isa_bus_new(&dev
->qdev
, pci_address_space_io(dev
));
236 i82378_init(&dev
->qdev
, s
);
241 static Property i82378_properties
[] = {
242 DEFINE_PROP_HEX32("iobase", PCIi82378State
, isa_io_base
, 0x80000000),
243 DEFINE_PROP_HEX32("membase", PCIi82378State
, isa_mem_base
, 0xc0000000),
244 DEFINE_PROP_END_OF_LIST()
247 static void pci_i82378_class_init(ObjectClass
*klass
, void *data
)
249 PCIDeviceClass
*k
= PCI_DEVICE_CLASS(klass
);
250 DeviceClass
*dc
= DEVICE_CLASS(klass
);
252 k
->init
= pci_i82378_init
;
253 k
->vendor_id
= PCI_VENDOR_ID_INTEL
;
254 k
->device_id
= PCI_DEVICE_ID_INTEL_82378
;
256 k
->class_id
= PCI_CLASS_BRIDGE_ISA
;
257 k
->subsystem_vendor_id
= 0x0;
258 k
->subsystem_id
= 0x0;
259 dc
->vmsd
= &vmstate_pci_i82378
;
260 dc
->props
= i82378_properties
;
263 static TypeInfo pci_i82378_info
= {
265 .parent
= TYPE_PCI_DEVICE
,
266 .instance_size
= sizeof(PCIi82378State
),
267 .class_init
= pci_i82378_class_init
,
270 static void i82378_register_devices(void)
272 type_register_static(&pci_i82378_info
);
275 device_init(i82378_register_devices
)