mirror of
				https://gitlab.com/qemu-project/qboot.git
				synced 2024-02-13 08:33:40 +08:00 
			
		
		
		
	 947bade344
			
		
	
	947bade344
	
	
	
		
			
			Instead of panicing when a PCI bus isn't found, continue booting without PCI nor ACPI initialization. Signed-off-by: Sergio Lopez <slp@redhat.com>
		
			
				
	
	
		
			161 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			161 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #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 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(0x4d1, 0x0c);
 | |
| }
 | |
| 
 | |
| void setup_pam(int bdf, int pambase)
 | |
| {
 | |
| 	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
 | |
| 	pci_config_writeb(bdf, pambase, 0x30);
 | |
| }
 | |
| 
 | |
| bool setup_hw(void)
 | |
| {
 | |
| 	const int bdf = 0;
 | |
| 	const uint8_t *bios_start = (void *)((uintptr_t)&stext + 0xfff00000);
 | |
| 	const uint8_t *init_start = (void *)((uintptr_t)&sinit + 0xfff00000);
 | |
| 	static volatile uint8_t rom_check;
 | |
| 	int rom_check_value;
 | |
| 	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 {
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	// Make ram from 0xc0000-0xf0000 read-write
 | |
| 	rom_check_value = rom_check;
 | |
| 	rom_check = rom_check_value + 1;
 | |
| 	if (rom_check == rom_check_value)
 | |
| 		setup_pam(bdf, pambase);
 | |
| 
 | |
| 	// Shadow BIOS; we're still running from 0xffff0000
 | |
| 	memcpy(&stext, bios_start, &edata - &stext);
 | |
| 	memcpy(&sinit, init_start, &einit - &sinit);
 | |
| 
 | |
| 	setup_pic();
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| #define Q35_HOST_BRIDGE_PCIEXBAREN      1
 | |
| #define Q35_HOST_BRIDGE_PCIEXBAR        0x60
 | |
| 
 | |
| static void setup_q35_mmconfig(void)
 | |
| {
 | |
| 	const int bdf = 0;
 | |
| 	uint64_t addr = PCIE_MMCONFIG_BASE;
 | |
| 	uint32_t upper = addr >> 32;
 | |
| 	uint32_t lower = (addr & 0xffffffff) | Q35_HOST_BRIDGE_PCIEXBAREN;
 | |
| 
 | |
| 	pci_config_writel(bdf, Q35_HOST_BRIDGE_PCIEXBAR, 0);
 | |
| 	pci_config_writel(bdf, Q35_HOST_BRIDGE_PCIEXBAR + 4, upper);
 | |
| 	pci_config_writel(bdf, Q35_HOST_BRIDGE_PCIEXBAR, lower);
 | |
| }
 | |
| 
 | |
| bool setup_mmconfig(void)
 | |
| {
 | |
| 	const int bdf = 0;
 | |
| 	uint32_t id = pci_config_readl(bdf, 0);
 | |
| 
 | |
| 	if (id == (PCI_VENDOR_ID_INTEL | (PCI_DEVICE_ID_INTEL_Q35_MCH << 16))) {
 | |
| 		setup_q35_mmconfig();
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	return false;
 | |
| }
 |