diff options
author | Paolo Bonzini <pbonzini@redhat.com> | 2015-05-25 11:06:01 +0200 |
---|---|---|
committer | Paolo Bonzini <pbonzini@redhat.com> | 2015-05-25 13:27:03 +0200 |
commit | 4a020bbc0dd5e6025f92b47f829561df249e6cc6 (patch) | |
tree | fef4bd8591e88242a0f03956815276b9d6f9b6e3 /hwsetup.c | |
parent | c4b724bd7ca65b8945ef683f33c0aa06152a381f (diff) | |
download | qboot-4a020bbc0dd5e6025f92b47f829561df249e6cc6.zip qboot-4a020bbc0dd5e6025f92b47f829561df249e6cc6.tar.gz qboot-4a020bbc0dd5e6025f92b47f829561df249e6cc6.tar.bz2 |
initialize the southbridge
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Diffstat (limited to 'hwsetup.c')
-rw-r--r-- | hwsetup.c | 121 |
1 files changed, 121 insertions, 0 deletions
diff --git a/hwsetup.c b/hwsetup.c new file mode 100644 index 0000000..77c6dec --- /dev/null +++ b/hwsetup.c @@ -0,0 +1,121 @@ +#include "bios.h" +#include "ioport.h" +#include "pci.h" +#include "string.h" + +// NOTE: this runs from ROM at 0xFFFF0000, so it is not possible to use any +// static data. + +#define PIIX_ISA_PIRQA_ROUT 0x60 +#define PIIX_PMBASE 0x40 +#define PIIX_PMREGMISC 0x80 +#define PIIX_SMBHSTBASE 0x90 +#define PIIX_SMBHSTCFG 0xd2 + +static void setup_piix(void) +{ + const int bdf = (1 << 3); + pci_config_writeb(bdf, PIIX_ISA_PIRQA_ROUT, 10); + pci_config_writeb(bdf, PIIX_ISA_PIRQA_ROUT+1, 10); + pci_config_writeb(bdf, PIIX_ISA_PIRQA_ROUT+2, 11); + pci_config_writeb(bdf, PIIX_ISA_PIRQA_ROUT+3, 11); +} + +static void setup_piix_pm(void) +{ + const int bdf = (1 << 3) | 3; + + pci_config_writel(bdf, PIIX_PMBASE, 0x601); + pci_config_writeb(bdf, PIIX_PMREGMISC, 0x01); + pci_config_writel(bdf, PIIX_SMBHSTBASE, 0x701); + pci_config_writeb(bdf, PIIX_SMBHSTCFG, 0x09); +} + +#define ICH9_LPC_PIRQA_ROUT 0x60 +#define ICH9_LPC_PIRQE_ROUT 0x68 +#define ICH9_LPC_PMBASE 0x40 +#define ICH9_LPC_ACPI_CTRL 0x44 + +static void setup_ich9(void) +{ + const int bdf = 0x1f << 3; + pci_config_writeb(bdf, ICH9_LPC_PIRQA_ROUT, 10); + pci_config_writeb(bdf, ICH9_LPC_PIRQA_ROUT+1, 10); + pci_config_writeb(bdf, ICH9_LPC_PIRQA_ROUT+2, 11); + pci_config_writeb(bdf, ICH9_LPC_PIRQA_ROUT+3, 11); + pci_config_writeb(bdf, ICH9_LPC_PIRQE_ROUT, 10); + pci_config_writeb(bdf, ICH9_LPC_PIRQE_ROUT+1, 10); + pci_config_writeb(bdf, ICH9_LPC_PIRQE_ROUT+2, 11); + pci_config_writeb(bdf, ICH9_LPC_PIRQE_ROUT+3, 11); +} + +static void setup_ich9_pm(void) +{ + const int bdf = 0x1f << 3; + pci_config_writel(bdf, ICH9_LPC_PMBASE, 0x601); + pci_config_writeb(bdf, ICH9_LPC_ACPI_CTRL, 0x80); +} + +#define PCI_VENDOR_ID_INTEL 0x8086 +#define PCI_DEVICE_ID_INTEL_82441 0x1237 +#define PCI_DEVICE_ID_INTEL_Q35_MCH 0x29c0 + +#define I440FX_PAM0 0x59 +#define Q35_HOST_BRIDGE_PAM0 0x90 + +static void setup_pic(void) +{ + /* Send ICW1 (select OCW1 + will send ICW4) */ + outb(0x20, 0x11); + outb(0xa0, 0x11); + /* Send ICW2 (base irqs: 0x08-0x0f for irq0-7, 0x70-0x77 for irq8-15) */ + outb(0x21, 8); + outb(0xa1, 0x70); + /* Send ICW3 (cascaded pic ids) */ + outb(0x21, 0x04); + outb(0xa1, 0x02); + /* Send ICW4 (enable 8086 mode) */ + outb(0x21, 0x01); + outb(0xa1, 0x01); + /* Mask all irqs (except cascaded PIC2 irq) */ + outb(0x21, ~(1 << 2)); + outb(0xa1, ~0); + + /* Set ELCR to IRQs 10 and 11 */ + outb(0x4d0, 0); + outb(0x4d0, 0x0c); +} + +void setup_hw(void) +{ + const int bdf = 0; + const uint8_t *bios_start = (uint8_t *)0xffff0000; + uint8_t *low_start = (uint8_t *)0xf0000; + int pambase; + + uint32_t id = pci_config_readl(bdf, 0); + if (id == (PCI_VENDOR_ID_INTEL | (PCI_DEVICE_ID_INTEL_82441 << 16))) { + setup_piix(); + setup_piix_pm(); + pambase = I440FX_PAM0; + } else if (id == (PCI_VENDOR_ID_INTEL | (PCI_DEVICE_ID_INTEL_Q35_MCH << 16))) { + setup_ich9(); + setup_ich9_pm(); + pambase = Q35_HOST_BRIDGE_PAM0; + } else + panic(); + + // Make ram from 0xc0000-0xf0000 read-write + int i; + for (i=0; i<6; i++) { + int pam = pambase + 1 + i; + pci_config_writeb(bdf, pam, 0x33); + } + + // Make ram from 0xf0000-0x100000 read-write and shadow BIOS + // We're still running from 0xffff0000 + pci_config_writeb(bdf, pambase, 0x30); + memcpy(low_start, bios_start, 0x10000); + + setup_pic(); +} |