EFM32: Integrate SPI DMA capability

This commit is contained in:
Gregory Nutt 2014-10-26 11:22:16 -06:00
parent 741f98fe0a
commit c392115fbd
5 changed files with 553 additions and 104 deletions

View File

@ -340,4 +340,38 @@ config LEUART1_2STOP
endmenu # LEUART1 Configuration endmenu # LEUART1 Configuration
if EFM32_USART_ISSPI
menu "SPI Configuration"
config EFM32_SPI_DMA
bool "SPI DMA support"
default n
depends on EFM32_DMA
---help---
Select to enable DMA SPI transfers
if EFM32_SPI_DMA
config EFM32_SPI_DMA_TIMEO_NSEC
int "Per word timer (nsec)"
default 500
---help---
A timeout will be be used to detect hung DMA transfers. The timeout
will vary as a function of the number of words transferred. This
value provides the per-word timeout value in nanoseconds.
config EFM32_SPI_DMA_MINSIZE
int "Minimum DMA size"
default 16
---help---
DMA is particularly helpful for the case of large SPI transfers.
Smaller SPI transfer may be more efficiently performed without DMA.
This option determines a threshold: For transfers of this size and
below, DMA will not be used. A value of zero will force all DMA-
based transfers.
endif # EFM32_SPI_DMA
endmenu # SPI Configuration
endif # EFM32_USART_ISSPI
endif # ARCH_CHIP_EFM32 endif # ARCH_CHIP_EFM32

View File

