diff options
author | Seungwon Jeon <tgih.jun@samsung.com> | 2013-03-14 02:17:13 -0400 |
---|---|---|
committer | Chris Ball <cjb@laptop.org> | 2013-03-22 13:29:36 -0400 |
commit | ef3a69c7a40d5350adeaed7f0f4b4dcf6362dbbd (patch) | |
tree | cb77517ba212670cb6234624848c4b3afc69eda6 /drivers/mmc/card | |
parent | 3500ed90b26a9935b943b5e2e4cd3226600d6b58 (diff) |
mmc: block: fix the host's claim-release in special request
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>
Diffstat (limited to 'drivers/mmc/card')
-rw-r--r-- | drivers/mmc/card/block.c | 10 | ||||
-rw-r--r-- | drivers/mmc/card/queue.c | 3 | ||||
-rw-r--r-- | drivers/mmc/card/queue.h | 2 |
3 files changed, 10 insertions, 5 deletions
diff --git a/drivers/mmc/card/block.c b/drivers/mmc/card/block.c index 5bab73b91c20..e12a03cc2a6e 100644 --- a/drivers/mmc/card/block.c +++ b/drivers/mmc/card/block.c | |||
@@ -1932,8 +1932,14 @@ static int mmc_blk_issue_rq(struct mmc_queue *mq, struct request *req) | |||
1932 | } | 1932 | } |
1933 | 1933 | ||
1934 | out: | 1934 | out: |
1935 | if (!req && !(mq->flags & MMC_QUEUE_NEW_REQUEST)) | 1935 | if ((!req && !(mq->flags & MMC_QUEUE_NEW_REQUEST)) || |
1936 | /* release host only when there are no more requests */ | 1936 | (req && (req->cmd_flags & MMC_REQ_SPECIAL_MASK))) |
1937 | /* | ||
1938 | * Release host when there are no more requests | ||
1939 | * and after special request(discard, flush) is done. | ||
1940 | * In case sepecial request, there is no reentry to | ||
1941 | * the 'mmc_blk_issue_rq' with 'mqrq_prev->req'. | ||
1942 | */ | ||
1937 | mmc_release_host(card->host); | 1943 | mmc_release_host(card->host); |
1938 | return ret; | 1944 | return ret; |
1939 | } | 1945 | } |
diff --git a/drivers/mmc/card/queue.c b/drivers/mmc/card/queue.c index fa4e44ee7961..9447a0e970d1 100644 --- a/drivers/mmc/card/queue.c +++ b/drivers/mmc/card/queue.c | |||
@@ -22,9 +22,6 @@ | |||
22 | 22 | ||
23 | #define MMC_QUEUE_BOUNCESZ 65536 | 23 | #define MMC_QUEUE_BOUNCESZ 65536 |
24 | 24 | ||
25 | |||
26 | #define MMC_REQ_SPECIAL_MASK (REQ_DISCARD | REQ_FLUSH) | ||
27 | |||
28 | /* | 25 | /* |
29 | * Prepare a MMC request. This just filters out odd stuff. | 26 | * Prepare a MMC request. This just filters out odd stuff. |
30 | */ | 27 | */ |
diff --git a/drivers/mmc/card/queue.h b/drivers/mmc/card/queue.h index 031bf6376c99..5752d50049a3 100644 --- a/drivers/mmc/card/queue.h +++ b/drivers/mmc/card/queue.h | |||
@@ -1,6 +1,8 @@ | |||
1 | #ifndef MMC_QUEUE_H | 1 | #ifndef MMC_QUEUE_H |
2 | #define MMC_QUEUE_H | 2 | #define MMC_QUEUE_H |
3 | 3 | ||
4 | #define MMC_REQ_SPECIAL_MASK (REQ_DISCARD | REQ_FLUSH) | ||
5 | |||
4 | struct request; | 6 | struct request; |
5 | struct task_struct; | 7 | struct task_struct; |
6 | 8 | ||