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
|
* 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 */
|
||||||
|
@ -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
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user