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:
parent
96e6835793
commit
8938550072
@ -47,6 +47,11 @@
|
||||
* 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) ******************************************/
|
||||
|
||||
#define LPC54_SDMMC_CTRL_OFFSET 0x0000 /* Control register */
|
||||
|
@ -135,6 +135,11 @@
|
||||
#define SDCARD_BUS_D4 1
|
||||
#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 */
|
||||
|
||||
#define SDCARD_RECV_MASK (SDMMC_INT_DCRC | SDMMC_INT_RCRC | SDMMC_INT_DRTO | \
|
||||
@ -187,7 +192,6 @@
|
||||
#define SDCARD_DMADONE_FLAG (2)
|
||||
#define SDCARD_ALLDONE (3)
|
||||
|
||||
|
||||
/* Card debounce time. Number of host clocks (SD_CLK) used by debounce
|
||||
* 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);
|
||||
|
||||
/* 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 */
|
||||
|
||||
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,
|
||||
size_t buflen)
|
||||
{
|
||||
struct lpc54_dev_s *priv = (struct lpc54_dev_s *)dev;
|
||||
struct lpc54_dev_s *priv;
|
||||
uint32_t regval;
|
||||
uint32_t ctrl = 0;
|
||||
uint32_t maxs = 0;
|
||||
int ret = OK;
|
||||
int i = 0;
|
||||
uint32_t ctrl;
|
||||
uint32_t maxs;
|
||||
int ret;
|
||||
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(((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 */
|
||||
|
||||
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 */
|
||||
|
||||
i = 0;
|
||||
while (buflen > 0)
|
||||
{
|
||||
/* 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].des3 = (uint32_t) &g_sdmmc_dmadd[i + 1];
|
||||
|
||||
i++;
|
||||
}
|
||||
|
||||
lpc54_putreg((uint32_t) &g_sdmmc_dmadd[0], LPC54_SDMMC_DBADDR);
|
||||
}
|
||||
|
||||
return ret;
|
||||
return OK;
|
||||
}
|
||||
#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;
|
||||
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);
|
||||
|
||||
priv = (struct lpc54_dev_s *)dev;
|
||||
DEBUGASSERT(priv != NULL && buffer != NULL && buflen > 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
|
||||
* handler.
|
||||
*/
|
||||
@ -2323,7 +2320,9 @@ static int lpc54_dmasendsetup(FAR struct sdio_dev_s *dev,
|
||||
regval = lpc54_getreg(LPC54_SDMMC_CTRL);
|
||||
regval |= SDMMC_CTRL_FIFORESET | SDMMC_CTRL_DMARESET;
|
||||
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 */
|
||||
|
||||
@ -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];
|
||||
|
||||
lpc54_putreg((uint32_t) &g_sdmmc_dmadd[0], LPC54_SDMMC_DBADDR);
|
||||
ret = OK;
|
||||
}
|
||||
|
||||
return ret;
|
||||
return OK;
|
||||
}
|
||||
#endif
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user