// 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 LGPLv3 license. #include "pci.h" // pci_config_writel #include "ioport.h" // outl #include "util.h" // dprintf #include "config.h" // CONFIG_* #include "pci_regs.h" // PCI_VENDOR_ID #include "pci_ids.h" // PCI_CLASS_DISPLAY_VGA 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)); } void pci_config_maskw(u16 bdf, u32 addr, u16 off, u16 on) { u16 val = pci_config_readw(bdf, addr); val = (val & ~off) | on; pci_config_writew(bdf, addr, val); } // Helper function for foreachpci() macro - return next device int pci_next(int bdf, int *pmax) { if (pci_bdf_to_fn(bdf) == 1 && (pci_config_readb(bdf-1, PCI_HEADER_TYPE) & 0x80) == 0) // Last found device wasn't a multi-function device - skip to // the next device. bdf += 7; int max = *pmax; for (;;) { if (bdf >= max) { if (CONFIG_PCI_ROOT1 && bdf <= (CONFIG_PCI_ROOT1 << 8)) bdf = CONFIG_PCI_ROOT1 << 8; else if (CONFIG_PCI_ROOT2 && bdf <= (CONFIG_PCI_ROOT2 << 8)) bdf = CONFIG_PCI_ROOT2 << 8; else return -1; *pmax = max = bdf + 0x0100; } u16 v = pci_config_readw(bdf, PCI_VENDOR_ID); if (v != 0x0000 && v != 0xffff) // Device is present. break; if (pci_bdf_to_fn(bdf) == 0) bdf += 8; else bdf += 1; } // Check if found device is a bridge. u32 v = pci_config_readb(bdf, PCI_HEADER_TYPE); v &= 0x7f; if (v == PCI_HEADER_TYPE_BRIDGE || v == PCI_HEADER_TYPE_CARDBUS) { v = pci_config_readl(bdf, PCI_PRIMARY_BUS); int newmax = (v & 0xff00) + 0x0100; if (newmax > max) *pmax = newmax; } return bdf; } // Find a vga device with legacy address decoding enabled. int pci_find_vga(void) { int bdf = 0x0000, max = 0x0100; for (;;) { if (bdf >= max) { if (CONFIG_PCI_ROOT1 && bdf <= (CONFIG_PCI_ROOT1 << 8)) bdf = CONFIG_PCI_ROOT1 << 8; else if (CONFIG_PCI_ROOT2 && bdf <= (CONFIG_PCI_ROOT2 << 8)) bdf = CONFIG_PCI_ROOT2 << 8; else return -1; max = bdf + 0x0100; } u16 cls = pci_config_readw(bdf, PCI_CLASS_DEVICE); if (cls == 0x0000 || cls == 0xffff) { // Device not present. if (pci_bdf_to_fn(bdf) == 0) bdf += 8; else bdf += 1; continue; } if (cls == PCI_CLASS_DISPLAY_VGA) { u16 cmd = pci_config_readw(bdf, PCI_COMMAND); if (cmd & PCI_COMMAND_IO && cmd & PCI_COMMAND_MEMORY) // Found active vga card return bdf; } // Check if device is a bridge. u8 hdr = pci_config_readb(bdf, PCI_HEADER_TYPE); u8 ht = hdr & 0x7f; if (ht == PCI_HEADER_TYPE_BRIDGE || ht == PCI_HEADER_TYPE_CARDBUS) { u32 ctrl = pci_config_readb(bdf, PCI_BRIDGE_CONTROL); if (ctrl & PCI_BRIDGE_CTL_VGA) { // Found a VGA enabled bridge. u32 pbus = pci_config_readl(bdf, PCI_PRIMARY_BUS); bdf = (pbus & 0xff00); max = bdf + 0x100; continue; } } if (pci_bdf_to_fn(bdf) == 0 && (hdr & 0x80) == 0) // Last found device wasn't a multi-function device - skip to // the next device. bdf += 8; else bdf += 1; } } // Search for a device with the specified vendor and device ids. int pci_find_device(u16 vendid, u16 devid) { u32 id = (devid << 16) | vendid; int bdf, max; foreachpci(bdf, max) { u32 v = pci_config_readl(bdf, PCI_VENDOR_ID); if (v == id) return bdf; } return -1; } // Search for a device with the specified class id. int pci_find_class(u16 classid) { int bdf, max; foreachpci(bdf, max) { u16 v = pci_config_readw(bdf, PCI_CLASS_DEVICE); if (v == classid) return bdf; } return -1; } int pci_init_device(const struct pci_device_id *ids, u16 bdf, void *arg) { u16 vendor_id = pci_config_readw(bdf, PCI_VENDOR_ID); u16 device_id = pci_config_readw(bdf, PCI_DEVICE_ID); u16 class = pci_config_readw(bdf, PCI_CLASS_DEVICE); while (ids->vendid || ids->class_mask) { if ((ids->vendid == PCI_ANY_ID || ids->vendid == vendor_id) && (ids->devid == PCI_ANY_ID || ids->devid == device_id) && !((ids->class ^ class) & ids->class_mask)) { if (ids->func) { ids->func(bdf, arg); } return 0; } ids++; } return -1; } int pci_find_init_device(const struct pci_device_id *ids, void *arg) { int bdf, max; foreachpci(bdf, max) { if (pci_init_device(ids, bdf, arg) == 0) { return bdf; } } return -1; } void pci_reboot(void) { u8 v = inb(PORT_PCI_REBOOT) & ~6; outb(v|2, PORT_PCI_REBOOT); /* Request hard reset */ udelay(50); outb(v|6, PORT_PCI_REBOOT); /* Actually do the reset */ udelay(50); }