Annotation of sys/arch/i386/pci/pci_bus_fixup.c, Revision 1.1.1.1
1.1 nbrk 1: /* $OpenBSD: pci_bus_fixup.c,v 1.9 2006/08/18 22:18:18 kettenis Exp $ */
2: /* $NetBSD: pci_bus_fixup.c,v 1.1 1999/11/17 07:32:58 thorpej Exp $ */
3:
4: /*
5: * Copyright (c) 1999, by UCHIYAMA Yasushi
6: * All rights reserved.
7: *
8: * Redistribution and use in source and binary forms, with or without
9: * modification, are permitted provided that the following conditions
10: * are met:
11: * 1. Redistributions of source code must retain the above copyright
12: * notice, this list of conditions and the following disclaimer.
13: * 2. The name of the developer may NOT be used to endorse or promote products
14: * derived from this software without specific prior written permission.
15: *
16: * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17: * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18: * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19: * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20: * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21: * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22: * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23: * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24: * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25: * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26: * SUCH DAMAGE.
27: */
28:
29: /*
30: * PCI bus renumbering support.
31: */
32:
33: #include <sys/param.h>
34: #include <sys/systm.h>
35: #include <sys/kernel.h>
36: #include <sys/device.h>
37:
38: #include <machine/bus.h>
39:
40: #include <dev/pci/pcireg.h>
41: #include <dev/pci/pcivar.h>
42: #include <dev/pci/pcidevs.h>
43: #include <dev/pci/ppbreg.h>
44:
45: #include <i386/pci/pcibiosvar.h>
46:
47: int pci_bus_check(pci_chipset_tag_t, int);
48: int pci_bus_assign(pci_chipset_tag_t, int);
49: void pcibus_print_devid(pci_chipset_tag_t, pcitag_t);
50:
51: int
52: pci_bus_check(pci_chipset_tag_t pc, int bus)
53: {
54: int device, maxdevs, function, nfuncs, bus_max, bus_sub;
55: const struct pci_quirkdata *qd;
56: pcireg_t reg;
57: pcitag_t tag;
58:
59: bus_max = bus;
60:
61: maxdevs = pci_bus_maxdevs(pc, bus);
62: for (device = 0; device < maxdevs; device++) {
63: tag = pci_make_tag(pc, bus, device, 0);
64: reg = pci_conf_read(pc, tag, PCI_ID_REG);
65:
66: /* can't be that many */
67: if (bus_max == 255)
68: break;
69:
70: /* Invalid vendor ID value? */
71: if (PCI_VENDOR(reg) == PCI_VENDOR_INVALID)
72: continue;
73: /* XXX Not invalid, but we've done this ~forever. */
74: if (PCI_VENDOR(reg) == 0)
75: continue;
76:
77: qd = pci_lookup_quirkdata(PCI_VENDOR(reg), PCI_PRODUCT(reg));
78:
79: reg = pci_conf_read(pc, tag, PCI_BHLC_REG);
80: if (PCI_HDRTYPE_MULTIFN(reg) ||
81: (qd != NULL &&
82: (qd->quirks & PCI_QUIRK_MULTIFUNCTION) != 0))
83: nfuncs = 8;
84: else
85: nfuncs = 1;
86:
87: for (function = 0; function < nfuncs; function++) {
88: tag = pci_make_tag(pc, bus, device, function);
89: reg = pci_conf_read(pc, tag, PCI_ID_REG);
90:
91: /* Invalid vendor ID value? */
92: if (PCI_VENDOR(reg) == PCI_VENDOR_INVALID)
93: continue;
94: /* XXX Not invalid, but we've done this ~forever. */
95: if (PCI_VENDOR(reg) == 0)
96: continue;
97:
98: reg = pci_conf_read(pc, tag, PCI_CLASS_REG);
99: if (PCI_CLASS(reg) == PCI_CLASS_BRIDGE &&
100: (PCI_SUBCLASS(reg) == PCI_SUBCLASS_BRIDGE_PCI ||
101: PCI_SUBCLASS(reg) == PCI_SUBCLASS_BRIDGE_CARDBUS)) {
102:
103: reg = pci_conf_read(pc, tag, PPB_REG_BUSINFO);
104: if (PPB_BUSINFO_PRIMARY(reg) != bus) {
105: if (pcibios_flags & PCIBIOS_VERBOSE) {
106: pcibus_print_devid(pc, tag);
107: printf("Mismatched primary bus: "
108: "primary %d, secondary %d, "
109: "subordinate %d\n",
110: PPB_BUSINFO_PRIMARY(reg),
111: PPB_BUSINFO_SECONDARY(reg),
112: PPB_BUSINFO_SUBORDINATE(reg));
113: }
114: return (-1);
115: }
116: if (PPB_BUSINFO_SECONDARY(reg) <= bus) {
117: if (pcibios_flags & PCIBIOS_VERBOSE) {
118: pcibus_print_devid(pc, tag);
119: printf("Incorrect secondary bus: "
120: "primary %d, secondary %d, "
121: "subordinate %d\n",
122: PPB_BUSINFO_PRIMARY(reg),
123: PPB_BUSINFO_SECONDARY(reg),
124: PPB_BUSINFO_SUBORDINATE(reg));
125: }
126: return (-1);
127: }
128:
129: /* Scan subordinate bus. */
130: bus_sub = pci_bus_check(pc,
131: PPB_BUSINFO_SECONDARY(reg));
132: if (bus_sub == -1)
133: return (-1);
134:
135: if (PPB_BUSINFO_SUBORDINATE(reg) < bus_sub) {
136: if (pcibios_flags & PCIBIOS_VERBOSE) {
137: pcibus_print_devid(pc, tag);
138: printf("Incorrect subordinate bus %d: "
139: "primary %d, secondary %d, "
140: "subordinate %d\n", bus_sub,
141: PPB_BUSINFO_PRIMARY(reg),
142: PPB_BUSINFO_SECONDARY(reg),
143: PPB_BUSINFO_SUBORDINATE(reg));
144: }
145: return (-1);
146: }
147:
148: bus_max = (bus_sub > bus_max) ?
149: bus_sub : bus_max;
150: }
151: }
152: }
153:
154: return (bus_max); /* last # of subordinate bus */
155: }
156:
157: int
158: pci_bus_assign(pci_chipset_tag_t pc, int bus)
159: {
160: static int bridge_cnt;
161: int bridge, device, maxdevs, function, nfuncs, bus_max, bus_sub;
162: const struct pci_quirkdata *qd;
163: pcireg_t reg;
164: pcitag_t tag;
165:
166: bus_max = bus;
167:
168: maxdevs = pci_bus_maxdevs(pc, bus);
169: for (device = 0; device < maxdevs; device++) {
170: tag = pci_make_tag(pc, bus, device, 0);
171: reg = pci_conf_read(pc, tag, PCI_ID_REG);
172:
173: /* can't be that many */
174: if (bus_max == 255)
175: break;
176:
177: /* Invalid vendor ID value? */
178: if (PCI_VENDOR(reg) == PCI_VENDOR_INVALID)
179: continue;
180: /* XXX Not invalid, but we've done this ~forever. */
181: if (PCI_VENDOR(reg) == 0)
182: continue;
183:
184: qd = pci_lookup_quirkdata(PCI_VENDOR(reg), PCI_PRODUCT(reg));
185:
186: reg = pci_conf_read(pc, tag, PCI_BHLC_REG);
187: if (PCI_HDRTYPE_MULTIFN(reg) ||
188: (qd != NULL &&
189: (qd->quirks & PCI_QUIRK_MULTIFUNCTION) != 0))
190: nfuncs = 8;
191: else
192: nfuncs = 1;
193:
194: for (function = 0; function < nfuncs; function++) {
195: tag = pci_make_tag(pc, bus, device, function);
196: reg = pci_conf_read(pc, tag, PCI_ID_REG);
197:
198: /* Invalid vendor ID value? */
199: if (PCI_VENDOR(reg) == PCI_VENDOR_INVALID)
200: continue;
201: /* XXX Not invalid, but we've done this ~forever. */
202: if (PCI_VENDOR(reg) == 0)
203: continue;
204:
205: reg = pci_conf_read(pc, tag, PCI_CLASS_REG);
206: if (PCI_CLASS(reg) == PCI_CLASS_BRIDGE &&
207: (PCI_SUBCLASS(reg) == PCI_SUBCLASS_BRIDGE_PCI ||
208: PCI_SUBCLASS(reg) == PCI_SUBCLASS_BRIDGE_CARDBUS)) {
209: /* Assign the bridge's secondary bus #. */
210: bus_max++;
211:
212: reg = pci_conf_read(pc, tag, PPB_REG_BUSINFO);
213: reg &= 0xff000000;
214: reg |= bus | (bus_max << 8) | (0xff << 16);
215: pci_conf_write(pc, tag, PPB_REG_BUSINFO, reg);
216:
217: /* Scan subordinate bus. */
218: bus_sub = pci_bus_assign(pc, bus_max);
219:
220: /* Configure the bridge. */
221: reg &= 0xff000000;
222: reg |= bus | (bus_max << 8) | (bus_sub << 16);
223: pci_conf_write(pc, tag, PPB_REG_BUSINFO, reg);
224:
225: if (pcibios_flags & PCIBIOS_VERBOSE) {
226: /* Assign the bridge #. */
227: bridge = bridge_cnt++;
228:
229: printf("PCI bridge %d: primary %d, "
230: "secondary %d, subordinate %d\n",
231: bridge, bus, bus_max, bus_sub);
232: }
233:
234: /* Next bridge's secondary bus #. */
235: bus_max = (bus_sub > bus_max) ?
236: bus_sub : bus_max;
237: }
238: }
239: }
240:
241: return (bus_max); /* last # of subordinate bus */
242: }
243:
244: int
245: pci_bus_fixup(pci_chipset_tag_t pc, int bus)
246: {
247: int bus_max;
248:
249: bus_max = pci_bus_check(pc, bus);
250: if (bus_max != -1)
251: return (bus_max);
252:
253: if (pcibios_flags & PCIBIOS_VERBOSE)
254: printf("PCI bus renumbering needed\n");
255: return pci_bus_assign(pc, bus);
256: }
257:
258: void
259: pcibus_print_devid(pci_chipset_tag_t pc, pcitag_t tag)
260: {
261: int bus, device, function;
262: pcireg_t id;
263:
264: id = pci_conf_read(pc, tag, PCI_ID_REG);
265: pci_decompose_tag(pc, tag, &bus, &device, &function);
266: printf("%03d:%02d:%d %04x:%04x\n", bus, device, function,
267: PCI_VENDOR(id), PCI_PRODUCT(id));
268: }
CVSweb