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
pci_cap_walk(struct pci_dev * dev,pci_cap_handler_t handler)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
pci_msi_set_enable(struct pci_dev * dev,bool enabled)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
pci_setup_msi(struct pci_dev * dev,uint64_t msi_addr,uint32_t msi_data)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) {
52fd6aada0SRadim 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
pci_cmd_set_clr(struct pci_dev * dev,uint16_t set,uint16_t clr)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
pci_dev_exists(pcidevaddr_t dev)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. */
pci_find_dev(uint16_t vendor_id,uint16_t device_id)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
pci_bar_mask(uint32_t bar)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
pci_bar_get(struct pci_dev * dev,int bar_num)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
__pci_bar_get_addr(struct pci_dev * dev,int bar_num)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
pci_bar_get_addr(struct pci_dev * dev,int bar_num)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
pci_bar_set_addr(struct pci_dev * dev,int bar_num,phys_addr_t addr)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 */
pci_bar_size_helper(struct pci_dev * dev,int bar_num)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
pci_bar_size(struct pci_dev * dev,int bar_num)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
pci_bar_is_memory(struct pci_dev * dev,int bar_num)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
pci_bar_is_valid(struct pci_dev * dev,int bar_num)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
pci_bar_is64(struct pci_dev * dev,int bar_num)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
pci_bar_print(struct pci_dev * dev,int bar_num)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 {
251*d26193a0SRoman Bolshakov printf("BAR#%d [%02" PRIx32 "-%02" PRIx32 " ",
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
pci_dev_print_id(struct pci_dev * dev)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
pci_cap_print(struct pci_dev * dev,int cap_offset,int cap_id)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:
298fd6aada0SRadim Krčmář printf("\tcapability %#04x ", cap_id);
29982f2f21aSAndrew Jones break;
30082f2f21aSAndrew Jones }
301fd6aada0SRadim Krčmář printf("at offset %#04x\n", cap_offset);
30282f2f21aSAndrew Jones }
30382f2f21aSAndrew Jones
pci_dev_print(struct pci_dev * dev)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
pci_print(void)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
pci_dev_init(struct pci_dev * dev,pcidevaddr_t bdf)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
pci_intx_line(struct pci_dev * dev)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
pci_cap_setup(struct pci_dev * dev,int cap_offset,int cap_id)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
pci_enable_defaults(struct pci_dev * dev)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