UPSTREAM: mmc: core: fix __mmc_switch timeout caused by preempt
authorChaotian Jing <chaotian.jing@mediatek.com>
Mon, 30 Nov 2015 01:27:30 +0000 (09:27 +0800)
committerGerrit Code Review <gerrit@rock-chips.com>
Fri, 25 Mar 2016 06:58:22 +0000 (14:58 +0800)
there is a time window between __mmc_send_status() and time_afer(),
on some eMMC chip, the timeout_ms is only 10ms, if this thread was
scheduled out during this period, then, even card has already changes
to transfer state by the result of CMD13, this part of code also treat
it to timeout error.
So, need calculate timeout first, then call __mmc_send_status(), if
already timeout and card still in programing state, then treat it to
the real timeout error.

Change-Id: I7499d3d41711ea5abe6baec780d2988dc60dfc5b
Signed-off-by: Chaotian Jing <chaotian.jing@mediatek.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
Signed-off-by: Huang, Tao <huangtao@rock-chips.com>
(cherry picked from commit 3bbb0deea6d5c6d5ed38ae927a5bf9b0cd7c8639)

drivers/mmc/core/mmc_ops.c

index 1f444269ebbe66027831757403d1bdd53d1b6a8d..2c90635c89afbb3782a48145fe830828dca4ec27 100644 (file)
@@ -489,6 +489,7 @@ int __mmc_switch(struct mmc_card *card, u8 set, u8 index, u8 value,
        unsigned long timeout;
        u32 status = 0;
        bool use_r1b_resp = use_busy_signal;
+       bool expired = false;
 
        mmc_retune_hold(host);
 
@@ -545,6 +546,12 @@ int __mmc_switch(struct mmc_card *card, u8 set, u8 index, u8 value,
        timeout = jiffies + msecs_to_jiffies(timeout_ms);
        do {
                if (send_status) {
+                       /*
+                        * Due to the possibility of being preempted after
+                        * sending the status command, check the expiration
+                        * time first.
+                        */
+                       expired = time_after(jiffies, timeout);
                        err = __mmc_send_status(card, &status, ignore_crc);
                        if (err)
                                goto out;
@@ -565,7 +572,7 @@ int __mmc_switch(struct mmc_card *card, u8 set, u8 index, u8 value,
                }
 
                /* Timeout if the device never leaves the program state. */
-               if (time_after(jiffies, timeout)) {
+               if (expired && R1_CURRENT_STATE(status) == R1_STATE_PRG) {
                        pr_err("%s: Card stuck in programming state! %s\n",
                                mmc_hostname(host), __func__);
                        err = -ETIMEDOUT;