STM32CUBEMX(6)--移植雅特力AT32F403AVGT7,双串口通过DMA方式接收不定长数据

概述

本篇文章主要介绍如何使用STM32CubeMX移植到雅特力AT32F403AVGT7,并通过双串口输出打印。
直接存储器访问(DMA)用于在外设与存储器之间以及存储器与存储器之间提供高速数据传输。可以在无需任何CPU操作的情况下通过DMA快速传输传输。这样节省的CPU资源可供其它操作使用。

DMA允许在后台执行数据传输,无需Cortex-M3处理器干预。在此操作过程中,主处理器可以执行其它任务,仅当整个数据块需要处理时,才会中断主处理器。这样即使传输大量数据也不会对系统性能造成太大影响。

DMA主要用于为不同的外设模块实现集中数据缓冲存储(通常在系统SRAM中)。与分布式解决方案(其中每个外设都需要实现自己的本地数据存储)相比,DMA解决方案在硅片成本和功耗方面的成本较低。

根据使用的产品型号的不同,有一个或两个DMA模块。

本篇文章主要介绍如何使用STM32CubeMX实现串口DMA读取,并且打印出去。
在这里插入图片描述

硬件准备

首先需要准备一个开发板,这里我准备的是雅特力AT32F403AVGT7的开发板:
在这里插入图片描述

选择芯片型号

雅特力AT32F403AVGT7兼容STM32F103系列,故选取STM32f103VG进行开发。
在这里插入图片描述

配置时钟源

HSE与LSE分别为外部高速时钟和低速时钟,在本文中使用内置的时钟源,故都选择Disable选项,如下所示:在这里插入图片描述

配置时钟树

雅特力AT32F403AVGT7最高频率到240M,但是STM32F1的最高主频到72M,同时使用不使用外部晶振时候,最高速度只能到64M,所以配置64即可:
在这里插入图片描述

串口配置

本次实验使用的串口2,串口3进行串口通信,波特率配置为115200。
在这里插入图片描述
在这里插入图片描述

配置DMA

在DMA 设置中,Mode有两种模式,一种是普通模式(Normal),使用一次发送语句就发一次,另一种是循环模式(Circula),使用一次发送会一直发送。这里接收选择循环模式。
在这里插入图片描述
在这里插入图片描述

中断

在中断设置里打开串口2,3的中断。
在这里插入图片描述

生成工程设置

注意在生产工程设置中不能出现中文,不然会报错。
在这里插入图片描述

代码生成设置

最后设置生成独立的初始化文件
在这里插入图片描述

生成代码

在这里插入图片描述

代码编写

在main.c中,添加头文件,若不添加会出现 identifier “FILE” is undefined报错。

/* USER CODE BEGIN Includes */
#include "stdio.h"
/* USER CODE END Includes */

变量定义 ,在main.c中添加。

/* USER CODE BEGIN PV */
#define BUFFERSIZE_usart2 1000           //可以接收的最大字符个数       
uint8_t ReceiveBuff_usart2[BUFFERSIZE_usart2]; //接收缓冲区
uint8_t recv_end_flag_usart2 = 0,Rx_len_usart2;//接收完成中断标志,接收到字符长度

#define BUFFERSIZE_usart3 1000           //可以接收的最大字符个数       
uint8_t ReceiveBuff_usart3[BUFFERSIZE_usart3]; //接收缓冲区
uint8_t recv_end_flag_usart3 = 0,Rx_len_usart3;//接收完成中断标志,接收到字符长度
int f1_u1 = 0;
/* USER CODE END PV */

函数声明和串口重定向,这里使用串口3进行打印,在main.c中添加。

/* USER CODE BEGIN PFP */
void uart2_data(void);					//接收函数
void uart3_data(void);					//接收函数
#ifdef __GNUC__									//串口重定向
#define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
#else
#define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
#endif 
PUTCHAR_PROTOTYPE
{
    HAL_UART_Transmit(&huart3 , (uint8_t *)&ch, 1, 0xFFFF);
    return ch;
}
/* USER CODE END PFP */

