diff options
| -rw-r--r-- | include/linux/irq.h | 3 | ||||
| -rw-r--r-- | kernel/irq/chip.c | 15 | ||||
| -rw-r--r-- | kernel/irq/handle.c | 16 |
3 files changed, 30 insertions, 4 deletions
diff --git a/include/linux/irq.h b/include/linux/irq.h index 0832149cdb18..00b6ef8b2f93 100644 --- a/include/linux/irq.h +++ b/include/linux/irq.h | |||
| @@ -348,8 +348,9 @@ extern int noirqdebug_setup(char *str); | |||
| 348 | /* Checks whether the interrupt can be requested by request_irq(): */ | 348 | /* Checks whether the interrupt can be requested by request_irq(): */ |
| 349 | extern int can_request_irq(unsigned int irq, unsigned long irqflags); | 349 | extern int can_request_irq(unsigned int irq, unsigned long irqflags); |
| 350 | 350 | ||
| 351 | /* Dummy irq-chip implementation: */ | 351 | /* Dummy irq-chip implementations: */ |
| 352 | extern struct irq_chip no_irq_chip; | 352 | extern struct irq_chip no_irq_chip; |
| 353 | extern struct irq_chip dummy_irq_chip; | ||
| 353 | 354 | ||
| 354 | extern void | 355 | extern void |
| 355 | set_irq_chip_and_handler(unsigned int irq, struct irq_chip *chip, | 356 | set_irq_chip_and_handler(unsigned int irq, struct irq_chip *chip, |
diff --git a/kernel/irq/chip.c b/kernel/irq/chip.c index 4a0952d9458b..54105bdfe20d 100644 --- a/kernel/irq/chip.c +++ b/kernel/irq/chip.c | |||
| @@ -462,9 +462,18 @@ __set_irq_handler(unsigned int irq, | |||
| 462 | if (!handle) | 462 | if (!handle) |
| 463 | handle = handle_bad_irq; | 463 | handle = handle_bad_irq; |
| 464 | 464 | ||
| 465 | if (is_chained && desc->chip == &no_irq_chip) | 465 | if (desc->chip == &no_irq_chip) { |
| 466 | printk(KERN_WARNING "Trying to install " | 466 | printk(KERN_WARNING "Trying to install %sinterrupt handler " |
| 467 | "chained interrupt type for IRQ%d\n", irq); | 467 | "for IRQ%d\n", is_chained ? "chained " : " ", irq); |
| 468 | /* | ||
| 469 | * Some ARM implementations install a handler for really dumb | ||
| 470 | * interrupt hardware without setting an irq_chip. This worked | ||
| 471 | * with the ARM no_irq_chip but the check in setup_irq would | ||
| 472 | * prevent us to setup the interrupt at all. Switch it to | ||
| 473 | * dummy_irq_chip for easy transition. | ||
| 474 | */ | ||
| 475 | desc->chip = &dummy_irq_chip; | ||
| 476 | } | ||
| 468 | 477 | ||
| 469 | spin_lock_irqsave(&desc->lock, flags); | 478 | spin_lock_irqsave(&desc->lock, flags); |
| 470 | 479 | ||
diff --git a/kernel/irq/handle.c b/kernel/irq/handle.c index 961b87591731..e71266c3803e 100644 --- a/kernel/irq/handle.c +++ b/kernel/irq/handle.c | |||
| @@ -96,6 +96,22 @@ struct irq_chip no_irq_chip = { | |||
| 96 | }; | 96 | }; |
| 97 | 97 | ||
| 98 | /* | 98 | /* |
| 99 | * Generic dummy implementation which can be used for | ||
| 100 | * real dumb interrupt sources | ||
| 101 | */ | ||
| 102 | struct irq_chip dummy_irq_chip = { | ||
| 103 | .name = "dummy", | ||
| 104 | .startup = noop_ret, | ||
| 105 | .shutdown = noop, | ||
| 106 | .enable = noop, | ||
| 107 | .disable = noop, | ||
| 108 | .ack = noop, | ||
| 109 | .mask = noop, | ||
| 110 | .unmask = noop, | ||
| 111 | .end = noop, | ||
| 112 | }; | ||
| 113 | |||
| 114 | /* | ||
| 99 | * Special, empty irq handler: | 115 | * Special, empty irq handler: |
| 100 | */ | 116 | */ |
| 101 | irqreturn_t no_action(int cpl, void *dev_id, struct pt_regs *regs) | 117 | irqreturn_t no_action(int cpl, void *dev_id, struct pt_regs *regs) |
