arch/arm/src/lpc54xx: In SDMMC driver, don't do DMA if the entire transfer will fit in the FIFO.

This commit is contained in:
Gregory Nutt 2017-12-20 14:38:09 -06:00
parent 96e6835793
commit 8938550072
2 changed files with 122 additions and 121 deletions

View File

@ -47,6 +47,11 @@
* Pre-processor Definitions * Pre-processor Definitions
************************************************************************************************/ ************************************************************************************************/
#define LPC54_TXFIFO_DEPTH 32
#define LPC54_TXFIFO_WIDTH 4
#define LPC54_RXFIFO_DEPTH 32
#define LPC54_RXFIFO_WIDTH 4
/* MCI register offsets (with respect to the MCI base) ******************************************/ /* MCI register offsets (with respect to the MCI base) ******************************************/
#define LPC54_SDMMC_CTRL_OFFSET 0x0000 /* Control register */ #define LPC54_SDMMC_CTRL_OFFSET 0x0000 /* Control register */

View File

@ -135,6 +135,11 @@
#define SDCARD_BUS_D4 1 #define SDCARD_BUS_D4 1
#define SDCARD_BUS_D8 0x100 #define SDCARD_BUS_D8 0x100
/* FIFO size in bytes */
#define LPC54_TXFIFO_SIZE (LPC54_TXFIFO_DEPTH | LPC54_TXFIFO_WIDTH)
#define LPC54_RXFIFO_SIZE (LPC54_RXFIFO_DEPTH | LPC54_RXFIFO_WIDTH)
/* Data transfer interrupt mask bits */ /* Data transfer interrupt mask bits */
#define SDCARD_RECV_MASK (SDMMC_INT_DCRC | SDMMC_INT_RCRC | SDMMC_INT_DRTO | \ #define SDCARD_RECV_MASK (SDMMC_INT_DCRC | SDMMC_INT_RCRC | SDMMC_INT_DRTO | \
@ -187,7 +192,6 @@
#define SDCARD_DMADONE_FLAG (2) #define SDCARD_DMADONE_FLAG (2)
#define SDCARD_ALLDONE (3) #define SDCARD_ALLDONE (3)
/* Card debounce time. Number of host clocks (SD_CLK) used by debounce /* Card debounce time. Number of host clocks (SD_CLK) used by debounce
* filter logic for card detect. typical debounce time is 5-25 ms. * filter logic for card detect. typical debounce time is 5-25 ms.
* *
@ -1595,20 +1599,6 @@ static int lpc54_cancel(FAR struct sdio_dev_s *dev)
(void)wd_cancel(priv->waitwdog); (void)wd_cancel(priv->waitwdog);
/* If this was a DMA transfer, make sure that DMA is stopped */
#ifdef CONFIG_LPC54_SDMMC_DMA
if (priv->dmamode)
{
/* Make sure that the DMA is stopped (it will be stopped automatically
* on normal transfers, but not necessarily when the transfer terminates
* on an error condition.
*/
//lpc54_dmastop(priv->dma);
}
#endif
/* Mark no transfer in progress */ /* Mark no transfer in progress */
priv->remaining = 0; priv->remaining = 0;
@ -2182,24 +2172,28 @@ static int lpc54_registercallback(FAR struct sdio_dev_s *dev,
static int lpc54_dmarecvsetup(FAR struct sdio_dev_s *dev, FAR uint8_t *buffer, static int lpc54_dmarecvsetup(FAR struct sdio_dev_s *dev, FAR uint8_t *buffer,
size_t buflen) size_t buflen)
{ {
struct lpc54_dev_s *priv = (struct lpc54_dev_s *)dev; struct lpc54_dev_s *priv;
uint32_t regval; uint32_t regval;
uint32_t ctrl = 0; uint32_t ctrl;
uint32_t maxs = 0; uint32_t maxs;
int ret = OK; int ret;
int i = 0; int i;
mcinfo("DMA size: %d\n", buflen); /* Don't bother with DMA if the entire transfer will fit in the RX FIFO or
* if we do not have a 4-bit wide bus.
*/
if (buflen <= LPC54_RXFIFO_SIZE || !priv->widebus)
{
return lpc54_recvsetup(dev, buffer, buflen);
}
mcinfo("buflen=%lu\n", (unsigned long)buflen);
priv = (struct lpc54_dev_s *)dev;
DEBUGASSERT(priv != NULL && buffer != NULL && buflen > 0); DEBUGASSERT(priv != NULL && buffer != NULL && buflen > 0);
DEBUGASSERT(((uint32_t)buffer & 3) == 0); DEBUGASSERT(((uint32_t)buffer & 3) == 0);
ctrl = MCI_DMADES0_OWN | MCI_DMADES0_CH;
/* Wide bus operation is required for DMA */
if (priv->widebus)
{
/* Save the destination buffer information for use by the interrupt handler */ /* Save the destination buffer information for use by the interrupt handler */
priv->buffer = (uint32_t *)buffer; priv->buffer = (uint32_t *)buffer;
@ -2218,6 +2212,7 @@ static int lpc54_dmarecvsetup(FAR struct sdio_dev_s *dev, FAR uint8_t *buffer,
/* Setup DMA list */ /* Setup DMA list */
i = 0;
while (buflen > 0) while (buflen > 0)
{ {
/* Limit size of the transfer to maximum buffer size */ /* Limit size of the transfer to maximum buffer size */
@ -2263,14 +2258,11 @@ static int lpc54_dmarecvsetup(FAR struct sdio_dev_s *dev, FAR uint8_t *buffer,
g_sdmmc_dmadd[i].des0 = ctrl; g_sdmmc_dmadd[i].des0 = ctrl;
g_sdmmc_dmadd[i].des3 = (uint32_t) &g_sdmmc_dmadd[i + 1]; g_sdmmc_dmadd[i].des3 = (uint32_t) &g_sdmmc_dmadd[i + 1];
i++; i++;
} }
lpc54_putreg((uint32_t) &g_sdmmc_dmadd[0], LPC54_SDMMC_DBADDR); lpc54_putreg((uint32_t) &g_sdmmc_dmadd[0], LPC54_SDMMC_DBADDR);
} return OK;
return ret;
} }
#endif #endif
@ -2299,17 +2291,22 @@ static int lpc54_dmasendsetup(FAR struct sdio_dev_s *dev,
{ {
struct lpc54_dev_s *priv = (struct lpc54_dev_s *)dev; struct lpc54_dev_s *priv = (struct lpc54_dev_s *)dev;
uint32_t regval; uint32_t regval;
int ret = -EPERM;
/* Don't bother with DMA if the entire transfer will fit in the TX FIFO or
* if we do not have a 4-bit wide bus.
*/
if (buflen <= LPC54_TXFIFO_SIZE || !priv->widebus)
{
return lpc54_sendsetup(dev, buffer, buflen);
}
mcinfo("buflen=%lu\n", (unsigned long)buflen); mcinfo("buflen=%lu\n", (unsigned long)buflen);
priv = (struct lpc54_dev_s *)dev;
DEBUGASSERT(priv != NULL && buffer != NULL && buflen > 0); DEBUGASSERT(priv != NULL && buffer != NULL && buflen > 0);
DEBUGASSERT(((uint32_t)buffer & 3) == 0); DEBUGASSERT(((uint32_t)buffer & 3) == 0);
/* Wide bus operation is required for DMA */
if (priv->widebus)
{
/* Save the destination buffer information for use by the interrupt /* Save the destination buffer information for use by the interrupt
* handler. * handler.
*/ */
@ -2323,7 +2320,9 @@ static int lpc54_dmasendsetup(FAR struct sdio_dev_s *dev,
regval = lpc54_getreg(LPC54_SDMMC_CTRL); regval = lpc54_getreg(LPC54_SDMMC_CTRL);
regval |= SDMMC_CTRL_FIFORESET | SDMMC_CTRL_DMARESET; regval |= SDMMC_CTRL_FIFORESET | SDMMC_CTRL_DMARESET;
lpc54_putreg(regval, LPC54_SDMMC_CTRL); lpc54_putreg(regval, LPC54_SDMMC_CTRL);
while (lpc54_getreg(LPC54_SDMMC_CTRL) & SDMMC_CTRL_DMARESET); while (lpc54_getreg(LPC54_SDMMC_CTRL) & SDMMC_CTRL_DMARESET)
{
}
/* Setup DMA descriptor list */ /* Setup DMA descriptor list */
@ -2333,10 +2332,7 @@ static int lpc54_dmasendsetup(FAR struct sdio_dev_s *dev,
g_sdmmc_dmadd[0].des3 = (uint32_t)&g_sdmmc_dmadd[1]; g_sdmmc_dmadd[0].des3 = (uint32_t)&g_sdmmc_dmadd[1];
lpc54_putreg((uint32_t) &g_sdmmc_dmadd[0], LPC54_SDMMC_DBADDR); lpc54_putreg((uint32_t) &g_sdmmc_dmadd[0], LPC54_SDMMC_DBADDR);
ret = OK; return OK;
}
return ret;
} }
#endif #endif