开启串口IDLE中断,在main.c中添加。

  /* USER CODE BEGIN 2 */
  __HAL_UART_ENABLE_IT(&huart2, UART_IT_IDLE);//使能串口2 IDLE中断 
	__HAL_UART_ENABLE_IT(&huart3, UART_IT_IDLE);//使能串口2 IDLE中断 
		printf("双串口DMA接收例程\r\n");
  /* USER CODE END 2 *

串口DMA获取,在main.c中添加。

void uart2_data(void)
{
	if(recv_end_flag_usart2 ==1)//接收完成标志
	{
		printf("串口2数据长度=%d\r\n",Rx_len_usart2);//打印接收到的数据长度 
		printf("串口2数据内容:");
		for(int i=0;i<Rx_len_usart2;i++)
		{
			printf("%c",ReceiveBuff_usart2[i]);//向串口打印接收到的数据
            }
    printf("\r\n");          
    for(int i = 0; i < Rx_len_usart2 ; i++) //清空接收缓存区
    ReceiveBuff_usart2[i]=0;//置0
    Rx_len_usart2=0;//接收数据长度清零
    recv_end_flag_usart2=0;//接收标志位清零
    }
			    //开启下一次接收
    HAL_UART_Receive_DMA(&huart2,(uint8_t*)ReceiveBuff_usart2,BUFFERSIZE_usart2);
}
void uart3_data(void)
{
	if(recv_end_flag_usart3 ==1)//接收完成标志
	{
		printf("串口3数据长度=%d\r\n",Rx_len_usart3);//打印接收到的数据长度 
		printf("串口3数据内容:");
		for(int i=0;i<Rx_len_usart3;i++)
		{
			printf("%c",ReceiveBuff_usart3[i]);//向串口打印接收到的数据
            }
    printf("\r\n");          
    for(int i = 0; i < Rx_len_usart3 ; i++) //清空接收缓存区
    ReceiveBuff_usart3[i]=0;//置0
    Rx_len_usart3=0;//接收数据长度清零
    recv_end_flag_usart3=0;//接收标志位清零
    }
			    //开启下一次接收
    HAL_UART_Receive_DMA(&huart3,(uint8_t*)ReceiveBuff_usart3,BUFFERSIZE_usart3);
}

主循环,在main.c中添加。

  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */
		uart2_data();
		uart3_data();
    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 *

中断外部变量引用,在stm32f0xx_it.c中添加。

/* USER CODE BEGIN 0 */
#define BUFFERSIZE_usart2 255	//可接收的最大数据量
extern uint8_t recv_end_flag_usart2,Rx_len_usart2;


#define BUFFERSIZE_usart3 255	//可接收的最大数据量
extern uint8_t recv_end_flag_usart3,Rx_len_usart3;
/* USER CODE END 0 */

串口2,3中断函数,在stm32f0xx_it.c中添加。

/**
  * @brief This function handles USART2 global interrupt.
  */
void USART2_IRQHandler(void)
{
  /* USER CODE BEGIN USART2_IRQn 0 */

  /* USER CODE END USART2_IRQn 0 */
  HAL_UART_IRQHandler(&huart2);
  /* USER CODE BEGIN USART2_IRQn 1 */
		uint32_t temp_usart2;
	if(USART2 == huart2.Instance)//判断是否为串口2中断

	{      
		if(RESET != __HAL_UART_GET_FLAG(&huart2,UART_FLAG_IDLE))//如果为串口2
		{
			__HAL_UART_CLEAR_IDLEFLAG(&huart2);//清除中断标志
      HAL_UART_DMAStop(&huart2);//停止DMA接收
     		temp_usart2  = __HAL_DMA_GET_COUNTER(&hdma_usart2_rx);//获取DMA当前还有多少未填充
      		Rx_len_usart2 =  BUFFERSIZE_usart2 - temp_usart2; //计算串口接收到的数据个数
      		recv_end_flag_usart2 = 1;
    	 }
        }
  /* USER CODE END USART2_IRQn 1 */
}

/**
  * @brief This function handles USART3 global interrupt.
  */
void USART3_IRQHandler(void)
{
  /* USER CODE BEGIN USART3_IRQn 0 */

  /* USER CODE END USART3_IRQn 0 */
  HAL_UART_IRQHandler(&huart3);
  /* USER CODE BEGIN USART3_IRQn 1 */
	uint32_t temp_usart3;
	if(USART3 == huart3.Instance)//判断是否为串口3中断

	{      
		if(RESET != __HAL_UART_GET_FLAG(&huart3,UART_FLAG_IDLE))//如果为串口3
		{
			__HAL_UART_CLEAR_IDLEFLAG(&huart3);//清除中断标志
      HAL_UART_DMAStop(&huart3);//停止DMA接收
     		temp_usart3  = __HAL_DMA_GET_COUNTER(&hdma_usart3_rx);//获取DMA当前还有多少未填充
      		Rx_len_usart3 =  BUFFERSIZE_usart3 - temp_usart3; //计算串口接收到的数据个数
      		recv_end_flag_usart3 = 1;
    	 }
        }
  /* USER CODE END USART3_IRQn 1 */
}

结果演示

开机会显示双串口DMA接收例程。
串口2发送数据会在串口3中打印,但是会显示接收的是串口2。
串口3发送数据会在串口3中打印,但是会显示接收的是串口3。
在这里插入图片描述

视频教学

https://www.bilibili.com/video/BV1ot4y1r7sW

最后

以上的代码会在Q群里分享。QQ群:615061293。
或者关注微信公众号『记贴』,持续更新文章和学习资料,可加作者的微信交流学习!
在这里插入图片描述

posted @ 2022-05-28 22:04  记帖  阅读(126)  评论(0编辑  收藏  举报