STM32F407 串口使用DMA方式通信

DMA的原理,就是利用寄存器方式进行读写,这样的好处就是相对于中断触发(往往一个字节字节的就中断一次),CPU中断次数大大降少,提高了效率,但也影响了实时性。总体来说,对于一般的应用,瑕不掩瑜,值得使用。

本文是基于串口1的,实际上串口1也是printf重定向接口,貌似没有什么冲突。

原则上:

1. 串口接收采用DMA+空闲中断的方式

2. 串口发送就是直接发送方式

这样的方式,符合一般项目需求。

也分两部分:STM32CubeMx端配置+代码的处理

STM32CubeMx端配置

CubeMX 这边在原先“串口”配置基础上,配置DMA方式,总体来说,跟网上大部分例子差不多。

在USART界面下选择 DMA Settings,如下图:

Mode选择Normal,一般都是。还有一种循环发送,大概意思就是开启后自动循环发送?

第4步骤,根据实际情况下选择,貌似没啥影响

RX:接收端配置

TX:发送端配置

在NVIC Settings。确认下是否中断开启?

Parameters Settings 跟常规串口配置一样,主要是一些波特率参数之类的

还要配置下DMA,在 System Core地方选择DMA

最后一项MEMTOMEM,默认是没有的,可通过下面的Add添加上去

其他都跟往常一样。

代码侧改动

好几个地方要改动:

直接上代码:

usart.h

/* USER CODE BEGIN Private defines */

/* xxxx 的意思是 你自己取个好点名字*/

#define XXXX_BUFFER_SIZE  128 

extern volatile uint8_t g_xxxxRxLen; 			/*接收一帧数据的长度*/
extern volatile uint8_t g_xxxxRecvEndFlag; 		/*一帧数据接收完成标志*/
extern uint8_t g_xxxxRxbuffer[XXXX_BUFFER_SIZE]; /*接收数据缓存数组*/

/* USER CODE END Private defines */

usart.c

/* USER CODE BEGIN 0 */

volatile uint8_t g_xxxxRxLen = 0;
volatile uint8_t g_xxxxRecvEndFlag = 0; 
uint8_t g_xxxxRxbuffer[XXXX_BUFFER_SIZE] = {0};

/* USER CODE END 0 */


在
void MX_USART1_UART_Init(void)
后面追加

/* USER CODE BEGIN USART1_Init 2 */
  __HAL_UART_ENABLE_IT(&huart1, UART_IT_RXNE); 
  __HAL_UART_ENABLE_IT(&huart1, UART_IT_IDLE); 

  /*DMA接收函数,此句一定要加,不加接收不到第一次传进来的实数据,*/
  /*是空的,且此时接收到的数据长度为缓存器的数据长度*/
  HAL_UART_Receive_DMA(&huart1,g_xxxxRxbuffer,XXXX_BUFFER_SIZE);


  /* USER CODE END USART1_Init 2 */

stm32f4xx_it.c  增加以下内容(10x系列,就是f1xx)

void USART1_IRQHandler(void)
{
  /* USER CODE BEGIN USART1_IRQn 0 */
	
  uint32_t temp;
	
	if(__HAL_UART_GET_FLAG(&huart1, UART_FLAG_IDLE) != RESET)
	{ 
可能是点个灯之类操作
		__HAL_UART_CLEAR_IDLEFLAG(&huart1);
		
		HAL_UART_DMAStop(&huart1); 
/* 获取DMA中未传输的数据个数 */
		temp = __HAL_DMA_GET_COUNTER(&hdma_usart1_rx);   

/*总计数减去未传输的数据个数,得到已经接收的数据个数 */
/*  这里有风险的,BUFFER_SIZE 如果小于当前帧的话,就有问题*/

/* 正常情况下,要考虑下此情况 */
		g_xxxxRxLen = XXXX_BUFFER_SIZE - temp; 

		g_xxxxRecvEndFlag = 1;	/* 标志位置1	*/
	}

  /* USER CODE END USART1_IRQn 0 */
  HAL_UART_IRQHandler(&huart1);
  /* USER CODE BEGIN USART1_IRQn 1 */

  /* USER CODE END USART1_IRQn 1 */
}

应用侧,需要一个task去轮询(是否收到数据)

比如说,我是放在Task3里面

