diff options
| -rw-r--r-- | include/linux/interrupt.h | 23 | ||||
| -rw-r--r-- | kernel/irq/manage.c | 37 |
2 files changed, 60 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 | ||
diff --git a/kernel/irq/manage.c b/kernel/irq/manage.c index 704e488730a5..84f32278ff1f 100644 --- a/kernel/irq/manage.c +++ b/kernel/irq/manage.c | |||
| @@ -1120,3 +1120,40 @@ int request_threaded_irq(unsigned int irq, irq_handler_t handler, | |||
| 1120 | return retval; | 1120 | return retval; |
| 1121 | } | 1121 | } |
| 1122 | EXPORT_SYMBOL(request_threaded_irq); | 1122 | EXPORT_SYMBOL(request_threaded_irq); |
| 1123 | |||
| 1124 | /** | ||
| 1125 | * request_any_context_irq - allocate an interrupt line | ||
| 1126 | * @irq: Interrupt line to allocate | ||
| 1127 | * @handler: Function to be called when the IRQ occurs. | ||
| 1128 | * Threaded handler for threaded interrupts. | ||
| 1129 | * @flags: Interrupt type flags | ||
| 1130 | * @name: An ascii name for the claiming device | ||
| 1131 | * @dev_id: A cookie passed back to the handler function | ||
| 1132 | * | ||
| 1133 | * This call allocates interrupt resources and enables the | ||
| 1134 | * interrupt line and IRQ handling. It selects either a | ||
| 1135 | * hardirq or threaded handling method depending on the | ||
| 1136 | * context. | ||
| 1137 | * | ||
| 1138 | * On failure, it returns a negative value. On success, | ||
| 1139 | * it returns either IRQC_IS_HARDIRQ or IRQC_IS_NESTED. | ||
| 1140 | */ | ||
| 1141 | int request_any_context_irq(unsigned int irq, irq_handler_t handler, | ||
| 1142 | unsigned long flags, const char *name, void *dev_id) | ||
| 1143 | { | ||
| 1144 | struct irq_desc *desc = irq_to_desc(irq); | ||
| 1145 | int ret; | ||
| 1146 | |||
| 1147 | if (!desc) | ||
| 1148 | return -EINVAL; | ||
| 1149 | |||
| 1150 | if (desc->status & IRQ_NESTED_THREAD) { | ||
| 1151 | ret = request_threaded_irq(irq, NULL, handler, | ||
| 1152 | flags, name, dev_id); | ||
| 1153 | return !ret ? IRQC_IS_NESTED : ret; | ||
| 1154 | } | ||
| 1155 | |||
| 1156 | ret = request_irq(irq, handler, flags, name, dev_id); | ||
| 1157 | return !ret ? IRQC_IS_HARDIRQ : ret; | ||
| 1158 | } | ||
| 1159 | EXPORT_SYMBOL_GPL(request_any_context_irq); | ||
