diff options
-rw-r--r-- | arch/i386/kernel/io_apic.c | 7 | ||||
-rw-r--r-- | arch/x86_64/kernel/io_apic.c | 7 | ||||
-rw-r--r-- | kernel/irq/chip.c | 5 | ||||
-rw-r--r-- | kernel/irq/resend.c | 7 |
4 files changed, 17 insertions, 9 deletions
diff --git a/arch/i386/kernel/io_apic.c b/arch/i386/kernel/io_apic.c index 893df8280756..4b8a8da4b2e0 100644 --- a/arch/i386/kernel/io_apic.c +++ b/arch/i386/kernel/io_apic.c | |||
@@ -1256,12 +1256,15 @@ static struct irq_chip ioapic_chip; | |||
1256 | static void ioapic_register_intr(int irq, int vector, unsigned long trigger) | 1256 | static void ioapic_register_intr(int irq, int vector, unsigned long trigger) |
1257 | { | 1257 | { |
1258 | if ((trigger == IOAPIC_AUTO && IO_APIC_irq_trigger(irq)) || | 1258 | if ((trigger == IOAPIC_AUTO && IO_APIC_irq_trigger(irq)) || |
1259 | trigger == IOAPIC_LEVEL) | 1259 | trigger == IOAPIC_LEVEL) { |
1260 | irq_desc[irq].status |= IRQ_LEVEL; | ||
1260 | set_irq_chip_and_handler_name(irq, &ioapic_chip, | 1261 | set_irq_chip_and_handler_name(irq, &ioapic_chip, |
1261 | handle_fasteoi_irq, "fasteoi"); | 1262 | handle_fasteoi_irq, "fasteoi"); |
1262 | else | 1263 | } else { |
1264 | irq_desc[irq].status &= ~IRQ_LEVEL; | ||
1263 | set_irq_chip_and_handler_name(irq, &ioapic_chip, | 1265 | set_irq_chip_and_handler_name(irq, &ioapic_chip, |
1264 | handle_edge_irq, "edge"); | 1266 | handle_edge_irq, "edge"); |
1267 | } | ||
1265 | set_intr_gate(vector, interrupt[irq]); | 1268 | set_intr_gate(vector, interrupt[irq]); |
1266 | } | 1269 | } |
1267 | 1270 | ||
diff --git a/arch/x86_64/kernel/io_apic.c b/arch/x86_64/kernel/io_apic.c index 050141c0602b..f57f8b901912 100644 --- a/arch/x86_64/kernel/io_apic.c +++ b/arch/x86_64/kernel/io_apic.c | |||
@@ -800,12 +800,15 @@ static struct irq_chip ioapic_chip; | |||
800 | 800 | ||
801 | static void ioapic_register_intr(int irq, unsigned long trigger) | 801 | static void ioapic_register_intr(int irq, unsigned long trigger) |
802 | { | 802 | { |
803 | if (trigger) | 803 | if (trigger) { |
804 | irq_desc[irq].status |= IRQ_LEVEL; | ||
804 | set_irq_chip_and_handler_name(irq, &ioapic_chip, | 805 | set_irq_chip_and_handler_name(irq, &ioapic_chip, |
805 | handle_fasteoi_irq, "fasteoi"); | 806 | handle_fasteoi_irq, "fasteoi"); |
806 | else | 807 | } else { |
808 | irq_desc[irq].status &= ~IRQ_LEVEL; | ||
807 | set_irq_chip_and_handler_name(irq, &ioapic_chip, | 809 | set_irq_chip_and_handler_name(irq, &ioapic_chip, |
808 | handle_edge_irq, "edge"); | 810 | handle_edge_irq, "edge"); |
811 | } | ||
809 | } | 812 | } |
810 | 813 | ||
811 | static void setup_IO_APIC_irq(int apic, int pin, unsigned int irq, | 814 | static void setup_IO_APIC_irq(int apic, int pin, unsigned int irq, |
diff --git a/kernel/irq/chip.c b/kernel/irq/chip.c index 615ce97c6cfd..f1a73f0b54e7 100644 --- a/kernel/irq/chip.c +++ b/kernel/irq/chip.c | |||
@@ -352,13 +352,10 @@ handle_level_irq(unsigned int irq, struct irq_desc *desc) | |||
352 | * keep it masked and get out of here | 352 | * keep it masked and get out of here |
353 | */ | 353 | */ |
354 | action = desc->action; | 354 | action = desc->action; |
355 | if (unlikely(!action || (desc->status & IRQ_DISABLED))) { | 355 | if (unlikely(!action || (desc->status & IRQ_DISABLED))) |
356 | desc->status |= IRQ_PENDING; | ||
357 | goto out_unlock; | 356 | goto out_unlock; |
358 | } | ||
359 | 357 | ||
360 | desc->status |= IRQ_INPROGRESS; | 358 | desc->status |= IRQ_INPROGRESS; |
361 | desc->status &= ~IRQ_PENDING; | ||
362 | spin_unlock(&desc->lock); | 359 | spin_unlock(&desc->lock); |
363 | 360 | ||
364 | action_ret = handle_IRQ_event(irq, action); | 361 | action_ret = handle_IRQ_event(irq, action); |
diff --git a/kernel/irq/resend.c b/kernel/irq/resend.c index 5bfeaed7e487..a8046791ba2d 100644 --- a/kernel/irq/resend.c +++ b/kernel/irq/resend.c | |||
@@ -62,7 +62,12 @@ void check_irq_resend(struct irq_desc *desc, unsigned int irq) | |||
62 | */ | 62 | */ |
63 | desc->chip->enable(irq); | 63 | desc->chip->enable(irq); |
64 | 64 | ||
65 | if ((status & (IRQ_PENDING | IRQ_REPLAY)) == IRQ_PENDING) { | 65 | /* |
66 | * We do not resend level type interrupts. Level type | ||
67 | * interrupts are resent by hardware when they are still | ||
68 | * active. | ||
69 | */ | ||
70 | if ((status & (IRQ_LEVEL | IRQ_PENDING | IRQ_REPLAY)) == IRQ_PENDING) { | ||
66 | desc->status = (status & ~IRQ_PENDING) | IRQ_REPLAY; | 71 | desc->status = (status & ~IRQ_PENDING) | IRQ_REPLAY; |
67 | 72 | ||
68 | if (!desc->chip || !desc->chip->retrigger || | 73 | if (!desc->chip || !desc->chip->retrigger || |