Commit 883f7c32 authored by Sergei Shtylyov's avatar Sergei Shtylyov Committed by Greg Kroah-Hartman

mmc: tmio_mmc_core: don't claim spurious interrupts

commit 5c27ff5d upstream.

I have encountered an interrupt storm during the eMMC chip probing (and
the chip finally didn't get detected).  It turned out that U-Boot left
the DMAC interrupts enabled while the Linux driver  didn't use those.
The SDHI driver's interrupt handler somehow assumes that, even if an
SDIO interrupt didn't happen, it should return IRQ_HANDLED.  I think
that if none of the enabled interrupts happened and got handled, we
should return IRQ_NONE -- that way the kernel IRQ code recoginizes
a spurious interrupt and masks it off pretty quickly...

Fixes: 7729c7a2 ("mmc: tmio: Provide separate interrupt handlers")
Signed-off-by: default avatarSergei Shtylyov <sergei.shtylyov@cogentembedded.com>
Reviewed-by: default avatarWolfram Sang <wsa+renesas@sang-engineering.com>
Tested-by: default avatarWolfram Sang <wsa+renesas@sang-engineering.com>
Reviewed-by: default avatarSimon Horman <horms+renesas@verge.net.au>
Cc: stable@vger.kernel.org
Signed-off-by: default avatarUlf Hansson <ulf.hansson@linaro.org>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 3decc9df
...@@ -675,7 +675,7 @@ static bool __tmio_mmc_sdcard_irq(struct tmio_mmc_host *host, ...@@ -675,7 +675,7 @@ static bool __tmio_mmc_sdcard_irq(struct tmio_mmc_host *host,
return false; return false;
} }
static void tmio_mmc_sdio_irq(int irq, void *devid) static bool tmio_mmc_sdio_irq(int irq, void *devid)
{ {
struct tmio_mmc_host *host = devid; struct tmio_mmc_host *host = devid;
struct mmc_host *mmc = host->mmc; struct mmc_host *mmc = host->mmc;
...@@ -684,7 +684,7 @@ static void tmio_mmc_sdio_irq(int irq, void *devid) ...@@ -684,7 +684,7 @@ static void tmio_mmc_sdio_irq(int irq, void *devid)
unsigned int sdio_status; unsigned int sdio_status;
if (!(pdata->flags & TMIO_MMC_SDIO_IRQ)) if (!(pdata->flags & TMIO_MMC_SDIO_IRQ))
return; return false;
status = sd_ctrl_read16(host, CTL_SDIO_STATUS); status = sd_ctrl_read16(host, CTL_SDIO_STATUS);
ireg = status & TMIO_SDIO_MASK_ALL & ~host->sdcard_irq_mask; ireg = status & TMIO_SDIO_MASK_ALL & ~host->sdcard_irq_mask;
...@@ -697,6 +697,8 @@ static void tmio_mmc_sdio_irq(int irq, void *devid) ...@@ -697,6 +697,8 @@ static void tmio_mmc_sdio_irq(int irq, void *devid)
if (mmc->caps & MMC_CAP_SDIO_IRQ && ireg & TMIO_SDIO_STAT_IOIRQ) if (mmc->caps & MMC_CAP_SDIO_IRQ && ireg & TMIO_SDIO_STAT_IOIRQ)
mmc_signal_sdio_irq(mmc); mmc_signal_sdio_irq(mmc);
return ireg;
} }
irqreturn_t tmio_mmc_irq(int irq, void *devid) irqreturn_t tmio_mmc_irq(int irq, void *devid)
...@@ -718,9 +720,10 @@ irqreturn_t tmio_mmc_irq(int irq, void *devid) ...@@ -718,9 +720,10 @@ irqreturn_t tmio_mmc_irq(int irq, void *devid)
if (__tmio_mmc_sdcard_irq(host, ireg, status)) if (__tmio_mmc_sdcard_irq(host, ireg, status))
return IRQ_HANDLED; return IRQ_HANDLED;
tmio_mmc_sdio_irq(irq, devid); if (tmio_mmc_sdio_irq(irq, devid))
return IRQ_HANDLED;
return IRQ_HANDLED; return IRQ_NONE;
} }
EXPORT_SYMBOL(tmio_mmc_irq); EXPORT_SYMBOL(tmio_mmc_irq);
......
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