CAN协议简介
CAN 是控制器局域网络(Controller Area Network)的简称, 它是由研发和生产汽车电子产品著称的德国 BOSCH 公司开发的,并最终成为国际标准(ISO11519) , 是国际上应用最普遍的现场总线之一。编程
CAN 总线协议已经成为汽车计算机控制系统和嵌入式工业控制局域网的标准总线,而且拥有以 CAN 为底层协议专为大型货车和重工机械车辆设计的 J1939 协议。近年来, 它具备的高可靠性和良好的错误检测能力受到重视,被普遍应用于汽车计算机控制系统和环境温度恶劣、电磁辐射强及振动大的工业环境。缓存
CAN物理层
与 I2C、 SPI 等具备时钟信号的同步通信方式不一样, CAN 通信并非以时钟信号来进行同步的,它是一种异步通信,只具备 CAN_High 和 CAN_Low 两条信号线,共同构成一组差分信号线,以差分信号的形式进行通信。网络
- 闭环总线网络
CAN 物理层的形式主要有两种, 图 43-1 中的 CAN 通信网络是一种遵循 ISO11898 标准的高速、短距离“闭环网络”,它的总线最大长度为 40m,通讯速度最高为 1Mbps,总线的两端各要求有一个“120 欧”的电阻。架构
- 开环总线网络
图 43-2 中的是遵循 ISO11519-2 标准的低速、远距离“开环网络”,它的最大传输距离为 1km,最高通信速率为 125kbps,两根总线是独立的、不造成闭环,要求每根总线上各串联有一个“2.2 千欧”的电阻。异步
- 通信节点
从 CAN 通信网络图可了解到, CAN 总线上能够挂载多个通信节点,节点之间的信号通过总线传输,实现节点间通信。因为 CAN 通信协议不对节点进行地址编码,而是对数据内容进行编码的,因此网络中的节点个数理论上不受限制,只要总线的负载足够便可,能够经过中继器加强负载。函数
CAN 通信节点由一个 CAN 控制器及 CAN 收发器组成,控制器与收发器之间经过CAN_Tx 及 CAN_Rx 信号线相连,收发器与 CAN 总线之间使用 CAN_High 及 CAN_Low信号线相连。其中 CAN_Tx 及 CAN_Rx 使用普通的相似 TTL 逻辑信号,而 CAN_High 及CAN_Low 是一对差分信号线,使用比较特别的差分信号,下一小节再详细说明。oop
当 CAN 节点须要发送数据时, 控制器把要发送的二进制编码经过 CAN_Tx 线发送到收发器,而后由收发器把这个普通的逻辑电平信号转化成差分信号,经过差分线CAN_High 和 CAN_Low 线输出到 CAN 总线网络。而经过收发器接收总线上的数据到控制器时,则是相反的过程,收发器把总线上收到的 CAN_High 及 CAN_Low 信号转化成普通的逻辑电平信号,经过 CAN_Rx 输出到控制器中。学习
例如, STM32 的 CAN 片上外设就是通信节点中的控制器,为了构成完整的节点,还要给它外接一个收发器,在咱们实验板中使用型号为 TJA1050 的芯片做为 CAN 收发器。CAN 控制器与 CAN 收发器的关系如同 TTL 串口与 MAX3232 电平转换芯片的关系,MAX3232 芯片把 TTL 电平的串口信号转换成 RS-232 电平的串口信号, CAN 收发器的做用则是把 CAN 控制器的 TTL 电平信号转换成差分信号(或者相反)。测试
- 差分信号
差分信号又称差模信号,与传统使用单根信号线电压表示逻辑的方式有区别,使用差分信号传输时,须要两根信号线,这两个信号线的振幅相等,相位相反,经过两根信号线的电压差值来表示逻辑 0 和逻辑 1。见图 43-3,它使用了 V+与 V-信号的差值表达出了图下方的信号。ui
相对于单信号线传输的方式,使用差分信号传输具备以下优势:
抗干扰能力强,当外界存在噪声干扰时,几乎会同时耦合到两条信号线上,而接收端只关心两个信号的差值,因此外界的共模噪声能够被彻底抵消。
能有效抑制它对外部的电磁干扰,一样的道理,因为两根信号的极性相反,他们对外辐射的电磁场能够相互抵消,耦合的越紧密,泄放到外界的电磁能量越少。
时序定位精确, 因为差分信号的开关变化是位于两个信号的交点,而不像普通单端信号依靠高低两个阈值电压判断,于是受工艺,温度的影响小,能下降时序上的偏差,同时也更适合于低幅度信号的电路。
因为差分信号线具备这些优势,因此在 USB 协议、 485 协议、以太网协议及 CAN 协议的物理层中,都使用了差分信号传输。
- CAN 协议中的差分信号
CAN 协议中对它使用的 CAN_High 及 CAN_Low 表示的差分信号作了规定,见表 43-1及图 43-4。以高速 CAN 协议为例,当表示逻辑 1 时(隐性电平), CAN_High 和 CAN_Low线上的电压均为 2.5v,即它们的电压差 VH-VL=0V;而表示逻辑 0 时(显性电平),CAN_High 的电平为 3.5V, CAN_Low 线的电平为 1.5V,即它们的电压差为 VH-VL=2V。例如,当 CAN 收发器从 CAN_Tx 线接收到来自 CAN 控制器的低电平信号时(逻辑 0),它会使 CAN_High 输出 3.5V,同时 CAN_Low 输出 1.5V,从而输出显性电平表示逻辑 0。
在 CAN 总线中,必须使它处于隐性电平(逻辑 1)或显性电平(逻辑 0)中的其中一个状态。假若有两个 CAN 通信节点,在同一时间,一个输出隐性电平,另外一个输出显性电平,相似 I2C 总线的“线与”特性将使它处于显性电平状态,显性电平的名字就是这样来的, 便可以认为显性具备优先的意味。
因为 CAN 总线协议的物理层只有 1 对差分线,在一个时刻只能表示一个信号,因此对通信节点来讲, CAN 通信是半双工的,收发数据须要分时进行。在 CAN 的通信网络中,由于共用总线,在整个网络中同一时刻只能有一个通信节点发送信号,其他的节点在该时刻都只能接收。
协议层
以上是 CAN 的物理层标准,约定了电气特性,如下介绍的协议层则规定了通信逻辑。
- CAN 的波特率及位同步
因为 CAN 属于异步通信,没有时钟信号线,链接在同一个总线网络中的各个节点会像串口异步通信那样,节点间使用约定好的波特率进行通信,特别地, CAN 还会使用“位同步” 的方式来抗干扰、吸取偏差,实现对总线电平信号进行正确的采样,确保通信正常。
位时序分解
为了实现位同步, CAN 协议把每个数据位的时序分解成如图 43-5 所示的 SS 段、PTS 段、 PBS1 段、 PBS2 段,这四段的长度加起来即为一个 CAN 数据位的长度。分解后最小的时间单位是 Tq,而一个完整的位由 8~25 个 Tq 组成。为方便表示, 图 43-5 中的高低电平直接表明信号逻辑 0 或逻辑 1(不是差分信号)。
该图中表示的 CAN 通信信号每个数据位的长度为 19Tq,其中 SS 段占 1Tq, PTS 段占 6Tq, PBS1 段占 5Tq, PBS2 段占 7Tq。信号的采样点位于 PBS1 段与 PBS2 段之间,经过控制各段的长度,能够对采样点的位置进行偏移,以便准确地采样。
各段的做用如介绍下:
SS 段(SYNC SEG)
SS 译为同步段, 若通信节点检测到总线上信号的跳变沿被包含在 SS 段的范围以内,则表示节点与总线的时序是同步的,当节点与总线同步时,采样点采集到的总线电平便可被肯定为该位的电平。 SS 段的大小固定为 1Tq。
PTS 段(PROP SEG)
PTS 译为传播时间段,这个时间段是用于补偿网络的物理延时时间。 是总线上输入比较器延时和输出驱动器延时总和的两倍。 PTS 段的大小能够为 1~8Tq。
PBS1 段(PHASE SEG1),PBS1 译为相位缓冲段,主要用来补偿边沿阶段的偏差,它的时间长度在从新同步的时候能够加长。 PBS1 段的初始大小能够为 1~8Tq。
PBS2 段(PHASE SEG2)
PBS2 这是另外一个相位缓冲段,也是用来补偿边沿阶段偏差的,它的时间长度在从新同步时能够缩短。 PBS2 段的初始大小能够为 2~8Tq。
通信的波特率
总线上的各个通信节点只要约定好 1 个 Tq 的时间长度以及每个数据位占据多少个Tq,就能够肯定 CAN 通信的波特率。
例如,假设上图中的 1Tq=1us,而每一个数据位由 19 个 Tq 组成,则传输一位数据须要时间 T1bit =19us,从而每秒能够传输的数据位个数为:
1x106/19 = 52631.6 (bps)
这个每秒可传输的数据位的个数即为通信中的波特率。
同步过程分析
波特率只是约定了每一个数据位的长度,数据同步还涉及到相位的细节,这个时候就须要用到数据位内的 SS、 PTS、 PBS1 及 PBS2 段了。
根据对段的应用方式差别, CAN 的数据同步分为硬同步和从新同步。其中硬同步只是当存在“帧起始信号”时起做用,没法确保后续一连串的位时序都是同步的,而从新同步方式可解决该问题,这两种方式具体介绍以下:
(1) 硬同步
若某个 CAN 节点经过总线发送数据时,它会发送一个表示通信起始的信号(即下一小节介绍的帧起始信号),该信号是一个由高变低的降低沿。而挂载到 CAN 总线上的通信节点在不发送数据时,会时刻检测总线上的信号。
见图 43-6,能够看到当总线出现帧起始信号时,某节点检测到总线的帧起始信号不在节点内部时序的 SS 段范围,因此判断它本身的内部时序与总线不一样步,于是这个状态的采样点采集得的数据是不正确的。因此节点以硬同步的方式调整,把本身的位时序中的 SS段平移至总线出现降低沿的部分,得到同步,同步后采样点就能够采集得正确数据了。
(2) 从新同步
前面的硬同步只是当存在帧起始信号时才起做用,若是在一帧很长的数据内,节点信号与总线信号相位有偏移时,这种同步方式就无能为力了。于是须要引入从新同步方式,它利用普通数据位的高至低电平的跳变沿来同步(帧起始信号是特殊的跳变沿)。从新同步与硬同步方式类似的地方是它们都使用 SS 段来进行检测,同步的目的都是使节点内的 SS段把跳变沿包含起来。
从新同步的方式分为超前和滞后两种状况,以总线跳变沿与 SS 段的相对位置进行区分。第一种相位超前的状况如图 43-7,节点从总线的边沿跳变中,检测到它内部的时序比总线的时序相对超前 2Tq,这时控制器在下一个位时序中的 PBS1 段增长 2Tq 的时间长度,使得节点与总线时序从新同步。
第二种相位滞后的状况如图 43-8,节点从总线的边沿跳变中,检测到它的时序比总线的时序相对滞后 2Tq,这时控制器在前一个位时序中的 PBS2 段减小 2Tq 的时间长度,得到同步。
在从新同步的时候, PBS1 和 PBS2 中增长或减小的这段时间长度被定义为“从新同步补偿宽度 SJW (reSynchronization Jump Width)”。通常来讲 CAN 控制器会限定 SJW 的最大值,如限定了最大 SJW=3Tq 时,单次同步调整的时候不能增长或减小超过 3Tq 的时间长度,如有须要,控制器会经过屡次小幅度调整来实现同步。当控制器设置的 SJW 极限值较大时,能够吸取的偏差加大,但通信的速度会降低。
- CAN 的报文种类及结构
在 SPI 通信中, 片选、时钟信号、数据输入及数据输出这 4 个信号都有单独的信号线,I2C 协议包含有时钟信号及数据信号 2 条信号线,异步串口包含接收与发送 2 条信号线,这些协议包含的信号都比 CAN 协议要丰富,它们能轻易进行数据同步或区分数据传输方向。而 CAN 使用的是两条差分信号线,只能表达一个信号,简洁的物理层决定了 CAN 必然要配上一套更复杂的协议,如何用一个信号通道实现一样、甚至更强大的功能呢? CAN协议给出的解决方案是对数据、操做命令(如读/写)以及同步信号进行打包,打包后的这些内容称为报文。
报文的种类
在原始数据段的前面加上传输起始标签、片选(识别)标签和控制标签,在数据的尾段加上 CRC 校验标签、应答标签和传输结束标签,把这些内容按特定的格式打包好,就能够用一个通道表达各类信号了, 各类各样的标签就如同 SPI 中各类通道上的信号,起到了协同传输的做用。当整个数据包被传输到其它设备时,只要这些设备按格式去解读,就能还原出原始数据,这样的报文就被称为 CAN 的“数据帧” 。
为了更有效地控制通信, CAN 一共规定了 5 种类型的帧,它们的类型及用途说明如表43-2。
<table><colgroup><col><col></colgroup><tbody><tr><th><div class="table-cell-line"><span>帧</span></div></th><th><div class="table-cell-line"><span>帧用途</span></div></th></tr><tr><td><div class="table-cell-line"><span>数据帧</span></div></td><td><div class="table-cell-line"><span>用于节点向外传送数据</span></div></td></tr><tr><td><div class="table-cell-line"><span>遥控帧</span></div></td><td><div class="table-cell-line"><span>用于向远端节点请求数据</span></div></td></tr><tr><td><div class="table-cell-line"><span>错误帧</span></div></td><td><div class="table-cell-line"><span>用于向远端节点通知校验错误,请求从新发送上一个数据</span></div></td></tr><tr><td><div class="table-cell-line"><span>过载帧</span></div></td><td><div class="table-cell-line"><span>用于通知远端节点:本节点还没有作好接收准备</span></div></td></tr><tr><td><div class="table-cell-line"><span>帧间隔</span></div></td><td><div class="table-cell-line"><span>用于将数据帧及遥控帧与前面的帧分离开来</span></div></td></tr></tbody></table>
数据帧的结构
数据帧是在 CAN 通信中最主要、最复杂的报文,咱们来了解它的结构,见图 43-9。
数据帧以一个显性位(逻辑 0)开始,以 7 个连续的隐性位(逻辑 1)结束,在它们之间,分别有仲裁段、控制段、数据段、 CRC 段和 ACK 段
帧起始
SOF 段(Start Of Frame),译为帧起始,帧起始信号只有一个数据位,是一个显性电平,它用于通知各个节点将有数据传输,其它节点经过帧起始信号的电平跳变沿来进行硬同步。
仲裁段
当同时有两个报文被发送时,总线会根据仲裁段的内容决定哪一个数据包能被传输,这也是它名称的由来。
仲裁段的内容主要为本数据帧的 ID 信息(标识符), 数据帧具备标准格式和扩展格式两种,区别就在于 ID 信息的长度,标准格式的 ID 为 11 位,扩展格式的 ID 为 29 位,它在标准 ID 的基础上多出 18 位。在 CAN 协议中, ID 起着重要的做用,它决定着数据帧发送的优先级,也决定着其它节点是否会接收这个数据帧。 CAN 协议不对挂载在它之上的节点分配优先级和地址,对总线的占有权是由信息的重要性决定的,即对于重要的信息,咱们会给它打包上一个优先级高的 ID,使它可以及时地发送出去。也正由于它这样的优先级分配原则,使得 CAN 的扩展性大大增强,在总线上增长或减小节点并不影响其它设备。报文的优先级,是经过对 ID 的仲裁来肯定的。根据前面对物理层的分析咱们知道若是总线上同时出现显性电平和隐性电平,总线的状态会被置为显性电平, CAN 正是利用这个特性进行仲裁。
若两个节点同时竞争 CAN 总线的占有权,当它们发送报文时, 若首先出现隐性电平,则会失去对总线的占有权,进入接收状态。见图 43-10,在开始阶段,两个设备发送的电平同样,因此它们一直继续发送数据。到了图中箭头所指的时序处,节点单元 1 发送的为隐性电平,而此时节点单元 2 发送的为显性电平,因为总线的“线与”特性使它表达出显示电平,所以单元 2 竞争总线成功,这个报文得以被继续发送出去。
仲裁段 ID 的优先级也影响着接收设备对报文的反应。由于在 CAN 总线上数据是以广播的形式发送的,全部链接在 CAN 总线的节点都会收到全部其它节点发出的有效数据,于是咱们的 CAN 控制器大多具备根据 ID 过滤报文的功能,它能够控制本身只接收某些 ID的报文。
回看图 43-9 中的数据帧格式,可看到仲裁段除了报文 ID 外,还有 RTR、 IDE 和 SRR位。
(1) RTR 位(Remote Transmission Request Bit),译做远程传输请求位,它是用于区分数据帧和遥控帧的,当它为显性电平时表示数据帧,隐性电平时表示遥控帧。
(2) IDE 位(Identifier Extension Bit),译做标识符扩展位,它是用于区分标准格式与扩展格式,当它为显性电平时表示标准格式,隐性电平时表示扩展格式。
(3) SRR 位(Substitute Remote Request Bit),只存在于扩展格式,它用于替代标准格式中的RTR 位。因为扩展帧中的 SRR 位为隐性位, RTR 在数据帧为显性位,因此在两个 ID相同的标准格式报文与扩展格式报文中,标准格式的优先级较高。
控制段
在控制段中的 r1 和 r0 为保留位,默认设置为显性位。它最主要的是 DLC 段(Data Length Code),译为数据长度码,它由 4 个数据位组成,用于表示本报文中的数据段含有多少个字节, DLC 段表示的数字为 0~8。
数据段
数据段为数据帧的核心内容,它是节点要发送的原始信息,由 0~8 个字节组成, MSB先行。
CRC 段
为了保证报文的正确传输, CAN 的报文包含了一段 15 位的 CRC 校验码,一旦接收节点算出的 CRC 码跟接收到的 CRC 码不一样, 则它会向发送节点反馈出错信息,利用错误帧请求它从新发送。 CRC 部分的计算通常由 CAN 控制器硬件完成,出错时的处理则由软件控制最大重发数。
在 CRC 校验码以后,有一个 CRC 界定符,它为隐性位,主要做用是把 CRC 校验码与后面的 ACK 段间隔起来。
ACK 段
ACK 段包括一个 ACK 槽位,和 ACK 界定符位。相似 I2C 总线,在 ACK 槽位中,发送节点发送的是隐性位,而接收节点则在这一位中发送显性位以示应答。在 ACK 槽和帧结束之间由 ACK 界定符间隔开。
帧结束
EOF 段(End Of Frame),译为帧结束,帧结束段由发送节点发送的 7 个隐性位表示结束。
其它报文的结构
关于其它的 CAN 报文结构,再也不展开讲解,其主要内容见图 43-11。
STM32的CAN外设简介
STM32 的芯片中具备 bxCAN 控制器 (Basic Extended CAN), 它支持 CAN 协议 2.0A 和2.0B 标准。
该 CAN 控制器支持最高的通信速率为 1Mb/s;能够自动地接收和发送 CAN 报文,支持使用标准 ID 和扩展 ID 的报文;外设中具备 3 个发送邮箱,发送报文的优先级可使用软件控制,还能够记录发送的时间;具备 2 个 3 级深度的接收 FIFO,可以使用过滤功能只接收或不接收某些 ID 号的报文;可配置成自动重发;不支持使用 DMA 进行数据收发。
STM32的CAN架构剖析
STM32 的有两组 CAN 控制器,其中 CAN1 是主设备,框图中的“存储访问控制器”是由 CAN1 控制的, CAN2 没法直接访问存储区域,因此使用 CAN2 的时候必须使能CAN1 外设的时钟。框图中主要包含 CAN 控制内核、发送邮箱、接收 FIFO 以及验收筛选器,下面对框图中的各个部分进行介。
- CAN 控制内核
框图中标号1处的 CAN 控制内核包含了各类控制寄存器及状态寄存器,咱们主要讲解其中的主控制寄存器 CAN_MCR 及位时序寄存器 CAN_BTR。
主控制寄存器 CAN_MCR
主控制寄存器 CAN_MCR 负责管理 CAN 的工做模式,它使用如下寄存器位实现控制。
(1) DBF 调试冻结功能
DBF(Debug freeze)调试冻结,使用它可设置 CAN 处于工做状态或禁止收发的状态,禁止收发时仍可访问接收 FIFO 中的数据。这两种状态是当 STM32 芯片处于程序调试模式时才使用的,平时使用并不影响。
(2) TTCM 时间触发模式
TTCM(Time triggered communication mode)时间触发模式,它用于配置 CAN 的时间触发通讯模式,在此模式下, CAN 使用它内部定时器产生时间戳, 并把它保存在CAN_RDTxR、CAN_TDTxR 寄存器中。内部定时器在每一个 CAN 位时间累加,在接收和发送的帧起始位被采样,并生成时间戳。 利用它能够实现 ISO 11898-4 CAN 标准的分时同步通讯功能。
(3) ABOM 自动离线管理
ABOM(Automatic bus-off management) 自动离线管理,它用于设置是否使用自动离线管理功能。 当节点检测到它发送错误或接收错误超过必定值时,会自动进入离线状态,在离线状态中, CAN 不能接收或发送报文。 处于离线状态的时候,能够软件控制恢复或者直接使用这个自动离线管理功能,它会在适当的时候自动恢复。
(4) AWUM 自动唤醒
AWUM(Automatic bus-off management), 自动唤醒功能, CAN 外设可使用软件进入低功耗的睡眠模式,若是使能了这个自动唤醒功能,当 CAN 检测到总线活动的时候,会自动唤醒。
(5) NART 自动重传
NART(No automatic retransmission)报文自动重传功能,设置这个功能后, 当报文发送失败时会自动重传至成功为止。若不使用这个功能,不管发送结果如何,消息只发送一次。
(6) RFLM 锁定模式
RFLM(Receive FIFO locked mode)FIFO 锁定模式,该功能用于锁定接收 FIFO。锁定后,当接收 FIFO 溢出时,会丢弃下一个接收的报文。若不锁定,则下一个接收到的报文会覆盖原报文。
(7) TXFP 报文发送优先级的断定方法
TXFP(Transmit FIFO priority)报文发送优先级的断定方法,当 CAN 外设的发送邮箱中有多个待发送报文时,本功能能够控制它是根据报文的 ID 优先级仍是报文存进邮箱的顺序来发送。
位时序寄存器(CAN_BTR)及波特率
CAN 外设中的位时序寄存器 CAN_BTR 用于配置测试模式、波特率以及各类位内的段参数。
(1) 测试模式
为方便调试, STM32 的 CAN 提供了测试模式,配置位时序寄存器 CAN_BTR 的 SILM及 LBKM 寄存器位能够控制使用正常模式、静默模式、回环模式及静默回环模式,见图43-13。
各个工做模式介绍以下:
正常模式
正常模式下就是一个正常的 CAN 节点,能够向总线发送数据和接收数据。
静默模式
静默模式下,它本身的输出端的逻辑 0 数据会直接传输到它本身的输入端,逻辑1 能够被发送到总线,因此它不能向总线发送显性位(逻辑 0),只能发送隐性位(逻辑 1)。输入端能够从总线接收内容。因为它只可发送的隐性位不会强制影响总线的状态,因此把它称为静默模式。这种模式通常用于监测,它能够用于分析总线上的流量,但又不会由于发送显性位而影响总线。
回环模式
回环模式下,它本身的输出端的全部内容都直接传输到本身的输入端,输出端的内容同时也会被传输到总线上,即也可以使用总线监测它的发送内容。输入端只接收本身发送端的内容,不接收来自总线上的内容。使用回环模式能够进行自检。
回环静默模式
回环静默模式是以上两种模式的结合,本身的输出端的全部内容都直接传输到本身的输入端,而且不会向总线发送显性位影响总线,不能经过总线监测它的发送内容。输入端只接收本身发送端的内容,不接收来自总线上的内容。这种方式能够在“热自检”时使用,即自我检查的时候,不会干扰总线。
以上说的各个模式,是不须要修改硬件接线的,如当输出直连输入时,它是在 STM32芯片内部链接的,传输路径不通过 STM32 的 CAN_Tx/Rx 引脚,更不通过外部链接的 CAN收发器,只有输出数据到总线或从总线接收的状况下才会通过 CAN_Tx/Rx 引脚和收发器。
(2) 位时序及波特率
STM32 外设定义的位时序与咱们前面解释的 CAN 标准时序有一点区别,见图 43-14。
STM32 的 CAN 外设位时序中只包含 3 段,分别是同步段 SYNC_SEG、位段 BS1 及位段 BS2,采样点位于 BS1 及 BS2 段的交界处。其中 SYNC_SEG 段固定长度为 1Tq,而BS1 及 BS2 段能够在位时序寄存器 CAN_BTR 设置它们的时间长度,它们能够在从新同步期间增加或缩短,该长度 SJW 也可在位时序寄存器中配置。
理解 STM32 的 CAN 外设的位时序时,能够把它的 BS1 段理解为是由前面介绍的CAN 标准协议中 PTS 段与 PBS1 段合在一块儿的,而 BS2 段就至关于 PBS2 段。
了解位时序后,咱们就能够配置波特率了。 经过配置位时序寄存器 CAN_BTR 的TS1[3:0]及 TS2[2:0]寄存器位设定 BS1 及 BS2 段的长度后,咱们就能够肯定每一个 CAN 数据位的时间:
BS1 段时间:
TS1=Tq x (TS1[3:0] + 1),
BS2 段时间:
TS2= Tq x (TS2[2:0] + 1),
一个数据位的时间:
T1bit =1Tq+TS1+TS2 =1+ (TS1[3:0] + 1)+ (TS2[2:0] + 1)= N Tq
其中单个时间片的长度 Tq 与 CAN 外设的所挂载的时钟总线及分频器配置有关,CAN1 和 CAN2 外设都是挂载在 APB1 总线上的,而位时序寄存器 CAN_BTR 中的 BRP[9:0]寄存器位能够设置 CAN 外设时钟的分频值 ,因此:
Tq = (BRP[9:0]+1) x TPCLK
其中的 PCLK 指 APB1 时钟,默认值为 45MHz。
最终能够计算出 CAN 通信的波特率:
BaudRate = 1/N Tq
例如表 43-3 说明了一种把波特率配置为 1Mbps 的方式。
- CAN 发送邮箱
回到图 24-5 中的 CAN 外设框图,在标号2处的是 CAN 外设的发送邮箱,它一共有 3个发送邮箱,即最多能够缓存 3 个待发送的报文。每一个发送邮箱中包含有标识符寄存器 CAN_TIxR、数据长度控制寄存器 CAN_TDTxR及 2 个数据寄存器 CAN_TDLxR、 CAN_TDHxR,它们的功能见表 43-5。
当咱们要使用 CAN 外设发送报文时,把报文的各个段分解,按位置写入到这些寄存器中,并对标识符寄存器 CAN_TIxR 中的发送请求寄存器位 TMIDxR_TXRQ 置 1,便可把数据发送出去。
其中标识符寄存器 CAN_TIxR 中的 STDID 寄存器位比较特别。咱们知道 CAN 的标准标识符的总位数为 11 位,而扩展标识符的总位数为 29 位的。当报文使用扩展标识符的时候,标识符寄存器 CAN_TIxR 中的 STDID[10:0]等效于 EXTID[18:28]位,它与 EXTID[17:0]共同组成完整的 29 位扩展标识符。
- CAN 接收 FIFO
图 24-5 中的 CAN 外设框图,在标号3处的是 CAN 外设的接收 FIFO,它一共有 2 个接收 FIFO,每一个 FIFO 中有 3 个邮箱,即最多能够缓存 6 个接收到的报文。当接收到报文时, FIFO 的报文计数器会自增,而 STM32 内部读取 FIFO 数据以后,报文计数器会自减,咱们经过状态寄存器可获知报文计数器的值,而经过前面主控制寄存器的 RFLM 位,可设置锁定模式,锁定模式下 FIFO 溢出时会丢弃新报文,非锁定模式下 FIFO 溢出时新报文会覆盖旧报文。
跟发送邮箱相似,每一个接收 FIFO 中包含有标识符寄存器 CAN_RIxR、数据长度控制寄存器CAN_RDTxR 及 2 个数据寄存器 CAN_RDLxR、 CAN_RDHxR,它们的功能见表43-5。
经过中断或状态寄存器知道接收 FIFO 有数据后,咱们再读取这些寄存器的值便可把接收到的报文加载到 STM32 的内存中。
- 验收筛选器
图 24-5 中的 CAN 外设框图,在标号4处的是 CAN 外设的验收筛选器,一共有 28 个筛选器组,每一个筛选器组有 2 个寄存器, CAN1 和 CAN2 共用的筛选器的。
在 CAN 协议中,消息的标识符与节点地址无关,但与消息内容有关。所以,发送节点将报文广播给全部接收器时,接收节点会根据报文标识符的值来肯定软件是否须要该消息,为了简化软件的工做, STM32 的 CAN 外设接收报文前会先使用验收筛选器检查,只接收须要的报文到 FIFO 中。
筛选器工做的时候,能够调整筛选 ID 的长度及过滤模式。根据筛选 ID 长度来分类有
有如下两种:
(1) 检查 STDID[10:0]、 EXTID[17:0]、 IDE 和 RTR 位,一共 31 位。
(2) 检查 STDID[10:0]、 RTR、 IDE 和 EXTID[17:15],一共 16 位。
经过配置筛选尺度寄存器 CAN_FS1R 的 FSCx 位能够设置筛选器工做在哪一个尺度。
而根据过滤的方法分为如下两种模式:
(1) 标识符列表模式,它把要接收报文的 ID 列成一个表,要求报文 ID 与列表中的某一个标识符彻底相同才能够接收,能够理解为白名单管理。
(2) 掩码模式,它把可接收报文 ID 的某几位做为列表,这几位被称为掩码,能够把它理解成关键字搜索,只要掩码(关键字)相同,就符合要求,报文就会被保存到接收 FIFO。
经过配置筛选模式寄存器 CAN_FM1R 的 FBMx 位能够设置筛选器工做在哪一个模式。
不一样的尺度和不一样的过滤方法可以使筛选器工做在图 43-15 的 4 种状态。
每组筛选器包含 2 个 32 位的寄存器,分别为 CAN_FxR1 和 CAN_FxR2,它们用来存储要筛选的 ID 或掩码,各个寄存器位表明的意义与图中两个寄存器下面“映射”的一栏一致,各个模式的说明见表 43-6。
例以下面的表格所示,在掩码模式时,第一个寄存器存储要筛选的 ID,第二个寄存器存储掩码,掩码为 1 的部分表示该位必须与 ID 中的内容一致,筛选的结果为表中第三行的ID 值,它是一组包含多个的 ID 值,其中 x 表示该位能够为 1 能够为 0。
而工做在标识符模式时, 2 个寄存器存储的都是要筛选的 ID,它只包含 2 个要筛选的ID 值(32 位模式时)。
若是使能了筛选器,且报文的 ID 与全部筛选器的配置都不匹配, CAN 外设会丢弃该报文,不存入接收 FIFO。
- 总体控制逻辑
回到图 24-5 结构框图,图中的标号5处表示的是 CAN2 外设的结构,它与 CAN1 外设是同样的,他们共用筛选器且因为存储访问控制器由 CAN1 控制,因此要使用 CAN2 的时候必需要使能 CAN1 的时钟。
CAN初始化结构体
从 STM32 的 CAN 外设咱们了解到它的功能很是多,控制涉及的寄存器也很是丰富,而使用STM32 标准库提供的各类结构体及库函数能够简化这些控制过程。跟其它外设同样,STM32 标准库提供了 CAN 初始化结构体及初始化函数来控制 CAN 的工做方式,提供了收发报文使用的结构体及收发函数,还有配置控制筛选器模式及 ID 的结构体。这些内容都定义在库文件“stm32f4xx_can.h”及“stm32f4xx_can.c”中,编程时咱们能够结合这两个文件内的注释使用或参考库帮助文档。
首先咱们来学习初始化结构体的内容,见代码清单 24-1。
代码清单 43-1 CAN 初始化结构体
/** * @brief CAN 初始化结构体 */ typedef struct { uint16_t CAN_Prescaler; /*配置 CAN 外设的时钟分频,可设置为 1-1024*/ uint8_t CAN_Mode; /*配置 CAN 的工做模式,回环或正常模式*/ uint8_t CAN_SJW; /*配置 SJW 极限值 */ uint8_t CAN_BS1; /*配置 BS1 段长度*/ uint8_t CAN_BS2; /*配置 BS2 段长度 */ FunctionalState CAN_TTCM; /*是否使能 TTCM 时间触发功能*/ FunctionalState CAN_ABOM; /*是否使能 ABOM 自动离线管理功能*/ FunctionalState CAN_AWUM; /*是否使能 AWUM 自动唤醒功能 */ FunctionalState CAN_NART; /*是否使能 NART 自动重传功能*/ FunctionalState CAN_RFLM; /*是否使能 RFLM 锁定 FIFO 功能*/ FunctionalState CAN_TXFP; /*配置 TXFP 报文优先级的断定方法*/ } CAN_InitTypeDef;
这些结构体成员说明以下,其中括号内的文字是对应参数在 STM32 标准库中定义的宏,这些结构体成员都是“43.2.1 1CAN 控制内核”小节介绍的内容,可对比阅读:
(1) CAN_Prescaler
本成员设置 CAN 外设的时钟分频,它可控制时间片 Tq 的时间长度,这里设置的值最终会减 1 后再写入 BRP 寄存器位,即前面介绍的 Tq 计算公式:
Tq = (BRP[9:0]+1) x TPCLK
等效于: Tq = CAN_Prescaler x TPCLK
(2) CAN_Mode
本成员设置 CAN 的工做模式,可设置为正常模式(CAN_Mode_Normal)、回环模式(CAN_Mode_LoopBack)、静默模式(CAN_Mode_Silent)以及回环静默模式(CAN_Mode_Silent_LoopBack)。
(3) CAN_SJW
本成员能够配置 SJW 的极限长度,即 CAN 从新同步时单次可增长或缩短的最大长度,它能够被配置为 1-4Tq(CAN_SJW_1/2/3/4tq)。
(4) CAN_BS1
本成员用于设置 CAN 位时序中的 BS1 段的长度,它能够被配置为 1-16 个 Tq 长度(CAN_BS1_1/2/3…16tq)。
(5) CAN_BS2
本成员用于设置 CAN 位时序中的 BS2 段的长度,它能够被配置为 1-8 个 Tq 长度(CAN_BS2_1/2/3…8tq)。SYNC_SEG、 BS1 段及 BS2 段的长度加起来即一个数据位的长度,即前面介绍的原来
计算公式:
T1bit =1Tq+TS1+TS2 =1+ (TS1[3:0] + 1)+ (TS2[2:0] + 1)
等效于: T1bit = 1Tq+CAN_BS1+CAN_BS2
(6) CAN_TTCM
本成员用于设置是否使用时间触发功能(ENABLE/DISABLE),时间触发功能在某些CAN 标准中会使用到。
(7) CAN_ABOM
本成员用于设置是否使用自动离线管理(ENABLE/DISABLE),使用自动离线管理能够在节点出错离线后适时自动恢复,不须要软件干预。
(8) CAN_ AWUM
本成员用于设置是否使用自动唤醒功能(ENABLE/DISABLE),使能自动唤醒功能后它会在监测到总线活动后自动唤醒。
(9) CAN_ABOM
本成员用于设置是否使用自动离线管理功能(ENABLE/DISABLE),使用自动离线管理能够在出错时离线后适时自动恢复,不须要软件干预。
(10) CAN_NART
本成员用于设置是否使用自动重传功能(ENABLE/DISABLE),使用自动重传功能时,会一直发送报文直到成功为止。
(11) CAN_RFLM
本成员用于设置是否使用锁定接收 FIFO(ENABLE/DISABLE),锁定接收 FIFO 后,若FIFO 溢出时会丢弃新数据,不然在 FIFO 溢出时以新数据覆盖旧数据。
(12) CAN_TXFP
本成员用于设置发送报文的优先级断定方法(ENABLE/DISABLE),使能时,以报文存入发送邮箱的前后顺序来发送,不然按照报文 ID 的优先级来发送。
配置完这些结构体成员后,咱们调用库函数 CAN_Init 便可把这些参数写入到 CAN 控制寄存器中,实现 CAN 的初始化。
CAN发送及接收结构体
在发送或接收报文时,须要往发送邮箱中写入报文信息或从接收 FIFO 中读取报文信息,利用 STM32 标准库的发送及接收结构体能够方便地完成这样的工做,它们的定义见代码清单 43-2。
代码清单 43-2 CAN 发送及接收结构体
/** * @brief CAN Tx message structure definition * 发送结构体 */ typedef struct { uint32_t StdId; /*存储报文的标准标识符 11 位, 0-0x7FF. */ uint32_t ExtId; /*存储报文的扩展标识符 29 位, 0-0x1FFFFFFF. */ uint8_t IDE; /*存储 IDE 扩展标志 */ uint8_t RTR; /*存储 RTR 远程帧标志*/ uint8_t DLC; /*存储报文数据段的长度, 0-8 */ uint8_t Data[8]; /*存储报文数据段的内容 */ } CanTxMsg; /** * @brief CAN Rx message structure definition * 接收结构体 */ typedef struct { uint32_t StdId; /*存储了报文的标准标识符 11 位, 0-0x7FF. */ uint32_t ExtId; /*存储了报文的扩展标识符 29 位, 0-0x1FFFFFFF. */ uint8_t IDE; /*存储了 IDE 扩展标志 */ uint8_t RTR; /*存储了 RTR 远程帧标志*/ uint8_t DLC; /*存储了报文数据段的长度, 0-8 */ uint8_t Data[8]; /*存储了报文数据段的内容 */ uint8_t FMI; /*存储了 本报文是由通过筛选器存储进 FIFO 的, 0-0xFF */ } CanRxMsg;
这些结构体成员都是“43.2.1 2CAN 发送邮箱及 CAN 接收 FIFO”小节介绍的内容,可对比阅读,发送结构体与接收结构体是相似的,只是接收结构体多了一个 FMI 成员,说明以下:
(1) StdId
本成员存储的是报文的 11 位标准标识符,范围是 0-0x7FF。
(2) ExtId
本成员存储的是报文的 29 位扩展标识符,范围是 0-0x1FFFFFFF。 ExtId 与 StdId 这两个成员根据下面的 IDE 位配置,只有一个是有效的。
(3) IDE
本成员存储的是扩展标志 IDE 位,当它的值为宏 CAN_ID_STD 时表示本报文是标准帧,使用 StdId 成员存储报文 ID;当它的值为宏 CAN_ID_EXT 时表示本报文是扩展帧,使用 ExtId 成员存储报文 ID。
(4) RTR
本成员存储的是报文类型标志 RTR 位,当它的值为宏 CAN_RTR_Data 时表示本报文是数据帧;当它的值为宏 CAN_RTR_Remote 时表示本报文是遥控帧,因为遥控帧没有数据段,因此当报文是遥控帧时,下面的 Data[8]成员的内容是无效的。
(5) DLC
本成员存储的是数据帧数据段的长度,它的值的范围是 0-8,当报文是遥控帧时 DLC值为 0。
(6) Data[8]
本成员存储的就是数据帧中数据段的数据。
(7) FMI
本成员只存在于接收结构体,它存储了筛选器的编号,表示本报文是通过哪一个筛选器存储进接收 FIFO 的,能够用它简化软件处理。
CAN筛选器结构体
CAN 的筛选器有多种工做模式,利用筛选器结构体可方便配置,它的定义见代码清单43-3。
代码清单 43-3 CAN 筛选器结构体
/** * @brief CAN filter init structure definition * CAN 筛选器结构体 */ typedef struct { uint16_t CAN_FilterIdHigh; /*CAN_FxR1 寄存器的高 16 位 */ uint16_t CAN_FilterIdLow; /*CAN_FxR1 寄存器的低 16 位*/ uint16_t CAN_FilterMaskIdHigh; /*CAN_FxR2 寄存器的高 16 位*/ uint16_t CAN_FilterMaskIdLow; /*CAN_FxR2 寄存器的低 16 位 */ uint16_t CAN_FilterFIFOAssignment; /*设置通过筛选后数据存储到哪一个接收 FIFO */ uint8_t CAN_FilterNumber; /*筛选器编号,范围 0-27*/ uint8_t CAN_FilterMode; /*筛选器模式 */ uint8_t CAN_FilterScale; /*设置筛选器的尺度 */ FunctionalState CAN_FilterActivation; /*是否使能本筛选器*/ } CAN_FilterInitTypeDef;
这些结构体成员都是“43.2.1 4 验收筛选器”小节介绍的内容,可对比阅读,各个结构体成员的介绍以下:
(1) CAN_FilterIdHigh
CAN_FilterIdHigh 成员用于存储要筛选的 ID,若筛选器工做在 32 位模式,它存储的是所筛选 ID 的高 16 位;若筛选器工做在 16 位模式,它存储的就是一个完整的要筛选的 ID。
(2) CAN_FilterIdLow
相似地, CAN_FilterIdLow 成员也是用于存储要筛选的 ID,若筛选器工做在 32 位模式,它存储的是所筛选 ID 的低 16 位;若筛选器工做在 16 位模式,它存储的就是一个完整的要筛选的 ID。
(3) CAN_FilterMaskIdHigh
CAN_FilterMaskIdHigh 存储的内容分两种状况,当筛选器工做在标识符列表模式时,它的功能与 CAN_FilterIdHigh 相同,都是存储要筛选的 ID; 而当筛选器工做在掩码模式时,它存储的是 CAN_FilterIdHigh 成员对应的掩码,与 CAN_FilterIdLow 组成一组筛选器。
(4) CAN_FilterMaskIdLow
相似地, CAN_FilterMaskIdLow 存储的内容也分两种状况,当筛选器工做在标识符列表模式时,它的功能与 CAN_FilterIdLow 相同,都是存储要筛选的 ID; 而当筛选器工做在掩码模式时,它存储的是 CAN_FilterIdLow 成员对应的掩码,与CAN_FilterIdLow 组成一组筛选器。
上面四个结构体的存储的内容很容易让人糊涂,请结合前面的图 43-14 和下面的表43-7 理解,若是还搞不清楚,再结合库函数 CAN_FilterInit 的源码来分析。
对这些结构体成员赋值的时候,还要注意寄存器位的映射,即注意哪部分表明 STID,哪部分表明 EXID 以及 IDE、 RTR 位。
(5) CAN_FilterFIFOAssignment
本成员用于设置当报文经过筛选器的匹配后,该报文会被存储到哪个接收 FIFO,它的可选值为 FIFO0 或 FIFO1(宏 CAN_Filter_FIFO0/1)。
(6) CAN_FilterNumber
本成员用于设置筛选器的编号,即本过滤器结构体配置的是哪一组筛选器, CAN 一共有 28 个筛选器,因此它的可输入参数范围为 0-27。
(7) CAN_FilterMode
本成员用于设置筛选器的工做模式,能够设置为列表模式(宏 CAN_FilterMode_IdList)及掩码模式(宏 CAN_FilterMode_IdMask)。
(8) CAN_FilterScale
本成员用于设置筛选器的尺度,能够设置为 32 位长(宏 CAN_FilterScale_32bit)及 16 位长(宏 CAN_FilterScale_16bit)。
(9) CAN_FilterActivation
本成员用于设置是否激活这个筛选器(宏 ENABLE/DISABLE)。
配置完这些结构体成员后,咱们调用库函数 CAN_FilterInit 便可把这些参数写入到筛选控制寄存器中, 从而使用筛选器。咱们前面说若是不理解那几个 ID 结构体成员存储的内容时,能够直接阅读库函数 CAN_FilterInit 的源代码理解,就是由于它直接对寄存器写入内容,代码的逻辑是很是清晰的。
参考引用:
- 野火---《零死角玩转STM32-F429挑战者》
- 《STM32F4xx中文参考手册》
- 《Cortex-M4内核编程手册》