STM32的HAL库开发系列 - 串口DMA接收
串口DMA接收函数:
HAL_StatusTypeDef HAL_UART_Receive_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)
串口空闲中断(IDLE):
当DMA串口接收开始后,DMA通道会不断的将发送来的数据转移到主存,那么问题来了,该如何判断串口接收是否完成从而及时关闭DMA通道?如何知道接收到数据的长度?答案便是使用串口空闲中断。
串口空闲中断,对应事件标志为IDLE。
检测到串口空闲线路时,该位由硬件置 1。如果USART_CR1
寄存器中 IDLEIE=1
,则会生成中断。
该位由软件序列清零(读入 USART_SR
寄存器,然后读入 USART_DR
寄存器)。
利用串口空闲中断,可以用如下流程实现DMA控制的任意长数据接收。流程如下:
1.开启串口DMA接收。
2.串口收到数据,DMA不断传输数据到存储buf。
3.一帧数据发送完毕,串口暂时空闲,触发串口空闲中断。
4.在中断服务函数中,可以计算刚才收到了多少个字节的数据。
5.解码存储buf,清除标志位,开始下一帧接收。
举例实现串口DMA不定长接收:
// 定义变量
uint8_t rx_buffer[100];//接收数组
uint8_t rx_len = 0; //接收到的数据长度
// 在main中开启IDLE中断以及串口DMA接收
__HAL_UART_ENABLE_IT(&huart1, UART_IT_IDLE);
HAL_UART_Receive_DMA(&huart1,rx_buffer,100);
// 串口中断服务函数
void USART1_IRQHandler(void)
{
/* USER CODE BEGIN USART1_IRQn 0 */
/* USER CODE END USART1_IRQn 0 */
HAL_UART_IRQHandler(&huart1);
/* USER CODE BEGIN USART1_IRQn 1 */
if (__HAL_UART_GET_FLAG(&huart1,UART_FLAG_IDLE) != RESET)// 通过IDLE标志位判断接收是否结束
{
__HAL_UART_CLEAR_IDLEFLAG(&huart1);//清除标志位
HAL_UART_DMAStop(&huart1);
rx_len = 100 - __HAL_DMA_GET_COUNTER(&hdma_usart1_rx); //计算出数据长度
HAL_UART_Transmit_DMA(&huart1, rx_buffer,rx_len);//将收到的数据发送出去
HAL_UART_Receive_DMA(&huart1,rx_buffer,100);//开启DMA接收,方便下一次接收数据
}
/* USER CODE END USART1_IRQn 1 */
}
// 串口中断接收回调函数
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
if (huart->Instance == USART1)
{
}
}