Skip to content

Commit

Permalink
mmc: core/mmci: restore pre/post_req behaviour
Browse files Browse the repository at this point in the history
commit 64b12a6
"mmc: core: fix prepared requests while doing bkops"
is fixing a bug in the wrong way. A bug in the MMCI
device driver is fixed by amending the MMC core.

Thinking about it: what the pre- and post-callbacks
are doing is to essentially map and unmap SG lists
for DMA transfers. Why would we not be able to do that
just because a BKOPS command is sent inbetween?
Having to unprepare/prepare the next asynchronous
request for DMA seems wrong.

Looking the backtrace in that commit we can see what
the real problem actually is:

mmci_data_irq() is calling mmci_dma_unmap() twice
which is goung to call arm_dma_unmap_sg() twice
and v7_dma_inv_range() twice for the same sglist
and that will crash.

This happens because a request is prepared, then
a BKOPS is sent. The IRQ completing the BKOPS command
goes through mmci_data_irq() and thinks that a DMA
operation has just been completed because
dma_inprogress() reports true. It then proceeds to
unmap the sglist.

But that was wrong! dma_inprogress() should NOT be
true because no DMA was actually in progress! We had
just prepared the sglist, and the DMA channel
dma_current has been configured, but NOT started!

Because of this, the sglist is already unmapped when
we get our actual data completion IRQ, and we are
unmapping the sglist once more, and we get this crash.

Therefore, we need to revert this solution pushing
the problem to the core and causing problems, and
instead augment the implementation such that
dma_inprogress() only reports true if some DMA has
actually been started.

After this we can keep the request prepared during the
BKOPS and we need not unprepare/reprepare it.

Fixes: 64b12a6 ("mmc: core: fix prepared requests while doing bkops")
Cc: Srinivas Kandagatla <[email protected]>
Signed-off-by: Linus Walleij <[email protected]>
Tested-by: Srinivas Kandagatla <[email protected]>
Signed-off-by: Ulf Hansson <[email protected]>
  • Loading branch information
linusw authored and storulf committed Feb 13, 2017
1 parent 43c15e9 commit e13934b
Show file tree
Hide file tree
Showing 3 changed files with 8 additions and 11 deletions.
9 changes: 0 additions & 9 deletions drivers/mmc/core/core.c
Original file line number Diff line number Diff line change
Expand Up @@ -676,16 +676,7 @@ struct mmc_async_req *mmc_start_req(struct mmc_host *host,
((mmc_resp_type(host->areq->mrq->cmd) == MMC_RSP_R1) ||
(mmc_resp_type(host->areq->mrq->cmd) == MMC_RSP_R1B)) &&
(host->areq->mrq->cmd->resp[0] & R1_EXCEPTION_EVENT)) {

/* Cancel the prepared request */
if (areq)
mmc_post_req(host, areq->mrq, -EINVAL);

mmc_start_bkops(host->card, true);

/* prepare the request again */
if (areq)
mmc_pre_req(host, areq->mrq);
}
}

Expand Down
7 changes: 6 additions & 1 deletion drivers/mmc/host/mmci.c
Original file line number Diff line number Diff line change
Expand Up @@ -507,6 +507,7 @@ static void mmci_dma_data_error(struct mmci_host *host)
{
dev_err(mmc_dev(host->mmc), "error during DMA transfer!\n");
dmaengine_terminate_all(host->dma_current);
host->dma_in_progress = false;
host->dma_current = NULL;
host->dma_desc_current = NULL;
host->data->host_cookie = 0;
Expand Down Expand Up @@ -565,6 +566,7 @@ static void mmci_dma_finalize(struct mmci_host *host, struct mmc_data *data)
mmci_dma_release(host);
}

host->dma_in_progress = false;
host->dma_current = NULL;
host->dma_desc_current = NULL;
}
Expand Down Expand Up @@ -665,6 +667,7 @@ static int mmci_dma_start_data(struct mmci_host *host, unsigned int datactrl)
dev_vdbg(mmc_dev(host->mmc),
"Submit MMCI DMA job, sglen %d blksz %04x blks %04x flags %08x\n",
data->sg_len, data->blksz, data->blocks, data->flags);
host->dma_in_progress = true;
dmaengine_submit(host->dma_desc_current);
dma_async_issue_pending(host->dma_current);

Expand Down Expand Up @@ -740,8 +743,10 @@ static void mmci_post_request(struct mmc_host *mmc, struct mmc_request *mrq,
if (host->dma_desc_current == next->dma_desc)
host->dma_desc_current = NULL;

if (host->dma_current == next->dma_chan)
if (host->dma_current == next->dma_chan) {
host->dma_in_progress = false;
host->dma_current = NULL;
}

next->dma_desc = NULL;
next->dma_chan = NULL;
Expand Down
3 changes: 2 additions & 1 deletion drivers/mmc/host/mmci.h
Original file line number Diff line number Diff line change
Expand Up @@ -245,8 +245,9 @@ struct mmci_host {
struct dma_chan *dma_tx_channel;
struct dma_async_tx_descriptor *dma_desc_current;
struct mmci_host_next next_data;
bool dma_in_progress;

#define dma_inprogress(host) ((host)->dma_current)
#define dma_inprogress(host) ((host)->dma_in_progress)
#else
#define dma_inprogress(host) (0)
#endif
Expand Down

0 comments on commit e13934b

Please sign in to comment.