STM32 CAN通信自定义数据包多帧连发乱序问题

场景:

can标准帧中每一帧只能传输8字节,而应用中传输一包的内容往往超过8字节,因此需要把一个包拆成多个帧发送,接收端才把收到的多帧重新组装成一个完整的包


问题描述

在一问一答的两块板间通信,多帧连发是能够按照顺序发送的。但是,在一个主板和多个从板之间轮询一问一答的通信中,偶尔出现持续一段时间或者长时间无法通信的情况,特别高帧率发送的情况下,几乎无法通信。

原因分析:

抓取can总线数据发现不同板之间的帧相互交叉乱序,导致接收到的包因为乱序无法还原。但是不能保证多帧连续发送的话,就会导致无法还原包。如图,7E和E7之间为一包,但其出现乱序

发送过程大致为

  1. 把包根据8字节拆分为多个帧
  2. 调用发送函数HAL_CAN_AddTxMessage塞第一帧进发送邮箱,id为本设备ID
  3. can自动从邮箱里面取出该帧发送
  4. 发送完毕触发邮箱发送完毕中断,在中断里面再调用HAL_CAN_AddTxMessage塞下一帧进发送邮箱,ID为0,以最高优先级占用CAN总线,循环直到最后一帧
    以上过程按道理很快就可以连续发送,但是就是会出现不连续的情况。因为采用的轮询方式,很难保证其他can总线的设备不也在同时在竞争can总线,在发送完毕进入中断塞数据进邮箱的空隙,尽管ID号是0,但是在竞争总线的时刻,还在中断里面塞数据进邮箱,并未参与can总线竞争,就会被其他的设备竞争掉总线了,待填充完邮箱,因为ID号为0,其又可以占用can进行发送,就出现了本设备的帧和其他设备的帧交叉了。

解决方案:

其原因就是发送中出现时间停顿,让其他can设备有了可乘之机,因此保证多帧之间在上一帧发送完毕立马竞争总线进入下一帧发送就可以保证该包是连续发送的。以下是我的改进:

  1. 开启邮箱按照填写顺序发送(而不是根据邮箱ID优先级发送)
  2. 一包多帧的数据持续塞满三个邮箱,确保多帧发送过程中没有出现三个邮箱都出现空的情况
    在裸机的时候,其可以连续高帧率发送不出现乱帧,但是开启FREERTOS后,还会出现乱帧的情况,其原因是can的发送中断被freertos管理,需要把use freertos function关掉,使用裸机的中断,我把其优先级设为1.

    但是!还是有但是!我的freertos任务多了以后,还是出现乱帧的情况,检查后发送,我的第一次填写邮箱是在代码里面进行的,也就是说我塞邮箱的过程中,freertos会打断我塞数据的过程,导致有概率不是连续塞数据进去邮箱的。如图,虽然我在代码调用的发送中断函数,但是其程序指针运行在psp下,而不是msp下,优先级就是普通的freerto任务,自然不能保证以高优先级连续塞数据进发送邮箱了。

    改进为如下,分包好后,通过HAL_NVIC_SetPendingIRQ追加一个发送中断,其不会执行发送过程就可以直接触发发送中断,这下子没其他东西在打扰塞满发送邮箱了吧!

总结:

经过测试,can总线每秒8000帧数据的情况下,有10%的错误率,检查后发现,顺序没乱,有些帧没能发送来就丢弃了,因为我设置的发送失败不重发。降低速率至6000,1%内的错误率,帧没发出现的数量大幅度下降,证明还是太快了。设置3000,0%错误率,为什么3000就是0%呢,看了一下can总线,发送我的每一包发送需要时间是1ms,一包大概3帧,也就是1s上限差不多是3000多帧。

相关推荐
秀秀更健康13 分钟前
STM32的程序下载不进去----VDDA悬空
stm32·单片机·嵌入式硬件
长安第一美人1 小时前
AI辅助下的嵌入式UI系统设计与实践(二)[代码阅读理解]
c++·嵌入式硬件·ui·显示屏·工业应用
我在人间贩卖青春2 小时前
DMA的应用
单片机·dma·gpdma
学嵌入式的小杨同学3 小时前
STM32 进阶封神之路(二十五):ESP8266 深度解析 —— 从 WiFi 通信原理到 AT 指令开发(底层逻辑 + 实战基础)
c++·vscode·stm32·单片机·嵌入式硬件·mcu·智能硬件
树爷只认钱3 小时前
ESP01S模块+串口底座 AT指令连接中移Onenet物联网全过程(第1篇)
单片机·嵌入式硬件·物联网·esp8266
学嵌入式的小杨同学4 小时前
STM32 进阶封神之路(二十六):ESP8266 实战全攻略 ——TCP 通信 + 数据上传 + 远程控制 + 透传模式(库函数 + 代码落地)
stm32·单片机·嵌入式硬件·mcu·硬件架构·硬件工程·智能硬件
Nice__J4 小时前
Mcu架构以及原理——7.寄存器编程与抽象
stm32·单片机·架构
嵌入式学习和实践4 小时前
当MCU遇上大模型:在单片机上实现AI对话的硬核玩法
人工智能·单片机·大模型
我不是程序猿儿4 小时前
【嵌入式】适合 STM32 初学者BootLoader 入门学习心得
linux·stm32·单片机·嵌入式硬件·学习
惶了个恐5 小时前
嵌入式硬件第五弹——ARM(1)
嵌入式硬件