一、CAN总线协议解析
9 f# |: [( ?! G1 ?7 S+ X5 i6 N
; A( e- Z- t( T. K4 z) u+ }* g
1 W- n9 D8 u: q7 b6 T+ x在讲解实现 CAN 总线的实例以前,读者需要具备有关 CAN 总线的基本知识。为此,将在这里简要介绍与实例相关的基础知识。
k8 E3 l" g/ d6 P' _6 c8 u4 ~
1.1 CAN 总线通信模型
8 ]" i3 i6 d8 L4 z' R参照 ISO/OSI 标准模型,CAN 总线的通信参考模型如图 1 所示。
% n3 j. z! J1 u3 r. D% [6 W
% D" V2 B1 N7 g% W
# H% G: i. {# M! o8 N# }2 ]' e
图 1 CAN 总线通信模型1 W* E: ?3 q2 u$ p+ d( Z' S, V- K
这 4 层结构的功能如下:
: O+ v; s2 d8 K6 S" H5 L: x0 Q
• 物理层规定了节点的全部电气特性,在一个网络里,要实现不同节点间的数据传输,所有节点的物理层必须是相同的。
• 传输层描述了 CAN 总线协议的内核,它负责位时序(bit timing)、同步、仲裁、应答、错误探测等。
• 对象层负责报文的过滤、状态和控制。
• 应用层完成用户指定的数据传输任务。
# Y0 V5 r1 w# C3 v
9 E1 T& s) F [/ D6 K: @( hCAN 总线的物理层为数据通信提供了物理连接,而实际的数据通信在其他 3 层中完成。
$ h' r' C8 b5 T
1.2 CAN 总线协议中的基本概念
1 u8 X; U# ~1 f7 m5 \5 z; Z
在讲解 CAN 总线协议之前,需要介绍有关协议中的基本概念。
+ F# h' U* n% i: d% b# L; K( R
1.报文(Messages)
2 e2 b# s# w( u6 L4 f9 i
在 CAN 总线传输的数据具有固定的格式和有限的长度,称为报文。
$ D1 y \1 {. |% Q
2.发送器(Transmitter)和接收器(Receiver)
( y- g: x3 u: H1 F. F; O
在 CAN 总线的数据传输过程中,发出报文的节点称为发送器。节点在总线进入空闲状态前或丢失仲裁前为发送器。如果一个节点不是报文发送器,并且总线不处于空闲状态,则该节点为接收器。
+ @. G2 m$ ^0 [2 I2 K) Q9 O
3.比特率(bit rate)
4 K# j* R9 L! l+ C0 Z7 @ v, Q; ^; E
CAN 总线的输出速度以单位时间内传输的位来衡量,称为比特率。CAN 总线在不同的系统中可以有不同的比特率。但是在给定的系统中,比特率是统一的和固定的。
& ^/ a% v6 w. I" \% D9 s& V. h
4.优先级(Priorities)
1 ?* v+ ?' q# c( ]" s. n
优先级表示总线传输中一个报文的优先级别。
( `' c3 E- _% e: r
5.远程数据请求(Remote Data Request)
. F- A* h: E1 c
当一个节点向另一个节点请求数据时,需要首先发送一个远程帧(Remote Frame),然后发送一个和远程帧相符的数据帧(Data Frame)。远程帧和数据帧具有相同的标识符。
5 |: t' H- m3 Y: ^( M0 R6 F
6.位流(Bit Stream)
5 W$ j8 S# y/ r+ W9 P; I9 k
CAN 总线通信过程中的数据流。
* _9 T8 _4 [4 p7 n" `8 ?
7.编码方式CAN 总线通信协议规定,报文中的位流按照非归零(Non-Return to Zero)码的方法编码,一个完整的电平要么是显性,要么是隐性。
u/ G) F; J" B- M0 Z+ t# p
8.非归零编码(Non-Return to Zero encoding,简称 NRZ)
" z- t& u. M1 c5 j
: Y, H) T6 i1 I9 y非归零编码是一种用在低速通信接口中的编码方式,同时提供同步和非同步两种方式。在非归零编码方式中,逻辑“1”在传输过程中用一位高电平表示,逻辑“0”用一位低电平表示。非归零编码方式如图 2 所示。
$ a/ C4 B8 D% ]1 c5 p# u
) m4 S/ G# z0 A- l& k0 Z8 u8 S
% h9 n" d: C" [/ e7 L# L
图 2 非归零编码方式7 A5 M( O8 H2 j9 d
9.总线数值
! m! ^( l; y+ ` v: G
在数据传输时,CAN 总线有两种逻辑值:显性值(dominant)和隐性值(recessive)。如果同时传输显性值和隐性值时,总线上的最终结果是显性值。在线与(wired-AND)总线连接方式中,显性值用逻辑“0”表示,隐性值用逻辑“1”表示。
. U3 Q* C- n0 u% m L& h
! a9 Q1 ^1 ^* T( {5 h
1.3 报文的数据结构( g' S! H9 e7 F: _ `. N) ?
2 e; ~& t2 P" z) |CAN 总线的报文传输是通过 4 种不同类型的帧(Frame)来表示和控制的:
$ Z0 e/ A; q: ^- P2 t
• 数据帧(Data Frame) 用来在数据传输过程中携带数据。
• 远程帧(Remoter Frame) 接收器发送远程帧来请求发送器发送数据,具有和数据帧同样的标识符。
• 出错帧(Error Frame) 用来检测 CAN 总线数据传输过程中的错误。
• 超载帧(Overload Frame) 用于提供当前和后续数据帧或远程帧之间的附加延迟。
: B% c5 c& i; g2 d' f
a.数据帧
! d) d! E8 L- W2 V' T7 r' L2 J% U数据帧的具体组成如图 3 所示。
8 u. ^* }4 j' A4 c! h
! I, F( ^6 L" T0 N8 h7 E3 }2 h+ h& o- D {3 _7 J0 |8 u& T
- F( v+ H4 n% W+ K
0 K7 S/ L) W4 s4 k, R4 Y+ b图 3 数据帧的具体组成
& {5 ?& H/ ]4 P. o: v6 U; Y* @5 f7 v数据帧由帧起始(Start of Frame,SOF)、仲裁字段、控制字段、数据字段、CRC 字段、ACK 字段(应答字段)和帧尾组成。
' x, d+ a, a; t: K/ {帧起始标志数据帧的开始(远程帧同样具有帧起始),它仅由一个显性值组成。只有在总线处于空闲时,才允许节点开始发送。所有节点必须与首先开始发送的那个节点的帧起始位前沿同步。
S" f! g( U3 r仲裁字段由标识符和远程发送请求位(RTR 位)组成,如图 4 所示。标识符的长度为 11位。远程发送请求位在数据帧中必须是显性值,在远程帧中必须是隐性值。
, C; O/ A8 m2 V8 @1 W. r: n- K* t
' F( r6 F& ]; Y4 D" l# A1 M4 E
' m4 o: I6 T$ \( h' W' ?
2 {1 I6 N/ }" o图 4 仲裁字段的组成
9 t G' G* |0 o! Q1 V
! ^. x% u# b5 l9 y" {
控制字段由保留位和数据长度码组成,如图 5 所示。数据长度码表示数据字段的长度。
# ^5 Q8 c' o+ Z8 L6 L
# _9 T3 |/ ~# z! P
+ P2 C: f0 l/ k5 q7 P0 K图 5 控制字段的组成
1 |; |' c& n# e p2 [, ]1 B8 ?) O6 E
数据字段由数据帧中被发送的数据组成,它可以包括 0~8 个字节,每个字节 8 位。首先发送的是最高有效位。
( o# _% o( e7 N: u$ |5 m
CRC 字段包括 CRC 序列和 CRC 界定符。CRC 序列用来实现 CRC 计算,CRC 界定符只包括一个隐性值。应答字段为两位,包括应答间隙和应答界定符。帧尾由 7 个连续的隐性值组成,作为数据帧和远程帧的结束标志。
$ t. ]4 v( ~6 c! r2 `0 s. }& v3 t
b.远程帧
) f, N+ X: s* S+ w q/ g
3 V( S! v& d9 |3 a- i9 [
作为接收器的节点可以通过向相应的数据源节点发送一个远程帧来激活该节点,让它把数据发送给接收器。远程帧由帧起始、仲裁字段、控制字段、CRC 字段、应答字段和帧尾 6 个不同的字段组成。远程帧的组成如图 6 所示。
1 D2 x6 P4 B! S
) K* k) r! H" x1 N! Y
! A0 {' T" G/ }) ?/ t图 6 远程帧的组成
5 d6 ]- \# t. ?# y
7 P! h& B2 i3 v3 ?' ?) y! Ac.出错帧
5 t5 m" @) P) h
出错帧由出错叠加标志和错误界定符组成。出错叠加标志包括了多个出错信息的标志。
& E5 t* v% y' z) bd.超载帧
% T2 O8 W9 I" H/ U超载帧包括超载标志和超载界定符。超载发生在两种情况下:一个是接收器因内部条件要求推迟下一个数据帧或者远程帧的发送;另一个是在间歇字段检测到显性值时。
& ^# _1 i5 n8 O. I) Z: n8 |; g
e.帧间空间
5 ?1 v# H/ H4 q/ ]$ e5 Q- {
数据帧或远程帧通过帧间空间与前一帧隔开,而不管前一帧是何种类型的帧。而在超载帧与出错帧前面不需要帧间空间,多个超载帧之间也不需要帧间空间来作分隔。
% g+ x' z1 R4 q$ u: o' ` s( n, ?
1.4 位时序(Bit Timing)& t* z4 x* F2 e
5 @& H3 V; H1 [& z. L$ v' ]* C: o
CAN 总线协议规定,报文传输的同步或者非同步方式的选择通过位时序来实现。CAN 总线中位时序包括正常位速率和正常位时间两个参数。
6 v2 x" r+ B' w
• 正常位速率(Nominal Bit Rate):在非重同步情况下,借助理想发送器每秒发送的位数。
8 R# A( x& I3 ?• 正常位时间(Nominal Bit Time):正常位速率的倒数。正常位时间由几个不同的时间段组成,它们是同步段(SYNC_SEG)、传播段(PROP_SEG)、相位缓冲段 1(PHASE_SEG1)、相位缓冲段 2(PHASE_SEG2),如图 7 所示。
, Q7 J8 C6 ~4 V: c- g/ q6 |5 I) n3 v图 7 正常位时间的组成
9 d' R# E" k2 H S2 f• 同步段:在这段时间内,完成总线上各个节点的同步,需要一个跳变沿。
• 传播段:这个时间段是指网络上传输的延迟时间,它是信号在总线上传播时间、输入比较器延迟和输出驱动器延迟之和的两倍。
• 相位缓冲段 1 和相位缓冲段 2:它们用于弥补跳变沿的相位误差造成的影响。通过重同步,这两个时间段可以被延长或缩短。
5 e* E6 }, L! [* `! t( S1 c4 T• 采样点:这是读取总线电平并理解该位数值的时刻,它位于相位缓冲段 1 的终点。
: ]( {3 F/ m4 [" s4 J1.5 同步(Synchronization)
5 e8 z8 w- D, l; r/ s" w
a.硬同步(Hard Synchronization)
! V+ v# x' v- D A6 V& o
硬同步后,内部位时间从同步段(SYNC_SEG)重新开始,它迫使触发该硬同步的跳变沿处于新的位时间的同步段(SYNC_SEG)之内。
% F, M5 M$ A0 k+ J
b.重同步(Resynchronization)
6 a( ^0 v' o, K7 \4 C( `) ^" T当引起重同步沿的相位误差小于或等于重同步跳转宽度编程值时,重同步的作用和硬同步相同。若相位误差大于重同步跳转宽度且相位误差为正时,则相位缓冲段 1(PHASE_SEG1)延长总数为重同步跳转宽度。若相位误差大于重同步跳转宽度且相位误差为负时,则相位缓冲段2(PHASE_SEG2)缩短总数为重同步跳转宽度。
' l1 O* M8 S' ~% c
c.重同步跳转宽度(Resynchronization Jump Width)
* B! E3 j5 U' ?$ d2 Q
由于重同步的结果,PHASE_SEG1 可被延长或 PHASE_SEG2 可被缩短。相位缓冲段长度的改变量不应大于重同步跳转宽度。
3 O! w9 ~9 v8 P3 P0 U L9 jd.同步的规则CAN
. E2 ?8 n3 ~; J! q0 Q通信协议规定,同步包括硬同步和重同步两种形式。它们遵从下列几条规则:
{4 C( n1 P6 A- 在一个位时间内仅允许一种同步。
- 对于一个跳变沿,仅当它前面的第一个采样点数值与紧跟该跳变沿之后的总线值不相同时,才把该跳变沿用于同步。
- 在总线空闲期间,若出现一个从隐性值到显性值的跳变沿,则执行一次硬同步。
- 符合规则前两条规则的从隐性值到显性值的跳变沿都被用于重同步(在低比特率时也可选择从显性值到隐性值的跳变沿),例外的情况是具有正相位误差的隐性值到显性值的跳变沿将不会导致重同步。8 z+ z6 X" N& s, A8 x. x5 x
, x. W& [: r* O3 H! O" M. X! X二、CAN通信控制器程序基本框架
# d* J- @8 d* k2 t3 _! ~8 q( l. t, z
CAN 总线的通信协议由 CAN 通信控制器完成。CAN 通信控制器由实现 CAN 总线协议部分和微控制器部分的电路组成。下面将通过一个实例讲解如何用 FPGA 实现 CAN 通信控制器的功能。这个实例从功能和结构上完全参照 SJA 1000 CAN 通信控制器。
9 j) i, b+ W$ o( c2.1 SJA1000CAN 通信控制器
+ R. R( c* s' b" w3 z7 {SJA1000 是 Philips 公司于 1997 年推出的一种独立 CAN 总线控制器。它实现了 CAN 总线物理层和数据链路层的所有功能。SJA 1000 通信控制器的功能框图如图 8 所示。
" l5 X6 L5 }& h2 p3 N& J+ h C
SJA 1000 主要由以下几部分构成:
6 k8 l5 I r0 ]4 } ]3 D2 Z* P
• 接口管理逻辑 处理来自主 CPU 的命令,控制 CAN 寄存器的寻址,并为主 CPU 提供中断和状态信息。
• 发送缓冲器 它是 CPU 和位数据流处理器(BSP)之间的接口,能存储一条可发送到 CAN总线上的完整报文。报文由 CPU 写入,由位数据流处理器读出。
5 Y! w7 J" R( {2 g5 Y9 l
$ Z8 H" U- ?, \" b4 E/ V2 u
( _5 E; M1 V6 y: ?2 w; A3 E# h图 8 SJA1000 通信控制器功能框图
0 Y) L5 H7 o7 U! `8 h! J• 接收缓冲器 它是接收 FIFO 的一个可被 CPU 访问的窗口。在接收 FIFO 的支持下,CPU可以在处理当前信息的同时接收总线上的其他信息。
• 接收滤波器 它把收到的报文标识符和接收滤波器寄存器中的内容进行比较,以判断该报文是否应该被接收。如果符合接收条件,则报文被存入接收 FIFO 中。
• 位数据流处理器 它是一个序列发生器,控制发送缓冲器、接收 FIFO 和 CAN 总线之间的数据流,同时它也执行错误检测、仲裁、位填充和 CAN 总线错误处理功能。
• 位时序逻辑 它监视串行 CAN 总线并处理与总线相关的位时序。它在报文开始发送,总线电平从隐性值跳变到显性值时同步于 CAN 总线上的位数据流(硬同步),并在该报文的传送过程中,每遇到一次从隐性值到显性值的跳变沿就进行一次重同步(软同步)。位时序逻辑还提供可编程的时间段来补偿传播延迟时间和相位漂移(如晶振导致的漂移),还能定义采样点以及每一个位时间内的采样次数。
- d8 D5 F. a c1 V• 错误管理逻辑 它按照 CAN 协议完成传输错误界定。它接收来自位数据流处理器 BSP 的出错通知,并向位数据流处理器 BSP 和接口管理逻辑提供出错统计。
0 v2 h. g8 J: G1 k6 B, Z/ b% B6 j8 u4 g, p
2.2 CAN 通信控制器程序框架) l8 x" z9 l4 K# V% I
) j8 W; _ y" g+ K6 r5 P
/ I' N5 |$ m$ u! A% h5 e; K
实现的 CAN 通信控制器参照 SJA1000 CAN 通信控制器的结构,程序基本框架如图 9 所示。
3 V$ m1 [- |0 s) Y( @* ~1 a1 u
7 e! ?% a$ V m8 s, l* L
1 B# D) p- ^# z& U9 V' O' i
# X, [% H/ s3 f* U: E# D
图 9 CAN 通信控制器结构框图
0 j* }5 `! O1 @# J& \9 x6 l3 [
本篇到此结束,下一篇带来基于FPGA的CAN总线控制器的设计(中),会介绍CAN 通信控制器的具体实现,包括顶层控制程序、寄存器控制、 位时序逻辑(Bit Timing Logic)、位数据流处理器(Bit Stream Processor)、CRC 校验 、FIFO等相关内容。
9 m( a; e9 V4 V. g# q3 D& c- C* U
2 E- k* T" z4 L& C