diff options
| author | Marc Zyngier <maz@misterjones.org> | 2010-03-15 18:56:33 -0400 |
|---|---|---|
| committer | Thomas Gleixner <tglx@linutronix.de> | 2010-04-13 10:36:39 -0400 |
| commit | ae731f8d0785ccd3380f511bae888933b6562e45 (patch) | |
| tree | 07db03ce79231153a4ae5df75c0ca4dcd96307c2 /include/linux | |
| parent | 7c7145f6acc68100dbdc5d3c5c64fe3af1c99c89 (diff) | |
genirq: Introduce request_any_context_irq()
Now that we enjoy threaded interrupts, we're starting to see irq_chip
implementations (wm831x, pca953x) that make use of threaded interrupts
for the controller, and nested interrupts for the client interrupt. It
all works very well, with one drawback:
Drivers requesting an IRQ must now know whether the handler will
run in a thread context or not, and call request_threaded_irq() or
request_irq() accordingly.
The problem is that the requesting driver sometimes doesn't know
about the nature of the interrupt, specially when the interrupt
controller is a discrete chip (typically a GPIO expander connected
over I2C) that can be connected to a wide variety of otherwise perfectly
supported hardware.
This patch introduces the request_any_context_irq() function that mostly
mimics the usual request_irq(), except that it checks whether the irq
level is configured as nested or not, and calls the right backend.
On success, it also returns either IRQC_IS_HARDIRQ or IRQC_IS_NESTED.
[ tglx: Made return value an enum, simplified code and made the export
of request_any_context_irq GPL ]
Signed-off-by: Marc Zyngier <maz@misterjones.org>
Cc: <joachim.eastwood@jotron.com>
LKML-Reference: <927ea285bd0c68934ddae1a47e44a9ba@localhost>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Diffstat (limited to 'include/linux')
| -rw-r--r-- | include/linux/interrupt.h | 23 |
1 files changed, 23 insertions, 0 deletions
diff --git a/include/linux/interrupt.h b/include/linux/interrupt.h index 75f3f00ac1e5..d7e7a7660c6c 100644 --- a/include/linux/interrupt.h +++ b/include/linux/interrupt.h | |||
| @@ -77,6 +77,18 @@ enum { | |||
| 77 | IRQTF_AFFINITY, | 77 | IRQTF_AFFINITY, |
| 78 | }; | 78 | }; |
| 79 | 79 | ||
| 80 | /** | ||
| 81 | * These values can be returned by request_any_context_irq() and | ||
| 82 | * describe the context the interrupt will be run in. | ||
| 83 | * | ||
| 84 | * IRQC_IS_HARDIRQ - interrupt runs in hardirq context | ||
| 85 | * IRQC_IS_NESTED - interrupt runs in a nested threaded context | ||
| 86 | */ | ||
| 87 | enum { | ||
| 88 | IRQC_IS_HARDIRQ = 0, | ||
| 89 | IRQC_IS_NESTED, | ||
| 90 | }; | ||
| 91 | |||
| 80 | typedef irqreturn_t (*irq_handler_t)(int, void *); | 92 | typedef irqreturn_t (*irq_handler_t)(int, void *); |
| 81 | 93 | ||
| 82 | /** | 94 | /** |
| @@ -120,6 +132,10 @@ request_irq(unsigned int irq, irq_handler_t handler, unsigned long flags, | |||
| 120 | return request_threaded_irq(irq, handler, NULL, flags, name, dev); | 132 | return request_threaded_irq(irq, handler, NULL, flags, name, dev); |
| 121 | } | 133 | } |
| 122 | 134 | ||
| 135 | extern int __must_check | ||
| 136 | request_any_context_irq(unsigned int irq, irq_handler_t handler, | ||
| 137 | unsigned long flags, const char *name, void *dev_id); | ||
| 138 | |||
| 123 | extern void exit_irq_thread(void); | 139 | extern void exit_irq_thread(void); |
| 124 | #else | 140 | #else |
| 125 | 141 | ||
| @@ -141,6 +157,13 @@ request_threaded_irq(unsigned int irq, irq_handler_t handler, | |||
| 141 | return request_irq(irq, handler, flags, name, dev); | 157 | return request_irq(irq, handler, flags, name, dev); |
| 142 | } | 158 | } |
| 143 | 159 | ||
| 160 | static inline int __must_check | ||
| 161 | request_any_context_irq(unsigned int irq, irq_handler_t handler, | ||
| 162 | unsigned long flags, const char *name, void *dev_id) | ||
| 163 | { | ||
| 164 | return request_irq(irq, handler, flags, name, dev_id); | ||
| 165 | } | ||
| 166 | |||
| 144 | static inline void exit_irq_thread(void) { } | 167 | static inline void exit_irq_thread(void) { } |
| 145 | #endif | 168 | #endif |
| 146 | 169 | ||
