mmc: davinci: Eliminate spurious interrupts
authorIdo Yariv <ido@wizery.com>
Sun, 11 Mar 2012 21:39:58 +0000 (23:39 +0200)
committerChris Ball <cjb@laptop.org>
Tue, 27 Mar 2012 16:20:11 +0000 (12:20 -0400)
The davinci mmc interrupt handler fills the fifo, as long as the DXRDY
or DRRDY bits are set in the status register.

If interrupts fire during this loop, they will be handled by the
handler, but the interrupt controller will still buffer these. As a
result, the handler will be called again to serve these needlessly. In
order to avoid these spurious interrupts, keep interrupts masked while
filling the fifo.

Signed-off-by: Ido Yariv <ido@wizery.com>
Tested-by: Rajashekhara, Sudhakar <sudhakar.raj@ti.com>
Signed-off-by: Chris Ball <cjb@laptop.org>
drivers/mmc/host/davinci_mmc.c

index 8de9c9bdba0baf668f43d569ee0f59772bc9f0a2..1e076fb06ac36029f6068d8994d4e9d53e7c8586 100644 (file)
@@ -1009,12 +1009,33 @@ static irqreturn_t mmc_davinci_irq(int irq, void *dev_id)
         * by read. So, it is not unbouned loop even in the case of
         * non-dma.
         */
-       while (host->bytes_left && (status & (MMCST0_DXRDY | MMCST0_DRRDY))) {
-               davinci_fifo_data_trans(host, rw_threshold);
-               status = readl(host->base + DAVINCI_MMCST0);
-               if (!status)
-                       break;
-               qstatus |= status;
+       if (host->bytes_left && (status & (MMCST0_DXRDY | MMCST0_DRRDY))) {
+               unsigned long im_val;
+
+               /*
+                * If interrupts fire during the following loop, they will be
+                * handled by the handler, but the PIC will still buffer these.
+                * As a result, the handler will be called again to serve these
+                * needlessly. In order to avoid these spurious interrupts,
+                * keep interrupts masked during the loop.
+                */
+               im_val = readl(host->base + DAVINCI_MMCIM);
+               writel(0, host->base + DAVINCI_MMCIM);
+
+               do {
+                       davinci_fifo_data_trans(host, rw_threshold);
+                       status = readl(host->base + DAVINCI_MMCST0);
+                       qstatus |= status;
+               } while (host->bytes_left &&
+                        (status & (MMCST0_DXRDY | MMCST0_DRRDY)));
+
+               /*
+                * If an interrupt is pending, it is assumed it will fire when
+                * it is unmasked. This assumption is also taken when the MMCIM
+                * is first set. Otherwise, writing to MMCIM after reading the
+                * status is race-prone.
+                */
+               writel(im_val, host->base + DAVINCI_MMCIM);
        }
 
        if (qstatus & MMCST0_DATDNE) {