在CANopen通讯开发中,我们经常遇到这样的需求:当TPDO映射的对象字典(OD)数值发生变化时,从站需要即刻发送该PDO数据。
本文将结合CiA301标准,详细解析“事件触发(Event-driven)”模式的配置细节,并演示如何通过应用层代码(回调函数)实现这一功能。

默认情况下,PDO可能被配置为同步(Sync)模式。但对于状态改变、报警等关键数据,我们需要它具有“实时性”,即:值变即发,无需等待SYNC帧。这就涉及到了PDO的传输类型(Transmission Type)配置。
查阅CiA301标准或相关资料,我们可以看到Sub-index 02的定义如下表:

根据功能需求,由于无需同步周期触发或远程帧触发,我们将研究重心放在“事件驱动(Event-driven)”模式上。基于此,首先需要了解什么是事件触发。
参阅CiA301文档中7.2.2.3 Triggering modes章节所述,协议将PDO的传输触发分为三大类:
我们着重看Event-and timer-driven的解释:

在参考CiA相关协议文档时,并未发现“PDO映射数值变化即自动触发传输”的直接定义。 换言之,仅配置Transmission Type(传输类型)尚不足以实现该功能。
若需将对象字典(OD)的变化定义为触发事件,通常属于“用户设定”范畴,需要通过应用层代码配合实现。
在CANopen协议中,254与255均属于“事件驱动型(Event-driven)”,但其应用场景存在差异:
在确定将Transmission Type设置为254后,需同步核查PDO的约束参数,以确保通信的稳定性。特别是在高频触发场景下,抑制时间(Inhibit Time)的设置对于防止总线拥塞至关重要。
至此,对象字典(OD)层面的配置已准备就绪。

配置完OD后,我们需要在代码层面实现“监控数据变化”并“请求发送”的逻辑。
注册一个 Object Dictionary Callback(对象字典回调函数)。
监控目标对象(例如 0x2000:01)的写入操作。
当检测到值发生变化时,调用协议栈的发送函数(如coPdoReqNr或类似 API)。
假设TPDO 1映射的值为0x2000:01。
我们在初始化阶段注册回调,监控索引2000,子索引01的变化:

在回调函数中,判断变化的对象,并手动触发PDO发送请求。

TPDO 1映射对象:0x2000:01
应用逻辑:在Demo中设置一个5s定时器,每隔5秒让0x2000:01自增。
Demo逻辑代码:



通过CAN分析仪抓包可以看到:

当应用层逻辑修改0x2000:01的数值时,回调函数被触发。
coPdoReqNr(1)被执行。
总线上即刻观测到TPDO 1的报文发出,且数据已更新。
通过Transmission Type = 254配合Inhibit Time以及应用层回调机制,我们成功实现了CANopen PDO的事件驱动发送功能。
全部0条评论
快来发表一下你的评论吧 !