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 10cdccea7cSAndrew Jones void pci_cap_walk(struct pci_dev *dev, pci_cap_handler_t handler) 11903b0516SPeter Xu { 12903b0516SPeter Xu uint8_t cap_offset; 13903b0516SPeter Xu uint8_t cap_id; 14903b0516SPeter Xu int count = 0; 15903b0516SPeter Xu 16903b0516SPeter Xu cap_offset = pci_config_readb(dev->bdf, PCI_CAPABILITY_LIST); 17903b0516SPeter Xu while (cap_offset) { 18903b0516SPeter Xu cap_id = pci_config_readb(dev->bdf, cap_offset); 19903b0516SPeter Xu assert(cap_id < PCI_CAP_ID_MAX + 1); 20cdccea7cSAndrew Jones handler(dev, cap_offset, cap_id); 21903b0516SPeter Xu cap_offset = pci_config_readb(dev->bdf, cap_offset + 1); 22903b0516SPeter Xu /* Avoid dead loop during cap walk */ 23903b0516SPeter Xu assert(++count <= 255); 24903b0516SPeter Xu } 25903b0516SPeter Xu } 26903b0516SPeter Xu 2719daf1c5SPeter Xu void pci_msi_set_enable(struct pci_dev *dev, bool enabled) 2819daf1c5SPeter Xu { 2919daf1c5SPeter Xu uint16_t msi_control; 3019daf1c5SPeter Xu uint16_t offset; 3119daf1c5SPeter Xu 3219daf1c5SPeter Xu offset = dev->msi_offset; 3319daf1c5SPeter Xu msi_control = pci_config_readw(dev->bdf, offset + PCI_MSI_FLAGS); 3419daf1c5SPeter Xu 3519daf1c5SPeter Xu if (enabled) 3619daf1c5SPeter Xu msi_control |= PCI_MSI_FLAGS_ENABLE; 3719daf1c5SPeter Xu else 3819daf1c5SPeter Xu msi_control &= ~PCI_MSI_FLAGS_ENABLE; 3919daf1c5SPeter Xu 4019daf1c5SPeter Xu pci_config_writew(dev->bdf, offset + PCI_MSI_FLAGS, msi_control); 4119daf1c5SPeter Xu } 4219daf1c5SPeter Xu 43903b0516SPeter Xu bool pci_setup_msi(struct pci_dev *dev, uint64_t msi_addr, uint32_t msi_data) 44903b0516SPeter Xu { 45903b0516SPeter Xu uint16_t msi_control; 46903b0516SPeter Xu uint16_t offset; 47903b0516SPeter Xu pcidevaddr_t addr; 48903b0516SPeter Xu 49903b0516SPeter Xu assert(dev); 50903b0516SPeter Xu 51903b0516SPeter Xu if (!dev->msi_offset) { 52*fd6aada0SRadim Krčmář printf("MSI: dev %#x does not support MSI.\n", dev->bdf); 53903b0516SPeter Xu return false; 54903b0516SPeter Xu } 55903b0516SPeter Xu 56903b0516SPeter Xu addr = dev->bdf; 57903b0516SPeter Xu offset = dev->msi_offset; 58903b0516SPeter Xu msi_control = pci_config_readw(addr, offset + PCI_MSI_FLAGS); 59903b0516SPeter Xu pci_config_writel(addr, offset + PCI_MSI_ADDRESS_LO, 60903b0516SPeter Xu msi_addr & 0xffffffff); 61903b0516SPeter Xu 62903b0516SPeter Xu if (msi_control & PCI_MSI_FLAGS_64BIT) { 63903b0516SPeter Xu pci_config_writel(addr, offset + PCI_MSI_ADDRESS_HI, 64903b0516SPeter Xu (uint32_t)(msi_addr >> 32)); 65903b0516SPeter Xu pci_config_writel(addr, offset + PCI_MSI_DATA_64, msi_data); 66903b0516SPeter Xu } else { 67903b0516SPeter Xu pci_config_writel(addr, offset + PCI_MSI_DATA_32, msi_data); 68903b0516SPeter Xu } 69903b0516SPeter Xu 7019daf1c5SPeter Xu pci_msi_set_enable(dev, true); 71903b0516SPeter Xu 72903b0516SPeter Xu return true; 73903b0516SPeter Xu } 74903b0516SPeter Xu 7566082ed6SPeter Xu void pci_cmd_set_clr(struct pci_dev *dev, uint16_t set, uint16_t clr) 7666082ed6SPeter Xu { 7766082ed6SPeter Xu uint16_t val = pci_config_readw(dev->bdf, PCI_COMMAND); 7866082ed6SPeter Xu 7966082ed6SPeter Xu /* No overlap is allowed */ 8066082ed6SPeter Xu assert((set & clr) == 0); 8166082ed6SPeter Xu val |= set; 8266082ed6SPeter Xu val &= ~clr; 8366082ed6SPeter Xu 8466082ed6SPeter Xu pci_config_writew(dev->bdf, PCI_COMMAND, val); 8566082ed6SPeter Xu } 8666082ed6SPeter Xu 87e1cad5c8SAlexander Gordeev bool pci_dev_exists(pcidevaddr_t dev) 88e1cad5c8SAlexander Gordeev { 89e1cad5c8SAlexander Gordeev return (pci_config_readw(dev, PCI_VENDOR_ID) != 0xffff && 90e1cad5c8SAlexander Gordeev pci_config_readw(dev, PCI_DEVICE_ID) != 0xffff); 91e1cad5c8SAlexander Gordeev } 92e1cad5c8SAlexander Gordeev 934932b58aSMichael S. Tsirkin /* Scan bus look for a specific device. Only bus 0 scanned for now. */ 944932b58aSMichael S. Tsirkin pcidevaddr_t pci_find_dev(uint16_t vendor_id, uint16_t device_id) 954932b58aSMichael S. Tsirkin { 96ebb58e7eSAlexander Gordeev pcidevaddr_t dev; 97ebb58e7eSAlexander Gordeev 984d6cefa9SPeter Xu for (dev = 0; dev < PCI_DEVFN_MAX; ++dev) { 99d8369c77SAlexander Gordeev if (pci_config_readw(dev, PCI_VENDOR_ID) == vendor_id && 100d8369c77SAlexander Gordeev pci_config_readw(dev, PCI_DEVICE_ID) == device_id) 1014932b58aSMichael S. Tsirkin return dev; 1024932b58aSMichael S. Tsirkin } 103ebb58e7eSAlexander Gordeev 1044932b58aSMichael S. Tsirkin return PCIDEVADDR_INVALID; 1054932b58aSMichael S. Tsirkin } 1064932b58aSMichael S. Tsirkin 10733d78b07SAlexander Gordeev uint32_t pci_bar_mask(uint32_t bar) 1082455ef20SAlexander Gordeev { 1092455ef20SAlexander Gordeev return (bar & PCI_BASE_ADDRESS_SPACE_IO) ? 1102455ef20SAlexander Gordeev PCI_BASE_ADDRESS_IO_MASK : PCI_BASE_ADDRESS_MEM_MASK; 1112455ef20SAlexander Gordeev } 1122455ef20SAlexander Gordeev 1134d6cefa9SPeter Xu uint32_t pci_bar_get(struct pci_dev *dev, int bar_num) 1147aa83307SAlexander Gordeev { 11568cf12d2SAlexander Gordeev ASSERT_BAR_NUM(bar_num); 11668cf12d2SAlexander Gordeev 1174d6cefa9SPeter Xu return pci_config_readl(dev->bdf, PCI_BASE_ADDRESS_0 + 1184d6cefa9SPeter Xu bar_num * 4); 1197aa83307SAlexander Gordeev } 1207aa83307SAlexander Gordeev 12178609f0eSAlexander Gordeev static phys_addr_t __pci_bar_get_addr(struct pci_dev *dev, int bar_num) 1224932b58aSMichael S. Tsirkin { 1237aa83307SAlexander Gordeev uint32_t bar = pci_bar_get(dev, bar_num); 1242455ef20SAlexander Gordeev uint32_t mask = pci_bar_mask(bar); 1252455ef20SAlexander Gordeev uint64_t addr = bar & mask; 126647d2ab7SAlexander Gordeev phys_addr_t phys_addr; 127ebb58e7eSAlexander Gordeev 1282455ef20SAlexander Gordeev if (pci_bar_is64(dev, bar_num)) 1292455ef20SAlexander Gordeev addr |= (uint64_t)pci_bar_get(dev, bar_num + 1) << 32; 1302455ef20SAlexander Gordeev 1314d6cefa9SPeter Xu phys_addr = pci_translate_addr(dev->bdf, addr); 132647d2ab7SAlexander Gordeev assert(phys_addr != INVALID_PHYS_ADDR); 133647d2ab7SAlexander Gordeev 134647d2ab7SAlexander Gordeev return phys_addr; 1352455ef20SAlexander Gordeev } 1362455ef20SAlexander Gordeev 13778609f0eSAlexander Gordeev phys_addr_t pci_bar_get_addr(struct pci_dev *dev, int bar_num) 13878609f0eSAlexander Gordeev { 13968cf12d2SAlexander Gordeev ASSERT_BAR_NUM(bar_num); 14068cf12d2SAlexander Gordeev 14178609f0eSAlexander Gordeev return dev->resource[bar_num]; 14278609f0eSAlexander Gordeev } 14378609f0eSAlexander Gordeev 1444d6cefa9SPeter Xu void pci_bar_set_addr(struct pci_dev *dev, int bar_num, phys_addr_t addr) 145647f92c7SAlexander Gordeev { 146647f92c7SAlexander Gordeev int off = PCI_BASE_ADDRESS_0 + bar_num * 4; 147647f92c7SAlexander Gordeev 14868cf12d2SAlexander Gordeev assert(addr != INVALID_PHYS_ADDR); 14968cf12d2SAlexander Gordeev assert(dev->resource[bar_num] != INVALID_PHYS_ADDR); 15068cf12d2SAlexander Gordeev 15168cf12d2SAlexander Gordeev ASSERT_BAR_NUM(bar_num); 15268cf12d2SAlexander Gordeev if (pci_bar_is64(dev, bar_num)) 15368cf12d2SAlexander Gordeev ASSERT_BAR_NUM(bar_num + 1); 15468cf12d2SAlexander Gordeev else 15568cf12d2SAlexander Gordeev assert((addr >> 32) == 0); 15668cf12d2SAlexander Gordeev 1574d6cefa9SPeter Xu pci_config_writel(dev->bdf, off, (uint32_t)addr); 15878609f0eSAlexander Gordeev dev->resource[bar_num] = addr; 159647f92c7SAlexander Gordeev 16078609f0eSAlexander Gordeev if (pci_bar_is64(dev, bar_num)) { 16178609f0eSAlexander Gordeev pci_config_writel(dev->bdf, off + 4, (uint32_t)(addr >> 32)); 16278609f0eSAlexander Gordeev dev->resource[bar_num + 1] = dev->resource[bar_num]; 16378609f0eSAlexander Gordeev } 164647f92c7SAlexander Gordeev } 165647f92c7SAlexander Gordeev 1662455ef20SAlexander Gordeev /* 1672455ef20SAlexander Gordeev * To determine the amount of address space needed by a PCI device, 1682455ef20SAlexander Gordeev * one must save the original value of the BAR, write a value of 1692455ef20SAlexander Gordeev * all 1's to the register, and then read it back. The amount of 1702455ef20SAlexander Gordeev * memory can be then determined by masking the information bits, 1712455ef20SAlexander Gordeev * performing a bitwise NOT, and incrementing the value by 1. 1722455ef20SAlexander Gordeev * 1732455ef20SAlexander Gordeev * The following pci_bar_size_helper() and pci_bar_size() functions 1742455ef20SAlexander Gordeev * implement the algorithm. 1752455ef20SAlexander Gordeev */ 1764d6cefa9SPeter Xu static uint32_t pci_bar_size_helper(struct pci_dev *dev, int bar_num) 1772455ef20SAlexander Gordeev { 1782455ef20SAlexander Gordeev int off = PCI_BASE_ADDRESS_0 + bar_num * 4; 1794d6cefa9SPeter Xu uint16_t bdf = dev->bdf; 1802455ef20SAlexander Gordeev uint32_t bar, val; 1812455ef20SAlexander Gordeev 1824d6cefa9SPeter Xu bar = pci_config_readl(bdf, off); 1834d6cefa9SPeter Xu pci_config_writel(bdf, off, ~0u); 1844d6cefa9SPeter Xu val = pci_config_readl(bdf, off); 1854d6cefa9SPeter Xu pci_config_writel(bdf, off, bar); 1862455ef20SAlexander Gordeev 1872455ef20SAlexander Gordeev return val; 1882455ef20SAlexander Gordeev } 1892455ef20SAlexander Gordeev 1904d6cefa9SPeter Xu phys_addr_t pci_bar_size(struct pci_dev *dev, int bar_num) 1912455ef20SAlexander Gordeev { 1922455ef20SAlexander Gordeev uint32_t bar, size; 1932455ef20SAlexander Gordeev 1942455ef20SAlexander Gordeev size = pci_bar_size_helper(dev, bar_num); 1952455ef20SAlexander Gordeev if (!size) 1962455ef20SAlexander Gordeev return 0; 1972455ef20SAlexander Gordeev 1982455ef20SAlexander Gordeev bar = pci_bar_get(dev, bar_num); 1992455ef20SAlexander Gordeev size &= pci_bar_mask(bar); 2002455ef20SAlexander Gordeev 2012455ef20SAlexander Gordeev if (pci_bar_is64(dev, bar_num)) { 2022455ef20SAlexander Gordeev phys_addr_t size64 = pci_bar_size_helper(dev, bar_num + 1); 2032455ef20SAlexander Gordeev size64 = (size64 << 32) | size; 2042455ef20SAlexander Gordeev 2052455ef20SAlexander Gordeev return ~size64 + 1; 2062455ef20SAlexander Gordeev } else { 2072455ef20SAlexander Gordeev return ~size + 1; 2082455ef20SAlexander Gordeev } 2094932b58aSMichael S. Tsirkin } 2104932b58aSMichael S. Tsirkin 2114d6cefa9SPeter Xu bool pci_bar_is_memory(struct pci_dev *dev, int bar_num) 2124932b58aSMichael S. Tsirkin { 2137aa83307SAlexander Gordeev uint32_t bar = pci_bar_get(dev, bar_num); 214ebb58e7eSAlexander Gordeev 2154932b58aSMichael S. Tsirkin return !(bar & PCI_BASE_ADDRESS_SPACE_IO); 2164932b58aSMichael S. Tsirkin } 2174932b58aSMichael S. Tsirkin 2184d6cefa9SPeter Xu bool pci_bar_is_valid(struct pci_dev *dev, int bar_num) 2194932b58aSMichael S. Tsirkin { 2209e292d10SAlexander Gordeev return dev->resource[bar_num] != INVALID_PHYS_ADDR; 2214932b58aSMichael S. Tsirkin } 2222455ef20SAlexander Gordeev 2234d6cefa9SPeter Xu bool pci_bar_is64(struct pci_dev *dev, int bar_num) 2242455ef20SAlexander Gordeev { 2252455ef20SAlexander Gordeev uint32_t bar = pci_bar_get(dev, bar_num); 2262455ef20SAlexander Gordeev 2272455ef20SAlexander Gordeev if (bar & PCI_BASE_ADDRESS_SPACE_IO) 2282455ef20SAlexander Gordeev return false; 2292455ef20SAlexander Gordeev 2302455ef20SAlexander Gordeev return (bar & PCI_BASE_ADDRESS_MEM_TYPE_MASK) == 2312455ef20SAlexander Gordeev PCI_BASE_ADDRESS_MEM_TYPE_64; 2322455ef20SAlexander Gordeev } 233e4611520SAlexander Gordeev 2344d6cefa9SPeter Xu void pci_bar_print(struct pci_dev *dev, int bar_num) 235e4611520SAlexander Gordeev { 236e4611520SAlexander Gordeev phys_addr_t size, start, end; 237e4611520SAlexander Gordeev uint32_t bar; 238e4611520SAlexander Gordeev 2399e292d10SAlexander Gordeev if (!pci_bar_is_valid(dev, bar_num)) 240e4611520SAlexander Gordeev return; 241e4611520SAlexander Gordeev 242e4611520SAlexander Gordeev bar = pci_bar_get(dev, bar_num); 2439e292d10SAlexander Gordeev size = pci_bar_size(dev, bar_num); 244e4611520SAlexander Gordeev start = pci_bar_get_addr(dev, bar_num); 245e4611520SAlexander Gordeev end = start + size - 1; 246e4611520SAlexander Gordeev 247e4611520SAlexander Gordeev if (pci_bar_is64(dev, bar_num)) { 248e4611520SAlexander Gordeev printf("BAR#%d,%d [%" PRIx64 "-%" PRIx64 " ", 249e4611520SAlexander Gordeev bar_num, bar_num + 1, start, end); 250e4611520SAlexander Gordeev } else { 251e4611520SAlexander Gordeev printf("BAR#%d [%02x-%02x ", 252e4611520SAlexander Gordeev bar_num, (uint32_t)start, (uint32_t)end); 253e4611520SAlexander Gordeev } 254e4611520SAlexander Gordeev 255e4611520SAlexander Gordeev if (bar & PCI_BASE_ADDRESS_SPACE_IO) { 256e4611520SAlexander Gordeev printf("PIO"); 257e4611520SAlexander Gordeev } else { 258e4611520SAlexander Gordeev printf("MEM"); 259e4611520SAlexander Gordeev switch (bar & PCI_BASE_ADDRESS_MEM_TYPE_MASK) { 260e4611520SAlexander Gordeev case PCI_BASE_ADDRESS_MEM_TYPE_32: 261e4611520SAlexander Gordeev printf("32"); 262e4611520SAlexander Gordeev break; 263e4611520SAlexander Gordeev case PCI_BASE_ADDRESS_MEM_TYPE_1M: 264e4611520SAlexander Gordeev printf("1M"); 265e4611520SAlexander Gordeev break; 266e4611520SAlexander Gordeev case PCI_BASE_ADDRESS_MEM_TYPE_64: 267e4611520SAlexander Gordeev printf("64"); 268e4611520SAlexander Gordeev break; 269e4611520SAlexander Gordeev default: 270e4611520SAlexander Gordeev assert(0); 271e4611520SAlexander Gordeev } 272e4611520SAlexander Gordeev } 273e4611520SAlexander Gordeev 274e4611520SAlexander Gordeev if (bar & PCI_BASE_ADDRESS_MEM_PREFETCH) 275e4611520SAlexander Gordeev printf("/p"); 276e4611520SAlexander Gordeev 277e4611520SAlexander Gordeev printf("]"); 278e4611520SAlexander Gordeev } 279e4611520SAlexander Gordeev 280cb026028SAlexander Gordeev void pci_dev_print_id(struct pci_dev *dev) 281e4611520SAlexander Gordeev { 282cb026028SAlexander Gordeev pcidevaddr_t bdf = dev->bdf; 283cb026028SAlexander Gordeev 284cb026028SAlexander Gordeev printf("00.%02x.%1x %04x:%04x", bdf / 8, bdf % 8, 285cb026028SAlexander Gordeev pci_config_readw(bdf, PCI_VENDOR_ID), 286cb026028SAlexander Gordeev pci_config_readw(bdf, PCI_DEVICE_ID)); 287e4611520SAlexander Gordeev } 288e4611520SAlexander Gordeev 28982f2f21aSAndrew Jones static void pci_cap_print(struct pci_dev *dev, int cap_offset, int cap_id) 29082f2f21aSAndrew Jones { 29182f2f21aSAndrew Jones switch (cap_id) { 29282f2f21aSAndrew Jones case PCI_CAP_ID_MSI: { 29382f2f21aSAndrew Jones uint16_t control = pci_config_readw(dev->bdf, cap_offset + PCI_MSI_FLAGS); 29482f2f21aSAndrew Jones printf("\tMSI,%s-bit capability ", control & PCI_MSI_FLAGS_64BIT ? "64" : "32"); 29582f2f21aSAndrew Jones break; 29682f2f21aSAndrew Jones } 29782f2f21aSAndrew Jones default: 298*fd6aada0SRadim Krčmář printf("\tcapability %#04x ", cap_id); 29982f2f21aSAndrew Jones break; 30082f2f21aSAndrew Jones } 301*fd6aada0SRadim Krčmář printf("at offset %#04x\n", cap_offset); 30282f2f21aSAndrew Jones } 30382f2f21aSAndrew Jones 304cb026028SAlexander Gordeev void pci_dev_print(struct pci_dev *dev) 305e4611520SAlexander Gordeev { 306cb026028SAlexander Gordeev pcidevaddr_t bdf = dev->bdf; 307cb026028SAlexander Gordeev uint8_t header = pci_config_readb(bdf, PCI_HEADER_TYPE); 308cb026028SAlexander Gordeev uint8_t progif = pci_config_readb(bdf, PCI_CLASS_PROG); 309cb026028SAlexander Gordeev uint8_t subclass = pci_config_readb(bdf, PCI_CLASS_DEVICE); 310cb026028SAlexander Gordeev uint8_t class = pci_config_readb(bdf, PCI_CLASS_DEVICE + 1); 311e4611520SAlexander Gordeev int i; 312e4611520SAlexander Gordeev 313e4611520SAlexander Gordeev pci_dev_print_id(dev); 314e4611520SAlexander Gordeev printf(" type %02x progif %02x class %02x subclass %02x\n", 315e4611520SAlexander Gordeev header, progif, class, subclass); 316e4611520SAlexander Gordeev 317cb026028SAlexander Gordeev pci_cap_walk(dev, pci_cap_print); 31882f2f21aSAndrew Jones 319e4611520SAlexander Gordeev if ((header & PCI_HEADER_TYPE_MASK) != PCI_HEADER_TYPE_NORMAL) 320e4611520SAlexander Gordeev return; 321e4611520SAlexander Gordeev 322e954ce23SPeter Xu for (i = 0; i < PCI_BAR_NUM; i++) { 323cb026028SAlexander Gordeev if (pci_bar_is_valid(dev, i)) { 324e4611520SAlexander Gordeev printf("\t"); 325cb026028SAlexander Gordeev pci_bar_print(dev, i); 326e4611520SAlexander Gordeev printf("\n"); 327e4611520SAlexander Gordeev } 328cb026028SAlexander Gordeev if (pci_bar_is64(dev, i)) 329e4611520SAlexander Gordeev i++; 330e4611520SAlexander Gordeev } 331e4611520SAlexander Gordeev } 332e4611520SAlexander Gordeev 333e4611520SAlexander Gordeev void pci_print(void) 334e4611520SAlexander Gordeev { 335cb026028SAlexander Gordeev pcidevaddr_t devfn; 336cb026028SAlexander Gordeev struct pci_dev pci_dev; 337e4611520SAlexander Gordeev 338cb026028SAlexander Gordeev for (devfn = 0; devfn < PCI_DEVFN_MAX; ++devfn) { 339cb026028SAlexander Gordeev if (pci_dev_exists(devfn)) { 340cb026028SAlexander Gordeev pci_dev_init(&pci_dev, devfn); 341cb026028SAlexander Gordeev pci_dev_print(&pci_dev); 342cb026028SAlexander Gordeev } 343e4611520SAlexander Gordeev } 344e4611520SAlexander Gordeev } 345e954ce23SPeter Xu 34678609f0eSAlexander Gordeev void pci_dev_init(struct pci_dev *dev, pcidevaddr_t bdf) 347e954ce23SPeter Xu { 348e954ce23SPeter Xu int i; 349e954ce23SPeter Xu 35078609f0eSAlexander Gordeev memset(dev, 0, sizeof(*dev)); 35178609f0eSAlexander Gordeev dev->bdf = bdf; 35278609f0eSAlexander Gordeev 353e954ce23SPeter Xu for (i = 0; i < PCI_BAR_NUM; i++) { 35448543201SAlexander Gordeev if (pci_bar_size(dev, i)) { 35578609f0eSAlexander Gordeev dev->resource[i] = __pci_bar_get_addr(dev, i); 356e954ce23SPeter Xu if (pci_bar_is64(dev, i)) { 35748543201SAlexander Gordeev assert(i + 1 < PCI_BAR_NUM); 35848543201SAlexander Gordeev dev->resource[i + 1] = dev->resource[i]; 359e954ce23SPeter Xu i++; 36048543201SAlexander Gordeev } 36148543201SAlexander Gordeev } else { 36248543201SAlexander Gordeev dev->resource[i] = INVALID_PHYS_ADDR; 363e954ce23SPeter Xu } 364e954ce23SPeter Xu } 365e954ce23SPeter Xu } 36666082ed6SPeter Xu 367352096c7SPeter Xu uint8_t pci_intx_line(struct pci_dev *dev) 368352096c7SPeter Xu { 369352096c7SPeter Xu return pci_config_readb(dev->bdf, PCI_INTERRUPT_LINE); 370352096c7SPeter Xu } 371352096c7SPeter Xu 372cdccea7cSAndrew Jones static void pci_cap_setup(struct pci_dev *dev, int cap_offset, int cap_id) 373cdccea7cSAndrew Jones { 374cdccea7cSAndrew Jones switch (cap_id) { 375cdccea7cSAndrew Jones case PCI_CAP_ID_MSI: 376cdccea7cSAndrew Jones dev->msi_offset = cap_offset; 377cdccea7cSAndrew Jones break; 378cdccea7cSAndrew Jones } 379cdccea7cSAndrew Jones } 380cdccea7cSAndrew Jones 38166082ed6SPeter Xu void pci_enable_defaults(struct pci_dev *dev) 38266082ed6SPeter Xu { 38366082ed6SPeter Xu /* Enable device DMA operations */ 38466082ed6SPeter Xu pci_cmd_set_clr(dev, PCI_COMMAND_MASTER, 0); 385cdccea7cSAndrew Jones pci_cap_walk(dev, pci_cap_setup); 38666082ed6SPeter Xu } 387