sun4u: update PCI topology to include simba PCI bridges
This patch updates the sun4u model to being much closer to a real Ultra 5 by moving devices behind the 2 simba PCI bridges (A and B) as found on real hardware. The most noticeable change introduced by this patchset is that in-built devices are no longer attached to the PCI root bus, but instead behind PCI bridge A. Along with this the interrupt routing is updated accordingly to match the official documentation. Since the existing code currently bypasses the PCI bridge interrupt swizzling, the interrupt mapping functions are reorganised so that pci_pbm_map_irq() is used by the PCI bridges and pci_apb_map_irq() is used by the PCI host bridge. Behind the sabre PCI host bridge, the PCI IO space now needs to be split into two separate halves at 0x8000000. Therefore we also setup a new PCI IO space region of increased size on the PCI host bridge and enable 32-bit PCI IO accesses to allow IO accesses to reach devices behind PCI bridge B correctly. As part of this change we also combine the onboard sunhme NIC and the ebus into a single multi-function device as done on a real Ultra 5. For other NICs the existing behaviour is preserved, i.e. we initialise them and place them into the next free slot on PCI bus B. Finally we mark the physically unavailable slots (plus slot 0 in busA) as reserved to ensure that users can't plug devices into non-existent slots which will break interrupt routing. Note: since this commit changes PCI topology and interrupt routing, an updated openbios-sparc64 binary is included with this commit containing the associated changes to maintain bisectability. Signed-off-by: Mark Cave-Ayland <mark.cave-ayland@ilande.co.uk> Reviewed-by: Artyom Tarasenko <atar4qemu@gmail.com>
This commit is contained in:
parent
861cd431c9
commit
6864fa3897
@ -599,16 +599,35 @@ static uint64_t apb_pci_config_read(void *opaque, hwaddr addr,
|
|||||||
/* The APB host has an IRQ line for each IRQ line of each slot. */
|
/* The APB host has an IRQ line for each IRQ line of each slot. */
|
||||||
static int pci_apb_map_irq(PCIDevice *pci_dev, int irq_num)
|
static int pci_apb_map_irq(PCIDevice *pci_dev, int irq_num)
|
||||||
{
|
{
|
||||||
return ((pci_dev->devfn & 0x18) >> 1) + irq_num;
|
/* Return the irq as swizzled by the PBM */
|
||||||
|
return irq_num;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int pci_pbm_map_irq(PCIDevice *pci_dev, int irq_num)
|
static int pci_pbm_map_irq(PCIDevice *pci_dev, int irq_num)
|
||||||
{
|
{
|
||||||
|
PBMPCIBridge *br = PBM_PCI_BRIDGE(pci_bridge_get_device(
|
||||||
|
PCI_BUS(qdev_get_parent_bus(DEVICE(pci_dev)))));
|
||||||
|
|
||||||
int bus_offset;
|
int bus_offset;
|
||||||
if (pci_dev->devfn & 1)
|
if (br->busA) {
|
||||||
bus_offset = 16;
|
bus_offset = 0x0;
|
||||||
else
|
|
||||||
bus_offset = 0;
|
/* The on-board devices have fixed (legacy) OBIO intnos */
|
||||||
|
switch (PCI_SLOT(pci_dev->devfn)) {
|
||||||
|
case 1:
|
||||||
|
/* Onboard NIC */
|
||||||
|
return 0x21;
|
||||||
|
case 3:
|
||||||
|
/* Onboard IDE */
|
||||||
|
return 0x20;
|
||||||
|
|
||||||
|
default:
|
||||||
|
/* Normal intno, fall through */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
bus_offset = 0x10;
|
||||||
|
}
|
||||||
return (bus_offset + (PCI_SLOT(pci_dev->devfn) << 2) + irq_num) & 0x1f;
|
return (bus_offset + (PCI_SLOT(pci_dev->devfn) << 2) + irq_num) & 0x1f;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -669,6 +688,12 @@ static void apb_pci_bridge_realize(PCIDevice *dev, Error **errp)
|
|||||||
PCI_STATUS_FAST_BACK | PCI_STATUS_66MHZ |
|
PCI_STATUS_FAST_BACK | PCI_STATUS_66MHZ |
|
||||||
PCI_STATUS_DEVSEL_MEDIUM);
|
PCI_STATUS_DEVSEL_MEDIUM);
|
||||||
|
|
||||||
|
/* Allow 32-bit IO addresses */
|
||||||
|
pci_set_word(dev->config + PCI_IO_BASE, PCI_IO_RANGE_TYPE_32);
|
||||||
|
pci_set_word(dev->config + PCI_IO_LIMIT, PCI_IO_RANGE_TYPE_32);
|
||||||
|
pci_set_word(dev->wmask + PCI_IO_BASE_UPPER16, 0xffff);
|
||||||
|
pci_set_word(dev->wmask + PCI_IO_LIMIT_UPPER16, 0xffff);
|
||||||
|
|
||||||
pci_bridge_update_mappings(PCI_BRIDGE(br));
|
pci_bridge_update_mappings(PCI_BRIDGE(br));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -690,9 +715,9 @@ PCIBus *pci_apb_init(hwaddr special_base,
|
|||||||
d = APB_DEVICE(dev);
|
d = APB_DEVICE(dev);
|
||||||
phb = PCI_HOST_BRIDGE(dev);
|
phb = PCI_HOST_BRIDGE(dev);
|
||||||
phb->bus = pci_register_bus(DEVICE(phb), "pci",
|
phb->bus = pci_register_bus(DEVICE(phb), "pci",
|
||||||
pci_apb_set_irq, pci_pbm_map_irq, d,
|
pci_apb_set_irq, pci_apb_map_irq, d,
|
||||||
&d->pci_mmio,
|
&d->pci_mmio,
|
||||||
get_system_io(),
|
&d->pci_ioport,
|
||||||
0, 32, TYPE_PCI_BUS);
|
0, 32, TYPE_PCI_BUS);
|
||||||
qdev_init_nofail(dev);
|
qdev_init_nofail(dev);
|
||||||
s = SYS_BUS_DEVICE(dev);
|
s = SYS_BUS_DEVICE(dev);
|
||||||
@ -725,14 +750,14 @@ PCIBus *pci_apb_init(hwaddr special_base,
|
|||||||
pci_dev = pci_create_multifunction(phb->bus, PCI_DEVFN(1, 0), true,
|
pci_dev = pci_create_multifunction(phb->bus, PCI_DEVFN(1, 0), true,
|
||||||
TYPE_PBM_PCI_BRIDGE);
|
TYPE_PBM_PCI_BRIDGE);
|
||||||
br = PCI_BRIDGE(pci_dev);
|
br = PCI_BRIDGE(pci_dev);
|
||||||
pci_bridge_map_irq(br, "pciB", pci_apb_map_irq);
|
pci_bridge_map_irq(br, "pciB", pci_pbm_map_irq);
|
||||||
qdev_init_nofail(&pci_dev->qdev);
|
qdev_init_nofail(&pci_dev->qdev);
|
||||||
*busB = pci_bridge_get_sec_bus(br);
|
*busB = pci_bridge_get_sec_bus(br);
|
||||||
|
|
||||||
pci_dev = pci_create_multifunction(phb->bus, PCI_DEVFN(1, 1), true,
|
pci_dev = pci_create_multifunction(phb->bus, PCI_DEVFN(1, 1), true,
|
||||||
TYPE_PBM_PCI_BRIDGE);
|
TYPE_PBM_PCI_BRIDGE);
|
||||||
br = PCI_BRIDGE(pci_dev);
|
br = PCI_BRIDGE(pci_dev);
|
||||||
pci_bridge_map_irq(br, "pciA", pci_apb_map_irq);
|
pci_bridge_map_irq(br, "pciA", pci_pbm_map_irq);
|
||||||
qdev_prop_set_bit(DEVICE(pci_dev), "busA", true);
|
qdev_prop_set_bit(DEVICE(pci_dev), "busA", true);
|
||||||
qdev_init_nofail(&pci_dev->qdev);
|
qdev_init_nofail(&pci_dev->qdev);
|
||||||
*busA = pci_bridge_get_sec_bus(br);
|
*busA = pci_bridge_get_sec_bus(br);
|
||||||
@ -798,8 +823,8 @@ static int pci_pbm_init_device(SysBusDevice *dev)
|
|||||||
sysbus_init_mmio(dev, &s->pci_config);
|
sysbus_init_mmio(dev, &s->pci_config);
|
||||||
|
|
||||||
/* pci_ioport */
|
/* pci_ioport */
|
||||||
memory_region_init_alias(&s->pci_ioport, OBJECT(s), "apb-pci-ioport",
|
memory_region_init(&s->pci_ioport, OBJECT(s), "apb-pci-ioport", 0x1000000);
|
||||||
get_system_io(), 0, 0x10000);
|
|
||||||
/* at region 2 */
|
/* at region 2 */
|
||||||
sysbus_init_mmio(dev, &s->pci_ioport);
|
sysbus_init_mmio(dev, &s->pci_ioport);
|
||||||
|
|
||||||
|
@ -27,6 +27,7 @@
|
|||||||
#include "cpu.h"
|
#include "cpu.h"
|
||||||
#include "hw/hw.h"
|
#include "hw/hw.h"
|
||||||
#include "hw/pci/pci.h"
|
#include "hw/pci/pci.h"
|
||||||
|
#include "hw/pci/pci_bus.h"
|
||||||
#include "hw/pci-host/apb.h"
|
#include "hw/pci-host/apb.h"
|
||||||
#include "hw/i386/pc.h"
|
#include "hw/i386/pc.h"
|
||||||
#include "hw/char/serial.h"
|
#include "hw/char/serial.h"
|
||||||
@ -42,6 +43,7 @@
|
|||||||
#include "hw/nvram/fw_cfg.h"
|
#include "hw/nvram/fw_cfg.h"
|
||||||
#include "hw/sysbus.h"
|
#include "hw/sysbus.h"
|
||||||
#include "hw/ide.h"
|
#include "hw/ide.h"
|
||||||
|
#include "hw/ide/pci.h"
|
||||||
#include "hw/loader.h"
|
#include "hw/loader.h"
|
||||||
#include "elf.h"
|
#include "elf.h"
|
||||||
#include "qemu/cutils.h"
|
#include "qemu/cutils.h"
|
||||||
@ -440,7 +442,8 @@ static void sun4uv_init(MemoryRegion *address_space_mem,
|
|||||||
DeviceState *dev;
|
DeviceState *dev;
|
||||||
FWCfgState *fw_cfg;
|
FWCfgState *fw_cfg;
|
||||||
NICInfo *nd;
|
NICInfo *nd;
|
||||||
int onboard_nic_idx;
|
MACAddr macaddr;
|
||||||
|
bool onboard_nic;
|
||||||
|
|
||||||
/* init CPUs */
|
/* init CPUs */
|
||||||
cpu = sparc64_cpu_devinit(machine->cpu_model, hwdef->default_cpu_model,
|
cpu = sparc64_cpu_devinit(machine->cpu_model, hwdef->default_cpu_model,
|
||||||
@ -454,10 +457,17 @@ static void sun4uv_init(MemoryRegion *address_space_mem,
|
|||||||
ivec_irqs = qemu_allocate_irqs(sparc64_cpu_set_ivec_irq, cpu, IVEC_MAX);
|
ivec_irqs = qemu_allocate_irqs(sparc64_cpu_set_ivec_irq, cpu, IVEC_MAX);
|
||||||
pci_bus = pci_apb_init(APB_SPECIAL_BASE, APB_MEM_BASE, ivec_irqs, &pci_busA,
|
pci_bus = pci_apb_init(APB_SPECIAL_BASE, APB_MEM_BASE, ivec_irqs, &pci_busA,
|
||||||
&pci_busB, &pbm_irqs);
|
&pci_busB, &pbm_irqs);
|
||||||
pci_vga_init(pci_bus);
|
|
||||||
|
|
||||||
/* XXX Should be pci_busA */
|
/* Only in-built Simba PBMs can exist on the root bus, slot 0 on busA is
|
||||||
ebus = pci_create_simple(pci_bus, -1, "ebus");
|
reserved (leaving no slots free after on-board devices) however slots
|
||||||
|
0-3 are free on busB */
|
||||||
|
pci_bus->slot_reserved_mask = 0xfffffffc;
|
||||||
|
pci_busA->slot_reserved_mask = 0xfffffff1;
|
||||||
|
pci_busB->slot_reserved_mask = 0xfffffff0;
|
||||||
|
|
||||||
|
ebus = pci_create_multifunction(pci_busA, PCI_DEVFN(1, 0), true, "ebus");
|
||||||
|
qdev_init_nofail(DEVICE(ebus));
|
||||||
|
|
||||||
isa_bus = pci_ebus_init(ebus, pbm_irqs);
|
isa_bus = pci_ebus_init(ebus, pbm_irqs);
|
||||||
|
|
||||||
i = 0;
|
i = 0;
|
||||||
@ -470,27 +480,43 @@ static void sun4uv_init(MemoryRegion *address_space_mem,
|
|||||||
serial_hds_isa_init(isa_bus, i, MAX_SERIAL_PORTS);
|
serial_hds_isa_init(isa_bus, i, MAX_SERIAL_PORTS);
|
||||||
parallel_hds_isa_init(isa_bus, MAX_PARALLEL_PORTS);
|
parallel_hds_isa_init(isa_bus, MAX_PARALLEL_PORTS);
|
||||||
|
|
||||||
onboard_nic_idx = -1;
|
pci_dev = pci_create_simple(pci_busA, PCI_DEVFN(2, 0), "VGA");
|
||||||
|
|
||||||
|
memset(&macaddr, 0, sizeof(MACAddr));
|
||||||
|
onboard_nic = false;
|
||||||
for (i = 0; i < nb_nics; i++) {
|
for (i = 0; i < nb_nics; i++) {
|
||||||
nd = &nd_table[i];
|
nd = &nd_table[i];
|
||||||
|
|
||||||
if (onboard_nic_idx == -1 &&
|
if (!nd->model || strcmp(nd->model, "sunhme") == 0) {
|
||||||
(!nd->model || strcmp(nd->model, "sunhme") == 0)) {
|
if (!onboard_nic) {
|
||||||
pci_dev = pci_create(pci_bus, -1, "sunhme");
|
pci_dev = pci_create_multifunction(pci_busA, PCI_DEVFN(1, 1),
|
||||||
|
true, "sunhme");
|
||||||
|
memcpy(&macaddr, &nd->macaddr.a, sizeof(MACAddr));
|
||||||
|
onboard_nic = true;
|
||||||
|
} else {
|
||||||
|
pci_dev = pci_create_simple(pci_busB, -1, "sunhme");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
pci_dev = pci_create_simple(pci_busB, -1, nd->model);
|
||||||
|
}
|
||||||
|
|
||||||
dev = &pci_dev->qdev;
|
dev = &pci_dev->qdev;
|
||||||
qdev_set_nic_properties(dev, nd);
|
qdev_set_nic_properties(dev, nd);
|
||||||
qdev_init_nofail(dev);
|
qdev_init_nofail(dev);
|
||||||
|
}
|
||||||
|
|
||||||
onboard_nic_idx = i;
|
/* If we don't have an onboard NIC, grab a default MAC address so that
|
||||||
} else {
|
* we have a valid machine id */
|
||||||
pci_nic_init_nofail(nd, pci_bus, "ne2k_pci", NULL);
|
if (!onboard_nic) {
|
||||||
|
qemu_macaddr_default_if_unset(&macaddr);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
onboard_nic_idx = MAX(onboard_nic_idx, 0);
|
|
||||||
|
|
||||||
ide_drive_get(hd, ARRAY_SIZE(hd));
|
ide_drive_get(hd, ARRAY_SIZE(hd));
|
||||||
|
|
||||||
pci_cmd646_ide_init(pci_bus, hd, 1);
|
pci_dev = pci_create(pci_busA, PCI_DEVFN(3, 0), "cmd646-ide");
|
||||||
|
qdev_prop_set_uint32(&pci_dev->qdev, "secondary", 1);
|
||||||
|
qdev_init_nofail(&pci_dev->qdev);
|
||||||
|
pci_ide_create_devs(pci_dev, hd);
|
||||||
|
|
||||||
isa_create_simple(isa_bus, "i8042");
|
isa_create_simple(isa_bus, "i8042");
|
||||||
|
|
||||||
@ -531,7 +557,7 @@ static void sun4uv_init(MemoryRegion *address_space_mem,
|
|||||||
/* XXX: need an option to load a NVRAM image */
|
/* XXX: need an option to load a NVRAM image */
|
||||||
0,
|
0,
|
||||||
graphic_width, graphic_height, graphic_depth,
|
graphic_width, graphic_height, graphic_depth,
|
||||||
(uint8_t *)&nd_table[onboard_nic_idx].macaddr);
|
(uint8_t *)&macaddr);
|
||||||
|
|
||||||
dev = qdev_create(NULL, TYPE_FW_CFG_IO);
|
dev = qdev_create(NULL, TYPE_FW_CFG_IO);
|
||||||
qdev_prop_set_bit(dev, "dma_enabled", false);
|
qdev_prop_set_bit(dev, "dma_enabled", false);
|
||||||
|
Binary file not shown.
Loading…
Reference in New Issue
Block a user