STM32L4 串口通信(DMA+空闲中断方式)

芯片:STM32L452RE

 通过CubeMx生成工程文件,利用HAL库实现串口通信(DMA+空闲中断)任意长度的数据接收,

该程序不同于其它博客的写法,不用在主函数判断空闲中断再调用串口DMA接收函数,且解决第一次接收不到数据或数据不完整的情况。。

1.在STM32CubeMX里配置所需功能

1.1 时钟系统

 建议选择MSI作为时钟输入源,HSI反应有Bug(没有去尝试过,暂时省略)

 

1.2 设置串口中断和DMA

这里不做详细介绍,网上有大量教程可以参考

1.3 生成工程文件(keil5)

不熟悉的同学可以参考其它博客,有写得很详细的。

2. 程序编写

不用在主函数里调用DMA接收函数,且解决第一次接收不到数据或数据不完整的情况(不用环形缓冲区)。

2.1设计思路

首先开启串口中断接收函数,允许接收中断 HAL_UART_Receive_IT(&huart1, (uint8_t *)&Buftemp, 1);

当接收到第一个字节时产生中断(非空闲中断),此时的中断会调用HAL_UART_RxCpltCallback(),

在该函数里开启空闲中断和DMA接收,由于第一个中断已向缓冲区写入一个字节,此时的DMA接收区需要调整(若不设置,第一次接收的数据会丢失第一个字节);

在stm32l4xx_it.c里,编写中断服务程序。如果有空闲中断,编写空闲中断处理函数(清除标志位,停止DMA传输),再开启DMA接收

main.c

   主函数

unsigned char data[MAX_RCV_LEN];
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
 
  /* USER CODE END WHILE */

  /* USER CODE BEGIN 3 */
	printf("****start*****\n");
	HAL_Delay(1000);	
	GetRcvData(&huart1 ,data, sizeof(data));
	if(strlen(data)){
	printf("recv: %s \n",data);
	HAL_Delay(1000);
	}

  }
  /* USER CODE END 3 */

usart.h

/* Define to prevent recursive inclusion -------------------------------------*/
#ifndef __usart_H
#define __usart_H
#ifdef __cplusplus
 extern "C" {
#endif

/* Includes ------------------------------------------------------------------*/
#include "stm32l4xx_hal.h"
#include "main.h"

/* USER CODE BEGIN Includes */
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
/* USER CODE END Includes */

extern UART_HandleTypeDef huart1;
extern UART_HandleTypeDef huart2;
extern UART_HandleTypeDef huart3;

/* USER CODE BEGIN Private defines */
#define MAX_RCV_LEN 1024
extern uint8_t USART1RECV[MAX_RCV_LEN];	 //串口1
/* USER CODE END Private defines */

extern void _Error_Handler(char *, int);

void MX_USART1_UART_Init(void);
void MX_USART2_UART_Init(void);
void MX_USART3_UART_Init(void);

/* USER CODE BEGIN Prototypes */
/* 清空*/
void USART_Clear(UART_HandleTypeDef *huart);

extern void USART_IDLECallBack(void);

uint16_t GetRcvNum(UART_HandleTypeDef *huart);
extern void GetRcvData(UART_HandleTypeDef *huart, uint8_t *buf, uint16_t rcv_len);
void USART_Write(UART_HandleTypeDef *huart, uint8_t *Data, uint16_t len);
/* USER CODE END Prototypes */

#ifdef __cplusplus
}
#endif
#endif /*__ usart_H */

usart.c

/* USER CODE BEGIN 0 */
uint8_t Buftemp;

uint16_t usart1_recv_len;
uint8_t USART1RECV[MAX_RCV_LEN];
/* USER CODE END 0 */

 在串口1的初始化程序里,开启串口接收中断

  HAL_UART_Receive_IT(&huart1, (uint8_t *)&Buftemp, 1);

void MX_USART1_UART_Init(void)
{

  huart1.Instance = USART1;
  huart1.Init.BaudRate = 115200;
  huart1.Init.WordLength = UART_WORDLENGTH_8B;
  huart1.Init.StopBits = UART_STOPBITS_1;
  huart1.Init.Parity = UART_PARITY_NONE;
  huart1.Init.Mode = UART_MODE_TX_RX;
  huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE;
  huart1.Init.OverSampling = UART_OVERSAMPLING_16;
  huart1.Init.OneBitSampling = UART_ONE_BIT_SAMPLE_DISABLE;
  huart1.AdvancedInit.AdvFeatureInit = UART_ADVFEATURE_NO_INIT;
  if (HAL_UART_Init(&huart1) != HAL_OK)
  {
    _Error_Handler(__FILE__, __LINE__);
  }
	HAL_UART_Receive_IT(&huart1, (uint8_t *)&Buftemp, 1);//使能第一次中断
}

 由第一次中断调用 HAL_UART_RxCpltCallback()函数,再开启空闲中断和DMA接收

