aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/acpi/pci_irq.c93
1 files changed, 36 insertions, 57 deletions
diff --git a/drivers/acpi/pci_irq.c b/drivers/acpi/pci_irq.c
index 7978c97afb0a..28338227dd5e 100644
--- a/drivers/acpi/pci_irq.c
+++ b/drivers/acpi/pci_irq.c
@@ -339,8 +339,6 @@ void acpi_pci_irq_del_prt(int segment, int bus)
339/* -------------------------------------------------------------------------- 339/* --------------------------------------------------------------------------
340 PCI Interrupt Routing Support 340 PCI Interrupt Routing Support
341 -------------------------------------------------------------------------- */ 341 -------------------------------------------------------------------------- */
342typedef int (*irq_lookup_func) (struct acpi_prt_entry *, int *, int *, char **);
343
344static int 342static int
345acpi_pci_allocate_irq(struct acpi_prt_entry *entry, 343acpi_pci_allocate_irq(struct acpi_prt_entry *entry,
346 int *triggering, int *polarity, char **link) 344 int *triggering, int *polarity, char **link)
@@ -368,8 +366,7 @@ acpi_pci_allocate_irq(struct acpi_prt_entry *entry,
368} 366}
369 367
370static int 368static int
371acpi_pci_free_irq(struct acpi_prt_entry *entry, 369acpi_pci_free_irq(struct acpi_prt_entry *entry)
372 int *triggering, int *polarity, char **link)
373{ 370{
374 int irq; 371 int irq;
375 372
@@ -381,47 +378,29 @@ acpi_pci_free_irq(struct acpi_prt_entry *entry,
381 return irq; 378 return irq;
382} 379}
383 380
384/* 381static struct acpi_prt_entry *
385 * acpi_pci_irq_lookup 382acpi_pci_irq_lookup(struct pci_dev *dev, int pin)
386 * success: return IRQ >= 0
387 * failure: return -1
388 */
389static int
390acpi_pci_irq_lookup(struct pci_dev *dev, int pin,
391 int *triggering,
392 int *polarity, char **link, irq_lookup_func func)
393{ 383{
394 struct acpi_prt_entry *entry = NULL; 384 struct acpi_prt_entry *entry;
395 int ret;
396
397 385
398 entry = acpi_pci_irq_find_prt_entry(dev, pin); 386 entry = acpi_pci_irq_find_prt_entry(dev, pin);
399 if (!entry) { 387 if (!entry) {
400 ACPI_DEBUG_PRINT((ACPI_DB_INFO, "No %s[%c] _PRT entry\n", 388 ACPI_DEBUG_PRINT((ACPI_DB_INFO, "No %s[%c] _PRT entry\n",
401 pci_name(dev), pin_name(pin))); 389 pci_name(dev), pin_name(pin)));
402 return -1; 390 return NULL;
403 } 391 }
404 392
405 ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Found %s[%c] _PRT entry\n", 393 ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Found %s[%c] _PRT entry\n",
406 pci_name(dev), pin_name(pin))); 394 pci_name(dev), pin_name(pin)));
407 395
408 ret = func(entry, triggering, polarity, link); 396 return entry;
409 return ret;
410} 397}
411 398
412/* 399static struct acpi_prt_entry *
413 * acpi_pci_irq_derive 400acpi_pci_irq_derive(struct pci_dev *dev, int pin)
414 * success: return IRQ >= 0
415 * failure: return < 0
416 */
417static int
418acpi_pci_irq_derive(struct pci_dev *dev,
419 int pin,
420 int *triggering,
421 int *polarity, char **link, irq_lookup_func func)
422{ 401{
402 struct acpi_prt_entry *entry = NULL;
423 struct pci_dev *bridge = dev; 403 struct pci_dev *bridge = dev;
424 int irq = -1;
425 u8 bridge_pin = 0, orig_pin = pin; 404 u8 bridge_pin = 0, orig_pin = pin;
426 405
427 406
@@ -429,7 +408,7 @@ acpi_pci_irq_derive(struct pci_dev *dev,
429 * Attempt to derive an IRQ for this device from a parent bridge's 408 * Attempt to derive an IRQ for this device from a parent bridge's
430 * PCI interrupt routing entry (eg. yenta bridge and add-in card bridge). 409 * PCI interrupt routing entry (eg. yenta bridge and add-in card bridge).
431 */ 410 */
432 while (irq < 0 && bridge->bus->self) { 411 while (!entry && bridge->bus->self) {
433 pin = (((pin - 1) + PCI_SLOT(bridge->devfn)) % 4) + 1; 412 pin = (((pin - 1) + PCI_SLOT(bridge->devfn)) % 4) + 1;
434 bridge = bridge->bus->self; 413 bridge = bridge->bus->self;
435 414
@@ -440,26 +419,24 @@ acpi_pci_irq_derive(struct pci_dev *dev,
440 ACPI_DEBUG_PRINT((ACPI_DB_INFO, 419 ACPI_DEBUG_PRINT((ACPI_DB_INFO,
441 "No interrupt pin configured for device %s\n", 420 "No interrupt pin configured for device %s\n",
442 pci_name(bridge))); 421 pci_name(bridge)));
443 return -1; 422 return NULL;
444 } 423 }
445 pin = bridge_pin; 424 pin = bridge_pin;
446 } 425 }
447 426
448 irq = acpi_pci_irq_lookup(bridge, 427 entry = acpi_pci_irq_lookup(bridge, pin);
449 pin, triggering, polarity,
450 link, func);
451 } 428 }
452 429
453 if (irq < 0) { 430 if (!entry) {
454 dev_warn(&dev->dev, "can't derive routing for PCI INT %c\n", 431 dev_warn(&dev->dev, "can't derive routing for PCI INT %c\n",
455 pin_name(orig_pin)); 432 pin_name(orig_pin));
456 return -1; 433 return NULL;
457 } 434 }
458 435
459 ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Derived GSI %d for %s INT %c from %s\n", 436 ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Derived GSI for %s INT %c from %s\n",
460 irq, pci_name(dev), pin_name(orig_pin), pci_name(bridge))); 437 pci_name(dev), pin_name(orig_pin), pci_name(bridge)));
461 438
462 return irq; 439 return entry;
463} 440}
464 441
465/* 442/*
@@ -470,6 +447,7 @@ acpi_pci_irq_derive(struct pci_dev *dev,
470 447
471int acpi_pci_irq_enable(struct pci_dev *dev) 448int acpi_pci_irq_enable(struct pci_dev *dev)
472{ 449{
450 struct acpi_prt_entry *entry;
473 int gsi = 0; 451 int gsi = 0;
474 u8 pin = 0; 452 u8 pin = 0;
475 int triggering = ACPI_LEVEL_SENSITIVE; 453 int triggering = ACPI_LEVEL_SENSITIVE;
@@ -491,18 +469,20 @@ int acpi_pci_irq_enable(struct pci_dev *dev)
491 * First we check the PCI IRQ routing table (PRT) for an IRQ. PRT 469 * First we check the PCI IRQ routing table (PRT) for an IRQ. PRT
492 * values override any BIOS-assigned IRQs set during boot. 470 * values override any BIOS-assigned IRQs set during boot.
493 */ 471 */
494 gsi = acpi_pci_irq_lookup(dev, pin, 472 entry = acpi_pci_irq_lookup(dev, pin);
495 &triggering, &polarity, &link,
496 acpi_pci_allocate_irq);
497 473
498 /* 474 /*
499 * If no PRT entry was found, we'll try to derive an IRQ from the 475 * If no PRT entry was found, we'll try to derive an IRQ from the
500 * device's parent bridge. 476 * device's parent bridge.
501 */ 477 */
502 if (gsi < 0) 478 if (!entry)
503 gsi = acpi_pci_irq_derive(dev, pin, &triggering, 479 entry = acpi_pci_irq_derive(dev, pin);
504 &polarity, &link, 480
505 acpi_pci_allocate_irq); 481 if (entry)
482 gsi = acpi_pci_allocate_irq(entry, &triggering, &polarity,
483 &link);
484 else
485 gsi = -1;
506 486
507 if (gsi < 0) { 487 if (gsi < 0) {
508 /* 488 /*
@@ -559,10 +539,9 @@ void __attribute__ ((weak)) acpi_unregister_gsi(u32 i)
559 539
560void acpi_pci_irq_disable(struct pci_dev *dev) 540void acpi_pci_irq_disable(struct pci_dev *dev)
561{ 541{
542 struct acpi_prt_entry *entry;
562 int gsi = 0; 543 int gsi = 0;
563 u8 pin = 0; 544 u8 pin = 0;
564 int triggering = ACPI_LEVEL_SENSITIVE;
565 int polarity = ACPI_ACTIVE_LOW;
566 545
567 546
568 pin = dev->pin; 547 pin = dev->pin;
@@ -572,20 +551,20 @@ void acpi_pci_irq_disable(struct pci_dev *dev)
572 /* 551 /*
573 * First we check the PCI IRQ routing table (PRT) for an IRQ. 552 * First we check the PCI IRQ routing table (PRT) for an IRQ.
574 */ 553 */
575 gsi = acpi_pci_irq_lookup(dev, pin, 554 entry = acpi_pci_irq_lookup(dev, pin);
576 &triggering, &polarity, NULL, 555
577 acpi_pci_free_irq);
578 /* 556 /*
579 * If no PRT entry was found, we'll try to derive an IRQ from the 557 * If no PRT entry was found, we'll try to derive an IRQ from the
580 * device's parent bridge. 558 * device's parent bridge.
581 */ 559 */
582 if (gsi < 0) 560 if (!entry)
583 gsi = acpi_pci_irq_derive(dev, pin, 561 entry = acpi_pci_irq_derive(dev, pin);
584 &triggering, &polarity, NULL, 562
585 acpi_pci_free_irq); 563 if (!entry)
586 if (gsi < 0)
587 return; 564 return;
588 565
566 gsi = acpi_pci_free_irq(entry);
567
589 /* 568 /*
590 * TBD: It might be worth clearing dev->irq by magic constant 569 * TBD: It might be worth clearing dev->irq by magic constant
591 * (e.g. PCI_UNDEFINED_IRQ). 570 * (e.g. PCI_UNDEFINED_IRQ).