// PCI config space access functions. // // Copyright (C) 2008 Kevin O'Connor // Copyright (C) 2002 MandrakeSoft S.A. // // This file may be distributed under the terms of the GNU GPLv3 license. #include "pci.h" // MaxBDF #include "ioport.h" // outl #include "util.h" // dprintf #include "config.h" // CONFIG_* #include "pci_regs.h" // PCI_VENDOR_ID #include "farptr.h" // SET_VAR void pci_config_writel(u16 bdf, u32 addr, u32 val) { outl(0x80000000 | (bdf << 8) | (addr & 0xfc), PORT_PCI_CMD); outl(val, PORT_PCI_DATA); } void pci_config_writew(u16 bdf, u32 addr, u16 val) { outl(0x80000000 | (bdf << 8) | (addr & 0xfc), PORT_PCI_CMD); outw(val, PORT_PCI_DATA + (addr & 2)); } void pci_config_writeb(u16 bdf, u32 addr, u8 val) { outl(0x80000000 | (bdf << 8) | (addr & 0xfc), PORT_PCI_CMD); outb(val, PORT_PCI_DATA + (addr & 3)); } u32 pci_config_readl(u16 bdf, u32 addr) { outl(0x80000000 | (bdf << 8) | (addr & 0xfc), PORT_PCI_CMD); return inl(PORT_PCI_DATA); } u16 pci_config_readw(u16 bdf, u32 addr) { outl(0x80000000 | (bdf << 8) | (addr & 0xfc), PORT_PCI_CMD); return inw(PORT_PCI_DATA + (addr & 2)); } u8 pci_config_readb(u16 bdf, u32 addr) { outl(0x80000000 | (bdf << 8) | (addr & 0xfc), PORT_PCI_CMD); return inb(PORT_PCI_DATA + (addr & 3)); } #if MODE16 int MaxBDF VISIBLE16 = 0x0100; #endif // Find the maximum bus number. void pci_bus_setup() { dprintf(3, "Scan for max PCI bus\n"); int max = 0x0100; int bdf; for (bdf=0; bdf < max; bdf++) { u32 v = pci_config_readl(bdf, PCI_VENDOR_ID); if (v == 0xffffffff || v == 0x00000000 || v == 0x0000ffff || v == 0xffff0000) // No device present. continue; v = pci_config_readb(bdf, PCI_HEADER_TYPE); v &= 0x7f; if (v != PCI_HEADER_TYPE_BRIDGE && v != PCI_HEADER_TYPE_CARDBUS) // Not a bridge continue; v = pci_config_readl(bdf, PCI_PRIMARY_BUS); int newmax = (v & 0xff00) + 0x0100; if (newmax > max) max = newmax; } SET_VAR(CS, MaxBDF, max); dprintf(1, "Found %d PCI buses\n", pci_bdf_to_bus(max)); } // Search for a device with the specified vendor and device ids. int pci_find_device(u16 vendid, u16 devid, int start_bdf) { u32 id = (devid << 16) | vendid; int max = GET_VAR(CS, MaxBDF); int bdf; for (bdf=start_bdf; bdf < max; bdf++) { u32 v = pci_config_readl(bdf, PCI_VENDOR_ID); if (v != id) continue; // Found it. return bdf; } return -1; } // Search for a device with the specified class id and prog-if. int pci_find_classprog(u32 classprog, int start_bdf) { int max = GET_VAR(CS, MaxBDF); int bdf; for (bdf=start_bdf; bdf < max; bdf++) { u32 v = pci_config_readl(bdf, PCI_CLASS_REVISION); if ((v>>8) != classprog) continue; // Found it. return bdf; } return -1; } // Search for a device with the specified class id. int pci_find_class(u16 classid, int start_bdf) { int max = GET_VAR(CS, MaxBDF); int bdf; for (bdf=start_bdf; bdf < max; bdf++) { u16 v = pci_config_readw(bdf, PCI_CLASS_DEVICE); if (v != classid) continue; // Found it. return bdf; } return -1; }