Add qemu 2.4.0
[kvmfornfv.git] / qemu / roms / seabios / src / hw / pci.c
1 // PCI config space access functions.
2 //
3 // Copyright (C) 2008  Kevin O'Connor <kevin@koconnor.net>
4 // Copyright (C) 2002  MandrakeSoft S.A.
5 //
6 // This file may be distributed under the terms of the GNU LGPLv3 license.
7
8 #include "malloc.h" // malloc_tmp
9 #include "output.h" // dprintf
10 #include "pci.h" // pci_config_writel
11 #include "pci_regs.h" // PCI_VENDOR_ID
12 #include "romfile.h" // romfile_loadint
13 #include "string.h" // memset
14 #include "util.h" // udelay
15 #include "x86.h" // outl
16
17 void pci_config_writel(u16 bdf, u32 addr, u32 val)
18 {
19     outl(0x80000000 | (bdf << 8) | (addr & 0xfc), PORT_PCI_CMD);
20     outl(val, PORT_PCI_DATA);
21 }
22
23 void pci_config_writew(u16 bdf, u32 addr, u16 val)
24 {
25     outl(0x80000000 | (bdf << 8) | (addr & 0xfc), PORT_PCI_CMD);
26     outw(val, PORT_PCI_DATA + (addr & 2));
27 }
28
29 void pci_config_writeb(u16 bdf, u32 addr, u8 val)
30 {
31     outl(0x80000000 | (bdf << 8) | (addr & 0xfc), PORT_PCI_CMD);
32     outb(val, PORT_PCI_DATA + (addr & 3));
33 }
34
35 u32 pci_config_readl(u16 bdf, u32 addr)
36 {
37     outl(0x80000000 | (bdf << 8) | (addr & 0xfc), PORT_PCI_CMD);
38     return inl(PORT_PCI_DATA);
39 }
40
41 u16 pci_config_readw(u16 bdf, u32 addr)
42 {
43     outl(0x80000000 | (bdf << 8) | (addr & 0xfc), PORT_PCI_CMD);
44     return inw(PORT_PCI_DATA + (addr & 2));
45 }
46
47 u8 pci_config_readb(u16 bdf, u32 addr)
48 {
49     outl(0x80000000 | (bdf << 8) | (addr & 0xfc), PORT_PCI_CMD);
50     return inb(PORT_PCI_DATA + (addr & 3));
51 }
52
53 void
54 pci_config_maskw(u16 bdf, u32 addr, u16 off, u16 on)
55 {
56     u16 val = pci_config_readw(bdf, addr);
57     val = (val & ~off) | on;
58     pci_config_writew(bdf, addr, val);
59 }
60
61 // Helper function for foreachbdf() macro - return next device
62 int
63 pci_next(int bdf, int bus)
64 {
65     if (pci_bdf_to_fn(bdf) == 0
66         && (pci_config_readb(bdf, PCI_HEADER_TYPE) & 0x80) == 0)
67         // Last found device wasn't a multi-function device - skip to
68         // the next device.
69         bdf += 8;
70     else
71         bdf += 1;
72
73     for (;;) {
74         if (pci_bdf_to_bus(bdf) != bus)
75             return -1;
76
77         u16 v = pci_config_readw(bdf, PCI_VENDOR_ID);
78         if (v != 0x0000 && v != 0xffff)
79             // Device is present.
80             return bdf;
81
82         if (pci_bdf_to_fn(bdf) == 0)
83             bdf += 8;
84         else
85             bdf += 1;
86     }
87 }
88
89 struct hlist_head PCIDevices VARVERIFY32INIT;
90 int MaxPCIBus VARFSEG;
91
92 // Check if PCI is available at all
93 int
94 pci_probe_host(void)
95 {
96     outl(0x80000000, PORT_PCI_CMD);
97     if (inl(PORT_PCI_CMD) != 0x80000000) {
98         dprintf(1, "Detected non-PCI system\n");
99         return -1;
100     }
101     return 0;
102 }
103
104 // Find all PCI devices and populate PCIDevices linked list.
105 void
106 pci_probe_devices(void)
107 {
108     dprintf(3, "PCI probe\n");
109     struct pci_device *busdevs[256];
110     memset(busdevs, 0, sizeof(busdevs));
111     struct hlist_node **pprev = &PCIDevices.first;
112     int extraroots = romfile_loadint("etc/extra-pci-roots", 0);
113     int bus = -1, lastbus = 0, rootbuses = 0, count=0;
114     while (bus < 0xff && (bus < MaxPCIBus || rootbuses < extraroots)) {
115         bus++;
116         int bdf;
117         foreachbdf(bdf, bus) {
118             // Create new pci_device struct and add to list.
119             struct pci_device *dev = malloc_tmp(sizeof(*dev));
120             if (!dev) {
121                 warn_noalloc();
122                 return;
123             }
124             memset(dev, 0, sizeof(*dev));
125             hlist_add(&dev->node, pprev);
126             pprev = &dev->node.next;
127             count++;
128
129             // Find parent device.
130             int rootbus;
131             struct pci_device *parent = busdevs[bus];
132             if (!parent) {
133                 if (bus != lastbus)
134                     rootbuses++;
135                 lastbus = bus;
136                 rootbus = rootbuses;
137                 if (bus > MaxPCIBus)
138                     MaxPCIBus = bus;
139             } else {
140                 rootbus = parent->rootbus;
141             }
142
143             // Populate pci_device info.
144             dev->bdf = bdf;
145             dev->parent = parent;
146             dev->rootbus = rootbus;
147             u32 vendev = pci_config_readl(bdf, PCI_VENDOR_ID);
148             dev->vendor = vendev & 0xffff;
149             dev->device = vendev >> 16;
150             u32 classrev = pci_config_readl(bdf, PCI_CLASS_REVISION);
151             dev->class = classrev >> 16;
152             dev->prog_if = classrev >> 8;
153             dev->revision = classrev & 0xff;
154             dev->header_type = pci_config_readb(bdf, PCI_HEADER_TYPE);
155             u8 v = dev->header_type & 0x7f;
156             if (v == PCI_HEADER_TYPE_BRIDGE || v == PCI_HEADER_TYPE_CARDBUS) {
157                 u8 secbus = pci_config_readb(bdf, PCI_SECONDARY_BUS);
158                 dev->secondary_bus = secbus;
159                 if (secbus > bus && !busdevs[secbus])
160                     busdevs[secbus] = dev;
161                 if (secbus > MaxPCIBus)
162                     MaxPCIBus = secbus;
163             }
164             dprintf(4, "PCI device %02x:%02x.%x (vd=%04x:%04x c=%04x)\n"
165                     , pci_bdf_to_bus(bdf), pci_bdf_to_dev(bdf)
166                     , pci_bdf_to_fn(bdf)
167                     , dev->vendor, dev->device, dev->class);
168         }
169     }
170     dprintf(1, "Found %d PCI devices (max PCI bus is %02x)\n", count, MaxPCIBus);
171 }
172
173 // Search for a device with the specified vendor and device ids.
174 struct pci_device *
175 pci_find_device(u16 vendid, u16 devid)
176 {
177     struct pci_device *pci;
178     foreachpci(pci) {
179         if (pci->vendor == vendid && pci->device == devid)
180             return pci;
181     }
182     return NULL;
183 }
184
185 // Search for a device with the specified class id.
186 struct pci_device *
187 pci_find_class(u16 classid)
188 {
189     struct pci_device *pci;
190     foreachpci(pci) {
191         if (pci->class == classid)
192             return pci;
193     }
194     return NULL;
195 }
196
197 int pci_init_device(const struct pci_device_id *ids
198                     , struct pci_device *pci, void *arg)
199 {
200     while (ids->vendid || ids->class_mask) {
201         if ((ids->vendid == PCI_ANY_ID || ids->vendid == pci->vendor) &&
202             (ids->devid == PCI_ANY_ID || ids->devid == pci->device) &&
203             !((ids->class ^ pci->class) & ids->class_mask)) {
204             if (ids->func)
205                 ids->func(pci, arg);
206             return 0;
207         }
208         ids++;
209     }
210     return -1;
211 }
212
213 struct pci_device *
214 pci_find_init_device(const struct pci_device_id *ids, void *arg)
215 {
216     struct pci_device *pci;
217     foreachpci(pci) {
218         if (pci_init_device(ids, pci, arg) == 0)
219             return pci;
220     }
221     return NULL;
222 }
223
224 u8 pci_find_capability(struct pci_device *pci, u8 cap_id)
225 {
226     int i;
227     u8 cap;
228     u16 status = pci_config_readw(pci->bdf, PCI_STATUS);
229
230     if (!(status & PCI_STATUS_CAP_LIST))
231         return 0;
232
233     cap = pci_config_readb(pci->bdf, PCI_CAPABILITY_LIST);
234     for (i = 0; cap && i <= 0xff; i++) {
235         if (pci_config_readb(pci->bdf, cap + PCI_CAP_LIST_ID) == cap_id)
236             return cap;
237         cap = pci_config_readb(pci->bdf, cap + PCI_CAP_LIST_NEXT);
238     }
239
240     return 0;
241 }
242
243 /* Test whether bridge support forwarding of transactions
244  * of a specific type.
245  * Note: disables bridge's window registers as a side effect.
246  */
247 int pci_bridge_has_region(struct pci_device *pci,
248         enum pci_region_type region_type)
249 {
250     u8 base;
251
252     switch (region_type) {
253         case PCI_REGION_TYPE_IO:
254             base = PCI_IO_BASE;
255             break;
256         case PCI_REGION_TYPE_PREFMEM:
257             base = PCI_PREF_MEMORY_BASE;
258             break;
259         default:
260             /* Regular memory support is mandatory */
261             return 1;
262     }
263
264     pci_config_writeb(pci->bdf, base, 0xFF);
265
266     return pci_config_readb(pci->bdf, base) != 0;
267 }
268
269 void
270 pci_reboot(void)
271 {
272     u8 v = inb(PORT_PCI_REBOOT) & ~6;
273     outb(v|2, PORT_PCI_REBOOT); /* Request hard reset */
274     udelay(50);
275     outb(v|6, PORT_PCI_REBOOT); /* Actually do the reset */
276     udelay(50);
277 }