xref: /kvm-unit-tests/lib/pci.c (revision cb026028c8e80271208587a6e89e876aa2c6f11a)
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) {
52903b0516SPeter Xu 		printf("MSI: dev 0x%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 {
1154d6cefa9SPeter Xu 	return pci_config_readl(dev->bdf, PCI_BASE_ADDRESS_0 +
1164d6cefa9SPeter Xu 				bar_num * 4);
1177aa83307SAlexander Gordeev }
1187aa83307SAlexander Gordeev 
11978609f0eSAlexander Gordeev static phys_addr_t __pci_bar_get_addr(struct pci_dev *dev, int bar_num)
1204932b58aSMichael S. Tsirkin {
1217aa83307SAlexander Gordeev 	uint32_t bar = pci_bar_get(dev, bar_num);
1222455ef20SAlexander Gordeev 	uint32_t mask = pci_bar_mask(bar);
1232455ef20SAlexander Gordeev 	uint64_t addr = bar & mask;
124647d2ab7SAlexander Gordeev 	phys_addr_t phys_addr;
125ebb58e7eSAlexander Gordeev 
1262455ef20SAlexander Gordeev 	if (pci_bar_is64(dev, bar_num))
1272455ef20SAlexander Gordeev 		addr |= (uint64_t)pci_bar_get(dev, bar_num + 1) << 32;
1282455ef20SAlexander Gordeev 
1294d6cefa9SPeter Xu 	phys_addr = pci_translate_addr(dev->bdf, addr);
130647d2ab7SAlexander Gordeev 	assert(phys_addr != INVALID_PHYS_ADDR);
131647d2ab7SAlexander Gordeev 
132647d2ab7SAlexander Gordeev 	return phys_addr;
1332455ef20SAlexander Gordeev }
1342455ef20SAlexander Gordeev 
13578609f0eSAlexander Gordeev phys_addr_t pci_bar_get_addr(struct pci_dev *dev, int bar_num)
13678609f0eSAlexander Gordeev {
13778609f0eSAlexander Gordeev 	return dev->resource[bar_num];
13878609f0eSAlexander Gordeev }
13978609f0eSAlexander Gordeev 
1404d6cefa9SPeter Xu void pci_bar_set_addr(struct pci_dev *dev, int bar_num, phys_addr_t addr)
141647f92c7SAlexander Gordeev {
142647f92c7SAlexander Gordeev 	int off = PCI_BASE_ADDRESS_0 + bar_num * 4;
143647f92c7SAlexander Gordeev 
1444d6cefa9SPeter Xu 	pci_config_writel(dev->bdf, off, (uint32_t)addr);
14578609f0eSAlexander Gordeev 	dev->resource[bar_num] = addr;
146647f92c7SAlexander Gordeev 
14778609f0eSAlexander Gordeev 	if (pci_bar_is64(dev, bar_num)) {
14878609f0eSAlexander Gordeev 		assert(bar_num + 1 < PCI_BAR_NUM);
14978609f0eSAlexander Gordeev 		pci_config_writel(dev->bdf, off + 4, (uint32_t)(addr >> 32));
15078609f0eSAlexander Gordeev 		dev->resource[bar_num + 1] = dev->resource[bar_num];
15178609f0eSAlexander Gordeev 	}
152647f92c7SAlexander Gordeev }
153647f92c7SAlexander Gordeev 
1542455ef20SAlexander Gordeev /*
1552455ef20SAlexander Gordeev  * To determine the amount of address space needed by a PCI device,
1562455ef20SAlexander Gordeev  * one must save the original value of the BAR, write a value of
1572455ef20SAlexander Gordeev  * all 1's to the register, and then read it back. The amount of
1582455ef20SAlexander Gordeev  * memory can be then determined by masking the information bits,
1592455ef20SAlexander Gordeev  * performing a bitwise NOT, and incrementing the value by 1.
1602455ef20SAlexander Gordeev  *
1612455ef20SAlexander Gordeev  * The following pci_bar_size_helper() and pci_bar_size() functions
1622455ef20SAlexander Gordeev  * implement the algorithm.
1632455ef20SAlexander Gordeev  */
1644d6cefa9SPeter Xu static uint32_t pci_bar_size_helper(struct pci_dev *dev, int bar_num)
1652455ef20SAlexander Gordeev {
1662455ef20SAlexander Gordeev 	int off = PCI_BASE_ADDRESS_0 + bar_num * 4;
1674d6cefa9SPeter Xu 	uint16_t bdf = dev->bdf;
1682455ef20SAlexander Gordeev 	uint32_t bar, val;
1692455ef20SAlexander Gordeev 
1704d6cefa9SPeter Xu 	bar = pci_config_readl(bdf, off);
1714d6cefa9SPeter Xu 	pci_config_writel(bdf, off, ~0u);
1724d6cefa9SPeter Xu 	val = pci_config_readl(bdf, off);
1734d6cefa9SPeter Xu 	pci_config_writel(bdf, off, bar);
1742455ef20SAlexander Gordeev 
1752455ef20SAlexander Gordeev 	return val;
1762455ef20SAlexander Gordeev }
1772455ef20SAlexander Gordeev 
1784d6cefa9SPeter Xu phys_addr_t pci_bar_size(struct pci_dev *dev, int bar_num)
1792455ef20SAlexander Gordeev {
1802455ef20SAlexander Gordeev 	uint32_t bar, size;
1812455ef20SAlexander Gordeev 
1822455ef20SAlexander Gordeev 	size = pci_bar_size_helper(dev, bar_num);
1832455ef20SAlexander Gordeev 	if (!size)
1842455ef20SAlexander Gordeev 		return 0;
1852455ef20SAlexander Gordeev 
1862455ef20SAlexander Gordeev 	bar = pci_bar_get(dev, bar_num);
1872455ef20SAlexander Gordeev 	size &= pci_bar_mask(bar);
1882455ef20SAlexander Gordeev 
1892455ef20SAlexander Gordeev 	if (pci_bar_is64(dev, bar_num)) {
1902455ef20SAlexander Gordeev 		phys_addr_t size64 = pci_bar_size_helper(dev, bar_num + 1);
1912455ef20SAlexander Gordeev 		size64 = (size64 << 32) | size;
1922455ef20SAlexander Gordeev 
1932455ef20SAlexander Gordeev 		return ~size64 + 1;
1942455ef20SAlexander Gordeev 	} else {
1952455ef20SAlexander Gordeev 		return ~size + 1;
1962455ef20SAlexander Gordeev 	}
1974932b58aSMichael S. Tsirkin }
1984932b58aSMichael S. Tsirkin 
1994d6cefa9SPeter Xu bool pci_bar_is_memory(struct pci_dev *dev, int bar_num)
2004932b58aSMichael S. Tsirkin {
2017aa83307SAlexander Gordeev 	uint32_t bar = pci_bar_get(dev, bar_num);
202ebb58e7eSAlexander Gordeev 
2034932b58aSMichael S. Tsirkin 	return !(bar & PCI_BASE_ADDRESS_SPACE_IO);
2044932b58aSMichael S. Tsirkin }
2054932b58aSMichael S. Tsirkin 
2064d6cefa9SPeter Xu bool pci_bar_is_valid(struct pci_dev *dev, int bar_num)
2074932b58aSMichael S. Tsirkin {
2089e292d10SAlexander Gordeev 	return dev->resource[bar_num] != INVALID_PHYS_ADDR;
2094932b58aSMichael S. Tsirkin }
2102455ef20SAlexander Gordeev 
2114d6cefa9SPeter Xu bool pci_bar_is64(struct pci_dev *dev, int bar_num)
2122455ef20SAlexander Gordeev {
2132455ef20SAlexander Gordeev 	uint32_t bar = pci_bar_get(dev, bar_num);
2142455ef20SAlexander Gordeev 
2152455ef20SAlexander Gordeev 	if (bar & PCI_BASE_ADDRESS_SPACE_IO)
2162455ef20SAlexander Gordeev 		return false;
2172455ef20SAlexander Gordeev 
2182455ef20SAlexander Gordeev 	return (bar & PCI_BASE_ADDRESS_MEM_TYPE_MASK) ==
2192455ef20SAlexander Gordeev 		      PCI_BASE_ADDRESS_MEM_TYPE_64;
2202455ef20SAlexander Gordeev }
221e4611520SAlexander Gordeev 
2224d6cefa9SPeter Xu void pci_bar_print(struct pci_dev *dev, int bar_num)
223e4611520SAlexander Gordeev {
224e4611520SAlexander Gordeev 	phys_addr_t size, start, end;
225e4611520SAlexander Gordeev 	uint32_t bar;
226e4611520SAlexander Gordeev 
2279e292d10SAlexander Gordeev 	if (!pci_bar_is_valid(dev, bar_num))
228e4611520SAlexander Gordeev 		return;
229e4611520SAlexander Gordeev 
230e4611520SAlexander Gordeev 	bar = pci_bar_get(dev, bar_num);
2319e292d10SAlexander Gordeev 	size = pci_bar_size(dev, bar_num);
232e4611520SAlexander Gordeev 	start = pci_bar_get_addr(dev, bar_num);
233e4611520SAlexander Gordeev 	end = start + size - 1;
234e4611520SAlexander Gordeev 
235e4611520SAlexander Gordeev 	if (pci_bar_is64(dev, bar_num)) {
236e4611520SAlexander Gordeev 		printf("BAR#%d,%d [%" PRIx64 "-%" PRIx64 " ",
237e4611520SAlexander Gordeev 		       bar_num, bar_num + 1, start, end);
238e4611520SAlexander Gordeev 	} else {
239e4611520SAlexander Gordeev 		printf("BAR#%d [%02x-%02x ",
240e4611520SAlexander Gordeev 		       bar_num, (uint32_t)start, (uint32_t)end);
241e4611520SAlexander Gordeev 	}
242e4611520SAlexander Gordeev 
243e4611520SAlexander Gordeev 	if (bar & PCI_BASE_ADDRESS_SPACE_IO) {
244e4611520SAlexander Gordeev 		printf("PIO");
245e4611520SAlexander Gordeev 	} else {
246e4611520SAlexander Gordeev 		printf("MEM");
247e4611520SAlexander Gordeev 		switch (bar & PCI_BASE_ADDRESS_MEM_TYPE_MASK) {
248e4611520SAlexander Gordeev 		case PCI_BASE_ADDRESS_MEM_TYPE_32:
249e4611520SAlexander Gordeev 			printf("32");
250e4611520SAlexander Gordeev 			break;
251e4611520SAlexander Gordeev 		case PCI_BASE_ADDRESS_MEM_TYPE_1M:
252e4611520SAlexander Gordeev 			printf("1M");
253e4611520SAlexander Gordeev 			break;
254e4611520SAlexander Gordeev 		case PCI_BASE_ADDRESS_MEM_TYPE_64:
255e4611520SAlexander Gordeev 			printf("64");
256e4611520SAlexander Gordeev 			break;
257e4611520SAlexander Gordeev 		default:
258e4611520SAlexander Gordeev 			assert(0);
259e4611520SAlexander Gordeev 		}
260e4611520SAlexander Gordeev 	}
261e4611520SAlexander Gordeev 
262e4611520SAlexander Gordeev 	if (bar & PCI_BASE_ADDRESS_MEM_PREFETCH)
263e4611520SAlexander Gordeev 		printf("/p");
264e4611520SAlexander Gordeev 
265e4611520SAlexander Gordeev 	printf("]");
266e4611520SAlexander Gordeev }
267e4611520SAlexander Gordeev 
268*cb026028SAlexander Gordeev void pci_dev_print_id(struct pci_dev *dev)
269e4611520SAlexander Gordeev {
270*cb026028SAlexander Gordeev 	pcidevaddr_t bdf = dev->bdf;
271*cb026028SAlexander Gordeev 
272*cb026028SAlexander Gordeev 	printf("00.%02x.%1x %04x:%04x", bdf / 8, bdf % 8,
273*cb026028SAlexander Gordeev 		pci_config_readw(bdf, PCI_VENDOR_ID),
274*cb026028SAlexander Gordeev 		pci_config_readw(bdf, PCI_DEVICE_ID));
275e4611520SAlexander Gordeev }
276e4611520SAlexander Gordeev 
27782f2f21aSAndrew Jones static void pci_cap_print(struct pci_dev *dev, int cap_offset, int cap_id)
27882f2f21aSAndrew Jones {
27982f2f21aSAndrew Jones 	switch (cap_id) {
28082f2f21aSAndrew Jones 	case PCI_CAP_ID_MSI: {
28182f2f21aSAndrew Jones 		uint16_t control = pci_config_readw(dev->bdf, cap_offset + PCI_MSI_FLAGS);
28282f2f21aSAndrew Jones 		printf("\tMSI,%s-bit capability ", control & PCI_MSI_FLAGS_64BIT ? "64" : "32");
28382f2f21aSAndrew Jones 		break;
28482f2f21aSAndrew Jones 	}
28582f2f21aSAndrew Jones 	default:
28682f2f21aSAndrew Jones 		printf("\tcapability 0x%02x ", cap_id);
28782f2f21aSAndrew Jones 		break;
28882f2f21aSAndrew Jones 	}
28982f2f21aSAndrew Jones 	printf("at offset 0x%02x\n", cap_offset);
29082f2f21aSAndrew Jones }
29182f2f21aSAndrew Jones 
292*cb026028SAlexander Gordeev void pci_dev_print(struct pci_dev *dev)
293e4611520SAlexander Gordeev {
294*cb026028SAlexander Gordeev 	pcidevaddr_t bdf = dev->bdf;
295*cb026028SAlexander Gordeev 	uint8_t header = pci_config_readb(bdf, PCI_HEADER_TYPE);
296*cb026028SAlexander Gordeev 	uint8_t progif = pci_config_readb(bdf, PCI_CLASS_PROG);
297*cb026028SAlexander Gordeev 	uint8_t subclass = pci_config_readb(bdf, PCI_CLASS_DEVICE);
298*cb026028SAlexander Gordeev 	uint8_t class = pci_config_readb(bdf, PCI_CLASS_DEVICE + 1);
299e4611520SAlexander Gordeev 	int i;
300e4611520SAlexander Gordeev 
301e4611520SAlexander Gordeev 	pci_dev_print_id(dev);
302e4611520SAlexander Gordeev 	printf(" type %02x progif %02x class %02x subclass %02x\n",
303e4611520SAlexander Gordeev 	       header, progif, class, subclass);
304e4611520SAlexander Gordeev 
305*cb026028SAlexander Gordeev 	pci_cap_walk(dev, pci_cap_print);
30682f2f21aSAndrew Jones 
307e4611520SAlexander Gordeev 	if ((header & PCI_HEADER_TYPE_MASK) != PCI_HEADER_TYPE_NORMAL)
308e4611520SAlexander Gordeev 		return;
309e4611520SAlexander Gordeev 
310e954ce23SPeter Xu 	for (i = 0; i < PCI_BAR_NUM; i++) {
311*cb026028SAlexander Gordeev 		if (pci_bar_is_valid(dev, i)) {
312e4611520SAlexander Gordeev 			printf("\t");
313*cb026028SAlexander Gordeev 			pci_bar_print(dev, i);
314e4611520SAlexander Gordeev 			printf("\n");
315e4611520SAlexander Gordeev 		}
316*cb026028SAlexander Gordeev 		if (pci_bar_is64(dev, i))
317e4611520SAlexander Gordeev 			i++;
318e4611520SAlexander Gordeev 	}
319e4611520SAlexander Gordeev }
320e4611520SAlexander Gordeev 
321e4611520SAlexander Gordeev void pci_print(void)
322e4611520SAlexander Gordeev {
323*cb026028SAlexander Gordeev 	pcidevaddr_t devfn;
324*cb026028SAlexander Gordeev 	struct pci_dev pci_dev;
325e4611520SAlexander Gordeev 
326*cb026028SAlexander Gordeev 	for (devfn = 0; devfn < PCI_DEVFN_MAX; ++devfn) {
327*cb026028SAlexander Gordeev 		if (pci_dev_exists(devfn)) {
328*cb026028SAlexander Gordeev 			pci_dev_init(&pci_dev, devfn);
329*cb026028SAlexander Gordeev 			pci_dev_print(&pci_dev);
330*cb026028SAlexander Gordeev 		}
331e4611520SAlexander Gordeev 	}
332e4611520SAlexander Gordeev }
333e954ce23SPeter Xu 
33478609f0eSAlexander Gordeev void pci_dev_init(struct pci_dev *dev, pcidevaddr_t bdf)
335e954ce23SPeter Xu {
336e954ce23SPeter Xu 	int i;
337e954ce23SPeter Xu 
33878609f0eSAlexander Gordeev 	memset(dev, 0, sizeof(*dev));
33978609f0eSAlexander Gordeev 	dev->bdf = bdf;
34078609f0eSAlexander Gordeev 
341e954ce23SPeter Xu 	for (i = 0; i < PCI_BAR_NUM; i++) {
34248543201SAlexander Gordeev 		if (pci_bar_size(dev, i)) {
34378609f0eSAlexander Gordeev 			dev->resource[i] = __pci_bar_get_addr(dev, i);
344e954ce23SPeter Xu 			if (pci_bar_is64(dev, i)) {
34548543201SAlexander Gordeev 				assert(i + 1 < PCI_BAR_NUM);
34648543201SAlexander Gordeev 				dev->resource[i + 1] = dev->resource[i];
347e954ce23SPeter Xu 				i++;
34848543201SAlexander Gordeev 			}
34948543201SAlexander Gordeev 		} else {
35048543201SAlexander Gordeev 			dev->resource[i] = INVALID_PHYS_ADDR;
351e954ce23SPeter Xu 		}
352e954ce23SPeter Xu 	}
353e954ce23SPeter Xu }
35466082ed6SPeter Xu 
355352096c7SPeter Xu uint8_t pci_intx_line(struct pci_dev *dev)
356352096c7SPeter Xu {
357352096c7SPeter Xu 	return pci_config_readb(dev->bdf, PCI_INTERRUPT_LINE);
358352096c7SPeter Xu }
359352096c7SPeter Xu 
360cdccea7cSAndrew Jones static void pci_cap_setup(struct pci_dev *dev, int cap_offset, int cap_id)
361cdccea7cSAndrew Jones {
362cdccea7cSAndrew Jones 	switch (cap_id) {
363cdccea7cSAndrew Jones 	case PCI_CAP_ID_MSI:
364cdccea7cSAndrew Jones 		dev->msi_offset = cap_offset;
365cdccea7cSAndrew Jones 		break;
366cdccea7cSAndrew Jones 	}
367cdccea7cSAndrew Jones }
368cdccea7cSAndrew Jones 
36966082ed6SPeter Xu void pci_enable_defaults(struct pci_dev *dev)
37066082ed6SPeter Xu {
37166082ed6SPeter Xu 	/* Enable device DMA operations */
37266082ed6SPeter Xu 	pci_cmd_set_clr(dev, PCI_COMMAND_MASTER, 0);
373cdccea7cSAndrew Jones 	pci_cap_walk(dev, pci_cap_setup);
37466082ed6SPeter Xu }
375