SAM4E单片机之旅——19、CAN间通信
扫描二维码
随时随地手机看文章
CAN协议具有良好的可靠性,在工业中应用广泛。这次就先熟悉CAN的基本功能。
开发板有两个CAN,每个CAN有8个信箱。这次内容是从CAN0的信箱0发送数据到CAN1的信箱0。
除本次使用的功能外,CAN还有远程帧、强大的错误处理功能。
一、电路CAN总线上的逻辑数值是用显性电平和隐性电平表示的。“显性”的意思是指在同时传输显性电平和隐性电平时,总线上呈现的是显性电平。显性电平表示逻辑“0”,隐性电平表示逻辑“1”。
在使用CAN的过程中,需要使用一个CAN收发器进行电平的转换与解释。开发板使用的CAN收发器为SN65HVD234,其接线如下图所示:
其中CANTXx和CANRXx引脚可以复用为CAN的外设。而在使用该收发器时,需要将CANRXxEN驱动为高电平以启用收发器的接收功能,将CANTXxRS驱动求低电平以启用发送功能。
在实验的时候,需要将这两个口(J13和J14)使用线缆连接起来。当连接完成而未通电时,可以测得CANH和CANL是短路状态的。
二、CAN网络参数及波特率假设MCK为96 MHz,需要设置的CAN波特率为1000 Kbps。
CAN的波特率的设置不是那么的直接。CAN定义了一个名为“原子时间(TQ)”的最小时间单位;然后把一个比特的传输过程分为若干阶段(同步段、传播时间段、相位缓冲端1、相位缓冲段2),每个阶段的时间均是由TQ的数量表示。
SAM4中,时间TQ用“CAN系统时钟(CSC)”表示。波特率相关的参数均通过CAN波特率寄存器(CAN_BR)设置。
TQ(CSC)设置。组成每个位时间的TQ数量的范围为8—25。为取整,这里将数量选择为16。所以CAN系统时钟的频率为CAN波特率的16倍,即16 MHz。再所以需要将MCK进行6分频。根据BRP字段的作用方法,需要将BRP字段设置为5。
同时,可以计算出每个TQ的长度为62.5 ns。
同步段固定为1 TQ。
传播时间端PROP_SEG需要根据硬件相关的信息确定,用于吸收网络的物理(发送单元、总线、接收单元)延迟。该段的时间需要为总物理延迟的2倍。在芯片手册的示例中,该延迟为190 ns。所以该段的时长需要设置为380 ns,即约6 TQ。将PROPAG字段设置为5即可达到目的。
剩下的16-1-6=9 TQ,均用与相位缓冲段。在Atmel的CAN中,需要2 TQ确定总线的电平。因为采样点位于相位缓冲段2的起始,所以它的长度不能少于2 TQ。这里使两个阶段尽量等长,所以让相位缓冲段1设置为4 TQ,段2设置为5 TQ。将PHASE1和PHASE2分别设置为3和4即可。
再补偿宽度。最小可配置为1 TQ,最多可配置为相位缓冲段1和4 TQ间的较小值。这里配置为4 TQ。将SJW段设置为3即可。
具体设置代码如下:
12345678constuint32_t can_br = CAN_BR_BRP(5)| CAN_BR_PROPAG(5)| CAN_BR_PHASE1(3)| CAN_BR_PHASE2(4)| CAN_BR_SJW(3)| CAN_BR_SMP_ONCE;CAN0->CAN_BR = can_br;CAN1->CAN_BR = can_br;三、CAN初始化GPIO及PMC设置。注意将PE1和PE3驱动为高电平,PE0和PE2驱动为低电平。
网络参数设置。在启用CAN之前,需要设置好网络参数。
启用CAN。CAN使能后,需要和总线进行同步。在连续检测到11个隐性位时,CAN进入唤醒状态,且WAKEUP位置位:
1234CAN0->CAN_MR = CAN_MR_CANEN;CAN1->CAN_MR = CAN_MR_CANEN;while( ((CAN0->CAN_SR & CAN_SR_WAKEUP) == 0)|| ((CAN1->CAN_SR & CAN_SR_WAKEUP) == 0) );信箱设置。通过设置CAN_MMR的MOT字段即可设置信箱的类型。由于这个设置是立即生效的,所以在设置这个字段时,需要先(或同时)完成其他相关信息的设置。同时,在修改设置时,应该先关闭信箱。
发送信箱需要先设置好的只有优先级:
123#define TX_MB (CAN0->CAN_MB + 0)TX_MB->CAN_MMR = CAN_MMR_PRIOR(0)| CAN_MMR_MOT_MB_TX;接收信箱需要先设置好ID相关的信息。简单起见,这里只使用标准格式的帧,即只指定MIDvA部分,同时MIDE位指定为0(默认)。由于符合接收条件的ID设置为1个,即需要比较接收ID所有的位,所以将CAN_MAM的MIDvA字段全部置1。
12345#define RX_MB (CAN1->CAN_MB + 0)#define CAN_COMM_ID 5RX_MB->CAN_MID = CAN_MID_MIDvA(CAN_COMM_ID);RX_MB->CAN_MAM = CAN_MAM_MIDvA(~(uint32_t)0);RX_MB->CAN_MMR = CAN_MMR_MOT_MB_RX;四、数据传输通过UART读取一个数字:
12intnum;scanf("%d", &num);通过信箱发送数据。
假设int为4字节,则通过CAN_MDL即可表示所需信息。发送时,在确定信箱可用后,需要指定好信息ID。然后向CAN_MCR写入信息长度(用byte表示),同时写入MTCR位以开始发送操作。最后,在发送完成后,CAN_MSR的MRDY位重新置位。
123456789// 等待信箱可用while(!(TX_MB->CAN_MSR & CAN_MSR_MRDY));TX_MB->CAN_MID = CAN_MID_MIDvA(CAN_COMM_ID); // IDTX_MB->CAN_MDL = num; // 低4字节数据TX_MB->CAN_MCR = CAN_MCR_MDLC(4) // 数据长度| CAN_MCR_MTCR; // 开始尝试发送printf("-I- Sending message from TX mailbox...rn");// 等待发送完成while(!(TX_MB->CAN_MSR & CAN_MSR_MRDY));通过信箱接收数据。
通过查询CAN_MSR的MRDY位可以确定是否接收到了数据,然后在CAN_MSR的MDLC字段可以确定信息长度。在完成数据接收后,需要向CAN_MCR写入MTCR字段以完成本次接收,从而开始下一次信息接收工作。
123456789101112// 等待信息接收完成while(!(RX_MB->CAN_MSR & CAN_MSR_MRDY));// 检查信息长度constintrec_len =(RX_MB->CAN_MSR & CAN_MSR_MDLC_Msk) >> CAN_MSR_MDLC_Pos;if(rec_len == 4) {// 读取信息并打印printf("-I- Data read from RX mailbox: %d rn",(int)RX_MB->CAN_MDL);}// 开始下一次接收RX_MB->CAN_MCR =