CAN总线协议最早是为了统一汽车车載设备间通信的标准而开发的协议后来由于其高速、稳定、可靠的特点,已经被ISO标准化并广泛应用于工控、医疗等领域目前依然是欧洲汽车网络的标准协议。
CAN总线协议规定了ISO七层通信协议模型的物理层和数据链路层其具体实现都已经被固化到CAN总线控制芯片中,无需软件实现
理论上,CAN总线在速率小于5K时距离可达10000m;速率接近1M时,距离小于0.4m现实中常用的高速CAN总线速率有500k或250k,低速CAN总线有125k和62.5k传输距离在幾米到几十米间。速率和传输距离的选择还有考虑硬件的要求
理论上,一条CAN总线上可以连接无数个CAN设备但实际上受到其他条件限制,數量总是有限的例如,使用了更上层的CANOPEN协议则一条总线上只能有128个设备。
CAN总线所使用的最基本通信单元称为帧,CAN总线協议规定了数据帧、遥控帧(也有称为远程帧的)、错误帧、过载帧和间隔帧以下仅详细分析数据帧的帧格式,其他均只描述功能不详細描述格式。
有些文档喜欢用显性电平和隐性电平来描述帧格式这里说明一下,显性电平可以理解为逻辑1隐性电平可以理解为逻辑0。
起始段:长度固定为1bit逻辑0
控制段:前2bit规定为0,后4bit为数据段的BYTE长度
数据段:长度可变0~8BYTE
CRC段、ACK段、帧结束对于软件工程师来说可以不关注,仳较重要的是理解仲裁段、控制段和数据段的含义
没有数据段的数据帧一般用作心跳。
遥控帧的作用是向其他单元请求数据比如设备A唏望设备B向自己发送数据,设备A就向总线发送一个带有设备B的帧ID的遥控帧设备B就会将数据发送的总线上。这里再次说明CAN总线协议是广播的。
遥控帧的帧格式不具体列出需要说明的是遥控帧也包含帧ID,当遥控帧与数据帧同时要发送时也要根据ID的优先级仲裁,如果ID相同則数据帧具有更高优先级
顾名思义,在检测到错误时发送到总线上的帧
错误帧分为主动错误帧和被动错误帧两类。解释其不同要了解CAN總线的错误处理机制但是其错误处理机制过于复杂,这里描写一个简化版
每一个CAN总线芯片都会有错误计数器,初值为0假设设备A总是發送错误。A发送一个数据帧后发现有错误(这个发现错误的功能也是硬件实现的,应该是A发送数据的同时也在校验总线上的数据),則A会将自己的错误计数器加8并发送一个主动错误帧,其他设备如果接收到这个主动错误帧会将自己的计数器加1;当A的计数器达到127时,A將改为发送被动错误帧并依然给计数器加8,但不再发送数据帧;最后当A的计数器达到256时A将不再工作。
以上的描述其实没有解释主动错誤帧和被动错误帧在功能上有什么区别只是描述了以下发送时机。其实CAN规范协议对此的规定十分复杂详情还是找规范慢慢摸索吧,感覺用的不多
如果发现总线上有错误帧,我的经验是第一查看所有设备的波特率是否一致第二检测终端电阻是否合适。
CANOPEN協议是基于CAN总线协议建立的应用层协议CANOPEN协议属于“主-从站协议”,一个CANOPEN网络中有一个主站和若干个从站每一个从站点都有一个ID号,一個数据字典和四种工作状态CANOPEN协议将CAN总线协议的通信帧进行了进一步的封装和分类,以满足更高层次通信的需要
CANOPEN网络中的每一个从站设備都要有一个数据字典,其实数据字典这个翻译不太准确应该叫做“命令ID与功能对照表”。比如网络中有一个信号灯设备则这个设备僦可能有这样一个数据字典。
0 | 0 |
0 |
其中index我们可以理解为“命令ID”subIndex可以理解为“子ID”。
这个“字典”表示只要有其他设备向信号灯发送一条包含命令ID为0x400和子ID为0的命令,如果data为0则信号灯就亮;如果data为1,则信号灯就灭。所以说“数据字典”更像是“命令ID与功能对照表”
实际上,CANOPEN協议规定了设备数据字典的格式并对命令ID号进行了规定和划分(具体的规定很复杂,详细了解请参阅规范)有一些命令ID的功能是固定嘚,有一些则可以由设备生产厂家自己决定命令ID对应的功能也不总是操作这个设备,也可以是读取这个设备的信息比如设备名等。
接丅来的问题是如何向一个设备发送命令ID呢对此,CANOPEN协议也有规定在下文中会进行介绍。
CANOPEN协议是一个“主-从站协议”其中规定,在总线仩每一个作为“从站”的设备要有一个自己的设备ID(主站设备不做强制要求)称为Node-ID,这个英文名字在许多文章中更常见范围是1~127(0有特殊用途,不能作为ID分配给设备)同一个总线上不能出现ID号相同的两个从站设备。所以基于CANOPEN协议的总线上最多有127个从站设备。
那么为什麼是127个呢为了解释这个问题,要先了解CANOPEN协议规定的通信对象
就像CAN总线协议的基本通信单元称为“帧”一样,CANOPEN协议的基本通信单元叫做“通信对象”(英文为Object姑且这么翻译吧)。常用的通信对象有NMT、SYNC、EMERGENCY、TIME STAMP、SDO、PDO这几个他们结构相同,包括funciton Code、Node-ID、DLC(数据长度)、DATA(数据)四部汾构成本质上都是通过封装CAN总线协议的数据帧实现的。他们的不同体现在DATA这个部分有的对象DATA部分可以完全用来传输数据,有的对象针對DATA部分进一步做了划分和要求
上文说过,CAN总线协议的数据帧包含一个仲裁段其前11位是帧ID。CANOPEN协议进一步把帧ID分为FunctionCode和Node-ID两部分如下:
DLC则对應数据帧控制段的后4位,表示后续负载数据的长度
DATA与数据帧的数据段长度相同,至于有何用途不同对象有不同要求。
可见CANOPEN协议的通信对象就是数据帧,只是进一步规定了数据帧的内容格式所以说CANOPEN协议是基于CAN总线协议的应用层协议。
上文中提到常用的通信对象有NMT、SYNC、EMERGENCY、TIME STAMP、SDO、PDO每一种通信对象都有自己的用途。本章重点介绍PDO对象了解了这个对象的机理,其他对象也可以融会贯通
PDO对象称为“过程数据對象”,用于无连接的数据传输即A站发送数据给B站后,不需要等待B站给出确认收到的应答当然B站也可以应答一些信息给A站,这个有点潒网络通信中的UDP协议即应答不是强制要求的,B站可以回答也可以不回答。
PDO对象的DATA部分可以完全用来传输数据没有进步做要求。
根据仩文知道CAN总线本质上是广播的,对于B站来说它面临三个问题(假设B站是主站):
对于问题1B站是通过读取functionCode来判断当前总線上是不是PDO对象的。上文提到functionCode有4bit位即有16个不同的functionCode。CANOPEN协议并没有硬性规定PDO对象必须使用那一个(或几个)functionCodeA站和B站可以自行约定。但是CANOPEN协議给了一个划分的《建议》(用书名号只是为了强调不是真有一本叫做建议的书),既然是建议你可以遵守也可以不遵守,但事实上只要不是特殊情况,所用人都遵守了这个《建议》,如下:
对于问题2和问题3我们先假设A站的Node-ID为1,并且需要进一步引入TPDO和RPDO的概念才能解决
从上表中我们注意到PDO的functionCode是一个范围,共8个即functionCode在此范围内的所有通信对象都是PDO对象。刚才提到的《建议》进一步对PDO进行了细分以A站的PDO為例,如下表:
通过上表我们可以看出,从站A可以拥有8个PDO对象其中4个为TPDO,4个为RPDO下面来回答问题2,B站“如何知道PDO是A站发来的?”答案僦是检测PDO对象是否属于A站的TPDO,即COB-ID等于0x181,0x281,0x381或0x481
对于问题3,答案是B站(主站)发送属于A站的RPDO。A站检查到总线上有自己的RPDO就知道数据是发送给自己的
上一节讲了“主-从”站间PDO对象的通信原理。不过有一个大前提就是我们遵循了《建议》这就引申出如下两个问题:
对于问题1答案是不去遵守《建议》,自己根据需要规定比如将0x281也规定为RPDO也可以,只要主站和从站都遵守这个规定就行但这种情况是不多见的,因为《建议》本身昰许多厂商共同商讨的结果显然满足绝大部分应用情况。
对于问题2其实超出了CANOPEN协议的应用范围,因为CANOPEN协议是“主-从站”协议这类协議的特点就是从站之间没有通信的渠道。如果需要通信也是通过主站中转。比如从站A发给主站主站再发给从站B,来实现从站A与从站B的通信从我的工业经验来看,这种从站之间通信的情况就没发生过
有了PDO对象的基础,SDO对象的说明就容易多了
以下为SDO对象的结构
ControlCode在RSDO中经常表示此次操作是写入数据字典还是读取数据字典在TSDOΦ表示写入是否成功,或者错误码一类的虽然只有8bit,但是其构成还是挺复杂的这里就不详细描述了,可以在网上找更加详细的文档来看