mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2025-01-25 07:37:39 +08:00
555 lines
18 KiB
C
555 lines
18 KiB
C
|
/**
|
||
|
******************************************************************************
|
||
|
* @brief SPI functions of the firmware library.
|
||
|
******************************************************************************
|
||
|
*/
|
||
|
|
||
|
/* Includes ------------------------------------------------------------------*/
|
||
|
#include "gd32f10x_spi.h"
|
||
|
#include "gd32f10x_rcc.h"
|
||
|
|
||
|
/** @addtogroup GD32F10x_Firmware
|
||
|
* @{
|
||
|
*/
|
||
|
|
||
|
/** @defgroup SPI
|
||
|
* @brief SPI driver modules
|
||
|
* @{
|
||
|
*/
|
||
|
|
||
|
/** @defgroup SPI_Private_Defines
|
||
|
* @{
|
||
|
*/
|
||
|
|
||
|
/* SPI registers Masks */
|
||
|
#define CTLR1_CLEAR_MASK ((uint16_t)0x3040)
|
||
|
#define I2SCTLR_CLEAR_MASK ((uint16_t)0xF040)
|
||
|
|
||
|
/* I2S clock source selection Masks */
|
||
|
#define I2S2_CLOCK_SRC ((uint32_t)(0x00020000))
|
||
|
#define I2S3_CLOCK_SRC ((uint32_t)(0x00040000))
|
||
|
#define I2S_MUL_MASK ((uint32_t)(0x0000F000))
|
||
|
#define I2S_DIV_MASK ((uint32_t)(0x000000F0))
|
||
|
/**
|
||
|
* @}
|
||
|
*/
|
||
|
|
||
|
/** @defgroup SPI_Private_Functions
|
||
|
* @{
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* @brief Reset the SPIx and the I2Sx peripheral.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_I2S_DeInit(SPI_TypeDef *SPIx)
|
||
|
{
|
||
|
if (SPIx == SPI1) {
|
||
|
/* Reset SPI1 */
|
||
|
RCC_APB2PeriphReset_Enable(RCC_APB2PERIPH_SPI1RST, ENABLE);
|
||
|
|
||
|
RCC_APB2PeriphReset_Enable(RCC_APB2PERIPH_SPI1RST, DISABLE);
|
||
|
} else if (SPIx == SPI2) {
|
||
|
/* Reset SPI2 and I2S2 peripheral */
|
||
|
RCC_APB1PeriphReset_Enable(RCC_APB1PERIPH_SPI2RST, ENABLE);
|
||
|
RCC_APB1PeriphReset_Enable(RCC_APB1PERIPH_SPI2RST, DISABLE);
|
||
|
} else if (SPIx == SPI3) {
|
||
|
/* Reset SPI3 and I2S3 peripheral */
|
||
|
RCC_APB1PeriphReset_Enable(RCC_APB1PERIPH_SPI3RST, ENABLE);
|
||
|
RCC_APB1PeriphReset_Enable(RCC_APB1PERIPH_SPI3RST, DISABLE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Initialize the SPIx peripheral.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param SPI_InitParameter: The structuer contains configuration information.
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_Init(SPI_TypeDef *SPIx, SPI_InitPara *SPI_InitParameter)
|
||
|
{
|
||
|
uint16_t temp_ctrl1 = 0;
|
||
|
|
||
|
/* Configure SPIx CTRL1 according to the SPI_InitParameter */
|
||
|
temp_ctrl1 = SPIx->CTLR1;
|
||
|
temp_ctrl1 &= CTLR1_CLEAR_MASK;
|
||
|
temp_ctrl1 |= (uint16_t)((uint32_t)SPI_InitParameter->SPI_TransType | SPI_InitParameter->SPI_Mode |
|
||
|
SPI_InitParameter->SPI_FrameFormat | SPI_InitParameter->SPI_SCKPL |
|
||
|
SPI_InitParameter->SPI_SCKPH | SPI_InitParameter->SPI_SWNSSEN |
|
||
|
SPI_InitParameter->SPI_PSC | SPI_InitParameter->SPI_FirstBit);
|
||
|
SPIx->CTLR1 = temp_ctrl1;
|
||
|
|
||
|
/* Configure SPIx CRC Polynomial */
|
||
|
SPIx->CPR = SPI_InitParameter->SPI_CRCPOL;
|
||
|
|
||
|
/* Configure the I2SSEL bit in I2SCTLR register as SPI mode */
|
||
|
SPIx->I2SCTLR &= ~SPI_I2SCTLR_I2SSEL;
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Initialize the I2Sx peripheral.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 2 or 3.
|
||
|
* @param I2S_InitParameter: The structuer contains configuration information.
|
||
|
* @retval None
|
||
|
*/
|
||
|
void I2S_Init(SPI_TypeDef *SPIx, I2S_InitPara *I2S_InitParameter)
|
||
|
{
|
||
|
uint16_t temp_i2sctrl = 0, div = 2, of = 0;
|
||
|
uint32_t temp = 0;
|
||
|
RCC_ClocksPara RCC_Clocks;
|
||
|
uint32_t sourceclock = 0;
|
||
|
|
||
|
/* SPIx I2SCTLR & I2SCKP Configuration */
|
||
|
/* Deinit I2SCTLR I2SCKP register */
|
||
|
SPIx->I2SCTLR &= I2SCTLR_CLEAR_MASK;
|
||
|
SPIx->I2SCKP = 0x0002;
|
||
|
|
||
|
/* Default config of the prescaler*/
|
||
|
if (I2S_InitParameter->I2S_AudioFreq == I2S_AUDIOFREQ_DEFAULT) {
|
||
|
of = (uint16_t)0;
|
||
|
div = (uint16_t)2;
|
||
|
} else {
|
||
|
/* Get the I2S clock source */
|
||
|
if (((uint32_t)SPIx) == SPI2_BASE) {
|
||
|
temp = I2S2_CLOCK_SRC;
|
||
|
} else {
|
||
|
temp = I2S3_CLOCK_SRC;
|
||
|
}
|
||
|
|
||
|
/* I2S clock source configuration depend on different device */
|
||
|
#ifdef GD32F10X_CL
|
||
|
if ((RCC->GCFGR2 & temp) != 0) {
|
||
|
/* Get RCC PLL3 multiplier */
|
||
|
temp = (uint32_t)((RCC->GCFGR2 & I2S_MUL_MASK) >> 12);
|
||
|
|
||
|
if ((temp > 5) && (temp < 15)) {
|
||
|
/* Multiplier is between 8 and 14 (value 15 is forbidden) */
|
||
|
temp += 2;
|
||
|
} else {
|
||
|
if (temp == 15) {
|
||
|
/* Multiplier is 20 */
|
||
|
temp = 20;
|
||
|
}
|
||
|
}
|
||
|
/* Get the PREDV2 value */
|
||
|
sourceclock = (uint32_t)(((RCC->GCFGR2 & I2S_DIV_MASK) >> 4) + 1);
|
||
|
|
||
|
/* Calculate sourceclock based on PLL3 and PREDV2 */
|
||
|
sourceclock = (uint32_t)((HSE_VALUE / sourceclock) * temp * 2);
|
||
|
} else {
|
||
|
/* Get system clock */
|
||
|
RCC_GetClocksFreq(&RCC_Clocks);
|
||
|
sourceclock = RCC_Clocks.CK_SYS_Frequency;
|
||
|
}
|
||
|
#else
|
||
|
/* Get system clock */
|
||
|
RCC_GetClocksFreq(&RCC_Clocks);
|
||
|
sourceclock = RCC_Clocks.CK_SYS_Frequency;
|
||
|
#endif /* GD32F10X_CL */
|
||
|
|
||
|
/* Calculate the prescaler depending on the MCLK output state and the data format with a flaoting point. */
|
||
|
if (I2S_InitParameter->I2S_MCKOE == I2S_MCK_ENABLE) {
|
||
|
temp = (uint16_t)(((((sourceclock / 256) * 10) / I2S_InitParameter->I2S_AudioFreq)) + 5);
|
||
|
} else {
|
||
|
if (I2S_InitParameter->I2S_FrameFormat == I2S_FRAMEFORMAT_DL16b_CL16b) {
|
||
|
temp = (uint16_t)(((((sourceclock / 32) * 10) / I2S_InitParameter->I2S_AudioFreq)) + 5);
|
||
|
} else {
|
||
|
temp = (uint16_t)(((((sourceclock / 64) * 10) / I2S_InitParameter->I2S_AudioFreq)) + 5);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* Remove the flaoting point */
|
||
|
temp = temp / 10;
|
||
|
of = (uint16_t)(temp & (uint16_t)0x0001);
|
||
|
div = (uint16_t)((temp - of) / 2);
|
||
|
of = (uint16_t)(of << 8);
|
||
|
}
|
||
|
|
||
|
/* Inappropriate prescaler, Set the default values */
|
||
|
if ((div < 1) || (div > 0xFF)) {
|
||
|
div = 2;
|
||
|
of = 0;
|
||
|
}
|
||
|
|
||
|
/* Configure SPIx I2SCKP */
|
||
|
SPIx->I2SCKP = (uint16_t)(div | (uint16_t)(of | (uint16_t)I2S_InitParameter->I2S_MCKOE));
|
||
|
|
||
|
/* Configure SPIx I2SCTLR according to the I2S_InitParameter */
|
||
|
temp_i2sctrl = SPIx->I2SCTLR;
|
||
|
temp_i2sctrl |= (uint16_t)(SPI_I2SCTLR_I2SSEL | (uint16_t)(I2S_InitParameter->I2S_Mode | \
|
||
|
(uint16_t)(I2S_InitParameter->I2S_STD | (uint16_t)(I2S_InitParameter->I2S_FrameFormat | \
|
||
|
(uint16_t)I2S_InitParameter->I2S_CKPL))));
|
||
|
SPIx->I2SCTLR = temp_i2sctrl;
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Initial SPI_InitParameter members.
|
||
|
* @param SPI_InitParameter : pointer to a SPI_InitPara structure.
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_ParaInit(SPI_InitPara *SPI_InitParameter)
|
||
|
{
|
||
|
/* Reset SPI init structure parameters values */
|
||
|
SPI_InitParameter->SPI_TransType = SPI_TRANSTYPE_FULLDUPLEX;
|
||
|
SPI_InitParameter->SPI_Mode = SPI_MODE_SLAVE;
|
||
|
SPI_InitParameter->SPI_FrameFormat = SPI_FRAMEFORMAT_8BIT;
|
||
|
SPI_InitParameter->SPI_SCKPL = SPI_SCKPL_LOW;
|
||
|
SPI_InitParameter->SPI_SCKPH = SPI_SCKPH_1EDGE;
|
||
|
SPI_InitParameter->SPI_SWNSSEN = SPI_SWNSS_HARD;
|
||
|
SPI_InitParameter->SPI_PSC = SPI_PSC_2;
|
||
|
SPI_InitParameter->SPI_FirstBit = SPI_FIRSTBIT_MSB;
|
||
|
SPI_InitParameter->SPI_CRCPOL = 7;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Initial I2S_InitParameter members.
|
||
|
* @param I2S_InitParameter : pointer to a I2S_InitPara structure.
|
||
|
* @retval None
|
||
|
*/
|
||
|
void I2S_ParaInit(I2S_InitPara *I2S_InitParameter)
|
||
|
{
|
||
|
/* Reset I2S init structure parameters values */
|
||
|
I2S_InitParameter->I2S_Mode = I2S_MODE_SLAVETX;
|
||
|
I2S_InitParameter->I2S_STD = I2S_STD_PHILLIPS;
|
||
|
I2S_InitParameter->I2S_FrameFormat = I2S_FRAMEFORMAT_DL16b_CL16b;
|
||
|
I2S_InitParameter->I2S_MCKOE = I2S_MCK_DISABLE;
|
||
|
I2S_InitParameter->I2S_AudioFreq = I2S_AUDIOFREQ_DEFAULT;
|
||
|
I2S_InitParameter->I2S_CKPL = I2S_CKPL_LOW;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Enable or disable the SPI peripheral.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param NewValue: ENABLE or DISABLE.
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_Enable(SPI_TypeDef *SPIx, TypeState NewValue)
|
||
|
{
|
||
|
if (NewValue != DISABLE) {
|
||
|
SPIx->CTLR1 |= SPI_CTLR1_SPIEN;
|
||
|
} else {
|
||
|
SPIx->CTLR1 &= ~SPI_CTLR1_SPIEN;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Enable or disable the I2S peripheral.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 2 or 3.
|
||
|
* @param NewValue: ENABLE or DISABLE.
|
||
|
* @retval None
|
||
|
*/
|
||
|
void I2S_Enable(SPI_TypeDef *SPIx, TypeState NewValue)
|
||
|
{
|
||
|
if (NewValue != DISABLE) {
|
||
|
SPIx->I2SCTLR |= SPI_I2SCTLR_I2SEN;
|
||
|
} else {
|
||
|
SPIx->I2SCTLR &= ~SPI_I2SCTLR_I2SEN;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Enable or disable the SPI or I2S interrupts.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param SPI_I2S_INT: specifies the SPI or I2S interrupt source. Select one of the follwing values :
|
||
|
* @arg SPI_I2S_INT_TBE: Tx buffer empty interrupt mask
|
||
|
* @arg SPI_I2S_INT_RBNE: Rx buffer not empty interrupt mask
|
||
|
* @arg SPI_I2S_INT_ERR: Error interrupt mask
|
||
|
* @param NewValue: ENABLE or DISABLE.
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_I2S_INTConfig(SPI_TypeDef *SPIx, uint8_t SPI_I2S_INT, TypeState NewValue)
|
||
|
{
|
||
|
uint16_t intmask = 0 ;
|
||
|
|
||
|
/* Get the interrupt enable bit in CTRL2 */
|
||
|
intmask = (uint16_t)1 << (uint16_t)(SPI_I2S_INT >> 4);
|
||
|
|
||
|
/* Enable or disable the selected interrupt */
|
||
|
if (NewValue != DISABLE) {
|
||
|
SPIx->CTLR2 |= intmask;
|
||
|
} else {
|
||
|
SPIx->CTLR2 &= (uint16_t)~intmask;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Enable or disable the SPIx or I2Sx DMA transfer request.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param SPI_I2S_DMAReq: Select the SPI or I2S DMA transfer request. Select one of the follwing values :
|
||
|
* @arg SPI_I2S_DMA_TX: Tx buffer DMA transfer request
|
||
|
* @arg SPI_I2S_DMA_RX: Rx buffer DMA transfer request
|
||
|
* @param NewValue: ENABLE or DISABLE.
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_I2S_DMA_Enable(SPI_TypeDef *SPIx, uint16_t SPI_I2S_DMAReq, TypeState NewValue)
|
||
|
{
|
||
|
if (NewValue != DISABLE) {
|
||
|
SPIx->CTLR2 |= SPI_I2S_DMAReq;
|
||
|
} else {
|
||
|
SPIx->CTLR2 &= (uint16_t)~SPI_I2S_DMAReq;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Send a Data by the SPIx or I2Sx peripheral.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param Data : Data to be Send.
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_I2S_SendData(SPI_TypeDef *SPIx, uint16_t Data)
|
||
|
{
|
||
|
SPIx->DTR = Data;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Return the received data by the SPIx or I2Sx peripheral.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @retval The value of the received data.
|
||
|
*/
|
||
|
uint16_t SPI_I2S_ReceiveData(SPI_TypeDef *SPIx)
|
||
|
{
|
||
|
return SPIx->DTR;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief NSS pin internal software management.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param SPI_SWNSS: specifies the SPI NSS internal state. Select one of the follwing values :
|
||
|
* @arg SPI_SWNSS_SET: Set NSS pin internally
|
||
|
* @arg SPI_SWNSS_RESET: Reset NSS pin internally
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_SWNSSConfig(SPI_TypeDef *SPIx, uint16_t SPI_SWNSS)
|
||
|
{
|
||
|
if (SPI_SWNSS != SPI_SWNSS_RESET) {
|
||
|
/* Set NSS pin */
|
||
|
SPIx->CTLR1 |= SPI_CTLR1_SWNSS;
|
||
|
} else {
|
||
|
/* Reset NSS pin */
|
||
|
SPIx->CTLR1 &= ~SPI_CTLR1_SWNSS;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Enable or disable the NSS output.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param NewValue: ENABLE or DISABLE.
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_NSSDRV(SPI_TypeDef *SPIx, TypeState NewValue)
|
||
|
{
|
||
|
if (NewValue != DISABLE) {
|
||
|
SPIx->CTLR2 |= SPI_CTLR2_NSSDRV;
|
||
|
} else {
|
||
|
SPIx->CTLR2 &= ~SPI_CTLR2_NSSDRV;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Configure the data frame format.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param SPI_FrameFormat: Select the data frame format. Select one of the follwing values :
|
||
|
* @arg SPI_FRAMEFORMAT_16BIT: Set data frame format to 16bit
|
||
|
* @arg SPI_FRAMEFORMAT_8BIT: Set data frame format to 8bit
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_FrameFormatConfig(SPI_TypeDef *SPIx, uint16_t SPI_FrameFormat)
|
||
|
{
|
||
|
/* Clear FF16 bit */
|
||
|
SPIx->CTLR1 &= (uint16_t)~SPI_FRAMEFORMAT_16BIT;
|
||
|
/* Set new FF16 bit value */
|
||
|
SPIx->CTLR1 |= SPI_FrameFormat;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Send the SPIx CRC value.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_SendCRCNext(SPI_TypeDef *SPIx)
|
||
|
{
|
||
|
/* Enable the CRC transmission */
|
||
|
SPIx->CTLR1 |= SPI_CTLR1_CRCNT;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Enable or disable the CRC calculation.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param NewValue: ENABLE or DISABLE.
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_CRC_Enable(SPI_TypeDef *SPIx, TypeState NewValue)
|
||
|
{
|
||
|
if (NewValue != DISABLE) {
|
||
|
/* Enable the CRC calculation */
|
||
|
SPIx->CTLR1 |= SPI_CTLR1_CRCEN;
|
||
|
} else {
|
||
|
/* Disable the CRC calculation */
|
||
|
SPIx->CTLR1 &= ~SPI_CTLR1_CRCEN;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Get the transmit or the receive CRC register value.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param SPI_CRC: Select the transmit or the receive CRC register. Select one of the follwing values :
|
||
|
* @arg SPI_CRC_TX: Selects Tx CRC register
|
||
|
* @arg SPI_CRC_RX: Selects Rx CRC register
|
||
|
* @retval The selected CRC register value.
|
||
|
*/
|
||
|
uint16_t SPI_GetCRC(SPI_TypeDef *SPIx, uint8_t SPI_CRC)
|
||
|
{
|
||
|
if (SPI_CRC != SPI_CRC_RX) {
|
||
|
/* Transmit CRC value */
|
||
|
return SPIx->TCR;
|
||
|
} else {
|
||
|
/* Receive CRC value */
|
||
|
return SPIx->RCR;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Get the CRC Polynomial value.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @retval The CRC Polynomial value.
|
||
|
*/
|
||
|
uint16_t SPI_GetCRCPolynomial(SPI_TypeDef *SPIx)
|
||
|
{
|
||
|
return SPIx->CPR;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Select the transfer direction in bidirectional mode.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param SPI_BDOE: The transfer direction in bi-directional mode. Select one of the follwing values :
|
||
|
* @arg SPI_BDOE_TX: Selects Tx transmission direction
|
||
|
* @arg SPI_BDOE_RX: Selects Rx receive direction
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_BDOEConfig(SPI_TypeDef *SPIx, uint16_t SPI_BDOE)
|
||
|
{
|
||
|
if (SPI_BDOE == SPI_BDOE_TX) {
|
||
|
/* Transmit only mode*/
|
||
|
SPIx->CTLR1 |= SPI_BDOE_TX;
|
||
|
} else {
|
||
|
/* Receive only mode */
|
||
|
SPIx->CTLR1 &= SPI_BDOE_RX;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Check whether the flag is set or not.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param SPI_I2S_FLAG: Select the flag. Select one of the follwing values :
|
||
|
* @arg SPI_FLAG_TBE: Transmit buffer empty flag.
|
||
|
* @arg SPI_FLAG_RBNE: Receive buffer not empty flag.
|
||
|
* @arg SPI_FLAG_BSY: Busy flag.
|
||
|
* @arg SPI_FLAG_OVR: Overrun flag.
|
||
|
* @arg SPI_FLAG_MODF: Mode Fault flag.
|
||
|
* @arg SPI_FLAG_CRCERR: CRC Error flag.
|
||
|
* @arg I2S_FLAG_TBE: Transmit buffer empty flag.
|
||
|
* @arg I2S_FLAG_RBNE: Receive buffer not empty flag.
|
||
|
* @arg I2S_FLAG_BSY: Busy flag.
|
||
|
* @arg I2S_FLAG_OVR: Overrun flag.
|
||
|
* @arg I2S_FLAG_UDR: Underrun Error flag.
|
||
|
* @arg I2S_FLAG_CHSIDE: Channel Side flag.
|
||
|
* @retval The new state of SPI_I2S_FLAG.
|
||
|
*/
|
||
|
TypeState SPI_I2S_GetBitState(SPI_TypeDef *SPIx, uint16_t SPI_I2S_FLAG)
|
||
|
{
|
||
|
/* Check the status of the selected flag */
|
||
|
if ((SPIx->STR & SPI_I2S_FLAG) != (uint16_t)RESET) {
|
||
|
return SET;
|
||
|
} else {
|
||
|
return RESET;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Clear the flag, only used for clear CRCERR flag.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param SPI_I2S_FLAG: Select the flag. This parametric only can be SPI_FLAG_CRCERR.
|
||
|
* @note
|
||
|
* The other flags are cleared by software sequences:
|
||
|
* - OVR (OverRun error) flag is cleared by software sequence: a read
|
||
|
* operation to SPI_DTR register (SPI_I2S_ReceiveData()) followed by a read
|
||
|
* operation to SPI_STR register (SPI_I2S_GetBitState()).
|
||
|
* - UDR (UnderRun error) flag is cleared by a read operation to
|
||
|
* SPI_STR register (SPI_I2S_GetBitState()).
|
||
|
* - MODF (Mode Fault) flag is cleared by software sequence: a read/write
|
||
|
* operation to SPI_STR register (SPI_I2S_GetBitState()) followed by a
|
||
|
* write operation to SPI_CTLR1 register (SPI_Enable() to enable the SPI).
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_I2S_ClearBitState(SPI_TypeDef *SPIx, uint16_t SPI_I2S_FLAG)
|
||
|
{
|
||
|
SPIx->STR = (uint16_t)~SPI_I2S_FLAG;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Check whether interrupt has occurred.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param SPI_I2S_INT: Select the SPI/I2S interrupt. Select one of the follwing values :
|
||
|
* @arg SPI_I2S_INT_TBE: Transmit buffer empty interrupt.
|
||
|
* @arg SPI_I2S_INT_RBNE: Receive buffer not empty interrupt.
|
||
|
* @arg SPI_I2S_INT_OVR: Overrun interrupt.
|
||
|
* @arg SPI_INT_MODF: Mode Fault interrupt.
|
||
|
* @arg SPI_INT_CRCERR: CRC Error interrupt.
|
||
|
* @arg I2S_INT_UDR: Underrun Error interrupt.
|
||
|
* @retval The new state of SPI_I2S_INT.
|
||
|
*/
|
||
|
TypeState SPI_I2S_GetIntBitState(SPI_TypeDef *SPIx, uint8_t SPI_I2S_INT)
|
||
|
{
|
||
|
uint16_t intposition = 0, intmask = 0;
|
||
|
/* Get the interrupt pending bit and enable bit */
|
||
|
intposition = 0x01 << (SPI_I2S_INT & 0x0F);
|
||
|
intmask = 0x01 << (SPI_I2S_INT >> 4);
|
||
|
|
||
|
/* Check the status of the interrupt */
|
||
|
if (((SPIx->STR & intposition) != (uint16_t)RESET) && (SPIx->CTLR2 & intmask)) {
|
||
|
return SET;
|
||
|
} else {
|
||
|
return RESET;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Clear the SPIx or I2S interrupt pending bit, only uesd for clear CRCERR interrupt.
|
||
|
* @param SPIx: the SPI/I2S peripheral where x can be 1..3.
|
||
|
* @param SPI_I2S_INT: Select the SPI or I2S interrupt. This parametric only can be SPI_INT_CRCERR.
|
||
|
* @note
|
||
|
* The other flags are cleared by software sequences:
|
||
|
* - OVR (OverRun error) flag is cleared by software sequence: a read
|
||
|
* operation to SPI_DTR register (SPI_I2S_ReceiveData()) followed by a read
|
||
|
* operation to SPI_STR register (SPI_I2S_GetBitState()).
|
||
|
* - UDR (UnderRun error) flag is cleared by a read operation to
|
||
|
* SPI_STR register (SPI_I2S_GetBitState()).
|
||
|
* - MODF (Mode Fault) flag is cleared by software sequence: a read/write
|
||
|
* operation to SPI_STR register (SPI_I2S_GetBitState()) followed by a
|
||
|
* write operation to SPI_CTLR1 register (SPI_Enable() to enable the SPI).
|
||
|
* @retval None
|
||
|
*/
|
||
|
void SPI_I2S_ClearIntBitState(SPI_TypeDef *SPIx, uint8_t SPI_I2S_INT)
|
||
|
{
|
||
|
uint16_t itpos = 0;
|
||
|
|
||
|
/* Clear the select interrupt pending bit. */
|
||
|
itpos = 0x01 << (SPI_I2S_INT & 0x0F);
|
||
|
SPIx->STR = (uint16_t)~itpos;
|
||
|
}
|
||
|
/**
|
||
|
* @}
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* @}
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* @}
|
||
|
*/
|
||
|
|