SAMA5: SPI driver now supports both SPI0 and SPI1

This commit is contained in:
Gregory Nutt 2013-08-04 12:50:20 -06:00
parent 487866b2b6
commit d516baa73f
4 changed files with 379 additions and 177 deletions

View File

@ -85,7 +85,11 @@
#endif #endif
#ifdef CONFIG_SAM34_SPI1 #ifdef CONFIG_SAM34_SPI1
# error Support for SPI1 has not yet been implemented /* NOTE: See arch/arm/sama5/sam_spi.c. That is the same SPI IP and that
* version on the driver has been extended to support both SPI0 and SPI1
*/
# error Support for SPI1 has not yet been implemented (see NOTE)
#endif #endif
/* Debug *******************************************************************/ /* Debug *******************************************************************/

View File

@ -115,7 +115,7 @@
struct sam_dmach_s struct sam_dmach_s
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
uint8_t dmac; /* DMA controller number (0-1) */ uint8_t dmac; /* DMA controller number (0-1) */
#endif #endif
uint8_t chan; /* DMA channel number (0-6) */ uint8_t chan; /* DMA channel number (0-6) */
@ -189,7 +189,7 @@ static struct sam_dmach_s g_dmach0[SAM_NDMACHAN] =
{ {
#if SAM_NDMACHAN > 0 #if SAM_NDMACHAN > 0
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 0, .dmac = 0,
#endif #endif
.chan = 0, .chan = 0,
@ -198,7 +198,7 @@ static struct sam_dmach_s g_dmach0[SAM_NDMACHAN] =
#endif #endif
#if SAM_NDMACHAN > 1 #if SAM_NDMACHAN > 1
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 0, .dmac = 0,
#endif #endif
.chan = 1, .chan = 1,
@ -207,7 +207,7 @@ static struct sam_dmach_s g_dmach0[SAM_NDMACHAN] =
#endif #endif
#if SAM_NDMACHAN > 2 #if SAM_NDMACHAN > 2
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 0, .dmac = 0,
#endif #endif
.chan = 2, .chan = 2,
@ -216,7 +216,7 @@ static struct sam_dmach_s g_dmach0[SAM_NDMACHAN] =
#endif #endif
#if SAM_NDMACHAN > 3 #if SAM_NDMACHAN > 3
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 0, .dmac = 0,
#endif #endif
.chan = 3, .chan = 3,
@ -225,7 +225,7 @@ static struct sam_dmach_s g_dmach0[SAM_NDMACHAN] =
#endif #endif
#if SAM_NDMACHAN > 4 #if SAM_NDMACHAN > 4
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 0, .dmac = 0,
#endif #endif
.chan = 4, .chan = 4,
@ -234,7 +234,7 @@ static struct sam_dmach_s g_dmach0[SAM_NDMACHAN] =
#endif #endif
#if SAM_NDMACHAN > 5 #if SAM_NDMACHAN > 5
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 0, .dmac = 0,
#endif #endif
.chan = 5, .chan = 5,
@ -243,7 +243,7 @@ static struct sam_dmach_s g_dmach0[SAM_NDMACHAN] =
#endif #endif
#if SAM_NDMACHAN > 6 #if SAM_NDMACHAN > 6
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 0, .dmac = 0,
#endif #endif
.chan = 6, .chan = 6,
@ -252,7 +252,7 @@ static struct sam_dmach_s g_dmach0[SAM_NDMACHAN] =
#endif #endif
#if SAM_NDMACHAN > 7 #if SAM_NDMACHAN > 7
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 0, .dmac = 0,
#endif #endif
.chan = 7, .chan = 7,
@ -291,7 +291,7 @@ static struct sam_dmach_s g_dmach1[SAM_NDMACHAN] =
{ {
#if SAM_NDMACHAN > 0 #if SAM_NDMACHAN > 0
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 1, .dmac = 1,
#endif #endif
.chan = 0, .chan = 0,
@ -300,7 +300,7 @@ static struct sam_dmach_s g_dmach1[SAM_NDMACHAN] =
#endif #endif
#if SAM_NDMACHAN > 1 #if SAM_NDMACHAN > 1
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 1, .dmac = 1,
#endif #endif
.chan = 1, .chan = 1,
@ -309,7 +309,7 @@ static struct sam_dmach_s g_dmach1[SAM_NDMACHAN] =
#endif #endif
#if SAM_NDMACHAN > 2 #if SAM_NDMACHAN > 2
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 1, .dmac = 1,
#endif #endif
.chan = 2, .chan = 2,
@ -318,7 +318,7 @@ static struct sam_dmach_s g_dmach1[SAM_NDMACHAN] =
#endif #endif
#if SAM_NDMACHAN > 3 #if SAM_NDMACHAN > 3
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 1, .dmac = 1,
#endif #endif
.chan = 3, .chan = 3,
@ -327,7 +327,7 @@ static struct sam_dmach_s g_dmach1[SAM_NDMACHAN] =
#endif #endif
#if SAM_NDMACHAN > 4 #if SAM_NDMACHAN > 4
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 1, .dmac = 1,
#endif #endif
.chan = 4, .chan = 4,
@ -336,7 +336,7 @@ static struct sam_dmach_s g_dmach1[SAM_NDMACHAN] =
#endif #endif
#if SAM_NDMACHAN > 5 #if SAM_NDMACHAN > 5
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 1, .dmac = 1,
#endif #endif
.chan = 5, .chan = 5,
@ -345,7 +345,7 @@ static struct sam_dmach_s g_dmach1[SAM_NDMACHAN] =
#endif #endif
#if SAM_NDMACHAN > 6 #if SAM_NDMACHAN > 6
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 1, .dmac = 1,
#endif #endif
.chan = 6, .chan = 6,
@ -354,7 +354,7 @@ static struct sam_dmach_s g_dmach1[SAM_NDMACHAN] =
#endif #endif
#if SAM_NDMACHAN > 7 #if SAM_NDMACHAN > 7
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
.dmac = 1, .dmac = 1,
#endif #endif
.chan = 7, .chan = 7,
@ -500,14 +500,14 @@ static inline void sam_putdmach(struct sam_dmach_s *dmach, uint32_t value,
* Name: sam_controller * Name: sam_controller
* *
* Description: * Description:
* Given a DMA channel instrance, return a pointer to the parent DMA * Given a DMA channel instance, return a pointer to the parent DMA
* controller instance. * controller instance.
* *
****************************************************************************/ ****************************************************************************/
static inline struct sam_dmac_s *sam_controller(struct sam_dmach_s *dmach) static inline struct sam_dmac_s *sam_controller(struct sam_dmach_s *dmach)
{ {
#if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC01) #if defined(CONFIG_SAMA5_DMAC0) && defined(CONFIG_SAMA5_DMAC1)
return dmach->dmac ? &g_dmac1 : &g_dmac0; return dmach->dmac ? &g_dmac1 : &g_dmac0;
#elif defined(CONFIG_SAMA5_DMAC0) #elif defined(CONFIG_SAMA5_DMAC0)
return &g_dmac0; return &g_dmac0;

View File

@ -80,10 +80,6 @@
#define SAM_SPI_CLOCK BOARD_MCK_FREQUENCY #define SAM_SPI_CLOCK BOARD_MCK_FREQUENCY
#ifdef CONFIG_SAMA5_SPI1
# error Support for SPI1 has not yet been implemented
#endif
/* Debug *******************************************************************/ /* Debug *******************************************************************/
/* Check if SPI debut is enabled (non-standard.. no support in /* Check if SPI debut is enabled (non-standard.. no support in
* include/debug.h * include/debug.h
@ -110,18 +106,36 @@
* Private Types * Private Types
****************************************************************************/ ****************************************************************************/
/* The state of the one chip select */ /* The state of the one SPI chip select */
struct sam_spidev_s struct sam_spics_s
{ {
struct spi_dev_s spidev; /* Externally visible part of the SPI interface */ struct spi_dev_s spidev; /* Externally visible part of the SPI interface */
#ifndef CONFIG_SPI_OWNBUS #ifndef CONFIG_SPI_OWNBUS
uint32_t frequency; /* Requested clock frequency */ uint32_t frequency; /* Requested clock frequency */
uint32_t actual; /* Actual clock frequency */ uint32_t actual; /* Actual clock frequency */
uint8_t nbits; /* Width of word in bits (8 to 16) */ uint8_t nbits; /* Width of word in bits (8 to 16) */
uint8_t mode; /* Mode 0,1,2,3 */ uint8_t mode; /* Mode 0,1,2,3 */
#endif #endif
uint8_t cs; /* Chip select number */ #if defined(CONFIG_SAMA5_SPI0) || defined(CONFIG_SAMA5_SPI1)
uint8_t spino; /* SPI controller number (0 or 1) */
#endif
uint8_t cs; /* Chip select number */
};
/* Type of board-specific SPI status fuction */
typedef void (*select_t)(enum spi_dev_e devid, bool selected);
/* The overall state of one SPI controller */
struct sam_spidev_s
{
uint32_t base; /* SPI controller register base address */
sem_t spisem; /* Assures mutually exclusive acess to SPI */
bool initialized; /* TRUE: Controller has been initialized */
const uint32_t *csraddr; /* Addresses of CSR register */
select_t select; /* SPI select callout */
}; };
/**************************************************************************** /****************************************************************************
@ -131,41 +145,42 @@ struct sam_spidev_s
/* Helpers */ /* Helpers */
#if defined(CONFIG_DEBUG_SPI) && defined(CONFIG_DEBUG_VERBOSE) #if defined(CONFIG_DEBUG_SPI) && defined(CONFIG_DEBUG_VERBOSE)
static void spi_dumpregs(FAR const char *msg); static void spi_dumpregs(struct sam_spidev_s *spi, const char *msg);
#else #else
# define spi_dumpregs(msg) # define spi_dumpregs(spi,msg)
#endif #endif
static inline void spi_flush(void); static inline void spi_flush(struct sam_spidev_s *spi);
static inline uint32_t spi_cs2pcs(FAR struct sam_spidev_s *priv); static inline uint32_t spi_cs2pcs(struct sam_spics_s *spics);
/* SPI methods */ /* SPI methods */
#ifndef CONFIG_SPI_OWNBUS #ifndef CONFIG_SPI_OWNBUS
static int spi_lock(FAR struct spi_dev_s *dev, bool lock); static int spi_lock(struct spi_dev_s *dev, bool lock);
#endif #endif
static void spi_select(FAR struct spi_dev_s *dev, enum spi_dev_e devid, static void spi_select(struct spi_dev_s *dev, enum spi_dev_e devid,
bool selected); bool selected);
static uint32_t spi_setfrequency(FAR struct spi_dev_s *dev, static uint32_t spi_setfrequency(struct spi_dev_s *dev, uint32_t frequency);
uint32_t frequency); static void spi_setmode(struct spi_dev_s *dev, enum spi_mode_e mode);
static void spi_setmode(FAR struct spi_dev_s *dev, static void spi_setbits(struct spi_dev_s *dev, int nbits);
enum spi_mode_e mode); static uint16_t spi_send(struct spi_dev_s *dev, uint16_t ch);
static void spi_setbits(FAR struct spi_dev_s *dev, int nbits); static void spi_exchange(struct spi_dev_s *dev, const void *txbuffer,
static uint16_t spi_send(FAR struct spi_dev_s *dev, uint16_t ch); void *rxbuffer, size_t nwords);
static void spi_exchange(FAR struct spi_dev_s *dev,
FAR const void *txbuffer, FAR void *rxbuffer, size_t nwords);
#ifndef CONFIG_SPI_EXCHANGE #ifndef CONFIG_SPI_EXCHANGE
static void spi_sndblock(FAR struct spi_dev_s *dev, FAR const void *buffer, size_t nwords); static void spi_sndblock(struct spi_dev_s *dev,
static void spi_recvblock(FAR struct spi_dev_s *dev, FAR void *buffer, size_t nwords); const void *buffer, size_t nwords);
static void spi_recvblock(struct spi_dev_s *dev, void *buffer,
size_t nwords);
#endif #endif
/**************************************************************************** /****************************************************************************
* Private Data * Private Data
****************************************************************************/ ****************************************************************************/
/* SPI driver operations */ #ifdef CONFIG_SAMA5_SPI0
/* SPI0 driver operations */
static const struct spi_ops_s g_spiops = static const struct spi_ops_s g_spi0ops =
{ {
#ifndef CONFIG_SPI_OWNBUS #ifndef CONFIG_SPI_OWNBUS
.lock = spi_lock, .lock = spi_lock,
@ -174,9 +189,9 @@ static const struct spi_ops_s g_spiops =
.setfrequency = spi_setfrequency, .setfrequency = spi_setfrequency,
.setmode = spi_setmode, .setmode = spi_setmode,
.setbits = spi_setbits, .setbits = spi_setbits,
.status = sam_spistatus, .status = sam_spi0status,
#ifdef CONFIG_SPI_CMDDATA #ifdef CONFIG_SPI_CMDDATA
.cmddata = sam_spicmddata, .cmddata = sam_spi0cmddata,
#endif #endif
.send = spi_send, .send = spi_send,
#ifdef CONFIG_SPI_EXCHANGE #ifdef CONFIG_SPI_EXCHANGE
@ -188,25 +203,66 @@ static const struct spi_ops_s g_spiops =
.registercallback = 0, /* Not implemented */ .registercallback = 0, /* Not implemented */
}; };
#ifdef CONFIG_SPI_OWNBUS
/* Single chip select device structure */
static struct sam_spidev_s g_spidev;
#else
/* Held while chip is selected for mutual exclusion */
static sem_t g_spisem;
static bool g_spinitialized = false;
#endif
/* This array maps chip select numbers (0-3) to CSR register addresses */ /* This array maps chip select numbers (0-3) to CSR register addresses */
static const uint32_t g_csraddr[4] = static const uint32_t g_csraddr0[4] =
{ {
SAM_SPI0_CSR0, SAM_SPI0_CSR1, SAM_SPI0_CSR2, SAM_SPI0_CSR3 SAM_SPI0_CSR0, SAM_SPI0_CSR1, SAM_SPI0_CSR2, SAM_SPI0_CSR3
}; };
/* This is the overall state of the SPI0 controller */
static struct sam_spidev_s g_spi0dev =
{
.base = SAM_SPI0_VBASE,
.csraddr = g_csraddr0,
.select = sam_spi0select,
};
#endif
#ifdef CONFIG_SAMA5_SPI1
/* SPI1 driver operations */
static const struct spi_ops_s g_spi1ops =
{
#ifndef CONFIG_SPI_OWNBUS
.lock = spi_lock,
#endif
.select = spi_select,
.setfrequency = spi_setfrequency,
.setmode = spi_setmode,
.setbits = spi_setbits,
.status = sam_spi1status,
#ifdef CONFIG_SPI_CMDDATA
.cmddata = sam_spi1cmddata,
#endif
.send = spi_send,
#ifdef CONFIG_SPI_EXCHANGE
.exchange = spi_exchange,
#else
.sndblock = spi_sndblock,
.recvblock = spi_recvblock,
#endif
.registercallback = 0, /* Not implemented */
};
/* This array maps chip select numbers (0-3) to CSR register addresses */
static const uint32_t g_csraddr1[4] =
{
SAM_SPI1_CSR0, SAM_SPI1_CSR1, SAM_SPI1_CSR2, SAM_SPI1_CSR3
};
/* This is the overall state of the SPI0 controller */
static struct sam_spidev_s g_spi1dev =
{
.base = SAM_SPI1_VBASE,
.csraddr = g_csraddr1,
.select = sam_spi1select,
};
#endif
/**************************************************************************** /****************************************************************************
* Public Data * Public Data
****************************************************************************/ ****************************************************************************/
@ -215,6 +271,34 @@ static const uint32_t g_csraddr[4] =
* Private Functions * Private Functions
****************************************************************************/ ****************************************************************************/
/****************************************************************************
* Name: spi_getreg
*
* Description:
* Read an SPI register
*
****************************************************************************/
static inline uint32_t spi_getreg(struct sam_spidev_s *spi,
unsigned int offset)
{
return getreg32(spi->base + offset);
}
/****************************************************************************
* Name: spi_putreg
*
* Description:
* Write a value to an SPI register
*
****************************************************************************/
static inline void spi_putreg(struct sam_spidev_s *spi, uint32_t value,
unsigned int offset)
{
spi_putreg(spi, value, spi->base + offset);
}
/**************************************************************************** /****************************************************************************
* Name: spi_dumpregs * Name: spi_dumpregs
* *
@ -222,6 +306,7 @@ static const uint32_t g_csraddr[4] =
* Dump the contents of all SPI registers * Dump the contents of all SPI registers
* *
* Input Parameters: * Input Parameters:
* spi - The SPI controller to dump
* msg - Message to print before the register data * msg - Message to print before the register data
* *
* Returned Value: * Returned Value:
@ -230,20 +315,44 @@ static const uint32_t g_csraddr[4] =
****************************************************************************/ ****************************************************************************/
#if defined(CONFIG_DEBUG_SPI) && defined(CONFIG_DEBUG_VERBOSE) #if defined(CONFIG_DEBUG_SPI) && defined(CONFIG_DEBUG_VERBOSE)
static void spi_dumpregs(FAR const char *msg) static void spi_dumpregs(const struct sam_spidev_s *spi, const char *msg)
{ {
spivdbg("%s:\n", msg); spivdbg("%s:\n", msg);
spivdbg(" MR:%08x SR:%08x IMR:%08x\n", spivdbg(" MR:%08x SR:%08x IMR:%08x\n",
getreg32(SAM_SPI0_MR), getreg32(SAM_SPI0_SR), spi_getreg(spi, SAM_SPI_MR_OFFSET),
getreg32(SAM_SPI0_IMR)); spi_getreg(spi, SAM_SPI_SR_OFFSET),
spi_getreg(spi, SAM_SPI_IMR_OFFSET));
spivdbg(" CSR0:%08x CSR1:%08x CSR2:%08x CSR3:%08x\n", spivdbg(" CSR0:%08x CSR1:%08x CSR2:%08x CSR3:%08x\n",
getreg32(SAM_SPI0_CSR0), getreg32(SAM_SPI0_CSR1), spi_getreg(spi, SAM_SPI_CSR0_OFFSET),
getreg32(SAM_SPI0_CSR2), getreg32(SAM_SPI0_CSR3)); spi_getreg(spi, SAM_SPI_CSR1_OFFSET),
spi_getreg(spi, SAM_SPI_CSR2_OFFSET),
spi_getreg(spi, SAM_SPI_CSR3_OFFSET));
spivdbg(" WPCR:%08x WPSR:%08x\n", spivdbg(" WPCR:%08x WPSR:%08x\n",
getreg32(SAM_SPI0_WPCR), getreg32(SAM_SPI0_WPSR)); spi_getreg(spi, SAM_SPI_WPCR_OFFSET),
spi_getreg(spi, SAM_SPI_WPSR_OFFSET));
} }
#endif #endif
/****************************************************************************
* Name: spi_device
*
* Description:
* Given a chip select instance, return a pointer to the parent SPI
* controller instance.
*
****************************************************************************/
static inline struct sam_spidev_s *spi_device(struct sam_spics_s *spics)
{
#if defined(CONFIG_SAMA5_SPI0) && defined(CONFIG_SAMA5_SPI1)
return spics->spino ? &g_spi1dev : &g_spi0dev;
#elif defined(CONFIG_SAMA5_SPI0)
return &g_spi0dev;
#else
return &g_spi1dev;
#endif
}
/**************************************************************************** /****************************************************************************
* Name: spi_flush * Name: spi_flush
* *
@ -251,26 +360,26 @@ static void spi_dumpregs(FAR const char *msg)
* Make sure that there are now dangling SPI transfer in progress * Make sure that there are now dangling SPI transfer in progress
* *
* Input Parameters: * Input Parameters:
* priv - Device-specific state data * spi - SPI controller state
* *
* Returned Value: * Returned Value:
* None * None
* *
****************************************************************************/ ****************************************************************************/
static inline void spi_flush(void) static inline void spi_flush(struct sam_spidev_s *spi)
{ {
/* Make sure the no TX activity is in progress... waiting if necessary */ /* Make sure the no TX activity is in progress... waiting if necessary */
while ((getreg32(SAM_SPI0_SR) & SPI_INT_TXEMPTY) == 0); while ((spi_getreg(spi, SAM_SPI_SR_OFFSET) & SPI_INT_TXEMPTY) == 0);
/* Then make sure that there is no pending RX data .. reading as /* Then make sure that there is no pending RX data .. reading as
* discarding as necessary. * discarding as necessary.
*/ */
while ((getreg32(SAM_SPI0_SR) & SPI_INT_RDRF) != 0) while ((spi_getreg(spi, SAM_SPI_SR_OFFSET) & SPI_INT_RDRF) != 0)
{ {
(void)getreg32(SAM_SPI0_RDR); (void)spi_getreg(spi, SAM_SPI_RDR_OFFSET);
} }
} }
@ -293,16 +402,16 @@ static inline void spi_flush(void)
* 3 0111 0111 0111 * 3 0111 0111 0111
* *
* Input Parameters: * Input Parameters:
* priv - Device-specific state data * spics - Device-specific state data
* *
* Returned Value: * Returned Value:
* None * None
* *
****************************************************************************/ ****************************************************************************/
static inline uint32_t spi_cs2pcs(FAR struct sam_spidev_s *priv) static inline uint32_t spi_cs2pcs(struct sam_spics_s *spics)
{ {
return ((uint32_t)1 << (priv->cs)) - 1; return ((uint32_t)1 << (spics->cs)) - 1;
} }
/**************************************************************************** /****************************************************************************
@ -327,14 +436,17 @@ static inline uint32_t spi_cs2pcs(FAR struct sam_spidev_s *priv)
****************************************************************************/ ****************************************************************************/
#ifndef CONFIG_SPI_OWNBUS #ifndef CONFIG_SPI_OWNBUS
static int spi_lock(FAR struct spi_dev_s *dev, bool lock) static int spi_lock(struct spi_dev_s *dev, bool lock)
{ {
struct sam_spics_s *spics = (struct sam_spics_s *)dev;
struct sam_spidev_s *spi = spi_device(spics);
spivdbg("lock=%d\n", lock); spivdbg("lock=%d\n", lock);
if (lock) if (lock)
{ {
/* Take the semaphore (perhaps waiting) */ /* Take the semaphore (perhaps waiting) */
while (sem_wait(&g_spisem) != 0) while (sem_wait(&spi->spisem) != 0)
{ {
/* The only case that an error should occur here is if the wait was awakened /* The only case that an error should occur here is if the wait was awakened
* by a signal. * by a signal.
@ -345,7 +457,7 @@ static int spi_lock(FAR struct spi_dev_s *dev, bool lock)
} }
else else
{ {
(void)sem_post(&g_spisem); (void)sem_post(&spi->spisem);
} }
return OK; return OK;
@ -369,10 +481,11 @@ static int spi_lock(FAR struct spi_dev_s *dev, bool lock)
* *
****************************************************************************/ ****************************************************************************/
static void spi_select(FAR struct spi_dev_s *dev, enum spi_dev_e devid, static void spi_select(struct spi_dev_s *dev, enum spi_dev_e devid,
bool selected) bool selected)
{ {
FAR struct sam_spidev_s *priv = (FAR struct sam_spidev_s *)dev; struct sam_spics_s *spics = (struct sam_spics_s *)dev;
struct sam_spidev_s *spi = spi_device(spics);
uint32_t regval; uint32_t regval;
/* Are we selecting or de-selecting the device? */ /* Are we selecting or de-selecting the device? */
@ -380,24 +493,24 @@ static int spi_lock(FAR struct spi_dev_s *dev, bool lock)
spivdbg("selected=%d\n", selected); spivdbg("selected=%d\n", selected);
if (selected) if (selected)
{ {
spivdbg("cs=%d\n", priv->cs); spivdbg("cs=%d\n", spics->cs);
/* Before writing the TDR, the PCS field in the SPI_MR register must be set /* Before writing the TDR, the PCS field in the SPI_MR register must be set
* in order to select a slave. * in order to select a slave.
*/ */
regval = getreg32(SAM_SPI0_MR); regval = spi_getreg(spi, SAM_SPI_MR_OFFSET);
regval &= ~SPI_MR_PCS_MASK; regval &= ~SPI_MR_PCS_MASK;
regval |= (spi_cs2pcs(priv) << SPI_MR_PCS_SHIFT); regval |= (spi_cs2pcs(spics) << SPI_MR_PCS_SHIFT);
putreg32(regval, SAM_SPI0_MR); spi_putreg(spi, regval, SAM_SPI_MR_OFFSET);
} }
/* Perform any board-specific chip select operations. PIO chip select /* Perform any board-specific chip select operations. PIO chip select
* pins may be programmed by the board specific logic in one of two * pins may be programmed by the board specific logic in one of two
* different ways. First, the pins may be programmed as SPI peripherals. * different ways. First, the pins may be programmed as SPI peripherals.
* In that case, the pins are completely controlled by the SPI driver. * In that case, the pins are completely controlled by the SPI driver.
* This sam_spiselect method still needs to be provided, but it may * The sam_spi[0|1]select methods still needs to be provided, but they
* be only a stub. * may be only stubs.
* *
* An alternative way to program the PIO chip select pins is as normal * An alternative way to program the PIO chip select pins is as normal
* GPIO outputs. In that case, the automatic control of the CS pins is * GPIO outputs. In that case, the automatic control of the CS pins is
@ -406,7 +519,7 @@ static int spi_lock(FAR struct spi_dev_s *dev, bool lock)
* same as the NPCS pin normal associated with the chip select number. * same as the NPCS pin normal associated with the chip select number.
*/ */
sam_spiselect(devid, selected); spi->select(devid, selected);
} }
/**************************************************************************** /****************************************************************************
@ -424,9 +537,10 @@ static int spi_lock(FAR struct spi_dev_s *dev, bool lock)
* *
****************************************************************************/ ****************************************************************************/
static uint32_t spi_setfrequency(FAR struct spi_dev_s *dev, uint32_t frequency) static uint32_t spi_setfrequency(struct spi_dev_s *dev, uint32_t frequency)
{ {
FAR struct sam_spidev_s *priv = (FAR struct sam_spidev_s *)dev; struct sam_spics_s *spics = (struct sam_spics_s *)dev;
struct sam_spidev_s *spi = spi_device(spics);
uint32_t actual; uint32_t actual;
uint32_t scbr; uint32_t scbr;
uint32_t dlybs; uint32_t dlybs;
@ -434,16 +548,16 @@ static uint32_t spi_setfrequency(FAR struct spi_dev_s *dev, uint32_t frequency)
uint32_t regval; uint32_t regval;
uint32_t regaddr; uint32_t regaddr;
spivdbg("cs=%d frequency=%d\n", priv->cs, frequency); spivdbg("cs=%d frequency=%d\n", spics->cs, frequency);
/* Check if the requested frequency is the same as the frequency selection */ /* Check if the requested frequency is the same as the frequency selection */
#ifndef CONFIG_SPI_OWNBUS #ifndef CONFIG_SPI_OWNBUS
if (priv->frequency == frequency) if (spics->frequency == frequency)
{ {
/* We are already at this frequency. Return the actual. */ /* We are already at this frequency. Return the actual. */
return priv->actual; return spics->actual;
} }
#endif #endif
@ -467,7 +581,7 @@ static uint32_t spi_setfrequency(FAR struct spi_dev_s *dev, uint32_t frequency)
/* Save the new scbr value */ /* Save the new scbr value */
regaddr = g_csraddr[priv->cs]; regaddr = spi->csraddr[spics->cs];
regval = getreg32(regaddr); regval = getreg32(regaddr);
regval &= ~(SPI_CSR_SCBR_MASK | SPI_CSR_DLYBS_MASK | SPI_CSR_DLYBCT_MASK); regval &= ~(SPI_CSR_SCBR_MASK | SPI_CSR_DLYBS_MASK | SPI_CSR_DLYBCT_MASK);
regval |= scbr << SPI_CSR_SCBR_SHIFT; regval |= scbr << SPI_CSR_SCBR_SHIFT;
@ -511,8 +625,8 @@ static uint32_t spi_setfrequency(FAR struct spi_dev_s *dev, uint32_t frequency)
/* Save the frequency setting */ /* Save the frequency setting */
#ifndef CONFIG_SPI_OWNBUS #ifndef CONFIG_SPI_OWNBUS
priv->frequency = frequency; spics->frequency = frequency;
priv->actual = actual; spics->actual = actual;
#endif #endif
spidbg("Frequency %d->%d\n", frequency, actual); spidbg("Frequency %d->%d\n", frequency, actual);
@ -534,18 +648,19 @@ static uint32_t spi_setfrequency(FAR struct spi_dev_s *dev, uint32_t frequency)
* *
****************************************************************************/ ****************************************************************************/
static void spi_setmode(FAR struct spi_dev_s *dev, enum spi_mode_e mode) static void spi_setmode(struct spi_dev_s *dev, enum spi_mode_e mode)
{ {
FAR struct sam_spidev_s *priv = (FAR struct sam_spidev_s *)dev; struct sam_spics_s *spics = (struct sam_spics_s *)dev;
struct sam_spidev_s *spi = spi_device(spics);
uint32_t regval; uint32_t regval;
uint32_t regaddr; uint32_t regaddr;
spivdbg("cs=%d mode=%d\n", priv->cs, mode); spivdbg("cs=%d mode=%d\n", spics->cs, mode);
/* Has the mode changed? */ /* Has the mode changed? */
#ifndef CONFIG_SPI_OWNBUS #ifndef CONFIG_SPI_OWNBUS
if (mode != priv->mode) if (mode != spics->mode)
{ {
#endif #endif
/* Yes... Set the mode appropriately: /* Yes... Set the mode appropriately:
@ -558,7 +673,7 @@ static void spi_setmode(FAR struct spi_dev_s *dev, enum spi_mode_e mode)
* 3 1 0 * 3 1 0
*/ */
regaddr = g_csraddr[priv->cs]; regaddr = spi->csraddr[spics->cs];
regval = getreg32(regaddr); regval = getreg32(regaddr);
regval &= ~(SPI_CSR_CPOL | SPI_CSR_NCPHA); regval &= ~(SPI_CSR_CPOL | SPI_CSR_NCPHA);
@ -590,7 +705,7 @@ static void spi_setmode(FAR struct spi_dev_s *dev, enum spi_mode_e mode)
/* Save the mode so that subsequent re-configurations will be faster */ /* Save the mode so that subsequent re-configurations will be faster */
#ifndef CONFIG_SPI_OWNBUS #ifndef CONFIG_SPI_OWNBUS
priv->mode = mode; spics->mode = mode;
} }
#endif #endif
} }
@ -610,14 +725,15 @@ static void spi_setmode(FAR struct spi_dev_s *dev, enum spi_mode_e mode)
* *
****************************************************************************/ ****************************************************************************/
static void spi_setbits(FAR struct spi_dev_s *dev, int nbits) static void spi_setbits(struct spi_dev_s *dev, int nbits)
{ {
FAR struct sam_spidev_s *priv = (FAR struct sam_spidev_s *)dev; struct sam_spics_s *spics = (struct sam_spics_s *)dev;
struct sam_spidev_s *spi = spi_device(spics);
uint32_t regaddr; uint32_t regaddr;
uint32_t regval; uint32_t regval;
spivdbg("cs=%d nbits=%d\n", priv->cs, nbits); spivdbg("cs=%d nbits=%d\n", spics->cs, nbits);
DEBUGASSERT(priv && nbits > 7 && nbits < 17); DEBUGASSERT(spics && nbits > 7 && nbits < 17);
/* NOTE: The logic in spi_send and in spi_exchange only handles 8-bit /* NOTE: The logic in spi_send and in spi_exchange only handles 8-bit
* data at the present time. So the following extra assertion is a * data at the present time. So the following extra assertion is a
@ -629,12 +745,12 @@ static void spi_setbits(FAR struct spi_dev_s *dev, int nbits)
/* Has the number of bits changed? */ /* Has the number of bits changed? */
#ifndef CONFIG_SPI_OWNBUS #ifndef CONFIG_SPI_OWNBUS
if (nbits != priv->nbits) if (nbits != spics->nbits)
{ {
#endif #endif
/* Yes... Set number of bits appropriately */ /* Yes... Set number of bits appropriately */
regaddr = g_csraddr[priv->cs]; regaddr = spi->csraddr[spics->cs];
regval = getreg32(regaddr); regval = getreg32(regaddr);
regval &= ~SPI_CSR_BITS_MASK; regval &= ~SPI_CSR_BITS_MASK;
regval |= SPI_CSR_BITS(nbits); regval |= SPI_CSR_BITS(nbits);
@ -645,7 +761,7 @@ static void spi_setbits(FAR struct spi_dev_s *dev, int nbits)
/* Save the selection so the subsequence re-configurations will be faster */ /* Save the selection so the subsequence re-configurations will be faster */
#ifndef CONFIG_SPI_OWNBUS #ifndef CONFIG_SPI_OWNBUS
priv->nbits = nbits; spics->nbits = nbits;
} }
#endif #endif
} }
@ -666,7 +782,7 @@ static void spi_setbits(FAR struct spi_dev_s *dev, int nbits)
* *
****************************************************************************/ ****************************************************************************/
static uint16_t spi_send(FAR struct spi_dev_s *dev, uint16_t wd) static uint16_t spi_send(struct spi_dev_s *dev, uint16_t wd)
{ {
uint8_t txbyte; uint8_t txbyte;
uint8_t rxbyte; uint8_t rxbyte;
@ -677,6 +793,7 @@ static uint16_t spi_send(FAR struct spi_dev_s *dev, uint16_t wd)
*/ */
txbyte = (uint8_t)wd; txbyte = (uint8_t)wd;
rxbyte = (uint8_t)0;
spi_exchange(dev, &txbyte, &rxbyte, 1); spi_exchange(dev, &txbyte, &rxbyte, 1);
spivdbg("Sent %02x received %02x\n", txbyte, rxbyte); spivdbg("Sent %02x received %02x\n", txbyte, rxbyte);
@ -704,13 +821,14 @@ static uint16_t spi_send(FAR struct spi_dev_s *dev, uint16_t wd)
* *
****************************************************************************/ ****************************************************************************/
static void spi_exchange(FAR struct spi_dev_s *dev, static void spi_exchange(struct spi_dev_s *dev,
FAR const void *txbuffer, FAR void *rxbuffer, const void *txbuffer, void *rxbuffer,
size_t nwords) size_t nwords)
{ {
FAR struct sam_spidev_s *priv = (FAR struct sam_spidev_s *)dev; struct sam_spics_s *spics = (struct sam_spics_s *)dev;
FAR uint8_t *rxptr = (FAR uint8_t*)rxbuffer; struct sam_spidev_s *spi = spi_device(spics);
FAR uint8_t *txptr = (FAR uint8_t*)txbuffer; uint8_t *rxptr = (uint8_t*)rxbuffer;
uint8_t *txptr = (uint8_t*)txbuffer;
uint32_t pcs; uint32_t pcs;
uint32_t data; uint32_t data;
@ -718,11 +836,11 @@ static void spi_exchange(FAR struct spi_dev_s *dev,
/* Set up PCS bits */ /* Set up PCS bits */
pcs = spi_cs2pcs(priv) << SPI_TDR_PCS_SHIFT; pcs = spi_cs2pcs(spics) << SPI_TDR_PCS_SHIFT;
/* Make sure that any previous transfer is flushed from the hardware */ /* Make sure that any previous transfer is flushed from the hardware */
spi_flush(); spi_flush(spi);
/* Loop, sending each word in the user-provied data buffer. /* Loop, sending each word in the user-provied data buffer.
* *
@ -785,24 +903,24 @@ static void spi_exchange(FAR struct spi_dev_s *dev,
* to the serializer. * to the serializer.
*/ */
while ((getreg32(SAM_SPI0_SR) & SPI_INT_TDRE) == 0); while ((spi_getreg(spi, SAM_SPI_SR_OFFSET) & SPI_INT_TDRE) == 0);
/* Write the data to transmitted to the Transmit Data Register (TDR) */ /* Write the data to transmitted to the Transmit Data Register (TDR) */
putreg32(data, SAM_SPI0_TDR); spi_putreg(spi, data, SAM_SPI_TDR_OFFSET);
/* Wait for the read data to be available in the RDR. /* Wait for the read data to be available in the RDR.
* TODO: Data transfer rates would be improved using the RX FIFO * TODO: Data transfer rates would be improved using the RX FIFO
* (and also DMA) * (and also DMA)
*/ */
while ((getreg32(SAM_SPI0_SR) & SPI_INT_RDRF) == 0); while ((spi_getreg(spi, SAM_SPI_SR_OFFSET) & SPI_INT_RDRF) == 0);
/* Read the received data from the SPI Data Register.. /* Read the received data from the SPI Data Register..
* TODO: The following only works if nbits <= 8. * TODO: The following only works if nbits <= 8.
*/ */
data = getreg32(SAM_SPI0_RDR); data = spi_getreg(spi, SAM_SPI_RDR_OFFSET);
if (rxptr) if (rxptr)
{ {
*rxptr++ = (uint8_t)data; *rxptr++ = (uint8_t)data;
@ -830,7 +948,7 @@ static void spi_exchange(FAR struct spi_dev_s *dev,
****************************************************************************/ ****************************************************************************/
#ifndef CONFIG_SPI_EXCHANGE #ifndef CONFIG_SPI_EXCHANGE
static void spi_sndblock(FAR struct spi_dev_s *dev, FAR const void *buffer, size_t nwords) static void spi_sndblock(struct spi_dev_s *dev, const void *buffer, size_t nwords)
{ {
/* spi_exchange can do this. */ /* spi_exchange can do this. */
@ -858,7 +976,7 @@ static void spi_sndblock(FAR struct spi_dev_s *dev, FAR const void *buffer, size
****************************************************************************/ ****************************************************************************/
#ifndef CONFIG_SPI_EXCHANGE #ifndef CONFIG_SPI_EXCHANGE
static void spi_recvblock(FAR struct spi_dev_s *dev, FAR void *buffer, size_t nwords) static void spi_recvblock(struct spi_dev_s *dev, void *buffer, size_t nwords)
{ {
/* spi_exchange can do this. */ /* spi_exchange can do this. */
@ -884,99 +1002,139 @@ static void spi_recvblock(FAR struct spi_dev_s *dev, FAR void *buffer, size_t nw
* *
****************************************************************************/ ****************************************************************************/
FAR struct spi_dev_s *up_spiinitialize(int cs) struct spi_dev_s *up_spiinitialize(int port)
{ {
FAR struct sam_spidev_s *priv; struct sam_spidev_s *spi;
struct sam_spics_s *spics;
int csno = (port & __SPI_CS_MASK) >> __SPI_CS_SHIFT;
int spino = (port & __SPI_SPI_MASK) >> __SPI_SPI_SHIFT;
irqstate_t flags; irqstate_t flags;
/* The support SAM parts have only a single SPI port */ /* The support SAM parts have only a single SPI port */
spivdbg("cs=%d\n", cs); spivdbg("port: %d csno: %d spino: %d\n", port, csno, spino);
DEBUGASSERT(cs >= 0 && cs <= SAM_SPI_NCS); DEBUGASSERT(csno >= 0 && csno <= SAM_SPI_NCS);
#ifdef CONFIG_SPI_OWNBUS
/* There is only one device on the bus and, therefore, there is only one
* supported chip select. In this case, use the single, pre-allocated
* chip select structure.
*/
priv = &g_spidev;
#if defined(CONFIG_SAMA5_SPI0) && defined(CONFIG_SAMA5_SPI1)
DEBUGASSERT(spino >= 0 && spino <= 1);
#elif defined(CONFIG_SAMA5_SPI0)
DEBUGASSERT(spino == 0);
#else #else
DEBUGASSERT(spino == 1);
#endif
/* Allocate a new state structure for this chip select. NOTE that there /* Allocate a new state structure for this chip select. NOTE that there
* is no protection if the same chip select is used in two different * is no protection if the same chip select is used in two different
* chip select structures. * chip select structures.
*/ */
priv = (FAR struct sam_spidev_s *)zalloc(sizeof(struct sam_spidev_s)); spics = (struct sam_spics_s *)zalloc(sizeof(struct sam_spics_s));
if (!priv) if (!spics)
{ {
spivdbg("ERROR: Failed to allocate a chip select structure\n", cs); spivdbg("ERROR: Failed to allocate a chip select structure\n");
return NULL; return NULL;
} }
#endif
/* Set up the initial state for this chip select structure. Other fields /* Set up the initial state for this chip select structure. Other fields
* were zeroed by zalloc(). * were zeroed by zalloc().
*/ */
priv->spidev.ops = &g_spiops; /* Select the SPI operations */
priv->cs = cs;
#if defined(CONFIG_SAMA5_SPI0) && defined(CONFIG_SAMA5_SPI1)
spics->spidev.ops = spino ? &g_spi1ops : &g_spi0ops;
#elif defined(CONFIG_SAMA5_SPI0)
spics->spidev.ops = &g_spi0ops;
#else
spics->spidev.ops = &g_spi1ops;
#endif
/* Save the chip select and SPI controller numbers */
spics->cs = csno;
#if defined(CONFIG_SAMA5_SPI0) || defined(CONFIG_SAMA5_SPI1)
spics->spino = spino;
#endif
/* Get the SPI device structure associated with the chip select */
spi = spi_device(spics);
#ifndef CONFIG_SPI_OWNBUS
/* Has the SPI hardware been initialized? */ /* Has the SPI hardware been initialized? */
if (!g_spinitialized) if (!spi->initialized)
#endif
{ {
/* Enable clocking to the SPI block */ /* Enable clocking to the SPI block */
flags = irqsave(); flags = irqsave();
sam_spi0_enableclk(); #if defined(CONFIG_SAMA5_SPI0) && defined(CONFIG_SAMA5_SPI1)
if (spino == 0)
#endif
#if defined(CONFIG_SAMA5_SPI0)
{
sam_spi0_enableclk();
/* Configure multiplexed pins as connected on the board. Chip select /* Configure multiplexed pins as connected on the board. Chip
* pins must be configured by board-specific logic. * select pins must be selected by board-specific logic.
*/ */
sam_configgpio(GPIO_SPI0_MISO); sam_configgpio(GPIO_SPI0_MISO);
sam_configgpio(GPIO_SPI0_MOSI); sam_configgpio(GPIO_SPI0_MOSI);
sam_configgpio(GPIO_SPI0_SPCK); sam_configgpio(GPIO_SPI0_SPCK);
}
#endif
#if defined(CONFIG_SAMA5_SPI0) && defined(CONFIG_SAMA5_SPI1)
else
#endif
#if defined(CONFIG_SAMA5_SPI1)
{
sam_spi1_enableclk();
/* Configure multiplexed pins as connected on the board. Chip
* select pins must be selected by board-specific logic.
*/
sam_configgpio(GPIO_SPI1_MISO);
sam_configgpio(GPIO_SPI1_MOSI);
sam_configgpio(GPIO_SPI1_SPCK);
}
#endif
/* Disable SPI clocking */ /* Disable SPI clocking */
putreg32(SPI_CR_SPIDIS, SAM_SPI0_CR); spi_putreg(spi, SPI_CR_SPIDIS, SAM_SPI_CR_OFFSET);
/* Execute a software reset of the SPI (twice) */ /* Execute a software reset of the SPI (twice) */
putreg32(SPI_CR_SWRST, SAM_SPI0_CR); spi_putreg(spi, SPI_CR_SWRST, SAM_SPI_CR_OFFSET);
putreg32(SPI_CR_SWRST, SAM_SPI0_CR); spi_putreg(spi, SPI_CR_SWRST, SAM_SPI_CR_OFFSET);
irqrestore(flags); irqrestore(flags);
/* Configure the SPI mode register */ /* Configure the SPI mode register */
putreg32(SPI_MR_MSTR | SPI_MR_MODFDIS, SAM_SPI0_MR); spi_putreg(spi, SPI_MR_MSTR | SPI_MR_MODFDIS, SAM_SPI_MR_OFFSET);
/* And enable the SPI */ /* And enable the SPI */
putreg32(SPI_CR_SPIEN, SAM_SPI0_CR); spi_putreg(spi, SPI_CR_SPIEN, SAM_SPI_CR_OFFSET);
up_mdelay(20); up_mdelay(20);
/* Flush any pending transfers */ /* Flush any pending transfers */
(void)getreg32(SAM_SPI0_SR); (void)spi_getreg(spi, SAM_SPI_SR_OFFSET);
(void)getreg32(SAM_SPI0_RDR); (void)spi_getreg(spi, SAM_SPI_RDR_OFFSET);
#ifndef CONFIG_SPI_OWNBUS #ifndef CONFIG_SPI_OWNBUS
/* Initialize the SPI semaphore that enforces mutually exclusive /* Initialize the SPI semaphore that enforces mutually exclusive
* access to the SPI registers. * access to the SPI registers.
*/ */
sem_init(&g_spisem, 0, 1); sem_init(&spi->spisem, 0, 1);
g_spinitialized = true; spi->initialized = true;
#endif #endif
spi_dumpregs("After initialization"); spi_dumpregs(spi, "After initialization");
} }
return &priv->spidev; return &spics->spidev;
} }
#endif /* CONFIG_SAMA5_SPI0 */ #endif /* CONFIG_SAMA5_SPI0 || CONFIG_SAMA5_SPI1 */

View File

@ -48,9 +48,34 @@
#include "chip.h" #include "chip.h"
/************************************************************************************ /************************************************************************************
* Definitions * Pre-processor Definitions
************************************************************************************/ ************************************************************************************/
/* The SPI port number used as an input to up_spiinitialize encodes information
* about the SPI controller (0 or 1) and the SPI chip select (0-3)
*/
#define __SPI_CS_SHIFT (0) /* Bits 0-1: SPI chip select number */
#define __SPI_CS_MASK (3 << __SPI_CS_SHIFT)
# define __SPI_CS0 (0 << __SPI_CS_SHIFT)
# define __SPI_CS1 (1 << __SPI_CS_SHIFT)
# define __SPI_CS2 (2 << __SPI_CS_SHIFT)
# define __SPI_CS3 (3 << __SPI_CS_SHIFT)
#define __SPI_SPI_SHIFT (2) /* Bit 2: SPI controller number */
#define __SPI_SPI_MASK (1 << __SPI_SPI_SHIFT)
# define __SPI_SPI0 (0 << __SPI_SPI_SHIFT) /* SPI0 */
# define __SPI_SPI1 (1 << __SPI_SPI_SHIFT) /* SPI1 */
#define SPI0_CS0 (__SPI_SPI0 | __SPI_CS0)
#define SPI0_CS1 (__SPI_SPI0 | __SPI_CS1)
#define SPI0_CS2 (__SPI_SPI0 | __SPI_CS2)
#define SPI0_CS3 (__SPI_SPI0 | __SPI_CS3)
#define SPI1_CS0 (__SPI_SPI1 | __SPI_CS0)
#define SPI1_CS1 (__SPI_SPI1 | __SPI_CS1)
#define SPI1_CS2 (__SPI_SPI1 | __SPI_CS2)
#define SPI1_CS3 (__SPI_SPI1 | __SPI_CS3)
/************************************************************************************ /************************************************************************************
* Public Types * Public Types
************************************************************************************/ ************************************************************************************/
@ -79,14 +104,14 @@ extern "C"
************************************************************************************/ ************************************************************************************/
/**************************************************************************** /****************************************************************************
* Name: sam_spiselect, sam_spistatus, and sam_spicmddata * Name: sam_spi[0|1]select, sam_spi[0|1]status, and sam_spi[0|1]cmddata
* *
* Description: * Description:
* These external functions must be provided by board-specific logic. They * These external functions must be provided by board-specific logic. They
* include: * include:
* *
* o sam_spiselect is a functions tomanage the board-specific chip selects * o sam_spi[0|1]select is a functions tomanage the board-specific chip selects
* o sam_spistatus and sam_spicmddata: Implementations of the status * o sam_spi[0|1]status and sam_spi[0|1]cmddata: Implementations of the status
* and cmddata methods of the SPI interface defined by struct spi_ops_ * and cmddata methods of the SPI interface defined by struct spi_ops_
* (see include/nuttx/spi/spi.h). All other methods including * (see include/nuttx/spi/spi.h). All other methods including
* up_spiinitialize()) are provided by common SAM3/4 logic. * up_spiinitialize()) are provided by common SAM3/4 logic.
@ -95,11 +120,11 @@ extern "C"
* *
* 1. Provide logic in sam_boardinitialize() to configure SPI chip select * 1. Provide logic in sam_boardinitialize() to configure SPI chip select
* pins. * pins.
* 2. Provide sam_spiselect() and sam_spistatus() functions in your board- * 2. Provide sam_spi[0|1]select() and sam_spi[0|1]status() functions in your board-
* specific logic. These functions will perform chip selection and * specific logic. These functions will perform chip selection and
* status operations using GPIOs in the way your board is configured. * status operations using GPIOs in the way your board is configured.
* 2. If CONFIG_SPI_CMDDATA is defined in the NuttX configuration, provide * 2. If CONFIG_SPI_CMDDATA is defined in the NuttX configuration, provide
* sam_spicmddata() functions in your board-specific logic. This * sam_spi[0|1]cmddata() functions in your board-specific logic. This
* function will perform cmd/data selection operations using GPIOs in * function will perform cmd/data selection operations using GPIOs in
* the way your board is configured. * the way your board is configured.
* 3. Add a call to up_spiinitialize() in your low level application * 3. Add a call to up_spiinitialize() in your low level application
@ -116,7 +141,7 @@ struct spi_dev_s;
enum spi_dev_e; enum spi_dev_e;
/**************************************************************************** /****************************************************************************
* Name: sam_spiselect * Name: sam_spi[0|1]select
* *
* Description: * Description:
* PIO chip select pins may be programmed by the board specific logic in * PIO chip select pins may be programmed by the board specific logic in
@ -141,10 +166,15 @@ enum spi_dev_e;
* *
****************************************************************************/ ****************************************************************************/
void sam_spiselect(enum spi_dev_e devid, bool selected); #ifdef CONFIG_SAMA5_SPI0
void sam_spi0select(enum spi_dev_e devid, bool selected);
#endif
#ifdef CONFIG_SAMA5_SPI1
void sam_spi1select(enum spi_dev_e devid, bool selected);
#endif
/**************************************************************************** /****************************************************************************
* Name: sam_spistatus * Name: sam_spi[0|1]status
* *
* Description: * Description:
* Return status information associated with the SPI device. * Return status information associated with the SPI device.
@ -158,10 +188,15 @@ void sam_spiselect(enum spi_dev_e devid, bool selected);
* *
****************************************************************************/ ****************************************************************************/
uint8_t sam_spistatus(FAR struct spi_dev_s *dev, enum spi_dev_e devid); #ifdef CONFIG_SAMA5_SPI0
uint8_t sam_spi0status(FAR struct spi_dev_s *dev, enum spi_dev_e devid);
#endif
#ifdef CONFIG_SAMA5_SPI1
uint8_t sam_spi1status(FAR struct spi_dev_s *dev, enum spi_dev_e devid);
#endif
/**************************************************************************** /****************************************************************************
* Name: sam_spicmddata * Name: sam_spi[0|1]cmddata
* *
* Description: * Description:
* Some SPI devices require an additional control to determine if the SPI * Some SPI devices require an additional control to determine if the SPI
@ -185,7 +220,12 @@ uint8_t sam_spistatus(FAR struct spi_dev_s *dev, enum spi_dev_e devid);
****************************************************************************/ ****************************************************************************/
#ifdef CONFIG_SPI_CMDDATA #ifdef CONFIG_SPI_CMDDATA
int sam_spicmddata(FAR struct spi_dev_s *dev, enum spi_dev_e devid, bool cmd); #ifdef CONFIG_SAMA5_SPI0
int sam_spi0cmddata(FAR struct spi_dev_s *dev, enum spi_dev_e devid, bool cmd);
#endif
#ifdef CONFIG_SAMA5_SPI1
int sam_spi1cmddata(FAR struct spi_dev_s *dev, enum spi_dev_e devid, bool cmd);
#endif
#endif #endif
#endif /* CONFIG_SAMA5_SPI0 */ #endif /* CONFIG_SAMA5_SPI0 */