1 /* 2 * Copyright (c) 1997, Stefan Esser <se@freebsd.org> 3 * Copyright (c) 2000, Michael Smith <msmith@freebsd.org> 4 * Copyright (c) 2000, BSDi 5 * All rights reserved. 6 * 7 * Redistribution and use in source and binary forms, with or without 8 * modification, are permitted provided that the following conditions 9 * are met: 10 * 1. Redistributions of source code must retain the above copyright 11 * notice unmodified, this list of conditions, and the following 12 * disclaimer. 13 * 2. Redistributions in binary form must reproduce the above copyright 14 * notice, this list of conditions and the following disclaimer in the 15 * documentation and/or other materials provided with the distribution. 16 * 17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 27 * 28 * $FreeBSD$ 29 * 30 */ 31 32 #include <sys/param.h> /* XXX trim includes */ 33 #include <sys/systm.h> 34 #include <sys/bus.h> 35 #include <sys/kernel.h> 36 #include <sys/module.h> 37 #include <sys/malloc.h> 38 #include <vm/vm.h> 39 #include <vm/pmap.h> 40 #include <machine/md_var.h> 41 #include <pci/pcivar.h> 42 #include <pci/pcireg.h> 43 #include <isa/isavar.h> 44 #include <machine/nexusvar.h> 45 #include <machine/pci_cfgreg.h> 46 #include <machine/segments.h> 47 #include <machine/pc/bios.h> 48 49 #ifdef APIC_IO 50 #include <machine/smp.h> 51 #endif /* APIC_IO */ 52 53 #include "pcib_if.h" 54 55 #define PRVERB(a) printf a 56 57 static int cfgmech; 58 static int devmax; 59 static int usebios; 60 static int enable_pcibios = 0; 61 62 TUNABLE_INT("hw.pci.enable_pcibios", &enable_pcibios); 63 64 static int pci_cfgintr_unique(struct PIR_entry *pe, int pin); 65 static int pci_cfgintr_linked(struct PIR_entry *pe, int pin); 66 static int pci_cfgintr_search(struct PIR_entry *pe, int bus, int device, int matchpin, int pin); 67 static int pci_cfgintr_virgin(struct PIR_entry *pe, int pin); 68 69 static int pcibios_cfgread(int bus, int slot, int func, int reg, int bytes); 70 static void pcibios_cfgwrite(int bus, int slot, int func, int reg, int data, int bytes); 71 static int pcibios_cfgopen(void); 72 static int pcireg_cfgread(int bus, int slot, int func, int reg, int bytes); 73 static void pcireg_cfgwrite(int bus, int slot, int func, int reg, int data, int bytes); 74 static int pcireg_cfgopen(void); 75 76 static struct PIR_table *pci_route_table; 77 static int pci_route_count; 78 79 /* 80 * Some BIOS writers seem to want to ignore the spec and put 81 * 0 in the intline rather than 255 to indicate none. Some use 82 * numbers in the range 128-254 to indicate something strange and 83 * apparently undocumented anywhere. Assume these are completely bogus 84 * and map them to 255, which means "none". 85 */ 86 static __inline__ int 87 pci_i386_map_intline(int line) 88 { 89 if (line == 0 || line >= 128) 90 return (255); 91 return (line); 92 } 93 94 int 95 pci_pcibios_active(void) 96 { 97 return usebios; 98 } 99 100 int 101 pci_kill_pcibios(void) 102 { 103 usebios = 0; 104 return pcireg_cfgopen() != 0; 105 } 106 107 static u_int16_t 108 pcibios_get_version(void) 109 { 110 struct bios_regs args; 111 112 if (PCIbios.entry == 0) { 113 PRVERB(("pcibios: No call entry point\n")); 114 return (0); 115 } 116 args.eax = PCIBIOS_BIOS_PRESENT; 117 if (bios32(&args, PCIbios.ventry, GSEL(GCODE_SEL, SEL_KPL))) { 118 PRVERB(("pcibios: BIOS_PRESENT call failed\n")); 119 return (0); 120 } 121 if (args.edx != 0x20494350) { 122 PRVERB(("pcibios: BIOS_PRESENT didn't return 'PCI ' in edx\n")); 123 return (0); 124 } 125 return (args.ebx & 0xffff); 126 } 127 128 /* 129 * Initialise access to PCI configuration space 130 */ 131 int 132 pci_cfgregopen(void) 133 { 134 static int opened = 0; 135 u_long sigaddr; 136 static struct PIR_table *pt; 137 u_int8_t ck, *cv; 138 int i; 139 140 if (opened) 141 return(1); 142 143 if (pcibios_cfgopen() != 0) { 144 usebios = 1; 145 } else if (pcireg_cfgopen() != 0) { 146 usebios = 0; 147 } else { 148 return(0); 149 } 150 151 /* 152 * Look for the interrupt routing table. 153 * 154 * We use PCI BIOS's PIR table if it's available $PIR is the 155 * standard way to do this. Sadly, some machines are not 156 * standards conforming and have _PIR instead. We shrug and cope 157 * by looking for both. 158 */ 159 if (pcibios_get_version() >= 0x0210 && pt == NULL) { 160 sigaddr = bios_sigsearch(0, "$PIR", 4, 16, 0); 161 if (sigaddr == 0) 162 sigaddr = bios_sigsearch(0, "_PIR", 4, 16, 0); 163 if (sigaddr != 0) { 164 pt = (struct PIR_table *)(uintptr_t)BIOS_PADDRTOVADDR(sigaddr); 165 for (cv = (u_int8_t *)pt, ck = 0, i = 0; 166 i < (pt->pt_header.ph_length); i++) { 167 ck += cv[i]; 168 } 169 if (ck == 0) { 170 pci_route_table = pt; 171 pci_route_count = (pt->pt_header.ph_length - 172 sizeof(struct PIR_header)) / sizeof(struct PIR_entry); 173 printf("Using $PIR table, %d entries at %p\n", 174 pci_route_count, pci_route_table); 175 } 176 } 177 } 178 opened = 1; 179 return(1); 180 } 181 182 /* 183 * Read configuration space register 184 */ 185 static u_int32_t 186 pci_do_cfgregread(int bus, int slot, int func, int reg, int bytes) 187 { 188 return(usebios ? 189 pcibios_cfgread(bus, slot, func, reg, bytes) : 190 pcireg_cfgread(bus, slot, func, reg, bytes)); 191 } 192 193 u_int32_t 194 pci_cfgregread(int bus, int slot, int func, int reg, int bytes) 195 { 196 uint32_t line, pin; 197 #ifdef APIC_IO 198 /* 199 * If we are using the APIC, the contents of the intline register will probably 200 * be wrong (since they are set up for use with the PIC. 201 * Rather than rewrite these registers (maybe that would be smarter) we trap 202 * attempts to read them and translate to our private vector numbers. 203 */ 204 if ((reg == PCIR_INTLINE) && (bytes == 1)) { 205 206 pin = pci_do_cfgregread(bus, slot, func, PCIR_INTPIN, 1); 207 line = pci_do_cfgregread(bus, slot, func, PCIR_INTLINE, 1); 208 209 if (pin != 0) { 210 int airq; 211 212 airq = pci_apic_irq(bus, slot, pin); 213 if (airq >= 0) { 214 /* PCI specific entry found in MP table */ 215 if (airq != line) 216 undirect_pci_irq(line); 217 return(airq); 218 } else { 219 /* 220 * PCI interrupts might be redirected to the 221 * ISA bus according to some MP tables. Use the 222 * same methods as used by the ISA devices 223 * devices to find the proper IOAPIC int pin. 224 */ 225 airq = isa_apic_irq(line); 226 if ((airq >= 0) && (airq != line)) { 227 /* XXX: undirect_pci_irq() ? */ 228 undirect_isa_irq(line); 229 return(airq); 230 } 231 } 232 } 233 return(line); 234 } 235 #else 236 /* 237 * Some BIOS writers seem to want to ignore the spec and put 238 * 0 in the intline rather than 255 to indicate none. The rest of 239 * the code uses 255 as an invalid IRQ. 240 */ 241 if (reg == PCIR_INTLINE && bytes == 1) { 242 line = pci_do_cfgregread(bus, slot, func, PCIR_INTLINE, 1); 243 pin = pci_do_cfgregread(bus, slot, func, PCIR_INTPIN, 1); 244 return pci_i386_map_intline(line); 245 } 246 #endif /* APIC_IO */ 247 return(pci_do_cfgregread(bus, slot, func, reg, bytes)); 248 } 249 250 /* 251 * Write configuration space register 252 */ 253 void 254 pci_cfgregwrite(int bus, int slot, int func, int reg, u_int32_t data, int bytes) 255 { 256 return(usebios ? 257 pcibios_cfgwrite(bus, slot, func, reg, data, bytes) : 258 pcireg_cfgwrite(bus, slot, func, reg, data, bytes)); 259 } 260 261 /* 262 * Route a PCI interrupt 263 * 264 * XXX we don't do anything "right" with the function number in the PIR table 265 * (because the consumer isn't currently passing it in). We don't care 266 * anyway, due to the way PCI interrupts are assigned. 267 */ 268 int 269 pci_cfgintr(int bus, int device, int pin) 270 { 271 struct PIR_entry *pe; 272 int i, irq; 273 struct bios_regs args; 274 u_int16_t v; 275 int already = 0; 276 277 v = pcibios_get_version(); 278 if (v < 0x0210) { 279 PRVERB(( 280 "pci_cfgintr: BIOS %x.%02x doesn't support interrupt routing\n", 281 (v & 0xff00) >> 8, v & 0xff)); 282 return (255); 283 } 284 if ((bus < 0) || (bus > 255) || (device < 0) || (device > 255) || 285 (pin < 1) || (pin > 4)) 286 return(255); 287 288 /* 289 * Scan the entry table for a contender 290 */ 291 for (i = 0, pe = &pci_route_table->pt_entry[0]; i < pci_route_count; i++, pe++) { 292 if ((bus != pe->pe_bus) || (device != pe->pe_device)) 293 continue; 294 295 irq = pci_cfgintr_linked(pe, pin); 296 if (irq == 255) 297 irq = pci_cfgintr_unique(pe, pin); 298 if (irq != 255) 299 already = 1; 300 if (irq == 255) 301 irq = pci_cfgintr_virgin(pe, pin); 302 if (irq == 255) 303 break; 304 305 /* 306 * Ask the BIOS to route the interrupt 307 */ 308 args.eax = PCIBIOS_ROUTE_INTERRUPT; 309 args.ebx = (bus << 8) | (device << 3); 310 args.ecx = (irq << 8) | (0xa + pin - 1); /* pin value is 0xa - 0xd */ 311 if (!already && bios32(&args, PCIbios.ventry, GSEL(GCODE_SEL, SEL_KPL))) { 312 /* 313 * XXX if it fails, we should try to smack the router 314 * hardware directly. 315 * XXX Also, there may be other choices that we can try that 316 * will work. 317 */ 318 PRVERB(("pci_cfgintr: ROUTE_INTERRUPT failed.\n")); 319 return(255); 320 } 321 printf("pci_cfgintr: %d:%d INT%c routed to irq %d\n", bus, device, 'A' + pin - 1, irq); 322 return(irq); 323 } 324 325 PRVERB(("pci_cfgintr: can't route an interrupt to %d:%d INT%c\n", bus, device, 'A' + pin - 1)); 326 return(255); 327 } 328 329 /* 330 * Look to see if the routing table claims this pin is uniquely routed. 331 */ 332 static int 333 pci_cfgintr_unique(struct PIR_entry *pe, int pin) 334 { 335 int irq; 336 uint32_t irqmask; 337 338 irqmask = pe->pe_intpin[pin - 1].irqs; 339 if (irqmask != 0 && powerof2(irqmask)) { 340 irq = ffs(irqmask) - 1; 341 PRVERB(("pci_cfgintr_unique: hard-routed to irq %d\n", irq)); 342 return(irq); 343 } 344 return(255); 345 } 346 347 /* 348 * Look for another device which shares the same link byte and 349 * already has a unique IRQ, or which has had one routed already. 350 */ 351 static int 352 pci_cfgintr_linked(struct PIR_entry *pe, int pin) 353 { 354 struct PIR_entry *oe; 355 struct PIR_intpin *pi; 356 int i, j, irq; 357 358 /* 359 * Scan table slots. 360 */ 361 for (i = 0, oe = &pci_route_table->pt_entry[0]; i < pci_route_count; i++, oe++) { 362 363 /* scan interrupt pins */ 364 for (j = 0, pi = &oe->pe_intpin[0]; j < 4; j++, pi++) { 365 366 /* don't look at the entry we're trying to match with */ 367 if ((pe == oe) && (i == (pin - 1))) 368 continue; 369 370 /* compare link bytes */ 371 if (pi->link != pe->pe_intpin[pin - 1].link) 372 continue; 373 374 /* link destination mapped to a unique interrupt? */ 375 if (pi->irqs != 0 && powerof2(pi->irqs)) { 376 irq = ffs(pi->irqs) - 1; 377 PRVERB(("pci_cfgintr_linked: linked (%x) to hard-routed irq %d\n", 378 pi->link, irq)); 379 return(irq); 380 } 381 382 /* look for the real PCI device that matches this table entry */ 383 if ((irq = pci_cfgintr_search(pe, oe->pe_bus, oe->pe_device, j, pin)) != 255) 384 return(irq); 385 } 386 } 387 return(255); 388 } 389 390 /* 391 * Scan for the real PCI device at (bus)/(device) using intpin (matchpin) and 392 * see if it has already been assigned an interrupt. 393 */ 394 static int 395 pci_cfgintr_search(struct PIR_entry *pe, int bus, int device, int matchpin, int pin) 396 { 397 devclass_t pci_devclass; 398 device_t *pci_devices; 399 int pci_count; 400 device_t *pci_children; 401 int pci_childcount; 402 device_t *busp, *childp; 403 int i, j, irq; 404 405 /* 406 * Find all the PCI busses. 407 */ 408 pci_count = 0; 409 if ((pci_devclass = devclass_find("pci")) != NULL) 410 devclass_get_devices(pci_devclass, &pci_devices, &pci_count); 411 412 /* 413 * Scan all the PCI busses/devices looking for this one. 414 */ 415 irq = 255; 416 for (i = 0, busp = pci_devices; (i < pci_count) && (irq == 255); i++, busp++) { 417 pci_childcount = 0; 418 device_get_children(*busp, &pci_children, &pci_childcount); 419 420 for (j = 0, childp = pci_children; j < pci_childcount; j++, childp++) { 421 if ((pci_get_bus(*childp) == bus) && 422 (pci_get_slot(*childp) == device) && 423 (pci_get_intpin(*childp) == matchpin)) { 424 irq = pci_i386_map_intline(pci_get_irq(*childp)); 425 if (irq != 255) 426 PRVERB(("pci_cfgintr_search: linked (%x) to configured irq %d at %d:%d:%d\n", 427 pe->pe_intpin[pin - 1].link, irq, 428 pci_get_bus(*childp), pci_get_slot(*childp), pci_get_function(*childp))); 429 break; 430 } 431 } 432 if (pci_children != NULL) 433 free(pci_children, M_TEMP); 434 } 435 if (pci_devices != NULL) 436 free(pci_devices, M_TEMP); 437 return(irq); 438 } 439 440 /* 441 * Pick a suitable IRQ from those listed as routable to this device. 442 */ 443 static int 444 pci_cfgintr_virgin(struct PIR_entry *pe, int pin) 445 { 446 int irq, ibit; 447 448 /* first scan the set of PCI-only interrupts and see if any of these are routable */ 449 for (irq = 0; irq < 16; irq++) { 450 ibit = (1 << irq); 451 452 /* can we use this interrupt? */ 453 if ((pci_route_table->pt_header.ph_pci_irqs & ibit) && 454 (pe->pe_intpin[pin - 1].irqs & ibit)) { 455 PRVERB(("pci_cfgintr_virgin: using routable PCI-only interrupt %d\n", irq)); 456 return(irq); 457 } 458 } 459 460 /* life is tough, so just pick an interrupt */ 461 for (irq = 0; irq < 16; irq++) { 462 ibit = (1 << irq); 463 464 if (pe->pe_intpin[pin - 1].irqs & ibit) { 465 PRVERB(("pci_cfgintr_virgin: using routable interrupt %d\n", irq)); 466 return(irq); 467 } 468 } 469 return(255); 470 } 471 472 473 /* 474 * Config space access using BIOS functions 475 */ 476 static int 477 pcibios_cfgread(int bus, int slot, int func, int reg, int bytes) 478 { 479 struct bios_regs args; 480 u_int mask; 481 482 switch(bytes) { 483 case 1: 484 args.eax = PCIBIOS_READ_CONFIG_BYTE; 485 mask = 0xff; 486 break; 487 case 2: 488 args.eax = PCIBIOS_READ_CONFIG_WORD; 489 mask = 0xffff; 490 break; 491 case 4: 492 args.eax = PCIBIOS_READ_CONFIG_DWORD; 493 mask = 0xffffffff; 494 break; 495 default: 496 return(-1); 497 } 498 args.ebx = (bus << 8) | (slot << 3) | func; 499 args.edi = reg; 500 bios32(&args, PCIbios.ventry, GSEL(GCODE_SEL, SEL_KPL)); 501 /* check call results? */ 502 return(args.ecx & mask); 503 } 504 505 static void 506 pcibios_cfgwrite(int bus, int slot, int func, int reg, int data, int bytes) 507 { 508 struct bios_regs args; 509 510 switch(bytes) { 511 case 1: 512 args.eax = PCIBIOS_WRITE_CONFIG_BYTE; 513 break; 514 case 2: 515 args.eax = PCIBIOS_WRITE_CONFIG_WORD; 516 break; 517 case 4: 518 args.eax = PCIBIOS_WRITE_CONFIG_DWORD; 519 break; 520 default: 521 return; 522 } 523 args.ebx = (bus << 8) | (slot << 3) | func; 524 args.ecx = data; 525 args.edi = reg; 526 bios32(&args, PCIbios.ventry, GSEL(GCODE_SEL, SEL_KPL)); 527 } 528 529 /* 530 * Determine whether there is a PCI BIOS present 531 */ 532 static int 533 pcibios_cfgopen(void) 534 { 535 u_int16_t v = 0; 536 537 if (PCIbios.entry != 0 && enable_pcibios) { 538 v = pcibios_get_version(); 539 if (v > 0) 540 printf("pcibios: BIOS version %x.%02x\n", (v & 0xff00) >> 8, 541 v & 0xff); 542 } 543 return (v > 0); 544 } 545 546 /* 547 * Configuration space access using direct register operations 548 */ 549 550 /* enable configuration space accesses and return data port address */ 551 static int 552 pci_cfgenable(unsigned bus, unsigned slot, unsigned func, int reg, int bytes) 553 { 554 int dataport = 0; 555 556 if (bus <= PCI_BUSMAX 557 && slot < devmax 558 && func <= PCI_FUNCMAX 559 && reg <= PCI_REGMAX 560 && bytes != 3 561 && (unsigned) bytes <= 4 562 && (reg & (bytes -1)) == 0) { 563 switch (cfgmech) { 564 case 1: 565 outl(CONF1_ADDR_PORT, (1 << 31) 566 | (bus << 16) | (slot << 11) 567 | (func << 8) | (reg & ~0x03)); 568 dataport = CONF1_DATA_PORT + (reg & 0x03); 569 break; 570 case 2: 571 outb(CONF2_ENABLE_PORT, 0xf0 | (func << 1)); 572 outb(CONF2_FORWARD_PORT, bus); 573 dataport = 0xc000 | (slot << 8) | reg; 574 break; 575 } 576 } 577 return (dataport); 578 } 579 580 /* disable configuration space accesses */ 581 static void 582 pci_cfgdisable(void) 583 { 584 switch (cfgmech) { 585 case 1: 586 outl(CONF1_ADDR_PORT, 0); 587 break; 588 case 2: 589 outb(CONF2_ENABLE_PORT, 0); 590 outb(CONF2_FORWARD_PORT, 0); 591 break; 592 } 593 } 594 595 static int 596 pcireg_cfgread(int bus, int slot, int func, int reg, int bytes) 597 { 598 int data = -1; 599 int port; 600 601 port = pci_cfgenable(bus, slot, func, reg, bytes); 602 603 if (port != 0) { 604 switch (bytes) { 605 case 1: 606 data = inb(port); 607 break; 608 case 2: 609 data = inw(port); 610 break; 611 case 4: 612 data = inl(port); 613 break; 614 } 615 pci_cfgdisable(); 616 } 617 return (data); 618 } 619 620 static void 621 pcireg_cfgwrite(int bus, int slot, int func, int reg, int data, int bytes) 622 { 623 int port; 624 625 port = pci_cfgenable(bus, slot, func, reg, bytes); 626 if (port != 0) { 627 switch (bytes) { 628 case 1: 629 outb(port, data); 630 break; 631 case 2: 632 outw(port, data); 633 break; 634 case 4: 635 outl(port, data); 636 break; 637 } 638 pci_cfgdisable(); 639 } 640 } 641 642 /* check whether the configuration mechanism has been correctly identified */ 643 static int 644 pci_cfgcheck(int maxdev) 645 { 646 u_char device; 647 648 if (bootverbose) 649 printf("pci_cfgcheck:\tdevice "); 650 651 for (device = 0; device < maxdev; device++) { 652 unsigned id, class, header; 653 if (bootverbose) 654 printf("%d ", device); 655 656 id = inl(pci_cfgenable(0, device, 0, 0, 4)); 657 if (id == 0 || id == -1) 658 continue; 659 660 class = inl(pci_cfgenable(0, device, 0, 8, 4)) >> 8; 661 if (bootverbose) 662 printf("[class=%06x] ", class); 663 if (class == 0 || (class & 0xf870ff) != 0) 664 continue; 665 666 header = inb(pci_cfgenable(0, device, 0, 14, 1)); 667 if (bootverbose) 668 printf("[hdr=%02x] ", header); 669 if ((header & 0x7e) != 0) 670 continue; 671 672 if (bootverbose) 673 printf("is there (id=%08x)\n", id); 674 675 pci_cfgdisable(); 676 return (1); 677 } 678 if (bootverbose) 679 printf("-- nothing found\n"); 680 681 pci_cfgdisable(); 682 return (0); 683 } 684 685 static int 686 pcireg_cfgopen(void) 687 { 688 unsigned long mode1res,oldval1; 689 unsigned char mode2res,oldval2; 690 691 oldval1 = inl(CONF1_ADDR_PORT); 692 693 if (bootverbose) { 694 printf("pci_open(1):\tmode 1 addr port (0x0cf8) is 0x%08lx\n", 695 oldval1); 696 } 697 698 if ((oldval1 & CONF1_ENABLE_MSK) == 0) { 699 700 cfgmech = 1; 701 devmax = 32; 702 703 outl(CONF1_ADDR_PORT, CONF1_ENABLE_CHK); 704 outb(CONF1_ADDR_PORT +3, 0); 705 mode1res = inl(CONF1_ADDR_PORT); 706 outl(CONF1_ADDR_PORT, oldval1); 707 708 if (bootverbose) 709 printf("pci_open(1a):\tmode1res=0x%08lx (0x%08lx)\n", 710 mode1res, CONF1_ENABLE_CHK); 711 712 if (mode1res) { 713 if (pci_cfgcheck(32)) 714 return (cfgmech); 715 } 716 717 outl(CONF1_ADDR_PORT, CONF1_ENABLE_CHK1); 718 mode1res = inl(CONF1_ADDR_PORT); 719 outl(CONF1_ADDR_PORT, oldval1); 720 721 if (bootverbose) 722 printf("pci_open(1b):\tmode1res=0x%08lx (0x%08lx)\n", 723 mode1res, CONF1_ENABLE_CHK1); 724 725 if ((mode1res & CONF1_ENABLE_MSK1) == CONF1_ENABLE_RES1) { 726 if (pci_cfgcheck(32)) 727 return (cfgmech); 728 } 729 } 730 731 oldval2 = inb(CONF2_ENABLE_PORT); 732 733 if (bootverbose) { 734 printf("pci_open(2):\tmode 2 enable port (0x0cf8) is 0x%02x\n", 735 oldval2); 736 } 737 738 if ((oldval2 & 0xf0) == 0) { 739 740 cfgmech = 2; 741 devmax = 16; 742 743 outb(CONF2_ENABLE_PORT, CONF2_ENABLE_CHK); 744 mode2res = inb(CONF2_ENABLE_PORT); 745 outb(CONF2_ENABLE_PORT, oldval2); 746 747 if (bootverbose) 748 printf("pci_open(2a):\tmode2res=0x%02x (0x%02x)\n", 749 mode2res, CONF2_ENABLE_CHK); 750 751 if (mode2res == CONF2_ENABLE_RES) { 752 if (bootverbose) 753 printf("pci_open(2a):\tnow trying mechanism 2\n"); 754 755 if (pci_cfgcheck(16)) 756 return (cfgmech); 757 } 758 } 759 760 cfgmech = 0; 761 devmax = 0; 762 return (cfgmech); 763 } 764 765