aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/linux/irq.h4
-rw-r--r--kernel/irq/chip.c48
-rw-r--r--kernel/irq/compat.h12
-rw-r--r--kernel/irq/internals.h4
-rw-r--r--kernel/irq/irqdesc.c2
-rw-r--r--kernel/irq/manage.c4
-rw-r--r--kernel/irq/migration.c2
-rw-r--r--kernel/irq/settings.h2
-rw-r--r--kernel/irq/spurious.c4
9 files changed, 57 insertions, 25 deletions
diff --git a/include/linux/irq.h b/include/linux/irq.h
index c38dbd506656..32efca71ce88 100644
--- a/include/linux/irq.h
+++ b/include/linux/irq.h
@@ -55,9 +55,9 @@ typedef void (*irq_flow_handler_t)(unsigned int irq,
55#define IRQ_INPROGRESS 0x00000100 /* DEPRECATED */ 55#define IRQ_INPROGRESS 0x00000100 /* DEPRECATED */
56#define IRQ_REPLAY 0x00000200 /* DEPRECATED */ 56#define IRQ_REPLAY 0x00000200 /* DEPRECATED */
57#define IRQ_WAITING 0x00000400 /* DEPRECATED */ 57#define IRQ_WAITING 0x00000400 /* DEPRECATED */
58#define IRQ_DISABLED 0x00000800 /* DEPRECATED */
58#endif 59#endif
59 60
60#define IRQ_DISABLED 0x00000800 /* IRQ disabled - do not enter! */
61#define IRQ_PENDING 0x00001000 /* IRQ pending - replay on enable */ 61#define IRQ_PENDING 0x00001000 /* IRQ pending - replay on enable */
62 62
63#define IRQ_LEVEL 0x00004000 /* IRQ level triggered */ 63#define IRQ_LEVEL 0x00004000 /* IRQ level triggered */
@@ -231,7 +231,7 @@ struct irq_chip {
231# define ARCH_IRQ_INIT_FLAGS 0 231# define ARCH_IRQ_INIT_FLAGS 0
232#endif 232#endif
233 233
234#define IRQ_DEFAULT_INIT_FLAGS (IRQ_DISABLED | ARCH_IRQ_INIT_FLAGS) 234#define IRQ_DEFAULT_INIT_FLAGS ARCH_IRQ_INIT_FLAGS
235 235
236struct irqaction; 236struct irqaction;
237extern int setup_irq(unsigned int irq, struct irqaction *new); 237extern int setup_irq(unsigned int irq, struct irqaction *new);
diff --git a/kernel/irq/chip.c b/kernel/irq/chip.c
index 59ae14527ecd..527df7ab1b05 100644
--- a/kernel/irq/chip.c
+++ b/kernel/irq/chip.c
@@ -164,9 +164,21 @@ struct irq_data *irq_get_irq_data(unsigned int irq)
164} 164}
165EXPORT_SYMBOL_GPL(irq_get_irq_data); 165EXPORT_SYMBOL_GPL(irq_get_irq_data);
166 166
167static void irq_state_clr_disabled(struct irq_desc *desc)
168{
169 desc->istate &= ~IRQS_DISABLED;
170 irq_compat_clr_disabled(desc);
171}
172
173static void irq_state_set_disabled(struct irq_desc *desc)
174{
175 desc->istate |= IRQS_DISABLED;
176 irq_compat_set_disabled(desc);
177}
178
167int irq_startup(struct irq_desc *desc) 179int irq_startup(struct irq_desc *desc)
168{ 180{
169 desc->status &= ~IRQ_DISABLED; 181 irq_state_clr_disabled(desc);
170 desc->depth = 0; 182 desc->depth = 0;
171 183
172 if (desc->irq_data.chip->irq_startup) { 184 if (desc->irq_data.chip->irq_startup) {
@@ -181,7 +193,7 @@ int irq_startup(struct irq_desc *desc)
181 193
182void irq_shutdown(struct irq_desc *desc) 194void irq_shutdown(struct irq_desc *desc)
183{ 195{
184 desc->status |= IRQ_DISABLED; 196 irq_state_set_disabled(desc);
185 desc->depth = 1; 197 desc->depth = 1;
186 if (desc->irq_data.chip->irq_shutdown) 198 if (desc->irq_data.chip->irq_shutdown)
187 desc->irq_data.chip->irq_shutdown(&desc->irq_data); 199 desc->irq_data.chip->irq_shutdown(&desc->irq_data);
@@ -194,7 +206,7 @@ void irq_shutdown(struct irq_desc *desc)
194 206
195void irq_enable(struct irq_desc *desc) 207void irq_enable(struct irq_desc *desc)
196{ 208{
197 desc->status &= ~IRQ_DISABLED; 209 irq_state_clr_disabled(desc);
198 if (desc->irq_data.chip->irq_enable) 210 if (desc->irq_data.chip->irq_enable)
199 desc->irq_data.chip->irq_enable(&desc->irq_data); 211 desc->irq_data.chip->irq_enable(&desc->irq_data);
200 else 212 else
@@ -204,7 +216,7 @@ void irq_enable(struct irq_desc *desc)
204 216
205void irq_disable(struct irq_desc *desc) 217void irq_disable(struct irq_desc *desc)
206{ 218{
207 desc->status |= IRQ_DISABLED; 219 irq_state_set_disabled(desc);
208 if (desc->irq_data.chip->irq_disable) { 220 if (desc->irq_data.chip->irq_disable) {
209 desc->irq_data.chip->irq_disable(&desc->irq_data); 221 desc->irq_data.chip->irq_disable(&desc->irq_data);
210 desc->status |= IRQ_MASKED; 222 desc->status |= IRQ_MASKED;
@@ -380,7 +392,7 @@ void handle_nested_irq(unsigned int irq)
380 kstat_incr_irqs_this_cpu(irq, desc); 392 kstat_incr_irqs_this_cpu(irq, desc);
381 393
382 action = desc->action; 394 action = desc->action;
383 if (unlikely(!action || (desc->status & IRQ_DISABLED))) 395 if (unlikely(!action || (desc->istate & IRQS_DISABLED)))
384 goto out_unlock; 396 goto out_unlock;
385 397
386 irq_compat_set_progress(desc); 398 irq_compat_set_progress(desc);
@@ -431,7 +443,7 @@ handle_simple_irq(unsigned int irq, struct irq_desc *desc)
431 desc->istate &= ~(IRQS_REPLAY | IRQS_WAITING); 443 desc->istate &= ~(IRQS_REPLAY | IRQS_WAITING);
432 kstat_incr_irqs_this_cpu(irq, desc); 444 kstat_incr_irqs_this_cpu(irq, desc);
433 445
434 if (unlikely(!desc->action || (desc->status & IRQ_DISABLED))) 446 if (unlikely(!desc->action || (desc->istate & IRQS_DISABLED)))
435 goto out_unlock; 447 goto out_unlock;
436 448
437 handle_irq_event(desc); 449 handle_irq_event(desc);
@@ -467,12 +479,12 @@ handle_level_irq(unsigned int irq, struct irq_desc *desc)
467 * If its disabled or no action available 479 * If its disabled or no action available
468 * keep it masked and get out of here 480 * keep it masked and get out of here
469 */ 481 */
470 if (unlikely(!desc->action || (desc->status & IRQ_DISABLED))) 482 if (unlikely(!desc->action || (desc->istate & IRQS_DISABLED)))
471 goto out_unlock; 483 goto out_unlock;
472 484
473 handle_irq_event(desc); 485 handle_irq_event(desc);
474 486
475 if (!(desc->status & IRQ_DISABLED) && !(desc->istate & IRQS_ONESHOT)) 487 if (!(desc->istate & (IRQS_DISABLED | IRQS_ONESHOT)))
476 unmask_irq(desc); 488 unmask_irq(desc);
477out_unlock: 489out_unlock:
478 raw_spin_unlock(&desc->lock); 490 raw_spin_unlock(&desc->lock);
@@ -505,7 +517,7 @@ handle_fasteoi_irq(unsigned int irq, struct irq_desc *desc)
505 * If its disabled or no action available 517 * If its disabled or no action available
506 * then mask it and get out of here: 518 * then mask it and get out of here:
507 */ 519 */
508 if (unlikely(!desc->action || (desc->status & IRQ_DISABLED))) { 520 if (unlikely(!desc->action || (desc->istate & IRQS_DISABLED))) {
509 desc->status |= IRQ_PENDING; 521 desc->status |= IRQ_PENDING;
510 mask_irq(desc); 522 mask_irq(desc);
511 goto out; 523 goto out;
@@ -543,8 +555,8 @@ handle_edge_irq(unsigned int irq, struct irq_desc *desc)
543 * we shouldn't process the IRQ. Mark it pending, handle 555 * we shouldn't process the IRQ. Mark it pending, handle
544 * the necessary masking and go out 556 * the necessary masking and go out
545 */ 557 */
546 if (unlikely((desc->istate & (IRQS_INPROGRESS) || 558 if (unlikely((desc->istate & (IRQS_DISABLED | IRQS_INPROGRESS) ||
547 (desc->status & IRQ_DISABLED) || !desc->action))) { 559 !desc->action))) {
548 if (!irq_check_poll(desc)) { 560 if (!irq_check_poll(desc)) {
549 desc->status |= IRQ_PENDING; 561 desc->status |= IRQ_PENDING;
550 mask_ack_irq(desc); 562 mask_ack_irq(desc);
@@ -567,15 +579,16 @@ handle_edge_irq(unsigned int irq, struct irq_desc *desc)
567 * one, we could have masked the irq. 579 * one, we could have masked the irq.
568 * Renable it, if it was not disabled in meantime. 580 * Renable it, if it was not disabled in meantime.
569 */ 581 */
570 if (unlikely((desc->status & 582 if (unlikely(desc->status & IRQ_PENDING)) {
571 (IRQ_PENDING | IRQ_MASKED | IRQ_DISABLED)) == 583 if (!(desc->istate & IRQS_DISABLED) &&
572 (IRQ_PENDING | IRQ_MASKED))) { 584 (desc->status & IRQ_MASKED))
573 unmask_irq(desc); 585 unmask_irq(desc);
574 } 586 }
575 587
576 handle_irq_event(desc); 588 handle_irq_event(desc);
577 589
578 } while ((desc->status & (IRQ_PENDING | IRQ_DISABLED)) == IRQ_PENDING); 590 } while ((desc->status & IRQ_PENDING) &&
591 !(desc->istate & IRQS_DISABLED));
579 592
580out_unlock: 593out_unlock:
581 raw_spin_unlock(&desc->lock); 594 raw_spin_unlock(&desc->lock);
@@ -639,7 +652,8 @@ __set_irq_handler(unsigned int irq, irq_flow_handler_t handle, int is_chained,
639 if (handle == handle_bad_irq) { 652 if (handle == handle_bad_irq) {
640 if (desc->irq_data.chip != &no_irq_chip) 653 if (desc->irq_data.chip != &no_irq_chip)
641 mask_ack_irq(desc); 654 mask_ack_irq(desc);
642 desc->status |= IRQ_DISABLED; 655 irq_compat_set_disabled(desc);
656 desc->istate |= IRQS_DISABLED;
643 desc->depth = 1; 657 desc->depth = 1;
644 } 658 }
645 desc->handle_irq = handle; 659 desc->handle_irq = handle;
diff --git a/kernel/irq/compat.h b/kernel/irq/compat.h
index aac6e400e608..bc0c2a501e82 100644
--- a/kernel/irq/compat.h
+++ b/kernel/irq/compat.h
@@ -11,7 +11,19 @@ static inline void irq_compat_clr_progress(struct irq_desc *desc)
11{ 11{
12 desc->status &= ~IRQ_INPROGRESS; 12 desc->status &= ~IRQ_INPROGRESS;
13} 13}
14static inline void irq_compat_set_disabled(struct irq_desc *desc)
15{
16 desc->status |= IRQ_DISABLED;
17}
18
19static inline void irq_compat_clr_disabled(struct irq_desc *desc)
20{
21 desc->status &= ~IRQ_DISABLED;
22}
14#else 23#else
15static inline void irq_compat_set_progress(struct irq_desc *desc) { } 24static inline void irq_compat_set_progress(struct irq_desc *desc) { }
16static inline void irq_compat_clr_progress(struct irq_desc *desc) { } 25static inline void irq_compat_clr_progress(struct irq_desc *desc) { }
26static inline void irq_compat_set_disabled(struct irq_desc *desc) { }
27static inline void irq_compat_clr_disabled(struct irq_desc *desc) { }
17#endif 28#endif
29
diff --git a/kernel/irq/internals.h b/kernel/irq/internals.h
index 54037533af7a..919d2dd0bb33 100644
--- a/kernel/irq/internals.h
+++ b/kernel/irq/internals.h
@@ -45,6 +45,7 @@ enum {
45 * IRQS_ONESHOT - irq is not unmasked in primary handler 45 * IRQS_ONESHOT - irq is not unmasked in primary handler
46 * IRQS_REPLAY - irq is replayed 46 * IRQS_REPLAY - irq is replayed
47 * IRQS_WAITING - irq is waiting 47 * IRQS_WAITING - irq is waiting
48 * IRQS_DISABLED - irq is disabled
48 */ 49 */
49enum { 50enum {
50 IRQS_AUTODETECT = 0x00000001, 51 IRQS_AUTODETECT = 0x00000001,
@@ -54,6 +55,7 @@ enum {
54 IRQS_ONESHOT = 0x00000020, 55 IRQS_ONESHOT = 0x00000020,
55 IRQS_REPLAY = 0x00000040, 56 IRQS_REPLAY = 0x00000040,
56 IRQS_WAITING = 0x00000080, 57 IRQS_WAITING = 0x00000080,
58 IRQS_DISABLED = 0x00000100,
57}; 59};
58 60
59#define irq_data_to_desc(data) container_of(data, struct irq_desc, irq_data) 61#define irq_data_to_desc(data) container_of(data, struct irq_desc, irq_data)
@@ -137,7 +139,6 @@ static inline void print_irq_desc(unsigned int irq, struct irq_desc *desc)
137 print_symbol("%s\n", (unsigned long)desc->action->handler); 139 print_symbol("%s\n", (unsigned long)desc->action->handler);
138 } 140 }
139 141
140 P(IRQ_DISABLED);
141 P(IRQ_PENDING); 142 P(IRQ_PENDING);
142 P(IRQ_LEVEL); 143 P(IRQ_LEVEL);
143 P(IRQ_MASKED); 144 P(IRQ_MASKED);
@@ -152,6 +153,7 @@ static inline void print_irq_desc(unsigned int irq, struct irq_desc *desc)
152 PS(IRQS_INPROGRESS); 153 PS(IRQS_INPROGRESS);
153 PS(IRQS_REPLAY); 154 PS(IRQS_REPLAY);
154 PS(IRQS_WAITING); 155 PS(IRQS_WAITING);
156 PS(IRQS_DISABLED);
155} 157}
156 158
157#undef P 159#undef P
diff --git a/kernel/irq/irqdesc.c b/kernel/irq/irqdesc.c
index 8b87f2ce0203..78866d050bc9 100644
--- a/kernel/irq/irqdesc.c
+++ b/kernel/irq/irqdesc.c
@@ -80,6 +80,7 @@ static void desc_set_defaults(unsigned int irq, struct irq_desc *desc, int node)
80 desc->irq_data.handler_data = NULL; 80 desc->irq_data.handler_data = NULL;
81 desc->irq_data.msi_desc = NULL; 81 desc->irq_data.msi_desc = NULL;
82 desc->status = _IRQ_DEFAULT_INIT_FLAGS; 82 desc->status = _IRQ_DEFAULT_INIT_FLAGS;
83 desc->istate = IRQS_DISABLED;
83 desc->handle_irq = handle_bad_irq; 84 desc->handle_irq = handle_bad_irq;
84 desc->depth = 1; 85 desc->depth = 1;
85 desc->irq_count = 0; 86 desc->irq_count = 0;
@@ -247,6 +248,7 @@ int __init early_irq_init(void)
247struct irq_desc irq_desc[NR_IRQS] __cacheline_aligned_in_smp = { 248struct irq_desc irq_desc[NR_IRQS] __cacheline_aligned_in_smp = {
248 [0 ... NR_IRQS-1] = { 249 [0 ... NR_IRQS-1] = {
249 .status = _IRQ_DEFAULT_INIT_FLAGS, 250 .status = _IRQ_DEFAULT_INIT_FLAGS,
251 .istate = IRQS_DISABLED,
250 .handle_irq = handle_bad_irq, 252 .handle_irq = handle_bad_irq,
251 .depth = 1, 253 .depth = 1,
252 .lock = __RAW_SPIN_LOCK_UNLOCKED(irq_desc->lock), 254 .lock = __RAW_SPIN_LOCK_UNLOCKED(irq_desc->lock),
diff --git a/kernel/irq/manage.c b/kernel/irq/manage.c
index 7971df53d6a9..77ff275b54cf 100644
--- a/kernel/irq/manage.c
+++ b/kernel/irq/manage.c
@@ -646,7 +646,7 @@ again:
646 goto again; 646 goto again;
647 } 647 }
648 648
649 if (!(desc->status & IRQ_DISABLED) && (desc->status & IRQ_MASKED)) { 649 if (!(desc->istate & IRQS_DISABLED) && (desc->status & IRQ_MASKED)) {
650 desc->status &= ~IRQ_MASKED; 650 desc->status &= ~IRQ_MASKED;
651 desc->irq_data.chip->irq_unmask(&desc->irq_data); 651 desc->irq_data.chip->irq_unmask(&desc->irq_data);
652 } 652 }
@@ -709,7 +709,7 @@ static int irq_thread(void *data)
709 atomic_inc(&desc->threads_active); 709 atomic_inc(&desc->threads_active);
710 710
711 raw_spin_lock_irq(&desc->lock); 711 raw_spin_lock_irq(&desc->lock);
712 if (unlikely(desc->status & IRQ_DISABLED)) { 712 if (unlikely(desc->istate & IRQS_DISABLED)) {
713 /* 713 /*
714 * CHECKME: We might need a dedicated 714 * CHECKME: We might need a dedicated
715 * IRQ_THREAD_PENDING flag here, which 715 * IRQ_THREAD_PENDING flag here, which
diff --git a/kernel/irq/migration.c b/kernel/irq/migration.c
index 441fd629ff04..8c68cb8555a7 100644
--- a/kernel/irq/migration.c
+++ b/kernel/irq/migration.c
@@ -61,7 +61,7 @@ void move_native_irq(int irq)
61 if (likely(!(desc->status & IRQ_MOVE_PENDING))) 61 if (likely(!(desc->status & IRQ_MOVE_PENDING)))
62 return; 62 return;
63 63
64 if (unlikely(desc->status & IRQ_DISABLED)) 64 if (unlikely(desc->istate & IRQS_DISABLED))
65 return; 65 return;
66 66
67 /* 67 /*
diff --git a/kernel/irq/settings.h b/kernel/irq/settings.h
index 2e7d08fff0ce..5e3411c7c62b 100644
--- a/kernel/irq/settings.h
+++ b/kernel/irq/settings.h
@@ -12,3 +12,5 @@ enum {
12#define IRQ_REPLAY GOT_YOU_MORON 12#define IRQ_REPLAY GOT_YOU_MORON
13#undef IRQ_WAITING 13#undef IRQ_WAITING
14#define IRQ_WAITING GOT_YOU_MORON 14#define IRQ_WAITING GOT_YOU_MORON
15#undef IRQ_DISABLED
16#define IRQ_DISABLED GOT_YOU_MORON
diff --git a/kernel/irq/spurious.c b/kernel/irq/spurious.c
index 51504837d8cc..367614f858ff 100644
--- a/kernel/irq/spurious.c
+++ b/kernel/irq/spurious.c
@@ -50,7 +50,7 @@ bool irq_wait_for_poll(struct irq_desc *desc)
50 raw_spin_lock(&desc->lock); 50 raw_spin_lock(&desc->lock);
51 } while (desc->istate & IRQS_INPROGRESS); 51 } while (desc->istate & IRQS_INPROGRESS);
52 /* Might have been disabled in meantime */ 52 /* Might have been disabled in meantime */
53 return !(desc->status & IRQ_DISABLED) && desc->action; 53 return !(desc->istate & IRQS_DISABLED) && desc->action;
54#else 54#else
55 return false; 55 return false;
56#endif 56#endif
@@ -75,7 +75,7 @@ static int try_one_irq(int irq, struct irq_desc *desc, bool force)
75 * Do not poll disabled interrupts unless the spurious 75 * Do not poll disabled interrupts unless the spurious
76 * disabled poller asks explicitely. 76 * disabled poller asks explicitely.
77 */ 77 */
78 if ((desc->status & IRQ_DISABLED) && !force) 78 if ((desc->istate & IRQS_DISABLED) && !force)
79 goto out; 79 goto out;
80 80
81 /* 81 /*