SAMA5: SPI driver now supports both SPI0 and SPI1
This commit is contained in:
parent
487866b2b6
commit
d516baa73f
@ -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 *******************************************************************/
|
||||||
|
@ -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;
|
||||||
|
@ -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 */
|
||||||
|
@ -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 */
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user