天天看点

STM32使用HAL库DMA+空闲中断实现串口不定长数据接收

环境:

  • STM32CubeIDE
  • STM32F103RB
  • WIN10
  • HAL库V1.8

首先配置串口:

STM32使用HAL库DMA+空闲中断实现串口不定长数据接收

开启串口中断:

STM32使用HAL库DMA+空闲中断实现串口不定长数据接收

开启DMA接收:

STM32使用HAL库DMA+空闲中断实现串口不定长数据接收

配置中断分组:

STM32使用HAL库DMA+空闲中断实现串口不定长数据接收

然后生成代码就可以了。

进入代码,添加串口1中断服务程序:

#define UART_RX_LEN 1024           // 一次最大接收的数据量
uint8_t UART_RX_BUF[UART_RX_LEN];  // DMA数据接收缓存
__IO uint16_t UART_RX_STA = 0;     // 第15bit表示一帧数据接收完成,第14~0位表示接收到的数据量

void USART1_IRQHandler(void)
{
	if(__HAL_UART_GET_FLAG(&huart1, UART_FLAG_IDLE) != RESET)  // 空闲中断标记被置位
	{
	    __HAL_UART_CLEAR_IDLEFLAG(&huart1);  // 清除中断标记
	    HAL_UART_DMAStop(&huart1);           // 停止DMA接收
	    UART_RX_STA = UART_RX_LEN - __HAL_DMA_GET_COUNTER(huart1.hdmarx);  // 总数据量减去未接收到的数据量为已经接收到的数据量
	    UART_RX_BUF[UART_RX_STA] = 0;  // 添加结束符
	    UART_RX_STA |= 0X8000;         // 标记接收结束
	    HAL_UART_Receive_DMA(&huart1, UART_RX_BUF, UART_RX_LEN);  // 重新启动DMA接收
	}
}
           

使能空闲中断和DMA接收:

HAL_UART_Receive_DMA(&huart1, UART_RX_BUF, UART_RX_LEN);  // 启动DMA接收
	__HAL_UART_ENABLE_IT(&huart1, UART_IT_IDLE);              // 使能空闲中断
           

while死循环中加入发回的代码:

if(UART_RX_STA & 0X8000)
{
	HAL_UART_Transmit(&huart1, UART_RX_BUF, UART_RX_STA & 0X7FFF, 100);    // 将接收到的数据发送回去
	UART_RX_STA = 0;  // 清除标记
}
           

完整代码如下:

/* USER CODE BEGIN 0 */

#define UART_RX_LEN 1024
uint8_t UART_RX_BUF[UART_RX_LEN];
__IO uint16_t UART_RX_STA = 0;

void USART1_IRQHandler(void)
{
	if(__HAL_UART_GET_FLAG(&huart1, UART_FLAG_IDLE) != RESET)  // 空闲中断标记被置位
	{
	    __HAL_UART_CLEAR_IDLEFLAG(&huart1);  // 清楚中断标记
	    HAL_UART_DMAStop(&huart1);           // 停止DMA接收
	    UART_RX_STA = UART_RX_LEN - __HAL_DMA_GET_COUNTER(huart1.hdmarx);  // 总数据量减去未接收到的数据量为已经接收到的数据量
	    UART_RX_BUF[UART_RX_STA] = 0;  // 添加结束符
	    UART_RX_STA |= 0X8000;         // 标记接收结束
	    HAL_UART_Receive_DMA(&huart1, UART_RX_BUF, UART_RX_LEN);  // 重新启动DMA接收
	}
}


int main(void)
{
	HAL_Init();
	SystemClock_Config();
	MX_GPIO_Init();
	MX_DMA_Init();
	MX_USART1_UART_Init();
	HAL_UART_Receive_DMA(&huart1, UART_RX_BUF, UART_RX_LEN);  // 启动DMA接收
	__HAL_UART_ENABLE_IT(&huart1, UART_IT_IDLE);              // 使能空闲中断
	while (1)
	{
		if(UART_RX_STA & 0X8000)
		{
			HAL_UART_Transmit(&huart1, UART_RX_BUF, UART_RX_STA & 0X7FFF, 100);    // 将接收到的数据发送回去
			UART_RX_STA = 0;
		}
	}
}

/* USER CODE END 0 */