xref: /kvm-unit-tests/lib/pci.c (revision 19daf1c56a314e672a9f2bc88dc4e94508b0ea62)
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