mmc: omap_hsmmc: Pass on the suspend failure to the PM core
authorVaibhav Bedia <vaibhav.bedia@ti.com>
Thu, 13 Sep 2012 06:31:03 +0000 (06:31 +0000)
committerChris Ball <cjb@laptop.org>
Wed, 3 Oct 2012 14:05:23 +0000 (10:05 -0400)
In some cases mmc_suspend_host() is not able to claim the
host and proceed with the suspend process. The core returns
-EBUSY to the host controller driver. Unfortunately, the
host controller driver does not pass on this information
to the PM core and hence the system suspend process continues.

ret = mmc_suspend_host(host->mmc);
if (ret) {
host->suspended = 0;
if (host->pdata->resume) {
ret = host->pdata->resume(dev, host->slot_id);

The return status from mmc_suspend_host() is overwritten by return
status from host->pdata->resume. So the original return status is lost.

In these cases the MMC core gets to an unexpected state
during resume and multiple issues related to MMC crop up.
1. Host controller driver starts accessing the device registers
before the clocks are enabled which leads to a prefetch abort.
2. A file copy thread which was launched before suspend gets
stuck due to the host not being reclaimed during resume.

To avoid such problems pass on the -EBUSY status to the PM core
from the host controller driver. With this change, MMC core
suspend might still fail but it does not end up making the
system unusable. Suspend gets aborted and the user can try
suspending the system again.

Signed-off-by: Vaibhav Bedia <vaibhav.bedia@ti.com>
Signed-off-by: Hebbar, Gururaja <gururaja.hebbar@ti.com>
Acked-by: Venkatraman S <svenkatr@ti.com>
Cc: stable <stable@vger.kernel.org>
Signed-off-by: Chris Ball <cjb@laptop.org>
drivers/mmc/host/omap_hsmmc.c

index 9afdd202b8735bf5e36509c1952f9f0e8a86d7be..d9af5f1463aaf74ccf208d42a23754fc136d8e0c 100644 (file)
@@ -2050,8 +2050,7 @@ static int omap_hsmmc_suspend(struct device *dev)
        if (ret) {
                host->suspended = 0;
                if (host->pdata->resume) {
-                       ret = host->pdata->resume(dev, host->slot_id);
-                       if (ret)
+                       if (host->pdata->resume(dev, host->slot_id))
                                dev_dbg(dev, "Unmask interrupt failed\n");
                }
                goto err;