在STM32F1中接收不定长数据都是使用空闲中断,STM32F0也支持这个中断,同样也可以用来接收不定长数据,不过F0系列的串口功能更加丰富,本次的话就主要介绍一下串口超时中断,实现和空闲中断同样的功能。
因为ST目前主推的是CubeMX + HAL + LL的结构,用习惯了SPL很难习惯HAL那种裹粽子的结构。LL(Low Layer)库的话,和SPL是差不多的,都是直观的操作寄存器,这个比较符合单片机的开发理念,所以就立马换上了LL库,但是LL目前Bug还是比较多,其次就是没有提供例程,使得所有的所有的功能都要自己码出来,不过你可以参考SPL的方式,实现起来都是大同小异的,只不过是换了套API而已。
- 硬件环境:STM32F051K6
- 软件环境:MDK5.25, CubeMX 5.0.1, Cube MCU Package for STM32F0 Series 1.9.0
CubeMX 配置
串口相关的部分只要打开串口的全局中断即可,着重于DMA配置,首先Add一个DMA Request为USART1_RX, 通道设置为DMA1 Channel 3,方向为外设到内存(本次是串口接收,所以是串口接收寄存器到RAM)优先级为中。
在生成代码之前还需要再设置一下,将所有的初始化放在独立的.c .h中,将生成的工程修改为MDK5,然后将所有的库都从HAL修改为LL,这边不在上图自行在软件中设置。
生成代码后有以下几处需要修改
LL_USART_EnableRxTimeout(USART1);
LL_USART_EnableIT_RTO(USART1);
LL_USART_EnableDMAReq_RX(USART1);
LL_DMA_EnableChannel(DMA1, LL_DMA_CHANNEL_3);
LL_USART_SetRxTimeout(USART1, 10); 设置超时时长,主要是设置USARTx_RTOR寄存器,其数值是以bit为单位,举例波特率是115200的话,那一个bit即为1/115200秒,如果设置为10就为(1/115200) * 10秒,时间的长度是跟随着波特率的变化而变化的,根据自己的需要来设置
LL_USART_EnableRxTimeout(USART1); 使能超时接收功能
LL_USART_EnableIT_RTO(USART1); 使能超时中断
LL_USART_EnableDMAReq_RX(USART1); 使能DMA接收
LL_DMA_EnableChannel(DMA1, LL_DMA_CHANNEL_3); 使能DMA通道
整个串口初始化代码
注意DMA在生成代码后还需要手动配置DMA的外设地址和内存地址。
void MX_USART1_UART_Init(void)
{
LL_USART_InitTypeDef USART_InitStruct = {0};
LL_GPIO_InitTypeDef GPIO_InitStruct = {0};
/* Peripheral clock enable */
LL_APB1_GRP2_EnableClock(LL_APB1_GRP2_PERIPH_USART1);
LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_GPIOA);
/**USART1 GPIO Configuration
PA9 ------> USART1_TX
PA10 ------> USART1_RX
*/
GPIO_InitStruct.Pin = LL_GPIO_PIN_9;
GPIO_InitStruct.Mode = LL_GPIO_MODE_ALTERNATE;
GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_HIGH;
GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_PUSHPULL;
GPIO_InitStruct.Pull = LL_GPIO_PULL_NO;
GPIO_InitStruct.Alternate = LL_GPIO_AF_1;
LL_GPIO_Init(GPIOA, &GPIO_InitStruct);
GPIO_InitStruct.Pin = LL_GPIO_PIN_10;
GPIO_InitStruct.Mode = LL_GPIO_MODE_ALTERNATE;
GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_HIGH;
GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_PUSHPULL;
GPIO_InitStruct.Pull = LL_GPIO_PULL_NO;
GPIO_InitStruct.Alternate = LL_GPIO_AF_1;
LL_GPIO_Init(GPIOA, &GPIO_InitStruct);
/* USART1 DMA Init */
LL_DMA_DeInit(DMA1, LL_DMA_CHANNEL_3);
/* USART1_RX Init */
//设置DMA外设基地址
LL_DMA_SetPeriphAddress(DMA1, LL_DMA_CHANNEL_3, (uint32_t)(&USART1->RDR));
//设置DMA内存基地址
LL_DMA_SetMemoryAddress(DMA1, LL_DMA_CHANNEL_3, (uint32_t)testBuf);
//设置DMA传输方向
LL_DMA_SetDataTransferDirection(DMA1, LL_DMA_CHANNEL_3, LL_DMA_DIRECTION_PERIPH_TO_MEMORY);
//设置DMA的优先级
LL_DMA_SetChannelPriorityLevel(DMA1, LL_DMA_CHANNEL_3, LL_DMA_PRIORITY_MEDIUM);
//设置DMA的模式
LL_DMA_SetMode(DMA1, LL_DMA_CHANNEL_3, LL_DMA_MODE_NORMAL);
//设置DMA缓冲区长度
LL_DMA_SetDataLength(DMA1, LL_DMA_CHANNEL_3, 100);
//设置DMA外设递增
LL_DMA_SetPeriphIncMode(DMA1, LL_DMA_CHANNEL_3, LL_DMA_PERIPH_NOINCREMENT);
//设置内存递增
LL_DMA_SetMemoryIncMode(DMA1, LL_DMA_CHANNEL_3, LL_DMA_MEMORY_INCREMENT);
//外设大小设置
LL_DMA_SetPeriphSize(DMA1, LL_DMA_CHANNEL_3, LL_DMA_PDATAALIGN_BYTE);
//内存大小设置
LL_DMA_SetMemorySize(DMA1, LL_DMA_CHANNEL_3, LL_DMA_MDATAALIGN_BYTE);
/* USART1 interrupt Init */
NVIC_SetPriority(USART1_IRQn, 0);
NVIC_EnableIRQ(USART1_IRQn);
USART_InitStruct.BaudRate = 115200;
USART_InitStruct.DataWidth = LL_USART_DATAWIDTH_8B;
USART_InitStruct.StopBits = LL_USART_STOPBITS_1;
USART_InitStruct.Parity = LL_USART_PARITY_NONE;
USART_InitStruct.TransferDirection = LL_USART_DIRECTION_TX_RX;
USART_InitStruct.HardwareFlowControl = LL_USART_HWCONTROL_NONE;
USART_InitStruct.OverSampling = LL_USART_OVERSAMPLING_16;
LL_USART_Init(USART1, &USART_InitStruct);
LL_USART_DisableIT_CTS(USART1);
LL_USART_DisableOverrunDetect(USART1);
LL_USART_ConfigAsyncMode(USART1);
LL_USART_Enable(USART1);
LL_USART_SetRxTimeout(USART1, 10);
LL_USART_EnableRxTimeout(USART1);
LL_USART_EnableIT_RTO(USART1);
//使能DMA接收
LL_USART_EnableDMAReq_RX(USART1);
//使能DMA通道
LL_DMA_EnableChannel(DMA1, LL_DMA_CHANNEL_3);
}
//DMA接收函数
void USART_DMA_Rx(DMA_TypeDef* DMAx,uint32_t DMA_CHx, uint8_t *pData, uint16_t Rcv_Len)
{
//失能DMA(设置DMA之前必须要关闭DMA)
LL_DMA_DisableChannel(DMAx, DMA_CHx);
//设置DMA接收缓冲区
LL_DMA_SetMemoryAddress(DMAx, DMA_CHx, (uint32_t)pData);
//设置DMA,通道,缓存区长度
LL_DMA_SetDataLength(DMAx, DMA_CHx, Rcv_Len);
//使能DMA
LL_DMA_EnableChannel(DMAx, DMA_CHx);
}
//串口中断函数
void USART1_IRQHandler(void)
{
if(LL_USART_IsActiveFlag_RTO(USART1))
{
LL_USART_ClearFlag_RTO(USART1);
Data_Cnt = LL_DMA_GetDataLength(DMA1, LL_DMA_CHANNEL_3);
// temp = LL_USART_ReceiveData8(USART1);
LL_USART_TransmitData8(USART1, Data_Cnt);
USART_DMA_Rx(DMA1, LL_DMA_CHANNEL_3, testBuf, 100);
}
}