STM32H7上使用CubeMX,HAL配置UART串口DMA的坑

本文记录了在STM32H7开发过程中遇到的串口DMA接收异常的问题,详细描述了从配置错误、内存访问限制、MPU设置到最终的解决方案。通过调整CubeMX配置,优化Cache策略,并使用串口空闲中断结合DMA实现高效数据接收。同时提到了中断接收的局限性和使用HAL_UARTEx_ReceiveToIdle_DMA函数的优点。
摘要由CSDN通过智能技术生成

近日调试一块H7的板子, 板上资源丰富,运行速度很快, 我移植了Lwip等应用,然后想简单打开一下串口, 使用DMA形式接收. 但是无论怎么设置都没有正常收发.

可以进入回调函数, 但是接收数组内容一直是0, 就像DMA没有搬运过来一样.

调试良久,终于通畅可以跑满串口带宽:
在这里插入图片描述
下面说我是怎么搞的:

  1. 配置UART, 再cubeMX中点选我想用得uart8 , 设置波特率.
    在这里插入图片描述
  2. DMA选项开启. RX要启用循环模式, 这样收到一次就会再启动一次.
    在这里插入图片描述
    ok,到这里cubeMX的配置就完了, 网上都是这么写的. 点击在CubeMX IDE 点击保存,就生成代码了.

然后我在代码端再加写一些代码:

  1. 在串口配置内, USER CODE 注释空间内添加 立刻启动接收. 接收1字节帮我DMA搬运去uart8_RXdata这个数组内. 在这里插入图片描述

下面写一个函数叫: HAL_UART_RxCpltCallback, 这个函数库函数其实已经声明好了, 就等着用户自己写逻辑, 每次串口接收完我指定的数目后就会自动进入.

进入这个回调函数后, 我就把uart8_Rxdata数据拿出来. 放去我逻辑缓冲区, 我在Freerots里面有个任务就是慢慢去解析这些数据的. 到这里没毛病.
在这里插入图片描述
下面就是把Ringbuffer内的东西打印出来.
在这里插入图片描述
到这整个过程就结束了. 可是结果很糟糕, 无论我发送什么字符, 都只收到0, 打印0.

开始查资料:

  1. H7 DMA 访问空间有要求, 无法访问DTCM 0x2000000区域的RAM, 如果编程环境设置了,我的变量再这个区域, DMA就搬运不了.

参考 : https://blog.csdn.net/qq_41544116/article/details/100155203
在这里插入图片描述

好, 我是CubeMX IDE, 是GCC环境, 查看LD文件, 查看MAP文件
bss段 指定是在 0x24000000区域, map显示我的uart8_RXdata也是在这个区域内, 所以没问题. DMA可以访问.
在这里插入图片描述

  1. 又折腾半天, 想起来配置LWIP的时候, 就是要配置MPU才能正常进行以太网收发. 由于H7内核达到480Mhz, CPU访问RAM都需要透过cache才能发挥性能. 所以H7芯片做了MPU这个部分来配置内存的访问策略. CPU访问SRAM 中间有CACHE的作用. 而DMA是直接操作SRAM空间. 所以要进行Cache策略配置.

在MPU设置中加入一个区块, 把整个0x2400000空间取消buffer.

在这里插入图片描述
生成代码, 串口收发正常了.

之前考虑过这个方面,使用了volatlie关键字, 使用ST提供的函数: SCB_InvalidateDCache_by_Addr((uint32_t *)uart8_Rxdata,2);
都不好使.

(完)

后记,在调试DMA发送的时候,需要在发送前,对DMA发送的数组做CACHE无效处理:

void send_by_DMA(UART_HandleTypeDef *huart,uint8_t *buf,int len,uint8_t *dma_buf)
{
	//检查串口是否忙
	while(huart->gState != HAL_UART_STATE_READY);
	
	// dma_buf数组必须使用 ALIGN_32BYTES修饰,进行32字节对齐。
	memcpy(dma_buf,buf,len);
	/*
		无效cache(从RAM获取真实值)+ 清除Cache(写入物理RAM),做一次同步
	*/
	SCB_CleanInvalidateDCache_by_Addr((uint32_t *)dma_buf, len);
	
	// 触发DMA发送
	HAL_UART_Transmit_DMA(huart,dma_buf,len);
}

后记2: 测试了:串口中断接受,缺点是占用cpu多,每个字节中断一次。 H7串口硬件 FIFO 。2-16字节接收是减少很多90%中断数量,但是如果发送端没有到规定字节数比如要求16,只发了15,会导致不进中断,只适合连续的数据流处理。

还是要使用IDLE空闲中断+接收DMA的方式,兼顾不定长数据+最少的中断。

配置也非常简单:

1.8以上的HAL库已经准备了完整的回调,不需要加任何配置代码。

1。 配置RX 的 DMA, 一切默认,普通模式即可。

在这里插入图片描述
2. 使用 HAL_UARTEx_ReceiveToIdle_DMA函数。 HAL 会帮我们配置好一切。

填入 准备好的 大 BUFFER, 以及最大字节数。注意最大字节数要大于2倍最长的数据,否则会异常。比如我们要最长一次接受 256字节,

准备512的buffer和512的接受长度。
在这里插入图片描述

  1. 编写 HAL_UARTEx_RxEventCallback

每当串口空闲,就会进入这里, Size 就是这次接受的长度,根据不同需要自己数据拷贝走,下一步处理即可。在这里插入图片描述

然后再调用 HAL_UARTEx_ReceiveToIdle_DMA 开启下一次接受。

串口空闲的定义是,串口硬件连续1字节没有数据,DMA也是全硬件处理。 可以节省接收途中MCU使用。

  • 18
    点赞
  • 70
    收藏
    觉得还不错? 一键收藏
  • 9
    评论
STM32F091中,可以通过使用HAL库提供的函数来配置UART使用DMA进行数据传输。以下是UART发送使用DMA配置步骤: 1. 开启DMA时钟 需要先开启DMA时钟,以DMA1为例: ```c __HAL_RCC_DMA1_CLK_ENABLE(); ``` 2. 配置DMA通道 使用HAL库提供的函数配置DMA通道,以DMA1通道2为例: ```c hdma_tx.Instance = DMA1_Channel2; hdma_tx.Init.Direction = DMA_MEMORY_TO_PERIPH; // 数据传输方向,从存储器到外设 hdma_tx.Init.PeriphInc = DMA_PINC_DISABLE; // 禁止外设地址自增 hdma_tx.Init.MemInc = DMA_MINC_ENABLE; // 允许存储器地址自增 hdma_tx.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE; // 外设数据宽度为8位 hdma_tx.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE; // 存储器数据宽度为8位 hdma_tx.Init.Mode = DMA_NORMAL; // DMA模式为普通模式 hdma_tx.Init.Priority = DMA_PRIORITY_HIGH; // DMA通道优先级为高 HAL_DMA_Init(&hdma_tx); ``` 3. 配置UART 使用HAL库提供的函数配置UART使用DMA进行数据发送,以USART1为例: ```c 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; huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE; huart1.Init.OverSampling = UART_OVERSAMPLING_16; HAL_UART_Init(&huart1); ``` 4. 启动DMA传输 使用HAL库提供的函数启动DMA传输,以发送数据为例: ```c HAL_UART_Transmit_DMA(&huart1, (uint8_t*)sendBuffer, sendSize); ``` 以上是UART发送使用DMA配置步骤,UART接收使用DMA配置步骤与此类似。需要注意的是,UART接收使用DMA时需要将数据传输方向配置为从外设到存储器。
评论 9
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值