解决STM32G0B1 串口DMA 空闲中断接收数据中途失效问题

问题:使用STM32G0B1 串口3 接收GPS 串口数据,频率1HZ,波特率9600,GPS 数据一帧约760字节,GPS 刚启动前面数据接收正常,中途某个时刻开始串口接收数据失败,进入不了串口中断,重启MCU后能恢复,过段时间问题仍会复现;

尝试办法:0.首先监控串口GPS数据发送没停止,一直在发数据

                   1.怀疑是中断优先级过低,调整优先级后无效;

                   2.怀疑串口上下拉配置问题,改变配置后无效;

                   3.怀疑是串口1、串口2 干扰到UART3,屏蔽掉仍然无效;

                    4.怀疑是其他初始化代码和任务创建干扰,屏蔽掉仍无效;

                    5.怀疑是DMA 接收Buffer长度不够足够长,导致某次中断没有及时产生,串口数据累积,造成DMA 异常,将DMA 长度设置成4K,问题仍然复现;                

                    6.屏蔽掉串口3的DMA 中断,问题仍复现;

                    7.怀疑是DMA 产生了某种错误没有清除,在DMA 中断处理中增加一些标记清除,问题仍复现;

                    8.监控GPS 串口数据,发现用串口工具接收到9600的波特率数据,稀稀拉拉,上一帧还没结束,下一帧就来了,担心是这样没有有效的空闲中断,所以设置GPS 串口波特率为115200,后数据发送变得比较整齐,帧与帧之间间隔比较明显,但是问题仍然会出现;

                    9.担心之前的工程不够纯净,重新构建CubeMx工程,初始化串口和GPIO,测试仍有问题。(此时楼主已经很崩溃,已经搞了两天,220728晚上10点多回到发现挂测的样机问题又复现了,睡不着觉,11点多又爬起来想办法,感觉所有的方法都试过了,已经黔驴技穷了,不过仍然想到是否要参考ST 官方例程的串口接收方式,或许比自己写的好,查到官方DMA 空闲中断例程,其中函数:HAL_UARTEx_ReceiveToIdle_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size),燃起一丝希望;

                    10.第二天 改用HAL_UARTEx_ReceiveToIdle_DMA 测试,终于跑起来能进中断了,

能跑起来了,结果中午买饭回来挂测又失败。已经崩溃,饭都没胃口吃;

                     11.午休,换脑子,冷静下,感觉蛮干已经走入绝境;

                      12.下午从头捋了下:1.GPS 数据发送没问题;2.串口开始接收正常.3.后面数据不接收了,中断进入不了,应该还是软件哪里问题;重新审视了CubeMx 串口配置Parameter Setting,发现里面有个DMA on Rx Err 参数,默认Enable,网上查了下,同时找到Overrun DMA 串口跑死的文章,感觉有点类似,做了测试,果然问题出现是是进入了Overrun,针对这个增加错误处理和重新开DMA接收,问题得到解决;

参考:https://www.csdn.net/tags/MtTaEg5sNjEwNDQzLWJsb2cO0O0O.html

 参考2:STM32串口溢出错误Overrun使用不当导致的串口死机_lljss2020的博客-CSDN博客_串口overrun

          

 这个帖子可能遇到类似的问题:

  • 28
    点赞
  • 79
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
针对STM32G4HAL库串口空闲中断接收不定长数据的问题,可以按照以下步骤进行解决: 1. 开启串口空闲中断,即在初始化串口时,将USART_CR1_REG中的USART_CR1_IDLEIE位设置为1。 2. 在串口空闲中断中,通过读取USART_ISR_REG寄存器中的位USART_ISR_RXNE和USART_ISR_IDLE来判断是接收到数据还是空闲中断。 3. 如果是接收到数据,则读取USART_RDR_REG寄存器中的数据,并将其存储到缓冲区中。 4. 如果是空闲中断,则通过计算接收到的数据长度来确定数据的长度,并将其存储到缓冲区中。 5. 在数据长度达到预定长度时,可以通过回调函数或者其他方式来通知数据接收已经完成。 下面是一个示例代码: ```c uint8_t rx_buffer[100]; uint8_t rx_counter = 0; uint8_t rx_length = 0; uint8_t rx_flag = 0; void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) { if(huart->Instance == USARTx) { if(rx_flag == 0) { //接收到数据 if((__HAL_UART_GET_FLAG(huart, UART_FLAG_RXNE) != RESET) && (__HAL_UART_GET_FLAG(huart, UART_FLAG_IDLE) == RESET)) { rx_buffer[rx_counter++] = (uint8_t)(huart->Instance->RDR & 0x00FF); } //空闲中断 else if(__HAL_UART_GET_FLAG(huart, UART_FLAG_IDLE) != RESET) { rx_length = rx_counter; rx_counter = 0; rx_flag = 1; } } } } ``` 在上面的代码中,当接收到数据时,将数据存储到缓冲区中,并将计数器rx_counter加1。当空闲中断发生时,计算接收到的数据长度,并将其存储到rx_length中。在接收完成后,将rx_flag设置为1,表示数据接收已经完成。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值