TMIO_STAT_CARD_REMOVE | TMIO_STAT_CARD_INSERT)
#define TMIO_MASK_IRQ (TMIO_MASK_READOP | TMIO_MASK_WRITEOP | TMIO_MASK_CMD)
+#define TMIO_MIN_DMA_LEN 8
+
#define enable_mmc_irqs(host, i) \
do { \
u32 mask;\
struct platform_device *pdev;
/* DMA support */
+ bool force_pio;
struct dma_chan *chan_rx;
struct dma_chan *chan_tx;
struct tasklet_struct dma_complete;
host->cmd = NULL;
host->data = NULL;
host->mrq = NULL;
+ host->force_pio = false;
spin_unlock_irqrestore(&host->lock, flags);
host->mrq = NULL;
host->cmd = NULL;
host->data = NULL;
+ host->force_pio = false;
cancel_delayed_work(&host->delayed_reset_work);
unsigned int count;
unsigned long flags;
- if (host->chan_tx || host->chan_rx) {
+ if ((host->chan_tx || host->chan_rx) && !host->force_pio) {
pr_err("PIO IRQ in DMA mode!\n");
return;
} else if (!data) {
*/
if (data->flags & MMC_DATA_READ) {
- if (!host->chan_rx)
- disable_mmc_irqs(host, TMIO_MASK_READOP);
- else
+ if (host->chan_rx && !host->force_pio)
tmio_check_bounce_buffer(host);
dev_dbg(&host->pdev->dev, "Complete Rx request %p\n",
host->mrq);
} else {
- if (!host->chan_tx)
- disable_mmc_irqs(host, TMIO_MASK_WRITEOP);
dev_dbg(&host->pdev->dev, "Complete Tx request %p\n",
host->mrq);
}
if (!data)
goto out;
- if (host->chan_tx && (data->flags & MMC_DATA_WRITE)) {
+ if (host->chan_tx && (data->flags & MMC_DATA_WRITE) && !host->force_pio) {
/*
* Has all data been written out yet? Testing on SuperH showed,
* that in most cases the first interrupt comes already with the
disable_mmc_irqs(host, TMIO_STAT_DATAEND);
tasklet_schedule(&host->dma_complete);
}
- } else if (host->chan_rx && (data->flags & MMC_DATA_READ)) {
+ } else if (host->chan_rx && (data->flags & MMC_DATA_READ) && !host->force_pio) {
disable_mmc_irqs(host, TMIO_STAT_DATAEND);
tasklet_schedule(&host->dma_complete);
} else {
tmio_mmc_do_data_irq(host);
+ disable_mmc_irqs(host, TMIO_MASK_READOP | TMIO_MASK_WRITEOP);
}
out:
spin_unlock(&host->lock);
*/
if (host->data && !cmd->error) {
if (host->data->flags & MMC_DATA_READ) {
- if (!host->chan_rx)
+ if (host->force_pio || !host->chan_rx)
enable_mmc_irqs(host, TMIO_MASK_READOP);
else
tasklet_schedule(&host->dma_issue);
} else {
- if (!host->chan_tx)
+ if (host->force_pio || !host->chan_tx)
enable_mmc_irqs(host, TMIO_MASK_WRITEOP);
else
tasklet_schedule(&host->dma_issue);
goto pio;
}
+ if (sg->length < TMIO_MIN_DMA_LEN) {
+ host->force_pio = true;
+ return;
+ }
+
disable_mmc_irqs(host, TMIO_STAT_RXRDY);
/* The only sg element can be unaligned, use our bounce buffer then */
goto pio;
}
+ if (sg->length < TMIO_MIN_DMA_LEN) {
+ host->force_pio = true;
+ return;
+ }
+
disable_mmc_irqs(host, TMIO_STAT_TXRQ);
/* The only sg element can be unaligned, use our bounce buffer then */
fail:
host->mrq = NULL;
+ host->force_pio = false;
mrq->cmd->error = ret;
mmc_request_done(mmc, mrq);
}