void StartTask03(void const * argument)
{
  /* USER CODE BEGIN StartTask03 */
  /* Infinite loop */

	uint8_t dmaSend[] = "this is DMA\n";
		
	for(;;)
	{
		if(g_xxxxRecvEndFlag == 1)
		{
            /* 此demo 简单处理,就应答一个固定内容,实际要根据实际情况处理*/
			DMA_Usart1_Send(dmaSend,sizeof(dmaSend)-1);		
            
           // 此处增加协议处理之类的。可以复杂
			
			g_xxxxRxLen = 0;
			g_xxxxRecvEndFlag = 0;
			
			memset(g_xxxxRxbuffer,0,g_xxxxRxLen);
		}
		
		HAL_UART_Receive_DMA(&huart1,g_xxxxRxbuffer,XXXX_BUFFER_SIZE);
		
		osDelay(1);
	}
  /* USER CODE END StartTask03 */
}
/*
*********************************************************************************************************
* 函 数 名: DMA_Usart_Send
* 功能说明: 串口发送功能函数
* 形  参: buf,len
* 返 回 值: 无
*********************************************************************************************************
*/
void DMA_Usart1_Send(uint8_t *buf,uint8_t len)
{
	while(HAL_DMA_GetState(&hdma_usart1_tx) == HAL_DMA_STATE_BUSY)  
	{
		osDelay(1);
	}
    
	/* 关闭DMA */
    __HAL_DMA_DISABLE(&hdma_usart1_tx);

	if(HAL_UART_Transmit_DMA(&huart1, buf, len) != HAL_OK)
	{
		Error_Handler();
	}
}



/*
*********************************************************************************************************
* 函 数 名: DMA_Usart1_Read
* 功能说明: 串口接收功能函数
* 形  参: Data,len
* 返 回 值: 无
*********************************************************************************************************
*/
void DMA_Usart1_Read(uint8_t *Data,uint8_t len)
{
	HAL_UART_Receive_DMA(&huart1,Data,len);
}

  • 2
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: stm32f407串口1dma发送的过程需要使用stm32f407芯片的内部dma控制器,将待发送的数据通过dma通道传输到串口1数据寄存器中,实现数据的自动发送使用dma传输数据的好处是可以减轻cpu的负担,同时提高数据传输的速度和稳定性。 在实现过程中,需要对串口1的初始化进行配置,包括波特率、数据位、校验位、停止位等参数。然后通过启用dma传输控制器,设置传输数据的起始地址和终止地址,以及传输数据的长度。在传输完成后,通过dma中断来触发发送完成的回调函数,可以实现发送完成后的操作,如清空缓存、关闭dma传输等。 需要注意的是,在使用dma传输数据时,需要预留足够的空间来存储数据,避免溢出。同时,对于长数据的发送,可以采用分包发送的方法,将数据分成多个小包进行发送,提高数据传输的成功率和稳定性。 总之,stm32f407串口1dma发送是一种高效、快速、稳定的数据传输方式,适用于需要频繁发送数据的应用场景,需要根据具体的实现需求进行配置和调试。 ### 回答2: STM32F407是一款强大的微控制器芯片,具有多种功能。其中,串口是一项非常重要的功能之一,可以用于与外设通信。利用DMA传输方式可以大大提高串口发送数据的效率。下面我们就来谈一下如何在STM32F407上使串口1实现DMA发送。 首先需要对DMA传输方式进行初始化配置。在HAL库中,可以使用HAL_UART_Transmit_DMA函数对串口DMA进行配置。 然后需要设置DMA的传输方向和传输长度。可以使用DMA_InitStructure结构体进行设置。在设置DMA传输方向时,应该选择DMA_MEMORY_TO_PERIPHERAL模式,表示数据从内存传输到外设。在设置DMA传输长度时,应该设置主存储区和DMA缓存区的地址以及数据的长度。 接下来需要设置自动请求值和DMA通道等参数。可以使用HAL_DMA_Init函数初始化并启动DMA传输。此时,DMA传输已经启动,可以等待传输完成。 完成DMA传输后,通过DMA的传输完成回调函数进行处理。在串口1DMA发送的应用中,可以发送下一次数据包或关闭DMA传输。这里需要注意的是,在完成DMA传输后,需要清除传输完成标志。 在STM32F407上使串口1实现DMA发送需要进行多项配置和设置。通过HAL库提供的函数可以更加方便地进行操作。当然,在实际应用中,还需要根据具体情况进行相应的修改。 ### 回答3: STM32F407串口1 DMA发送是一种高效的通信方式STM32F407开发板中,串口1的DMA数据发送是通过配置相应的寄存器,实现数据直接传输。该方法的优点在于,数据传输的速度相比中断方式和轮询方式更快,效率更高。同时,STM32F407 MCU也可以同时进行其他操作,不需要在发送数据时频繁地中断CPU。 实现STM32F407串口1 DMA发送,开发者需要按照以下步骤进行: 1. 配置串口1的GPIO引脚,并将串口1设置为DMA模式。 2. 配置DMA发送通道。 3. 配置串口的工作模式,包括停止位、校验位、波特率等。 4. 设置DMA传输的起始地址和传输的数据量。 5. 开启DMA传输和串口发送。 通过以上步骤的配置,即可实现STM32F407串口1 DMA发送。通过该方法,可以实现大量数据的快速传输,提高了数据传输的效率,同时还可以避免中断频率过高导致的CPU资源浪费。该方法在工业自动化、智能交通、智能家居等领域中得到了广泛的应用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值