@ -1,6 +1,6 @@
/******************************************************************************************************************************** /********************************************************************************************************************************
* arch/arm/src/efm32/EFM32GG/efm32_usart.h * arch/arm/src/efm32/EFM32GG/efm32_usart.h
* *
* (C) Copyright 2014 Silicon Labs, http://www.silabs.com * (C) Copyright 2014 Silicon Labs, http://www.silabs.com
* *
* Permission is granted to anyone to use this software for any purpose, * Permission is granted to anyone to use this software for any purpose,

View File

@ -457,6 +457,7 @@ void efm32_rxdmasetup(DMA_HANDLE handle, uintptr_t paddr, uintptr_t maddr,
unsigned int xfersize; unsigned int xfersize;
unsigned int shift; unsigned int shift;
uint32_t regval; uint32_t regval;
uint32_t incr;
uint32_t mask; uint32_t mask;
DEBUGASSERT(dmach != NULL && dmach->inuse); DEBUGASSERT(dmach != NULL && dmach->inuse);
@ -500,18 +501,28 @@ void efm32_rxdmasetup(DMA_HANDLE handle, uintptr_t paddr, uintptr_t maddr,
{ {
default: default:
case 0: /* Byte transfer */ case 0: /* Byte transfer */
regval = DMA_CTRL_DST_INC_BYTE | DMA_CTRL_DST_SIZE_BYTE | DMA_CTRL_SRC_SIZE_BYTE; regval |= DMA_CTRL_DST_SIZE_BYTE | DMA_CTRL_SRC_SIZE_BYTE;
incr = DMA_CTRL_DST_INC_BYTE;
break; break;
case 1: /* Half word transfer */ case 1: /* Half word transfer */
regval = DMA_CTRL_DST_INC_HALFWORD | DMA_CTRL_DST_SIZE_HALFWORD | DMA_CTRL_SRC_SIZE_HALFWORD; regval |= DMA_CTRL_DST_SIZE_HALFWORD | DMA_CTRL_SRC_SIZE_HALFWORD;
incr = DMA_CTRL_DST_INC_HALFWORD;
break; break;
case 2: /* Word transfer */ case 2: /* Word transfer */
regval = DMA_CTRL_DST_INC_WORD | DMA_CTRL_DST_SIZE_WORD | DMA_CTRL_SRC_SIZE_WORD; regval |= DMA_CTRL_DST_SIZE_WORD | DMA_CTRL_SRC_SIZE_WORD;
incr = DMA_CTRL_DST_INC_WORD;
break; break;
} }
/* Do we need to increment the memory address? */
if ((config & EFM32_DMA_MEMINCR_MASK) == EFM32_DMA_MEMINCR)
{
regval |= incr;
}
/* Set the number of transfers (minus 1) */ /* Set the number of transfers (minus 1) */
DEBUGASSERT((nbytes >> shift) < 1024); DEBUGASSERT((nbytes >> shift) < 1024);
@ -543,6 +554,7 @@ void efm32_txdmasetup(DMA_HANDLE handle, uintptr_t paddr, uintptr_t maddr,
unsigned int xfersize; unsigned int xfersize;
unsigned int shift; unsigned int shift;
uint32_t regval; uint32_t regval;
uint32_t incr;
uint32_t mask; uint32_t mask;
DEBUGASSERT(dmach != NULL && dmach->inuse); DEBUGASSERT(dmach != NULL && dmach->inuse);
@ -579,25 +591,35 @@ void efm32_txdmasetup(DMA_HANDLE handle, uintptr_t paddr, uintptr_t maddr,
*/ */
regval = DMA_CTRL_DST_INC_NONE | DMA_CTRL_DST_PROT_NON_PRIVILEGED | regval = DMA_CTRL_DST_INC_NONE | DMA_CTRL_DST_PROT_NON_PRIVILEGED |
DMA_CTRL_SRC_PROT_NON_PRIVILEGED | DMA_CTRL_R_POWER_1 | DMA_CTRL_SRC_PROT_NON_PRIVILEGED | DMA_CTRL_R_POWER_1 |
(0 << _DMA_CTRL_NEXT_USEBURST_SHIFT) | _DMA_CTRL_CYCLE_CTRL_BASIC; (0 << _DMA_CTRL_NEXT_USEBURST_SHIFT) | _DMA_CTRL_CYCLE_CTRL_BASIC;
switch (shift) switch (shift)
{ {
default: default:
case 0: /* Byte transfer */ case 0: /* Byte transfer */
regval = DMA_CTRL_DST_SIZE_BYTE | DMA_CTRL_SRC_INC_BYTE | DMA_CTRL_SRC_SIZE_BYTE; regval |= DMA_CTRL_DST_SIZE_BYTE | DMA_CTRL_SRC_SIZE_BYTE;
incr = DMA_CTRL_SRC_INC_BYTE;
break; break;
case 1: /* Half word transfer */ case 1: /* Half word transfer */
regval = DMA_CTRL_DST_SIZE_HALFWORD | DMA_CTRL_SRC_INC_HALFWORD | DMA_CTRL_SRC_SIZE_HALFWORD; regval |= DMA_CTRL_DST_SIZE_HALFWORD | DMA_CTRL_SRC_SIZE_HALFWORD;
incr = DMA_CTRL_SRC_INC_HALFWORD;
break; break;
case 2: /* Word transfer */ case 2: /* Word transfer */
regval = DMA_CTRL_DST_SIZE_WORD | DMA_CTRL_SRC_INC_WORD | DMA_CTRL_SRC_SIZE_WORD; regval |= DMA_CTRL_DST_SIZE_WORD | DMA_CTRL_SRC_SIZE_WORD;
incr = DMA_CTRL_SRC_INC_WORD;
break; break;
} }
/* Do we need to increment the memory address? */
if ((config & EFM32_DMA_MEMINCR_MASK) == EFM32_DMA_MEMINCR)
{
regval |= incr;
}
/* Set the number of transfers (minus 1) */ /* Set the number of transfers (minus 1) */
DEBUGASSERT((nbytes >> shift) < 1024); DEBUGASSERT((nbytes >> shift) < 1024);

View File

@ -63,23 +63,27 @@
* - Memory address is always incremented. * - Memory address is always incremented.
*/ */
#define EFM32_DMA_SIGSEL_SHIFT (0) /* Bits 0-3: _DMA_CH_CTRL_ * value */ #define EFM32_DMA_SIGSEL_SHIFT (0) /* Bits 0-3: _DMA_CH_CTRL_ * value */
#define EFM32_DMA_SIGSEL_MASK (15 << EFM32_DMA_SIGSEL_SHIFT) #define EFM32_DMA_SIGSEL_MASK (15 << EFM32_DMA_SIGSEL_SHIFT)
# define EFM32_DMA_SIGSEL(n) ((dma_config_t)(n) << EFM32_DMA_SIGSEL_SHIFT) # define EFM32_DMA_SIGSEL(n) ((dma_config_t)(n) << EFM32_DMA_SIGSEL_SHIFT)
#define EFM32_DMA_SOURCSEL_SHIFT (4) /* Bits 4-9: _DMA_CH_SOURCESEL_* value */ #define EFM32_DMA_SOURCSEL_SHIFT (4) /* Bits 4-9: _DMA_CH_SOURCESEL_* value */
#define EFM32_DMA_SOURCSEL_MASK (63 << EFM32_DMA_SOURCSEL_SHIFT) #define EFM32_DMA_SOURCSEL_MASK (63 << EFM32_DMA_SOURCSEL_SHIFT)
# define EFM32_DMA_SOURCSEL(n) (dma_config_t)(n) << EFM32_DMA_SOURCSEL_SHIFT) # define EFM32_DMA_SOURCSEL(n) ((dma_config_t)(n) << EFM32_DMA_SOURCSEL_SHIFT)
#define EFM32_DMA_XFERSIZE_SHIFT (10) /* Bits 10-11: Transfer size */ #define EFM32_DMA_XFERSIZE_SHIFT (10) /* Bits 10-11: Transfer size */
#define EFM32_DMA_XFERSIZE_MASK (3 << EFM32_DMA_XFERSIZE_SHIFT) #define EFM32_DMA_XFERSIZE_MASK (3 << EFM32_DMA_XFERSIZE_SHIFT)
# define EFM32_DMA_XFERSIZE_SHIFT_BYTE (0 << EFM32_DMA_SOURCSEL_SHIFT) # define EFM32_DMA_XFERSIZE_BYTE (0 << EFM32_DMA_SOURCSEL_SHIFT)
# define EFM32_DMA_XFERSIZE_SHIFT_HWORD (1 << EFM32_DMA_SOURCSEL_SHIFT) # define EFM32_DMA_XFERSIZE_HWORD (1 << EFM32_DMA_SOURCSEL_SHIFT)
# define EFM32_DMA_XFERSIZE_SHIFT_WORD (2 << EFM32_DMA_SOURCSEL_SHIFT) # define EFM32_DMA_XFERSIZE_WORD (2 << EFM32_DMA_SOURCSEL_SHIFT)
#define EFM32_DMA_SINGLE_MASK (1 << 12) /* Bit 12: Single or Buffer full request */ #define EFM32_DMA_SINGLE_MASK (1 << 12) /* Bit 12: Single or Buffer full request */
# define EFM32_DMA_SINGLE (1 << 12) /* 1=Buffer full request */ # define EFM32_DMA_SINGLE (1 << 12) /* 1=Buffer full request */
# define EFM32_DMA_BUFFER_FULL (0) /* 0=Buffer full request */ # define EFM32_DMA_BUFFER_FULL (0) /* 0=Buffer full request */
#define EFM32_DMA_MEMINCR_MASK (1 << 13) /* Bit 13: Increment memory address */
# define EFM32_DMA_MEMINCR (1 << 13) /* 1=Increment memory address */
# define EFM32_DMA_NOINCR (0) /* 0=No memory address increment */
/************************************************************************************ /************************************************************************************
* Public Types * Public Types

View File

@ -46,9 +46,12 @@
#include <stdbool.h> #include <stdbool.h>
#include <semaphore.h> #include <semaphore.h>
#include <errno.h> #include <errno.h>
#include <assert.h>
#include <debug.h> #include <debug.h>
#include <nuttx/arch.h> #include <nuttx/arch.h>
#include <nuttx/wdog.h>
#include <nuttx/clock.h>
#include <nuttx/spi/spi.h> #include <nuttx/spi/spi.h>
#include <arch/board/board.h> #include <arch/board/board.h>
@ -70,20 +73,31 @@
/* Configuration ********************************************************************/ /* Configuration ********************************************************************/
/* SPI interrupts */ /* SPI interrupts */
#ifdef CONFIG_EFM32_SPI_DMA
# error DMA driven SPI not yet supported
#endif
#ifdef CONFIG_EFM32_SPI_INTERRUPTS #ifdef CONFIG_EFM32_SPI_INTERRUPTS
# error Interrupt driven SPI not yet supported # error Interrupt driven SPI not yet supported
#endif #endif
#ifndef CONFIG_EFM32_SPI_DMA_TIMEO_NSEC
# define CONFIG_EFM32_SPI_DMA_TIMEO_NSEC 500
#endif
#ifndef CONFIG_EFM32_SPI_DMA_MINSIZE
# define CONFIG_EFM32_SPI_DMA_MINSIZE 16
#endif
/* Can't have both interrupt driven SPI and SPI DMA */ /* Can't have both interrupt driven SPI and SPI DMA */
#if defined(CONFIG_EFM32_SPI_INTERRUPTS) && defined(CONFIG_EFM32_SPI_DMA) #if defined(CONFIG_EFM32_SPI_INTERRUPTS) && defined(CONFIG_EFM32_SPI_DMA)
# error Cannot enable both interrupt mode and DMA mode for SPI # error Cannot enable both interrupt mode and DMA mode for SPI
#endif #endif
/* DMA definitions ******************************************************************/
#define SPI_DMA8_CONFIG (EFM32_DMA_XFERSIZE_BYTE| EFM32_DMA_MEMINCR)
#define SPI_DMA8NULL_CONFIG (EFM32_DMA_XFERSIZE_BYTE | EFM32_DMA_NOINCR)
#define SPI_DMA16_CONFIG (EFM32_DMA_XFERSIZE_HWORD | EFM32_DMA_MEMINCR)
#define SPI_DMA16NULL_CONFIG (EFM32_DMA_XFERSIZE_HWORD | EFM32_DMA_NOINCR)
/* Debug ****************************************************************************/ /* Debug ****************************************************************************/
/* Check if SPI debug is enabled */ /* Check if SPI debug is enabled */
@ -118,11 +132,11 @@ struct efm32_spiconfig_s
dma_config_t rxconfig; /* RX DMA configuration (excluding transfer width) */ dma_config_t rxconfig; /* RX DMA configuration (excluding transfer width) */
dma_config_t txconfig; /* TX DMA configuration (excluding transfer width) */ dma_config_t txconfig; /* TX DMA configuration (excluding transfer width) */
#endif #endif
void (*select)(FAR struct spi_dev_s *dev, enum spi_dev_e devid, void (*select)(struct spi_dev_s *dev, enum spi_dev_e devid,
bool selected); bool selected);
uint8_t (*status)(FAR struct spi_dev_s *dev, enum spi_dev_e devid); uint8_t (*status)(struct spi_dev_s *dev, enum spi_dev_e devid);
#ifdef CONFIG_SPI_CMDDATA #ifdef CONFIG_SPI_CMDDATA
int (*cmddata)(FAR struct spi_dev_s *dev, enum spi_dev_e devid, bool cmd); int (*cmddata)(struct spi_dev_s *dev, enum spi_dev_e devid, bool cmd);
#endif #endif
}; };
@ -134,8 +148,9 @@ struct efm32_spidev_s
const struct efm32_spiconfig_s *config; /* Constant SPI hardware configuration */ const struct efm32_spiconfig_s *config; /* Constant SPI hardware configuration */
#ifdef CONFIG_EFM32_SPI_DMA #ifdef CONFIG_EFM32_SPI_DMA
volatile int8_t rxresult; /* Result of the RX DMA */ WDOG_ID wdog; /* Timer to catch hung DMA */
volatile int8_t txresult; /* Result of the TX DMA */ volatile uint8_t rxresult; /* Result of the RX DMA */
volatile uint8_t txresult; /* Result of the TX DMA */
DMA_HANDLE rxdmach; /* RX DMA channel handle */ DMA_HANDLE rxdmach; /* RX DMA channel handle */
DMA_HANDLE txdmach; /* TX DMA channel handle */ DMA_HANDLE txdmach; /* TX DMA channel handle */
sem_t rxdmasem; /* Wait for RX DMA to complete */ sem_t rxdmasem; /* Wait for RX DMA to complete */
@ -164,6 +179,25 @@ static void spi_putreg(const struct efm32_spiconfig_s *config,
unsigned int regoffset, uint32_t regval); unsigned int regoffset, uint32_t regval);
static bool spi_16bitmode(struct efm32_spidev_s *priv); static bool spi_16bitmode(struct efm32_spidev_s *priv);
/* DMA support */
#ifdef CONFIG_EFM32_SPI_DMA
static void spi_dma_timeout(int argc, uint32_t arg1, ...);
static void spi_dmarxwait(struct efm32_spidev_s *priv);
static void spi_dmatxwait(struct efm32_spidev_s *priv);
static inline void spi_dmarxwakeup(struct efm32_spidev_s *priv);
static inline void spi_dmatxwakeup(struct efm32_spidev_s *priv);
static void spi_dmarxcallback(DMA_HANDLE handle, uint8_t status, void *arg);
static void spi_dmatxcallback(DMA_HANDLE handle, uint8_t status, void *arg);
static void spi_dmarxsetup(struct efm32_spidev_s *priv,
void *rxbuffer, void *rxdummy, size_t nwords);
static void spi_dmatxsetup(struct efm32_spidev_s *priv,
const void *txbuffer, const void *txdummy,
size_t nwords);
static inline void spi_dmarxstart(FAR struct efm32_spidev_s *priv);
static inline void spi_dmatxstart(FAR struct efm32_spidev_s *priv);
#endif
/* SPI methods */ /* SPI methods */
#ifndef CONFIG_SPI_OWNBUS #ifndef CONFIG_SPI_OWNBUS
@ -174,9 +208,9 @@ static void spi_select(struct spi_dev_s *dev, enum spi_dev_e devid,
static uint32_t spi_setfrequency(struct spi_dev_s *dev, uint32_t frequency); static uint32_t spi_setfrequency(struct spi_dev_s *dev, uint32_t frequency);
static void spi_setmode(struct spi_dev_s *dev, enum spi_mode_e mode); static void spi_setmode(struct spi_dev_s *dev, enum spi_mode_e mode);
static void spi_setbits(struct spi_dev_s *dev, int nbits); static void spi_setbits(struct spi_dev_s *dev, int nbits);
static uint8_t spi_status(FAR struct spi_dev_s *dev, enum spi_dev_e devid); static uint8_t spi_status(struct spi_dev_s *dev, enum spi_dev_e devid);
#ifdef CONFIG_SPI_CMDDATA #ifdef CONFIG_SPI_CMDDATA
static int spi_cmddata(FAR struct spi_dev_s *dev, enum spi_dev_e devid, static int spi_cmddata(struct spi_dev_s *dev, enum spi_dev_e devid,
bool cmd); bool cmd);
#endif #endif
static uint16_t spi_send(struct spi_dev_s *dev, uint16_t wd); static uint16_t spi_send(struct spi_dev_s *dev, uint16_t wd);
@ -191,7 +225,7 @@ static void spi_recvblock(struct spi_dev_s *dev, void *rxbuffer,
/* Initialization */ /* Initialization */
static void spi_portinitialize(struct efm32_spidev_s *priv); static int spi_portinitialize(struct efm32_spidev_s *priv);
/**************************************************************************** /****************************************************************************
* Private Data * Private Data
@ -357,7 +391,6 @@ static void spi_putreg(const struct efm32_spiconfig_s *config,
* True: 16-bit; false: 8-bit * True: 16-bit; false: 8-bit
* *
****************************************************************************/ ****************************************************************************/
/* helper SPI function */
static bool spi_16bitmode(struct efm32_spidev_s *priv) static bool spi_16bitmode(struct efm32_spidev_s *priv)
{ {
@ -373,16 +406,340 @@ static bool spi_16bitmode(struct efm32_spidev_s *priv)
#endif #endif
} }
/****************************************************************************
* Name: spi_dma_timeout
*
* Description:
* Invoked when a DMA timeout occurs
*
****************************************************************************/
#ifdef CONFIG_EFM32_SPI_DMA
static void spi_dma_timeout(int argc, uint32_t arg1, ...)
{
struct efm32_spidev_s *priv = (struct efm32_spidev_s *)((uintptr_t)arg1);
/* Mark DMA timeout error and wakeup form RX and TX waiters */
DEBUGASSERT(priv->rxresult == EINPROGRESS || priv->txresult == EINPROGRESS);
if (priv->rxresult == EINPROGRESS)
{
priv->rxresult = ETIMEDOUT;
spi_dmarxwakeup(priv);
}
if (priv->txresult == EINPROGRESS)
{
priv->txresult = ETIMEDOUT;
spi_dmatxwakeup(priv);
}
}
#endif
/****************************************************************************
* Name: spi_dmarxwait
*
* Description:
* Wait for RX DMA to complete.
*
****************************************************************************/
#ifdef CONFIG_EFM32_SPI_DMA
static void spi_dmarxwait(struct efm32_spidev_s *priv)
{
irqstate_t flags;
/* Take the semaphore (perhaps waiting). */
flags = irqsave();
while (sem_wait(&priv->rxdmasem) != 0)
{
/* The only case that an error should occur here is if the wait was awakened
* by a signal.
*/
DEBUGASSERT(errno == EINTR);
}
/* Cancel the timeout only if both the RX and TX transfers have completed */
DEBUGASSERT(priv->rxresult != EINPROGRESS);
if (priv->txresult != EINPROGRESS)
{
wd_cancel(priv->wdog);
}
irqrestore(flags);
}
#endif
/****************************************************************************
* Name: spi_dmatxwait
*
* Description:
* Wait for DMA to complete.
*
****************************************************************************/
#ifdef CONFIG_EFM32_SPI_DMA
static void spi_dmatxwait(struct efm32_spidev_s *priv)
{
irqstate_t flags;
/* Take the semaphore (perhaps waiting). */
flags = irqsave();
while (sem_wait(&priv->txdmasem) != 0)
{
/* The only case that an error should occur here is if the wait was awakened
* by a signal.
*/
DEBUGASSERT(errno == EINTR);
}
/* Cancel the timeout only if both the RX and TX transfers have completed */
DEBUGASSERT(priv->txresult != EINPROGRESS);
if (priv->rxresult != EINPROGRESS)
{
wd_cancel(priv->wdog);
}
irqrestore(flags);
}
#endif
/****************************************************************************
* Name: spi_dmarxwakeup
*
* Description:
* Signal that DMA is complete
*
****************************************************************************/
#ifdef CONFIG_EFM32_SPI_DMA
static inline void spi_dmarxwakeup(struct efm32_spidev_s *priv)
{
(void)sem_post(&priv->rxdmasem);
}
#endif
/****************************************************************************
* Name: spi_dmatxwakeup
*
* Description:
* Signal that DMA is complete
*
****************************************************************************/
#ifdef CONFIG_EFM32_SPI_DMA
static inline void spi_dmatxwakeup(struct efm32_spidev_s *priv)
{
(void)sem_post(&priv->txdmasem);
}
#endif
/****************************************************************************
* Name: spi_dmarxcallback
*
* Description:
* Called when the RX DMA completes
*
****************************************************************************/
#ifdef CONFIG_EFM32_SPI_DMA
static void spi_dmarxcallback(DMA_HANDLE handle, uint8_t status, void *arg)
{
struct efm32_spidev_s *priv = (struct efm32_spidev_s *)arg;
DEBUGASSERT(priv && status != EINPROGRESS);
/* Wake-up the SPI driver */
priv->rxresult = status;
spi_dmarxwakeup(priv);
}
#endif
/****************************************************************************
* Name: spi_dmatxcallback
*
* Description:
* Called when the RX DMA completes
*
****************************************************************************/
#ifdef CONFIG_EFM32_SPI_DMA
static void spi_dmatxcallback(DMA_HANDLE handle, uint8_t status, void *arg)
{
struct efm32_spidev_s *priv = (struct efm32_spidev_s *)arg;
DEBUGASSERT(priv && status != EINPROGRESS);
/* Wake-up the SPI driver */
priv->txresult = status;
spi_dmatxwakeup(priv);
}
#endif
/****************************************************************************
* Name: spi_dmarxsetup
*
* Description:
* Setup to perform RX DMA
*
****************************************************************************/
#ifdef CONFIG_EFM32_SPI_DMA
static void spi_dmarxsetup(struct efm32_spidev_s *priv, void *rxbuffer,
void *rxdummy, size_t nwords)
{
const struct efm32_spiconfig_s *config = priv->config;
dma_config_t dmaconfig = config->rxconfig;
size_t nbytes;
/* 8- or 16-bit mode? */
if (spi_16bitmode(priv))
{
/* 16-bit mode -- is there a buffer to receive data in? */
if (rxbuffer)
{
dmaconfig |= SPI_DMA16_CONFIG;
}
else
{
rxbuffer = rxdummy;
dmaconfig |= SPI_DMA16NULL_CONFIG;
}
nbytes = nwords << 1;
}
else
{
/* 8-bit mode -- is there a buffer to receive data in? */
if (rxbuffer)
{
dmaconfig |= SPI_DMA8_CONFIG;
}
else
{
rxbuffer = rxdummy;
dmaconfig |= SPI_DMA8NULL_CONFIG;
}
nbytes = nwords;
}
/* Configure the RX DMA */
efm32_rxdmasetup(priv->rxdmach, config->base + EFM32_USART_RXDATA_OFFSET,
(uintptr_t)rxbuffer, nbytes, dmaconfig);
}
#endif
/****************************************************************************
* Name: spi_dmatxsetup
*
* Description:
* Setup to perform TX DMA
*
****************************************************************************/
#ifdef CONFIG_EFM32_SPI_DMA
static void spi_dmatxsetup(struct efm32_spidev_s *priv, const void *txbuffer,
const void *txdummy, size_t nwords)
{
const struct efm32_spiconfig_s *config = priv->config;
dma_config_t dmaconfig = config->txconfig;
size_t nbytes;
/* 8- or 16-bit mode? */
if (spi_16bitmode(priv))
{
/* 16-bit mode -- is there a buffer to receive data in? */
if (txbuffer)
{
dmaconfig |= SPI_DMA16_CONFIG;
}
else
{
txbuffer = txdummy;
dmaconfig |= SPI_DMA16NULL_CONFIG;
}
nbytes = nwords << 1;
}
else
{
/* 8-bit mode -- is there a buffer to receive data in? */
if (txbuffer)
{
dmaconfig |= SPI_DMA8_CONFIG;
}
else
{
txbuffer = txdummy;
dmaconfig |= SPI_DMA8NULL_CONFIG;
}
nbytes = nwords;
}
/* Configure the RX DMA */
efm32_txdmasetup(priv->txdmach, config->base + EFM32_USART_TXDATA_OFFSET,
(uintptr_t)txbuffer, nbytes, dmaconfig);
}
#endif
/****************************************************************************
* Name: spi_dmarxstart
*
* Description:
* Start RX DMA
*
****************************************************************************/
#ifdef CONFIG_EFM32_SPI_DMA
static void spi_dmarxstart(FAR struct efm32_spidev_s *priv)
{
priv->rxresult = EINPROGRESS;
efm32_dmastart(priv->rxdmach, spi_dmarxcallback, priv);
}
#endif
/****************************************************************************
* Name: spi_dmatxstart
*
* Description:
* Start TX DMA
*
****************************************************************************/
#ifdef CONFIG_EFM32_SPI_DMA
static inline void spi_dmatxstart(FAR struct efm32_spidev_s *priv)
{
priv->txresult = EINPROGRESS;
efm32_dmastart(priv->txdmach, spi_dmatxcallback, priv);
}
#endif
/**************************************************************************** /****************************************************************************
* Name: spi_lock * Name: spi_lock
* *
* Description: * Description:
* On SPI busses where there are multiple devices, it will be necessary to * On SPI buses where there are multiple devices, it will be necessary to
* lock SPI to have exclusive access to the busses for a sequence of * lock SPI to have exclusive access to the buses for a sequence of
* transfers. The bus should be locked before the chip is selected. After * transfers. The bus should be locked before the chip is selected. After
* locking the SPI bus, the caller should then also call the setfrequency, * locking the SPI bus, the caller should then also call the setfrequency,
* setbits, and setmode methods to make sure that the SPI is properly * setbits, and setmode methods to make sure that the SPI is properly
* configured for the device. If the SPI buss is being shared, then it * configured for the device. If the SPI bus is being shared, then it
* may have been left in an incompatible state. * may have been left in an incompatible state.
* *
* Input Parameters: * Input Parameters:
@ -425,7 +782,7 @@ static int spi_lock(struct spi_dev_s *dev, bool lock)
* Name: spi_select * Name: spi_select
* *
* Description: * Description:
* Enable/disable the SPI chip select. * Enable/disable the SPI chip select.
* *
* Input Parameters: * Input Parameters:
* dev - Device-specific state data * dev - Device-specific state data
@ -743,7 +1100,7 @@ static void spi_setbits(struct spi_dev_s *dev, int nbits)
* *
****************************************************************************/ ****************************************************************************/
static uint8_t spi_status(FAR struct spi_dev_s *dev, enum spi_dev_e devid) static uint8_t spi_status(struct spi_dev_s *dev, enum spi_dev_e devid)
{ {
struct efm32_spidev_s *priv = (struct efm32_spidev_s *)dev; struct efm32_spidev_s *priv = (struct efm32_spidev_s *)dev;
const struct efm32_spiconfig_s *config; const struct efm32_spiconfig_s *config;
@ -781,7 +1138,7 @@ static uint8_t spi_status(FAR struct spi_dev_s *dev, enum spi_dev_e devid)
****************************************************************************/ ****************************************************************************/
#ifdef CONFIG_SPI_CMDDATA #ifdef CONFIG_SPI_CMDDATA
static int spi_cmddata(FAR struct spi_dev_s *dev, enum spi_dev_e devid, static int spi_cmddata(struct spi_dev_s *dev, enum spi_dev_e devid,
bool cmd); bool cmd);
{ {
struct efm32_spidev_s *priv = (struct efm32_spidev_s *)dev; struct efm32_spidev_s *priv = (struct efm32_spidev_s *)dev;
@ -859,12 +1216,12 @@ static uint16_t spi_send(struct spi_dev_s *dev, uint16_t wd)
* *
****************************************************************************/ ****************************************************************************/
#if !defined(CONFIG_EFM32_SPI_DMA) #if defined(CONFIG_EFM32_SPI_DMA) && CONFIG_EFM32_SPI_DMA_MINSIZE > 0
static void spi_exchange(struct spi_dev_s *dev, const void *txbuffer,
void *rxbuffer, size_t nwords)
#else
static void spi_exchange_nodma(struct spi_dev_s *dev, const void *txbuffer, static void spi_exchange_nodma(struct spi_dev_s *dev, const void *txbuffer,
void *rxbuffer, size_t nwords) void *rxbuffer, size_t nwords)
#else
static void spi_exchange(struct spi_dev_s *dev, const void *txbuffer,
void *rxbuffer, size_t nwords)
#endif #endif
{ {
struct efm32_spidev_s *priv = (struct efm32_spidev_s *)dev; struct efm32_spidev_s *priv = (struct efm32_spidev_s *)dev;
@ -989,7 +1346,7 @@ static void spi_exchange_nodma(struct spi_dev_s *dev, const void *txbuffer,
} }
} }
} }
DEBUGASSERT(unsent == 0); DEBUGASSERT(unsent == 0);
} }
@ -1018,27 +1375,30 @@ static void spi_exchange(struct spi_dev_s *dev, const void *txbuffer,
void *rxbuffer, size_t nwords) void *rxbuffer, size_t nwords)
{ {
struct efm32_spidev_s *priv = (struct efm32_spidev_s *)dev; struct efm32_spidev_s *priv = (struct efm32_spidev_s *)dev;
const struct efm32_spiconfig_s *config; static uint16_t rxdummy = 0xffff;
static const uint16_t txdummy = 0xffff;
irqstate_t flags;
uint64_t ticks;
int ret;
DEBUGASSERT(priv && priv->config); DEBUGASSERT(priv && priv->config);
config = priv->config;
#ifdef CONFIG_STM32_DMACAPABLE #if CONFIG_EFM32_SPI_DMA_MINSIZE > 0
if ((txbuffer && !stm32_dmacapable((uint32_t)txbuffer, nwords, priv->txccr)) || if (nwords <= CONFIG_EFM32_SPI_DMA_MINSIZE)
(rxbuffer && !stm32_dmacapable((uint32_t)rxbuffer, nwords, priv->rxccr)))
{ {
/* Unsupported memory region, fall back to non-DMA method. */ /* Small transfer, fall back to non-DMA method. */
spi_exchange_nodma(dev, txbuffer, rxbuffer, nwords); spi_exchange_nodma(dev, txbuffer, rxbuffer, nwords);
} }
else else
#endif #endif
{ {
static uint16_t rxdummy = 0xffff;
static const uint16_t txdummy = 0xffff;
spivdbg("txbuffer=%p rxbuffer=%p nwords=%d\n", txbuffer, rxbuffer, nwords); spivdbg("txbuffer=%p rxbuffer=%p nwords=%d\n", txbuffer, rxbuffer, nwords);
/* Pre-calculate the timerout value */
ticks = (CONFIG_EFM32_SPI_DMA_TIMEO_NSEC * nwords) / NSEC_PER_TICK;
/* Setup DMAs */ /* Setup DMAs */
spi_dmarxsetup(priv, rxbuffer, &rxdummy, nwords); spi_dmarxsetup(priv, rxbuffer, &rxdummy, nwords);
@ -1046,13 +1406,25 @@ static void spi_exchange(struct spi_dev_s *dev, const void *txbuffer,
/* Start the DMAs */ /* Start the DMAs */
flags = irqsave();
spi_dmarxstart(priv); spi_dmarxstart(priv);
spi_dmatxstart(priv); spi_dmatxstart(priv);
/* Then wait for each to complete */ /* Start a timer to catch hung DMA transfers. Timeout will be canceled
* when both RX and TX transfers complete.
*/
ret = wd_start(priv->wdog, (int)ticks, spi_dma_timeout, 1, (uint32_t)priv);
if (ret < 0)
{
spidbg("ERROR: Failed to start timeout\n");
}
/* Then wait for each to complete. TX should complete first */
spi_dmarxwait(priv);
spi_dmatxwait(priv); spi_dmatxwait(priv);
spi_dmarxwait(priv);
irqrestore(flags);
} }
} }
#endif /* CONFIG_EFM32_SPI_DMA */ #endif /* CONFIG_EFM32_SPI_DMA */
@ -1115,7 +1487,7 @@ static void spi_recvblock(struct spi_dev_s *dev, void *rxbuffer, size_t nwords)
* Name: spi_portinitialize * Name: spi_portinitialize
* *
* Description: * Description:
* Initialize the selected SPI port in its default state * Initialize the selected SPI port in its default state
* (Master, 8-bit, mode 0, etc.) * (Master, 8-bit, mode 0, etc.)
* *
* Input Parameter: * Input Parameter:
@ -1126,7 +1498,7 @@ static void spi_recvblock(struct spi_dev_s *dev, void *rxbuffer, size_t nwords)
* *
****************************************************************************/ ****************************************************************************/
static void spi_portinitialize(struct efm32_spidev_s *priv) static int spi_portinitialize(struct efm32_spidev_s *priv)
{ {
const struct efm32_spiconfig_s *config = priv->config; const struct efm32_spiconfig_s *config = priv->config;
uint32_t regval; uint32_t regval;
@ -1135,11 +1507,12 @@ static void spi_portinitialize(struct efm32_spidev_s *priv)
efm32_uart_reset(config->base); efm32_uart_reset(config->base);
/* Configure CR1. Default configuration: /* NOTES:
* Mode 0: USART_CTRL_CLKPOL_IDLELOW *
* USART_CTRL_CLKPHA_SAMPLELEADING * 1. USART GPIO pins were configured in efm32_lowsetup(). Chip select
* 8-bit: USART_FRAME_DATABITS_EIGHT * pins must be configured by board specific logic before
* MSB tranmitted first: ~USART_CTRL_MSBF * efm32_spi_initialize() is called.
* 2. Clocking for the USART as also enabled in up_lowsetup();
*/ */
/* Set bits for synchronous mode */ /* Set bits for synchronous mode */
@ -1179,15 +1552,57 @@ static void spi_portinitialize(struct efm32_spidev_s *priv)
sem_init(&priv->exclsem, 0, 1); sem_init(&priv->exclsem, 0, 1);
#endif #endif
/* Initialize the SPI semaphores that is used to wait for DMA completion */
#ifdef CONFIG_EFM32_SPI_DMA #ifdef CONFIG_EFM32_SPI_DMA
spi_portinitialize_dma(priv); /* Allocate two DMA channels... one for the RX and one for the TX side of
* the transfer.
*/
priv->rxdmach = efm32_dmachannel();
if (!priv->rxdmach)
{
spidbg("ERROR: Failed to allocate the RX DMA channel for SPI port: %d\n", port);
goto errout;
}
priv->txdmach = efm32_dmachannel();
if (!priv->txdmach)
{
spidbg("ERROR: Failed to allocate the TX DMA channel for SPI port: %d\n", port);
goto errout_with_rxdmach;
}
/* Allocate a timer to catch hung DMA transfers */
priv->wdog = wd_create();
if (!priv->wdog)
{
spidbg("ERROR: Failed to create a timer for SPI port: %d\n", port);
goto errout_with_txdmach;
}
/* Initialized semaphores used to wait for DMA completion */
(void)sem_init(&priv->rxdmasem, 0, 0);
(void)sem_init(&priv->txdmasem, 0, 0);
#endif #endif
/* Enable SPI */ /* Enable SPI */
spi_putreg(config, EFM32_USART_CMD_OFFSET, USART_CMD_RXEN | USART_CMD_TXEN); spi_putreg(config, EFM32_USART_CMD_OFFSET, USART_CMD_RXEN | USART_CMD_TXEN);
return OK;
#ifdef CONFIG_EFM32_SPI_DMA
errout_with_txdmach:
efm32_dmafree(priv->txdmach);
priv->txdmach = NULL;
errout_with_rxdmach:
efm32_dmafree(priv->rxdmach);
priv->rxdmach = NULL;
errout:
return -EBUSY;
#endif
} }
/**************************************************************************** /****************************************************************************
@ -1213,6 +1628,7 @@ struct spi_dev_s *efm32_spi_initialize(int port)
const struct efm32_spiconfig_s *config; const struct efm32_spiconfig_s *config;
struct efm32_spidev_s *priv; struct efm32_spidev_s *priv;
irqstate_t flags; irqstate_t flags;
int ret;
#ifdef CONFIG_EFM32_USART0_ISSPI #ifdef CONFIG_EFM32_USART0_ISSPI
if (port == 0) if (port == 0)
@ -1254,44 +1670,17 @@ struct spi_dev_s *efm32_spi_initialize(int port)
/* Initialize the state structure */ /* Initialize the state structure */
priv->spidev = &g_spiops; priv->spidev = &g_spiops;
priv->config = config; priv->config = config;
#ifdef CONFIG_EFM32_SPI_DMA /* Initialize the SPI device */
/* Allocate two DMA channels... one for the RX and one for the TX side
* of the transfer.
*/
priv->rxdmach = efm3_dmachannel(); ret = spi_portinitialize(priv);
if (!priv->rxdmach) if (ret < 0)
{ {
spidbg("ERROR: Failed to allocate the RX DMA channel for SPI port: %d\n", port); spidbg("ERROR: Failed to initialize SPI port %d\n", port);
return NULL; irqrestore(flags);
} return NULL;
}
priv-txdmach = efm3_dmachannel();
if (!priv->txdmach)
{
spidbg("ERROR: Failed to allocate the TX DMA channel for SPI port: %d\n", port);
efm32_dmafree(priv->rxdmach);
priv->rxdmach = NULL;
return NULL;
}
/* Initialized semaphores used to wait for DMA completion */
sem_init(&priv->rxdmasem, 0, 0);
sem_init(&priv->txdmasem, 0, 0);
#endif
/* NOTES:
*
* 1. USART GPIO pins were configured in efm32_lowsetup(). Chip
* select pins must be configured by board specific logic before
* efm32_spi_initialize() is called.
* 2. Clocking for the USART as also enabled in up_lowsetup();
*/
spi_portinitialize(priv);
/* Now we are initialized */ /* Now we are initialized */