mmc: block: fix the host's claim-release in special request
authorSeungwon Jeon <tgih.jun@samsung.com>
Thu, 14 Mar 2013 06:17:13 +0000 (15:17 +0900)
committerChris Ball <cjb@laptop.org>
Fri, 22 Mar 2013 17:29:36 +0000 (13:29 -0400)
For normal request mmc_blk_issue_rq is called twice with asynchronous
transfer(cur and prev). Host's claim and release can be done in each
mmc_blk_issue_rq. However, Special request is currently excluded in
asynchronous transfer. After special request is finished, if there is
no new request, mmc_release_host won't be called in mmc_blk_issue_rq.
The problem is founded during mmc_suspend.

[<c0541124>] (__schedule+0x0/0x78c) from [<c05419e8>] (schedule+0x38/0x78)
[<c05419b0>] (schedule+0x0/0x78) from [<c03a843c>] (__mmc_claim_host+0xac/0x1b4)
[<c03a8390>] (__mmc_claim_host+0x0/0x1b4) from [<c03ac98c>] (mmc_suspend+0x28/0x9c)
[<c03ac964>] (mmc_suspend+0x0/0x9c) from [<c03aad24>] (mmc_suspend_host+0xb4/0x194)
...

Reported-by: Johan Rudholm <jrudholm@gmail.com>
Signed-off-by: Seungwon Jeon <tgih.jun@samsung.com>
Tested-by: Johan Rudholm <johan.rudholm@stericsson.com>
Signed-off-by: Chris Ball <cjb@laptop.org>
drivers/mmc/card/block.c
drivers/mmc/card/queue.c
drivers/mmc/card/queue.h

index 5bab73b91c209a1cce2e92e5af29fa212f169ae5..e12a03cc2a6e924e6ef7cb2597a91c0383f8afdb 100644 (file)
@@ -1932,8 +1932,14 @@ static int mmc_blk_issue_rq(struct mmc_queue *mq, struct request *req)
        }
 
 out:
-       if (!req && !(mq->flags & MMC_QUEUE_NEW_REQUEST))
-               /* release host only when there are no more requests */
+       if ((!req && !(mq->flags & MMC_QUEUE_NEW_REQUEST)) ||
+            (req && (req->cmd_flags & MMC_REQ_SPECIAL_MASK)))
+               /*
+                * Release host when there are no more requests
+                * and after special request(discard, flush) is done.
+                * In case sepecial request, there is no reentry to
+                * the 'mmc_blk_issue_rq' with 'mqrq_prev->req'.
+                */
                mmc_release_host(card->host);
        return ret;
 }
index fa4e44ee79613760a47bf7a11578e7293923a831..9447a0e970d1bed4be023a6a96bc8366e10446da 100644 (file)
@@ -22,9 +22,6 @@
 
 #define MMC_QUEUE_BOUNCESZ     65536
 
-
-#define MMC_REQ_SPECIAL_MASK   (REQ_DISCARD | REQ_FLUSH)
-
 /*
  * Prepare a MMC request. This just filters out odd stuff.
  */
index 031bf6376c994ae566e4edbfc0a886c40f1728a6..5752d50049a34c2a9ee8f6585105990f1e018d05 100644 (file)
@@ -1,6 +1,8 @@
 #ifndef MMC_QUEUE_H
 #define MMC_QUEUE_H
 
+#define MMC_REQ_SPECIAL_MASK   (REQ_DISCARD | REQ_FLUSH)
+
 struct request;
 struct task_struct;