/* USER CODE BEGIN 1 */
void USART_Clear(UART_HandleTypeDef *huart)
{
	if(huart->Instance == USART1)
	{
		usart1_recv_len = 0;
		memset(USART1RECV, 0x0, sizeof(USART1RECV));
	}else if(huart->Instance == USART2)
	{
		
	}
}

void USART_IDLECallBack(void)
{
	unsigned int temp;
  __HAL_UART_CLEAR_IDLEFLAG(&huart1);//清除标志位
  temp = USART1->RDR;  //清除状态寄存器RDR
	temp = temp;
  HAL_UART_DMAStop(&huart1); //	
}
void GetRcvData(UART_HandleTypeDef *huart, uint8_t *buf, uint16_t rcv_len)
{ 
	if(huart->Instance == USART1)
	{	
		if(buf){
			memcpy(buf,USART1RECV, rcv_len);
		}
		USART_Clear(&huart1);
	}else if(huart->Instance == USART2){
		
	}
}

void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
	UNUSED(huart);
	if(huart->Instance == USART1)
	{//只有第一次中断会调用
		if(usart1_recv_len==0){
		USART1RECV[usart1_recv_len++]=Buftemp;//第一次中断的数据被写入USART1RECV[0]处
			__HAL_UART_CLEAR_IDLEFLAG(&huart1);//清除标志位
			__HAL_UART_ENABLE_IT(&huart1,UART_IT_IDLE);//使能空闲中断
			HAL_UART_Receive_DMA(&huart1, USART1RECV+1, MAX_RCV_LEN);	//设置第一次DMA接收缓冲区,
		}
	}
}

stm32l4xx_it.c 

判断是否为空闲中断,如果是,调用空闲中断处理函数USART_IDLECallBack(),如果需要处理串口收到的数据可以再该函数里进行;;

最后,开启DMA接收(不用调整接收缓冲区)

/**
* @brief This function handles USART1 global interrupt.
*/
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)
	{		
		USART_IDLECallBack();	
		HAL_UART_Receive_DMA(&huart1, USART1RECV, MAX_RCV_LEN);	
	}
			
  /* USER CODE END USART1_IRQn 1 */
}


  3 运行结果 

完美的情况

失败的情况

第一次的数据丢失第一个字节,后面的都正常

相关资源已上传

STM32L4系列 串口通信 空闲中断+DMA 实现任意长度的数据接收
包含两种方式 
1:不用在主函数调用DMA接收函数(推荐,非常实用)
2:许多教程的写法,需要在主函数调用判断空闲中断状态再调用DMA接收函数

https://download.csdn.net/download/sinat_37853238/10935731

stm32f103的串口DMA空闲中断接收功能是指通过DMA通道来实现串口接收,并通过空闲中断来触发数据接收完成的事件。下面是修复版的实现过程: 首先,要确保串口和DMA的时钟已经使能,并按照正确的引脚配置工作模式。然后,需要配置串口接收的参数,比如波特率、数据位、停止位、奇偶校验等。在这之前,还需要先初始化DMA的通道,并设置合适的传输方向和数据缓冲区。 接下来,在主函数中进行初始化操作。首先,要对串口进行初始化配置,调用相关库函数进行参数设置。然后,设置DMA的传输方向为从外设到存储器,并设置数据长度和数据缓冲区地址。 然后,要编写串口空闲中断的处理函数。当DMA传输完成时,触发空闲中断。在空闲中断处理函数中,需要判断是哪个串口触发的空闲中断。通过判断状态寄存器位来判断当前串口是否接收到了数据,并读取接收缓冲区的数据。在接收完数据后,通过设置DMA的数据长度和数据缓冲区地址来启动下一次接收。 最后,在主函数中,使能串口空闲中断,并启动DMA传输。等待空闲中断触发后,会自动执行空闲中断的处理函数。在空闲中断处理函数中,处理完数据后,再次启动DMA传输,实现连续的接收功能。 以上是stm32f103串口DMA空闲中断接收的修复版实现过程。通过合理的初始化配置和中断处理,可以保证串口接收的稳定性和可靠性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

我楚狂声

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值