嵌入式的工程师一般都知道CAN总线广泛应用到汽车中,其实船舰电子设备通信也广泛使用CAN,随着国家对海防的越来越重视,对CAN的需求也会越来越大。这个暑假,通过参加苏州社会实践,去某船舶电气公司实习几周,也借此机会,学习了一下CAN总线。
▍概述CAN(ControllerAreaNetwork)即控制器局域网,是一种能够实现分布式实时控制的串行通信网络。想到CAN就要想到德国的Bosch公司,因为CAN就是这个公司开发的(和Intel)CAN有很多优秀的特点,使得它能够被广泛的应用。比如:传输速度最高到1Mbps,通信距离最远到10km,无损位仲裁机制,多主结构。近些年来,CAN控制器价格越来越低,很多MCU也集成了CAN控制器。现在每一辆汽车上都装有CAN总线。一个典型的CAN应用场景:▍CAN总线标准CAN总线标准只规定了物理层和数据链路层,需要用户自定义应用层。不同的CAN标准仅物理层不同。CAN收发器负责逻辑电平和物理信号之间的转换。将逻辑信号转换成物理信号(差分电平),或者将物理信号转换成逻辑电平。CAN标准有两个,即IOS11898和IOS11519,两者差分电平特性不同。高低电平幅度低,对应的传输速度快;*双绞线共模消除干扰,是因为电平同时变化,电压差不变。物理层#defineREG_BASE_ADDR0xA000//寄存器基址unsignedchar*SJA_CS_Point=(unsignedchar*)REG_BASE_ADDR;//写SJA1000寄存器voidWriteSJAReg(unsignedcharRegAddr,unsignedcharValue){*(SJA_CS_Point+RegAddr)=Value;return;}//读SJA1000寄存器unsignedcharReadSJAReg(unsignedcharRegAddr){return(*(SJA_CS_Point+RegAddr));}
将缓存区的数据连续写入寄存器……for(i=0;i<len;i++){WriteSJAReg(RegAdr+i,ValueBuf[i]);}……
将连续多个寄存器连续读入缓存区……for(i=0;i<len;i++){ReadSJAReg(RegAdr+i,ValueBuf[i]);}……
头文件包含方案:- 每个程序包含用到的头文件
- 每个程序包含一个公用头文件,公用头文件包含所有其他头文件
#ifndef__CONFIG_H__//防止头文件被重复包含#define__CONFIG_H__#include<8051.h>//包含80C51寄存器定义头文件#include"SJA1000REG.h"//包含SJA1000寄存器定义头文件//定义取字节运算#defineLOW_BYTE(x)(unsignedchar)(x)#defineHIGH_BYTE(x)(unsignedchar)((unsignedint)(x)>>8)//定义振荡器时钟和处理器时钟频率(用户可以根据实际情况作出调整)#defineOSCCLK11059200UL//宏定义MCU的时钟频率#defineCPUCLK(OSCCLK/12)#endif//__CONFIG_H__
SJA1000上电后处于复位状态,必须初始化后才能工作。(1)置位模式寄存器Bit0位进入复位模式;(2)设置时钟分频寄存器选择时钟频率、CAN模式;(3)设置验收滤波,设定验证码和屏蔽码;(4)设置总线定时器寄存器0、1设定CAN波特率;(5)设置输出模式;(6)清零模式寄存器Bit0位退出复位模式;模式寄存器只检测模式:SJA1000发送CAN帧时不检查应答位;只听模式:此模式下SJA1000不会发送错误帧,用于自动检测波特率;SJA1000以不同的波特率接收CAN帧,当收到CAN帧时,表明当前波特率与总线波特率相同。波特率设置CAN总线无时钟,使用异步串行传输;波特率是1秒发送的数据位;CAN帧发送:发送CAN帧的步骤:1.检测状态寄存器,等待发送缓冲区可用;2.填充报文到发送缓冲区;3.启动发送。SJA1000具有一个12字节的缓冲区,要发送的报文可以通过寄存器16-28写入,也可通过寄存器96-108写入或读出设置发送模式
charSetSJASendCmd(unsignedcharcmd){unsignedcharret;switch(cmd){default:case0:ret=SetBitMask(REG_CAN_CMR,TR_BIT);//正常发送break;case1:ret=SetBitMask(REG_CAN_CMR,TR_BIT|AT_BIT);//单次发送break;case2:ret=SetBitMask(REG_CAN_CMR,TR_BIT|SRR_BIT);//自收自发break;case0xff:ret=SetBitMask(REG_CAN_CMR,AT_BIT);//终止发送break;}returnret;}
发送函数unsignedcharSJA_CAN_Filter[8]={//定义验收滤波器的参数,接收所有帧0x00,0x00,0x00,0x00,//ACR0~ACR30xff,0xff,0xff,0xff//AMR0~AMR3};unsignedcharSTD_SEND_BUFFER[11]={//CAN发送报文缓冲区0x08,//帧信息,标准数据帧,数据长度=80xEA,0x60,//帧ID=0x7530x55,0x55,0x55,0x55,0xaa,0xaa,0xaa,0xaa//帧数据};voidmain(void)//主函数,程序入口{TImerInit();//初始化D1=0;SJA1000_RST=1;//硬件复位SJA1000TImerDelay(50);//延时500msSJA1000_RST=0;SJA1000_Init(0x00,0x14,SJA_CAN_Filter);//初始化SJA1000,设置波特率为1Mbps//无限循环,main()函数不允许返回for(;;){SJASendData(STD_SEND_BUFFER,0x0);TImerDelay(100);//延时1000ms}}
为什么帧ID是0x753,这与CAN帧在缓冲区的存储格式有关。终端电阻非常重要,当波特率较高而且没加终端电阻时,信号过冲非常严重。SJA1000有64个字节的接收缓冲区(FIFO),这可以降低对MCU的要求。MCU可以通过查询或中断的方式确定SJA1000接收到报文后读取报文。