mirror of
https://github.com/qemu/qemu.git
synced 2024-12-13 14:33:31 +08:00
80bb2ff770
This is a DP264 variant, SMP capable, no unusual hardware present. The emulation does not currently include any PCI IOMMU code. Hopefully the generic support for that can be merged to HEAD soon. Signed-off-by: Richard Henderson <rth@twiddle.net>
135 lines
3.0 KiB
C
135 lines
3.0 KiB
C
/*
|
||
* QEMU Alpha PCI support functions.
|
||
*
|
||
* Some of this isn't very Alpha specific at all.
|
||
*
|
||
* ??? Sparse memory access not implemented.
|
||
*/
|
||
|
||
#include "config.h"
|
||
#include "alpha_sys.h"
|
||
#include "qemu-log.h"
|
||
#include "sysemu.h"
|
||
#include "vmware_vga.h"
|
||
|
||
|
||
/* PCI IO reads/writes, to byte-word addressable memory. */
|
||
/* ??? Doesn't handle multiple PCI busses. */
|
||
|
||
static uint64_t bw_io_read(void *opaque, target_phys_addr_t addr, unsigned size)
|
||
{
|
||
switch (size) {
|
||
case 1:
|
||
return cpu_inb(addr);
|
||
case 2:
|
||
return cpu_inw(addr);
|
||
case 4:
|
||
return cpu_inl(addr);
|
||
}
|
||
abort();
|
||
}
|
||
|
||
static void bw_io_write(void *opaque, target_phys_addr_t addr,
|
||
uint64_t val, unsigned size)
|
||
{
|
||
switch (size) {
|
||
case 1:
|
||
cpu_outb(addr, val);
|
||
break;
|
||
case 2:
|
||
cpu_outw(addr, val);
|
||
break;
|
||
case 4:
|
||
cpu_outl(addr, val);
|
||
break;
|
||
default:
|
||
abort();
|
||
}
|
||
}
|
||
|
||
const MemoryRegionOps alpha_pci_bw_io_ops = {
|
||
.read = bw_io_read,
|
||
.write = bw_io_write,
|
||
.endianness = DEVICE_LITTLE_ENDIAN,
|
||
.impl = {
|
||
.min_access_size = 1,
|
||
.max_access_size = 4,
|
||
},
|
||
};
|
||
|
||
/* PCI config space reads/writes, to byte-word addressable memory. */
|
||
static uint64_t bw_conf1_read(void *opaque, target_phys_addr_t addr,
|
||
unsigned size)
|
||
{
|
||
PCIBus *b = opaque;
|
||
return pci_data_read(b, addr, size);
|
||
}
|
||
|
||
static void bw_conf1_write(void *opaque, target_phys_addr_t addr,
|
||
uint64_t val, unsigned size)
|
||
{
|
||
PCIBus *b = opaque;
|
||
pci_data_write(b, addr, val, size);
|
||
}
|
||
|
||
const MemoryRegionOps alpha_pci_conf1_ops = {
|
||
.read = bw_conf1_read,
|
||
.write = bw_conf1_write,
|
||
.endianness = DEVICE_LITTLE_ENDIAN,
|
||
.impl = {
|
||
.min_access_size = 1,
|
||
.max_access_size = 4,
|
||
},
|
||
};
|
||
|
||
/* PCI/EISA Interrupt Acknowledge Cycle. */
|
||
|
||
static uint64_t iack_read(void *opaque, target_phys_addr_t addr, unsigned size)
|
||
{
|
||
return pic_read_irq(isa_pic);
|
||
}
|
||
|
||
static void special_write(void *opaque, target_phys_addr_t addr,
|
||
uint64_t val, unsigned size)
|
||
{
|
||
qemu_log("pci: special write cycle");
|
||
}
|
||
|
||
const MemoryRegionOps alpha_pci_iack_ops = {
|
||
.read = iack_read,
|
||
.write = special_write,
|
||
.endianness = DEVICE_LITTLE_ENDIAN,
|
||
.valid = {
|
||
.min_access_size = 4,
|
||
.max_access_size = 4,
|
||
},
|
||
.impl = {
|
||
.min_access_size = 4,
|
||
.max_access_size = 4,
|
||
},
|
||
};
|
||
|
||
void alpha_pci_vga_setup(PCIBus *pci_bus)
|
||
{
|
||
switch (vga_interface_type) {
|
||
#ifdef CONFIG_SPICE
|
||
case VGA_QXL:
|
||
pci_create_simple(pci_bus, -1, "qxl-vga");
|
||
return;
|
||
#endif
|
||
case VGA_CIRRUS:
|
||
pci_cirrus_vga_init(pci_bus);
|
||
return;
|
||
case VGA_VMWARE:
|
||
if (pci_vmsvga_init(pci_bus)) {
|
||
return;
|
||
}
|
||
break;
|
||
}
|
||
/* If VGA is enabled at all, and one of the above didn't work, then
|
||
fallback to Standard VGA. */
|
||
if (vga_interface_type != VGA_NONE) {
|
||
pci_vga_init(pci_bus);
|
||
}
|
||
}
|