1456c55bcSAndrew Jones /* 2456c55bcSAndrew Jones * Copyright (C) 2013, Red Hat Inc, Michael S. Tsirkin <mst@redhat.com> 3456c55bcSAndrew Jones * 4456c55bcSAndrew Jones * This work is licensed under the terms of the GNU LGPL, version 2. 5456c55bcSAndrew Jones */ 64932b58aSMichael S. Tsirkin #include <linux/pci_regs.h> 74932b58aSMichael S. Tsirkin #include "pci.h" 8456c55bcSAndrew Jones #include "asm/pci.h" 94932b58aSMichael S. Tsirkin 10903b0516SPeter Xu typedef void (*pci_cap_handler)(struct pci_dev *dev, int cap_offset); 11903b0516SPeter Xu 12903b0516SPeter Xu static void pci_cap_msi_handler(struct pci_dev *dev, int cap_offset) 13903b0516SPeter Xu { 14903b0516SPeter Xu printf("Detected MSI for device 0x%x offset 0x%x\n", 15903b0516SPeter Xu dev->bdf, cap_offset); 16903b0516SPeter Xu dev->msi_offset = cap_offset; 17903b0516SPeter Xu } 18903b0516SPeter Xu 19903b0516SPeter Xu static pci_cap_handler cap_handlers[PCI_CAP_ID_MAX + 1] = { 20903b0516SPeter Xu [PCI_CAP_ID_MSI] = pci_cap_msi_handler, 21903b0516SPeter Xu }; 22903b0516SPeter Xu 23903b0516SPeter Xu void pci_cap_walk(struct pci_dev *dev) 24903b0516SPeter Xu { 25903b0516SPeter Xu uint8_t cap_offset; 26903b0516SPeter Xu uint8_t cap_id; 27903b0516SPeter Xu int count = 0; 28903b0516SPeter Xu 29903b0516SPeter Xu cap_offset = pci_config_readb(dev->bdf, PCI_CAPABILITY_LIST); 30903b0516SPeter Xu while (cap_offset) { 31903b0516SPeter Xu cap_id = pci_config_readb(dev->bdf, cap_offset); 32903b0516SPeter Xu printf("PCI detected cap 0x%x\n", cap_id); 33903b0516SPeter Xu assert(cap_id < PCI_CAP_ID_MAX + 1); 34903b0516SPeter Xu if (cap_handlers[cap_id]) 35903b0516SPeter Xu cap_handlers[cap_id](dev, cap_offset); 36903b0516SPeter Xu cap_offset = pci_config_readb(dev->bdf, cap_offset + 1); 37903b0516SPeter Xu /* Avoid dead loop during cap walk */ 38903b0516SPeter Xu assert(++count <= 255); 39903b0516SPeter Xu } 40903b0516SPeter Xu } 41903b0516SPeter Xu 42*19daf1c5SPeter Xu void pci_msi_set_enable(struct pci_dev *dev, bool enabled) 43*19daf1c5SPeter Xu { 44*19daf1c5SPeter Xu uint16_t msi_control; 45*19daf1c5SPeter Xu uint16_t offset; 46*19daf1c5SPeter Xu 47*19daf1c5SPeter Xu offset = dev->msi_offset; 48*19daf1c5SPeter Xu msi_control = pci_config_readw(dev->bdf, offset + PCI_MSI_FLAGS); 49*19daf1c5SPeter Xu 50*19daf1c5SPeter Xu if (enabled) 51*19daf1c5SPeter Xu msi_control |= PCI_MSI_FLAGS_ENABLE; 52*19daf1c5SPeter Xu else 53*19daf1c5SPeter Xu msi_control &= ~PCI_MSI_FLAGS_ENABLE; 54*19daf1c5SPeter Xu 55*19daf1c5SPeter Xu pci_config_writew(dev->bdf, offset + PCI_MSI_FLAGS, msi_control); 56*19daf1c5SPeter Xu } 57*19daf1c5SPeter Xu 58903b0516SPeter Xu bool pci_setup_msi(struct pci_dev *dev, uint64_t msi_addr, uint32_t msi_data) 59903b0516SPeter Xu { 60903b0516SPeter Xu uint16_t msi_control; 61903b0516SPeter Xu uint16_t offset; 62903b0516SPeter Xu pcidevaddr_t addr; 63903b0516SPeter Xu 64903b0516SPeter Xu assert(dev); 65903b0516SPeter Xu 66903b0516SPeter Xu if (!dev->msi_offset) { 67903b0516SPeter Xu printf("MSI: dev 0x%x does not support MSI.\n", dev->bdf); 68903b0516SPeter Xu return false; 69903b0516SPeter Xu } 70903b0516SPeter Xu 71903b0516SPeter Xu addr = dev->bdf; 72903b0516SPeter Xu offset = dev->msi_offset; 73903b0516SPeter Xu msi_control = pci_config_readw(addr, offset + PCI_MSI_FLAGS); 74903b0516SPeter Xu pci_config_writel(addr, offset + PCI_MSI_ADDRESS_LO, 75903b0516SPeter Xu msi_addr & 0xffffffff); 76903b0516SPeter Xu 77903b0516SPeter Xu if (msi_control & PCI_MSI_FLAGS_64BIT) { 78903b0516SPeter Xu pci_config_writel(addr, offset + PCI_MSI_ADDRESS_HI, 79903b0516SPeter Xu (uint32_t)(msi_addr >> 32)); 80903b0516SPeter Xu pci_config_writel(addr, offset + PCI_MSI_DATA_64, msi_data); 81903b0516SPeter Xu printf("MSI: dev 0x%x init 64bit address: ", addr); 82903b0516SPeter Xu } else { 83903b0516SPeter Xu pci_config_writel(addr, offset + PCI_MSI_DATA_32, msi_data); 84903b0516SPeter Xu printf("MSI: dev 0x%x init 32bit address: ", addr); 85903b0516SPeter Xu } 86903b0516SPeter Xu printf("addr=0x%lx, data=0x%x\n", msi_addr, msi_data); 87903b0516SPeter Xu 88*19daf1c5SPeter Xu pci_msi_set_enable(dev, true); 89903b0516SPeter Xu 90903b0516SPeter Xu return true; 91903b0516SPeter Xu } 92903b0516SPeter Xu 9366082ed6SPeter Xu void pci_cmd_set_clr(struct pci_dev *dev, uint16_t set, uint16_t clr) 9466082ed6SPeter Xu { 9566082ed6SPeter Xu uint16_t val = pci_config_readw(dev->bdf, PCI_COMMAND); 9666082ed6SPeter Xu 9766082ed6SPeter Xu /* No overlap is allowed */ 9866082ed6SPeter Xu assert((set & clr) == 0); 9966082ed6SPeter Xu val |= set; 10066082ed6SPeter Xu val &= ~clr; 10166082ed6SPeter Xu 10266082ed6SPeter Xu pci_config_writew(dev->bdf, PCI_COMMAND, val); 10366082ed6SPeter Xu } 10466082ed6SPeter Xu 105e1cad5c8SAlexander Gordeev bool pci_dev_exists(pcidevaddr_t dev) 106e1cad5c8SAlexander Gordeev { 107e1cad5c8SAlexander Gordeev return (pci_config_readw(dev, PCI_VENDOR_ID) != 0xffff && 108e1cad5c8SAlexander Gordeev pci_config_readw(dev, PCI_DEVICE_ID) != 0xffff); 109e1cad5c8SAlexander Gordeev } 110e1cad5c8SAlexander Gordeev 1114d6cefa9SPeter Xu void pci_dev_init(struct pci_dev *dev, pcidevaddr_t bdf) 1124d6cefa9SPeter Xu { 1134d6cefa9SPeter Xu memset(dev, 0, sizeof(*dev)); 1144d6cefa9SPeter Xu dev->bdf = bdf; 1154d6cefa9SPeter Xu } 1164d6cefa9SPeter Xu 1174932b58aSMichael S. Tsirkin /* Scan bus look for a specific device. Only bus 0 scanned for now. */ 1184932b58aSMichael S. Tsirkin pcidevaddr_t pci_find_dev(uint16_t vendor_id, uint16_t device_id) 1194932b58aSMichael S. Tsirkin { 120ebb58e7eSAlexander Gordeev pcidevaddr_t dev; 121ebb58e7eSAlexander Gordeev 1224d6cefa9SPeter Xu for (dev = 0; dev < PCI_DEVFN_MAX; ++dev) { 123d8369c77SAlexander Gordeev if (pci_config_readw(dev, PCI_VENDOR_ID) == vendor_id && 124d8369c77SAlexander Gordeev pci_config_readw(dev, PCI_DEVICE_ID) == device_id) 1254932b58aSMichael S. Tsirkin return dev; 1264932b58aSMichael S. Tsirkin } 127ebb58e7eSAlexander Gordeev 1284932b58aSMichael S. Tsirkin return PCIDEVADDR_INVALID; 1294932b58aSMichael S. Tsirkin } 1304932b58aSMichael S. Tsirkin 13133d78b07SAlexander Gordeev uint32_t pci_bar_mask(uint32_t bar) 1322455ef20SAlexander Gordeev { 1332455ef20SAlexander Gordeev return (bar & PCI_BASE_ADDRESS_SPACE_IO) ? 1342455ef20SAlexander Gordeev PCI_BASE_ADDRESS_IO_MASK : PCI_BASE_ADDRESS_MEM_MASK; 1352455ef20SAlexander Gordeev } 1362455ef20SAlexander Gordeev 1374d6cefa9SPeter Xu uint32_t pci_bar_get(struct pci_dev *dev, int bar_num) 1387aa83307SAlexander Gordeev { 1394d6cefa9SPeter Xu return pci_config_readl(dev->bdf, PCI_BASE_ADDRESS_0 + 1404d6cefa9SPeter Xu bar_num * 4); 1417aa83307SAlexander Gordeev } 1427aa83307SAlexander Gordeev 1434d6cefa9SPeter Xu phys_addr_t pci_bar_get_addr(struct pci_dev *dev, int bar_num) 1444932b58aSMichael S. Tsirkin { 1457aa83307SAlexander Gordeev uint32_t bar = pci_bar_get(dev, bar_num); 1462455ef20SAlexander Gordeev uint32_t mask = pci_bar_mask(bar); 1472455ef20SAlexander Gordeev uint64_t addr = bar & mask; 148647d2ab7SAlexander Gordeev phys_addr_t phys_addr; 149ebb58e7eSAlexander Gordeev 1502455ef20SAlexander Gordeev if (pci_bar_is64(dev, bar_num)) 1512455ef20SAlexander Gordeev addr |= (uint64_t)pci_bar_get(dev, bar_num + 1) << 32; 1522455ef20SAlexander Gordeev 1534d6cefa9SPeter Xu phys_addr = pci_translate_addr(dev->bdf, addr); 154647d2ab7SAlexander Gordeev assert(phys_addr != INVALID_PHYS_ADDR); 155647d2ab7SAlexander Gordeev 156647d2ab7SAlexander Gordeev return phys_addr; 1572455ef20SAlexander Gordeev } 1582455ef20SAlexander Gordeev 1594d6cefa9SPeter Xu void pci_bar_set_addr(struct pci_dev *dev, int bar_num, phys_addr_t addr) 160647f92c7SAlexander Gordeev { 161647f92c7SAlexander Gordeev int off = PCI_BASE_ADDRESS_0 + bar_num * 4; 162647f92c7SAlexander Gordeev 1634d6cefa9SPeter Xu pci_config_writel(dev->bdf, off, (uint32_t)addr); 164647f92c7SAlexander Gordeev 165647f92c7SAlexander Gordeev if (pci_bar_is64(dev, bar_num)) 1664d6cefa9SPeter Xu pci_config_writel(dev->bdf, off + 4, 1674d6cefa9SPeter Xu (uint32_t)(addr >> 32)); 168647f92c7SAlexander Gordeev } 169647f92c7SAlexander Gordeev 1702455ef20SAlexander Gordeev /* 1712455ef20SAlexander Gordeev * To determine the amount of address space needed by a PCI device, 1722455ef20SAlexander Gordeev * one must save the original value of the BAR, write a value of 1732455ef20SAlexander Gordeev * all 1's to the register, and then read it back. The amount of 1742455ef20SAlexander Gordeev * memory can be then determined by masking the information bits, 1752455ef20SAlexander Gordeev * performing a bitwise NOT, and incrementing the value by 1. 1762455ef20SAlexander Gordeev * 1772455ef20SAlexander Gordeev * The following pci_bar_size_helper() and pci_bar_size() functions 1782455ef20SAlexander Gordeev * implement the algorithm. 1792455ef20SAlexander Gordeev */ 1804d6cefa9SPeter Xu static uint32_t pci_bar_size_helper(struct pci_dev *dev, int bar_num) 1812455ef20SAlexander Gordeev { 1822455ef20SAlexander Gordeev int off = PCI_BASE_ADDRESS_0 + bar_num * 4; 1834d6cefa9SPeter Xu uint16_t bdf = dev->bdf; 1842455ef20SAlexander Gordeev uint32_t bar, val; 1852455ef20SAlexander Gordeev 1864d6cefa9SPeter Xu bar = pci_config_readl(bdf, off); 1874d6cefa9SPeter Xu pci_config_writel(bdf, off, ~0u); 1884d6cefa9SPeter Xu val = pci_config_readl(bdf, off); 1894d6cefa9SPeter Xu pci_config_writel(bdf, off, bar); 1902455ef20SAlexander Gordeev 1912455ef20SAlexander Gordeev return val; 1922455ef20SAlexander Gordeev } 1932455ef20SAlexander Gordeev 1944d6cefa9SPeter Xu phys_addr_t pci_bar_size(struct pci_dev *dev, int bar_num) 1952455ef20SAlexander Gordeev { 1962455ef20SAlexander Gordeev uint32_t bar, size; 1972455ef20SAlexander Gordeev 1982455ef20SAlexander Gordeev size = pci_bar_size_helper(dev, bar_num); 1992455ef20SAlexander Gordeev if (!size) 2002455ef20SAlexander Gordeev return 0; 2012455ef20SAlexander Gordeev 2022455ef20SAlexander Gordeev bar = pci_bar_get(dev, bar_num); 2032455ef20SAlexander Gordeev size &= pci_bar_mask(bar); 2042455ef20SAlexander Gordeev 2052455ef20SAlexander Gordeev if (pci_bar_is64(dev, bar_num)) { 2062455ef20SAlexander Gordeev phys_addr_t size64 = pci_bar_size_helper(dev, bar_num + 1); 2072455ef20SAlexander Gordeev size64 = (size64 << 32) | size; 2082455ef20SAlexander Gordeev 2092455ef20SAlexander Gordeev return ~size64 + 1; 2102455ef20SAlexander Gordeev } else { 2112455ef20SAlexander Gordeev return ~size + 1; 2122455ef20SAlexander Gordeev } 2134932b58aSMichael S. Tsirkin } 2144932b58aSMichael S. Tsirkin 2154d6cefa9SPeter Xu bool pci_bar_is_memory(struct pci_dev *dev, int bar_num) 2164932b58aSMichael S. Tsirkin { 2177aa83307SAlexander Gordeev uint32_t bar = pci_bar_get(dev, bar_num); 218ebb58e7eSAlexander Gordeev 2194932b58aSMichael S. Tsirkin return !(bar & PCI_BASE_ADDRESS_SPACE_IO); 2204932b58aSMichael S. Tsirkin } 2214932b58aSMichael S. Tsirkin 2224d6cefa9SPeter Xu bool pci_bar_is_valid(struct pci_dev *dev, int bar_num) 2234932b58aSMichael S. Tsirkin { 2247aa83307SAlexander Gordeev return pci_bar_get(dev, bar_num); 2254932b58aSMichael S. Tsirkin } 2262455ef20SAlexander Gordeev 2274d6cefa9SPeter Xu bool pci_bar_is64(struct pci_dev *dev, int bar_num) 2282455ef20SAlexander Gordeev { 2292455ef20SAlexander Gordeev uint32_t bar = pci_bar_get(dev, bar_num); 2302455ef20SAlexander Gordeev 2312455ef20SAlexander Gordeev if (bar & PCI_BASE_ADDRESS_SPACE_IO) 2322455ef20SAlexander Gordeev return false; 2332455ef20SAlexander Gordeev 2342455ef20SAlexander Gordeev return (bar & PCI_BASE_ADDRESS_MEM_TYPE_MASK) == 2352455ef20SAlexander Gordeev PCI_BASE_ADDRESS_MEM_TYPE_64; 2362455ef20SAlexander Gordeev } 237e4611520SAlexander Gordeev 2384d6cefa9SPeter Xu void pci_bar_print(struct pci_dev *dev, int bar_num) 239e4611520SAlexander Gordeev { 240e4611520SAlexander Gordeev phys_addr_t size, start, end; 241e4611520SAlexander Gordeev uint32_t bar; 242e4611520SAlexander Gordeev 243e4611520SAlexander Gordeev size = pci_bar_size(dev, bar_num); 244e4611520SAlexander Gordeev if (!size) 245e4611520SAlexander Gordeev return; 246e4611520SAlexander Gordeev 247e4611520SAlexander Gordeev bar = pci_bar_get(dev, bar_num); 248e4611520SAlexander Gordeev start = pci_bar_get_addr(dev, bar_num); 249e4611520SAlexander Gordeev end = start + size - 1; 250e4611520SAlexander Gordeev 251e4611520SAlexander Gordeev if (pci_bar_is64(dev, bar_num)) { 252e4611520SAlexander Gordeev printf("BAR#%d,%d [%" PRIx64 "-%" PRIx64 " ", 253e4611520SAlexander Gordeev bar_num, bar_num + 1, start, end); 254e4611520SAlexander Gordeev } else { 255e4611520SAlexander Gordeev printf("BAR#%d [%02x-%02x ", 256e4611520SAlexander Gordeev bar_num, (uint32_t)start, (uint32_t)end); 257e4611520SAlexander Gordeev } 258e4611520SAlexander Gordeev 259e4611520SAlexander Gordeev if (bar & PCI_BASE_ADDRESS_SPACE_IO) { 260e4611520SAlexander Gordeev printf("PIO"); 261e4611520SAlexander Gordeev } else { 262e4611520SAlexander Gordeev printf("MEM"); 263e4611520SAlexander Gordeev switch (bar & PCI_BASE_ADDRESS_MEM_TYPE_MASK) { 264e4611520SAlexander Gordeev case PCI_BASE_ADDRESS_MEM_TYPE_32: 265e4611520SAlexander Gordeev printf("32"); 266e4611520SAlexander Gordeev break; 267e4611520SAlexander Gordeev case PCI_BASE_ADDRESS_MEM_TYPE_1M: 268e4611520SAlexander Gordeev printf("1M"); 269e4611520SAlexander Gordeev break; 270e4611520SAlexander Gordeev case PCI_BASE_ADDRESS_MEM_TYPE_64: 271e4611520SAlexander Gordeev printf("64"); 272e4611520SAlexander Gordeev break; 273e4611520SAlexander Gordeev default: 274e4611520SAlexander Gordeev assert(0); 275e4611520SAlexander Gordeev } 276e4611520SAlexander Gordeev } 277e4611520SAlexander Gordeev 278e4611520SAlexander Gordeev if (bar & PCI_BASE_ADDRESS_MEM_PREFETCH) 279e4611520SAlexander Gordeev printf("/p"); 280e4611520SAlexander Gordeev 281e4611520SAlexander Gordeev printf("]"); 282e4611520SAlexander Gordeev } 283e4611520SAlexander Gordeev 28433d78b07SAlexander Gordeev void pci_dev_print_id(pcidevaddr_t dev) 285e4611520SAlexander Gordeev { 286e4611520SAlexander Gordeev printf("00.%02x.%1x %04x:%04x", dev / 8, dev % 8, 287e4611520SAlexander Gordeev pci_config_readw(dev, PCI_VENDOR_ID), 288e4611520SAlexander Gordeev pci_config_readw(dev, PCI_DEVICE_ID)); 289e4611520SAlexander Gordeev } 290e4611520SAlexander Gordeev 291e4611520SAlexander Gordeev static void pci_dev_print(pcidevaddr_t dev) 292e4611520SAlexander Gordeev { 293e4611520SAlexander Gordeev uint8_t header = pci_config_readb(dev, PCI_HEADER_TYPE); 294e4611520SAlexander Gordeev uint8_t progif = pci_config_readb(dev, PCI_CLASS_PROG); 295e4611520SAlexander Gordeev uint8_t subclass = pci_config_readb(dev, PCI_CLASS_DEVICE); 296e4611520SAlexander Gordeev uint8_t class = pci_config_readb(dev, PCI_CLASS_DEVICE + 1); 2974d6cefa9SPeter Xu struct pci_dev pci_dev; 298e4611520SAlexander Gordeev int i; 299e4611520SAlexander Gordeev 3004d6cefa9SPeter Xu pci_dev_init(&pci_dev, dev); 3014d6cefa9SPeter Xu 302e4611520SAlexander Gordeev pci_dev_print_id(dev); 303e4611520SAlexander Gordeev printf(" type %02x progif %02x class %02x subclass %02x\n", 304e4611520SAlexander Gordeev header, progif, class, subclass); 305e4611520SAlexander Gordeev 306e4611520SAlexander Gordeev if ((header & PCI_HEADER_TYPE_MASK) != PCI_HEADER_TYPE_NORMAL) 307e4611520SAlexander Gordeev return; 308e4611520SAlexander Gordeev 309e954ce23SPeter Xu for (i = 0; i < PCI_BAR_NUM; i++) { 3104d6cefa9SPeter Xu if (pci_bar_size(&pci_dev, i)) { 311e4611520SAlexander Gordeev printf("\t"); 3124d6cefa9SPeter Xu pci_bar_print(&pci_dev, i); 313e4611520SAlexander Gordeev printf("\n"); 314e4611520SAlexander Gordeev } 3154d6cefa9SPeter Xu if (pci_bar_is64(&pci_dev, i)) 316e4611520SAlexander Gordeev i++; 317e4611520SAlexander Gordeev } 318e4611520SAlexander Gordeev } 319e4611520SAlexander Gordeev 320e4611520SAlexander Gordeev void pci_print(void) 321e4611520SAlexander Gordeev { 322e4611520SAlexander Gordeev pcidevaddr_t dev; 323e4611520SAlexander Gordeev 3244d6cefa9SPeter Xu for (dev = 0; dev < PCI_DEVFN_MAX; ++dev) { 325e4611520SAlexander Gordeev if (pci_dev_exists(dev)) 326e4611520SAlexander Gordeev pci_dev_print(dev); 327e4611520SAlexander Gordeev } 328e4611520SAlexander Gordeev } 329e954ce23SPeter Xu 330e954ce23SPeter Xu void pci_scan_bars(struct pci_dev *dev) 331e954ce23SPeter Xu { 332e954ce23SPeter Xu int i; 333e954ce23SPeter Xu 334e954ce23SPeter Xu for (i = 0; i < PCI_BAR_NUM; i++) { 335e954ce23SPeter Xu if (!pci_bar_is_valid(dev, i)) 336e954ce23SPeter Xu continue; 337e954ce23SPeter Xu dev->resource[i] = pci_bar_get_addr(dev, i); 338e954ce23SPeter Xu if (pci_bar_is64(dev, i)) { 339e954ce23SPeter Xu i++; 340e954ce23SPeter Xu dev->resource[i] = (phys_addr_t)0; 341e954ce23SPeter Xu } 342e954ce23SPeter Xu } 343e954ce23SPeter Xu } 34466082ed6SPeter Xu 345352096c7SPeter Xu uint8_t pci_intx_line(struct pci_dev *dev) 346352096c7SPeter Xu { 347352096c7SPeter Xu return pci_config_readb(dev->bdf, PCI_INTERRUPT_LINE); 348352096c7SPeter Xu } 349352096c7SPeter Xu 35066082ed6SPeter Xu void pci_enable_defaults(struct pci_dev *dev) 35166082ed6SPeter Xu { 35266082ed6SPeter Xu pci_scan_bars(dev); 35366082ed6SPeter Xu /* Enable device DMA operations */ 35466082ed6SPeter Xu pci_cmd_set_clr(dev, PCI_COMMAND_MASTER, 0); 355903b0516SPeter Xu pci_cap_walk(dev); 35666082ed6SPeter Xu } 357