Commit 2c1d2f34 authored by Scott Wood's avatar Scott Wood Committed by Paul Mackerras

[POWERPC] qe_ic: Do a sync when masking interrupts

This patch causes a sync do be done after masking a QE interrupt, to
ensure that the masking has completed before interrupts are enabled.
This allows the masking of the cascade IRQ to be removed without causing
spurious interrupts.

The mask_and_ack function is also removed and set to the mask function,
as the two are identical.
Signed-off-by: default avatarScott Wood <scottwood@freescale.com>
Signed-off-by: default avatarPaul Mackerras <paulus@samba.org>
parent d0e70341
...@@ -223,23 +223,15 @@ static void qe_ic_mask_irq(unsigned int virq) ...@@ -223,23 +223,15 @@ static void qe_ic_mask_irq(unsigned int virq)
qe_ic_write(qe_ic->regs, qe_ic_info[src].mask_reg, qe_ic_write(qe_ic->regs, qe_ic_info[src].mask_reg,
temp & ~qe_ic_info[src].mask); temp & ~qe_ic_info[src].mask);
spin_unlock_irqrestore(&qe_ic_lock, flags); /* Flush the above write before enabling interrupts; otherwise,
} * spurious interrupts will sometimes happen. To be 100% sure
* that the write has reached the device before interrupts are
static void qe_ic_mask_irq_and_ack(unsigned int virq) * enabled, the mask register would have to be read back; however,
{ * this is not required for correctness, only to avoid wasting
struct qe_ic *qe_ic = qe_ic_from_irq(virq); * time on a large number of spurious interrupts. In testing,
unsigned int src = virq_to_hw(virq); * a sync reduced the observed spurious interrupts to zero.
unsigned long flags; */
u32 temp; mb();
spin_lock_irqsave(&qe_ic_lock, flags);
temp = qe_ic_read(qe_ic->regs, qe_ic_info[src].mask_reg);
qe_ic_write(qe_ic->regs, qe_ic_info[src].mask_reg,
temp & ~qe_ic_info[src].mask);
/* There is nothing to do for ack here, ack is handled in ISR */
spin_unlock_irqrestore(&qe_ic_lock, flags); spin_unlock_irqrestore(&qe_ic_lock, flags);
} }
...@@ -248,7 +240,7 @@ static struct irq_chip qe_ic_irq_chip = { ...@@ -248,7 +240,7 @@ static struct irq_chip qe_ic_irq_chip = {
.typename = " QEIC ", .typename = " QEIC ",
.unmask = qe_ic_unmask_irq, .unmask = qe_ic_unmask_irq,
.mask = qe_ic_mask_irq, .mask = qe_ic_mask_irq,
.mask_ack = qe_ic_mask_irq_and_ack, .mask_ack = qe_ic_mask_irq,
}; };
static int qe_ic_host_match(struct irq_host *h, struct device_node *node) static int qe_ic_host_match(struct irq_host *h, struct device_node *node)
...@@ -331,34 +323,22 @@ unsigned int qe_ic_get_high_irq(struct qe_ic *qe_ic) ...@@ -331,34 +323,22 @@ unsigned int qe_ic_get_high_irq(struct qe_ic *qe_ic)
return irq_linear_revmap(qe_ic->irqhost, irq); return irq_linear_revmap(qe_ic->irqhost, irq);
} }
/* FIXME: We mask all the QE Low interrupts while handling. We should
* let other interrupt come in, but BAD interrupts are generated */
void fastcall qe_ic_cascade_low(unsigned int irq, struct irq_desc *desc) void fastcall qe_ic_cascade_low(unsigned int irq, struct irq_desc *desc)
{ {
struct qe_ic *qe_ic = desc->handler_data; struct qe_ic *qe_ic = desc->handler_data;
struct irq_chip *chip = irq_desc[irq].chip;
unsigned int cascade_irq = qe_ic_get_low_irq(qe_ic); unsigned int cascade_irq = qe_ic_get_low_irq(qe_ic);
chip->mask_ack(irq);
if (cascade_irq != NO_IRQ) if (cascade_irq != NO_IRQ)
generic_handle_irq(cascade_irq); generic_handle_irq(cascade_irq);
chip->unmask(irq);
} }
/* FIXME: We mask all the QE High interrupts while handling. We should
* let other interrupt come in, but BAD interrupts are generated */
void fastcall qe_ic_cascade_high(unsigned int irq, struct irq_desc *desc) void fastcall qe_ic_cascade_high(unsigned int irq, struct irq_desc *desc)
{ {
struct qe_ic *qe_ic = desc->handler_data; struct qe_ic *qe_ic = desc->handler_data;
struct irq_chip *chip = irq_desc[irq].chip;
unsigned int cascade_irq = qe_ic_get_high_irq(qe_ic); unsigned int cascade_irq = qe_ic_get_high_irq(qe_ic);
chip->mask_ack(irq);
if (cascade_irq != NO_IRQ) if (cascade_irq != NO_IRQ)
generic_handle_irq(cascade_irq); generic_handle_irq(cascade_irq);
chip->unmask(irq);
} }
void __init qe_ic_init(struct device_node *node, unsigned int flags) void __init qe_ic_init(struct device_node *node, unsigned int flags)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment