串行外设接口(SPI)是一种同步串行数据通信接口,常用于 MCU 与外部设备之间进行同步串行通信。SPI是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用四根线,节约了芯片的管脚,同时为PCB的布局上节省空间。CW32L083 内部集成 2 个串行外设 SPI 接口,支持双向全双工、单线半双工和单工通信模式,可配置 MCU 作为 主机或从机,支持多主机通信模式,支持直接内存访问(DMA)。
单工通信模式
SPI 支持单工通信模式,主机和从机通过一根单向数据线进行单发或单收通信。主机使用 MOSI 信号线进行单发通信,使用 MISO 信号线进行单收通信;从机使用 MOSI 信号线进行单收通信, 使用 MISO 信号线进行单发通信,未使用的信号线可供其它功能使用。
主机单发,从机单收应用场景下,连接框图如下:
主机单发配置:
设置 SPIx_CR1.MODE 为 0x1,SPI 工作于单工单发通信模式;
设置 SPIx_CR1.MSTR 为 1,SPI 工作于主机模式。
设置 SPIx_SSI.SSI 为 0,在从机选择 CS 引脚输出低电平,作为通信起始信号。
当发送缓冲器为空时,即 SPIx_ISR.TXE 标志位为 1,将待发送的一帧数据写入 SPIx_DR 寄存器,数据在同步移位时钟信号的控制下从 MOSI 引脚输出。
当写入最后一帧数据后,必须等待发送缓冲空标志位 SPIx_ISR.TXE 变为 1,同时 SPI 总线忙标志位 SPIx_ISR. BUSY 变为 0,以确保数据发送完毕。然后设置 SPIx_SSI.SSI 为 1,使从机选择 CS 引脚输出高电平,结束本次通信。
从机单收配置:
设置 SPIx_CR1.MODE 为 0x2,SPI 工作于单工单收通信模式;
设置 SPIx_CR1.MSTR 为 0,SPI 工作于从机模式。
当检测到 CS 引脚变为低电平时,从机开始与主机通信。当接收缓冲器非空时,即 SPIx_ISR.RXNE 标志位为 1,表示已经接收完成一帧数据,此时可以读取 SPIx_DR 寄存器。当检测到 CS 引脚变为高电平时,本次通信结束。
主机单收,从机单发应用场景下,连接框图如下:
具体设置与主机单发和从机单收类似,详情可查看用户手册。
SPI中断
SPI 控制器支持 8 个中断源,当 SPI 中断触发事件发生时,中断标志位会被硬件置位,如果设置了对应的中断使能控制位,将产生中断请求。
在用户 SPI 中断服务程序中,应查询相关 SPI 中断标志位,以进行相应的处理,在退出中断服务程序之前,要清除该中断标志位,避免重复进入中断程序。
实例演示: SPI单工模式进行主从机通信,主机单发,从机单收。
SPIy(主机)采用中断方式发送TxBuffer缓冲区中的数据,SPIz(从机)采用中断方式接收数据并存储到RxBuffer缓冲区,比较TxBuffer和RxBuffer,如果数据一致LED1亮,否则LED2亮。
1. 配置RCC
void RCC_Configuration(void)
{
RCC_HSI_Enable(RCC_HSIOSC_DIV2); //SYSCLK = HSI = 24MHz = HCLK = PCLK
RCC_AHBPeriphClk_Enable(SPIy_GPIO_CLK | SPIz_GPIO_CLK | RCC_AHB_PERIPH_GPIOC, ENABLE); //外设时钟使能
SPIy_APBClkENx(SPIy_CLK, ENABLE);
SPIz_APBClkENx(SPIz_CLK, ENABLE);
}
2. 配置GPIO
void GPIO_Configuration(void)
{
GPIO_InitTypeDef GPIO_InitStructure = {0};
SPIy_AF_SCK; //SPI SCK MOSI 复用
SPIy_AF_MOSI;
SPIz_AF_SCK;
SPIz_AF_MOSI;
GPIO_InitStructure.Pins = SPIy_SCK_PIN; //推挽输出
GPIO_InitStructure.Mode = GPIO_MODE_OUTPUT_PP;
GPIO_Init(SPIy_GPIO, &GPIO_InitStructure);
GPIO_InitStructure.Pins = SPIy_MOSI_PIN;
GPIO_Init(SPIy_GPIO, &GPIO_InitStructure);
GPIO_InitStructure.Pins = SPIz_SCK_PIN; //浮空输入
GPIO_InitStructure.Mode = GPIO_MODE_INPUT;
GPIO_Init(SPIz_GPIO, &GPIO_InitStructure);
GPIO_InitStructure.Pins = SPIz_MOSI_PIN;
GPIO_Init(SPIz_GPIO, &GPIO_InitStructure);
GPIO_InitStructure.Pins = GPIO_PIN_3 | GPIO_PIN_2; //PC3 LED1 / PC2 LED2
GPIO_InitStructure.Mode = GPIO_MODE_OUTPUT_PP;
GPIO_Init(CW_GPIOC, &GPIO_InitStructure);
PC03_SETLOW();//LED灭
PC02_SETLOW();
}
3. 配置SPI
void SPI_Configuration()
{
SPI_InitTypeDef SPI_InitStructure = {0};
SPI_InitStructure.SPI_Direction = SPI_Direction_1Line_TxOnly; // 单工单发
SPI_InitStructure.SPI_Mode = SPI_Mode_Master; // 主机模式
SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b; // 帧数据长度为8bit
SPI_InitStructure.SPI_CPOL = SPI_CPOL_Low; // 时钟空闲电平为低
SPI_InitStructure.SPI_CPHA = SPI_CPHA_1Edge; // 第一个边沿采样
SPI_InitStructure.SPI_NSS = SPI_NSS_Soft; // 片选信号由SSI寄存器控制
SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_8; // 波特率为PCLK的8分频
SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB; // 最高有效位 MSB 收发在前
SPI_InitStructure.SPI_Speed = SPI_Speed_Low; // 低速SPI
SPI_Init(SPIy, &SPI_InitStructure);
SPI_InitStructure.SPI_Direction = SPI_Direction_1Line_RxOnly; // 单工单收
SPI_InitStructure.SPI_Mode = SPI_Mode_Slave; // 从机模式
SPI_Init(SPIz, &SPI_InitStructure);
SPI_Cmd(SPIy, ENABLE);
SPI_Cmd(SPIz, ENABLE);
}
4. 配置NVIC中断函数
void NVIC_Configuration(void)
{
NVIC_SetPriority(SPIy_IRQ, 1); //优先级,无优先级分组
NVIC_SetPriority(SPIz_IRQ, 0);
NVIC_EnableIRQ(SPIy_IRQ); //SPI中断使能
NVIC_EnableIRQ(SPIz_IRQ);
}
void SPI1_IRQHandler(void)//SPI1中断函数
{
if(SPI_GetITStatus(CW_SPI1, SPI_IT_TXE) != RESET)
{
if(TxCounter == BufferSize - 1)
{
SPI_ITConfig(CW_SPI1, SPI_IT_TXE, DISABLE);
}
SPI_SendData(CW_SPI1, TxBuffer[TxCounter++]);
}
}
void SPI2_IRQHandler(void) //SPI2中断函数
{
if(SPI_GetITStatus(CW_SPI2, SPI_IT_RXNE) != RESET)
{
RxBuffer[RxCounter++] = SPI_ReceiveData(CW_SPI2);
}
}
5. 比较两个buffers区
TestStatus Buffercmp(uint8_t* pBuffer1, uint8_t* pBuffer2, uint16_t BufferLength)
{
while(BufferLength--)
{
if(*pBuffer1 != *pBuffer2)
{
return FAILED;
}
pBuffer1++;
pBuffer2++;
}
return PASSED;
}
6. 主程序
int32_t main(void)
{
RCC_Configuration();//配置RCC
GPIO_Configuration();//配置GPIO
SPI_Configuration();//配置SPI
NVIC_Configuration();//配置NVIC
SPI_ITConfig(SPIz, SPI_IT_RXNE, ENABLE); //使能 SPIz RXNE 中断
SPI_NSSInternalSoftwareConfig(SPIz, SPI_NSSInternalSoft_Reset); //软件NSS,选中SPIz
SPI_ITConfig(SPIy, SPI_IT_TXE, ENABLE); //使能 SPIy TXE 中断
while(RxCounter < BufferSize); //等待收发完成
SPI_NSSInternalSoftwareConfig(SPIz, SPI_NSSInternalSoft_Set); //释放SPIz
TransferStatus = Buffercmp(TxBuffer, RxBuffer, BufferSize); //检查收发数据一致性
if(TransferStatus == PASSED) //PASSED
{
PC03_SETHIGH();//LED1亮
}
else //FAILED
{
PC02_SETHIGH();//LED2亮
}
while(1)
{
}
}
7.实验结果显示:LED1亮起,buffers区数据相等,SPI单工模式主从通讯(中断方式)功能实现。
全部评论