STM32 串口DMA接收数据(高效接收数据)

关于重定义

极度不推荐在使用DMA的时候按照传统的方式进行重定义!!!
非常简单,轮询方式整个CPU 在串口发送时处于等待状态,但是使用DMA时无法确保当前DMA已经传输完成。
有同学可能会认为可以通过判断DMA的传输标志位来进行等待,但如果这样的话就丧失了DMA的设计意图:

再次使用经过优化的重定义:

找到usart.c这个c文件并打开:
先在这个文件里面添加头文件:#include <stdarg.h>和#include <stdio.h>
然后再用户代码区添加( UartTxBuf[128]需要设置全局,否则可能会出错):
 

unsigned char UartTxBuf[128]; 
void Usart1Printf(const char *format,...)
{
	
	uint16_t len;
	va_list args;	
	va_start(args,format);
	len = vsnprintf((char*)UartTxBuf,sizeof(UartTxBuf)+1,(char*)format,args);
	va_end(args);
	HAL_UART_Transmit_DMA(&huart1, UartTxBuf, len);
}

上述函数可以将Usart1Printf里面的要发送的数据打包后通过DMA一次完成传输,十分方便高效!!!

此函数用法和printf一致!!!

int main(void)
{
  HAL_Init();
  SystemClock_Config();
  MX_GPIO_Init();
  MX_DMA_Init();
  MX_USART1_UART_Init();
  unsigned char s_buf[]="hello world\r\n";
  
  HAL_UART_Transmit_DMA(&huart1,s_buf,sizeof(s_buf));
  HAL_Delay(200);
  Usart1Printf("HELLO\r\n");
  HAL_Delay(200);
  Usart1Printf("HELLO%d\r\n",10);
  HAL_Delay(200);
  while (1)
  {

  }
}

效果如下: 

 

转载:(49条消息) STM32CUBEMX配置教程(九)STM32串口DMA收发数据_cubemx uart dma发送和接收_怡步晓心l的博客-CSDN博客 

 

 

使用STM32DMA(直接内存访问)功能可以实现串口数据的快速接收。下面是一个简单的示例代码: ```c #include "stm32f10x.h" #define UART_RX_DMA_CH DMA1_Channel5 #define UART_RX_DMA_FLAG_TC DMA1_FLAG_TC5 uint8_t uart_rx_buf[64]; void uart_init(void) { USART_InitTypeDef usart_init; // 串口配置 usart_init.USART_BaudRate = 115200; usart_init.USART_WordLength = USART_WordLength_8b; usart_init.USART_StopBits = USART_StopBits_1; usart_init.USART_Parity = USART_Parity_No; usart_init.USART_Mode = USART_Mode_Rx; usart_init.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_Init(USART1, &usart_init); // 使能DMA接收模式 USART_DMACmd(USART1, USART_DMAReq_Rx, ENABLE); DMA_Cmd(UART_RX_DMA_CH, DISABLE); DMA_DeInit(UART_RX_DMA_CH); DMA_InitTypeDef dma_init; dma_init.DMA_PeripheralBaseAddr = (uint32_t)&USART1->DR; dma_init.DMA_MemoryBaseAddr = (uint32_t)uart_rx_buf; dma_init.DMA_DIR = DMA_DIR_PeripheralSRC; dma_init.DMA_BufferSize = sizeof(uart_rx_buf); dma_init.DMA_PeripheralInc = DMA_PeripheralInc_Disable; dma_init.DMA_MemoryInc = DMA_MemoryInc_Enable; dma_init.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; dma_init.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; dma_init.DMA_Mode = DMA_Mode_Circular; dma_init.DMA_Priority = DMA_Priority_High; dma_init.DMA_M2M = DMA_M2M_Disable; DMA_Init(UART_RX_DMA_CH, &dma_init); DMA_Cmd(UART_RX_DMA_CH, ENABLE); } int main(void) { uart_init(); while (1) { // 检查DMA接收是否完成 if (DMA_GetFlagStatus(DMA1_FLAG_TC5)) { DMA_ClearFlag(DMA1_FLAG_TC5); // DMA接收完成,处理接收到的数据 for (int i = 0; i < sizeof(uart_rx_buf); i++) { // 处理接收到的数据 } // 重新使能DMA接收 DMA_Cmd(UART_RX_DMA_CH, DISABLE); DMA_SetCurrDataCounter(UART_RX_DMA_CH, sizeof(uart_rx_buf)); DMA_Cmd(UART_RX_DMA_CH, ENABLE); } } } ``` 上述代码中,我们使用了USART1作为串口,并使能了DMA接收模式。在主循环中,我们检查DMA接收是否完成,如果完成,则处理接收到的数据,并重新使能DMA接收。注意,这里我们使用了循环模式的DMA,即DMA接收缓冲区满后会重新从头开始接收数据。如果需要处理较长的数据包,建议使用双缓冲模式的DMA
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Believeziwo

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

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

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

打赏作者

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

抵扣说明:

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

余额充值