CAN(Controller Area Network)即控制器区域网,是一种主要用于各种设备监测及控制的网络。CAN 具有独特的设计思想, 良好的功能特征和极高的可靠性,现场抗干扰能力强。其在国内外工业控制领域已经被广泛应用, 成为最有发展前途的现场总线之一。 美国微芯公司的PIC18F458 单片机集成了CAN 通信接口,执行Bosch 公司的CAN2.0A/B 协议。它能支持CAN1.2、CAN2.0A、CAN2.0B 协议的旧版本和CAN2.0B现行版本。使用PIC18F458 单片机的嵌入式系统, 可以很方便的利用CAN 总线与外界进行数据交换。它的优点是电路接口比较简单,只需很少的外围电路就可实现CAN 通信, 受硬件限制比较少;软件编程容易实现所需功能, 只需对相关寄存器进行正确设置即可。 1 硬件接口电路 PIC18F458 与CAN 驱动芯片PCA82C250T 的接口电路如图1 所示。PCA82C250T 是驱动CAN 控制器和物理总线间的接口, 提供对总线的差动发送和接收功能。电阻R 作为CAN 终端的匹配电阻;电感L 起滤波作用。 2 CAN 控制器的操作 2.1 初始化CAN 控制 在使用CAN 之前, 必须对它的一些内部寄存器进行设置, 如CAN 控制寄存器CANCON 、波特率寄存器BRGCONx 的设置以及对邮箱进行初始化。 波特率寄存器BRGCONx(x=1,2, 3)决定了CAN 控制器的波特率、采样次数、同步跳转宽度和重同步方式,对波特率寄存器的配置步骤如下: - 设置CANCON 寄存器中的C A N 操作方式请求位为1xx,即REQOP=1xx;
- 判断CANSTAT 寄存器中的操作方式状态位是否为100,即OPMODE 是否为100,如OPMODE=100 则进入下一步;
- 设置BRGCONx(x=1,2,3)寄存器,即配置正确的波特率, 同步跳转宽度、采样次数和重同步方式;
- 设置发送邮箱和接收邮箱的标识符、邮箱数据长度、屏蔽寄存器、滤波寄存器以及初始化邮箱的数据区;
- 设置CANCON 寄存器中的CAN 操作方式请求位为000,即REQOP=000,使CAN 模块进入正常工作方式;
- 判断CANSTAT 寄存器中的操作方式状态位是否为000,即OPMODE 是否为000,如OPMODE=000 则进入下一步;
- 寄存器配置和邮箱初始化完成后,进入正常工作方式。
初始化流程图如图2 所示。 2.2 信息的发送 PIC18F458 有3 个发送邮箱缓冲器,每一个发送缓冲器的数据长度可以设置为1 ~ 8 个字节长度, 信息发送的具体步骤如下: ① 初始化发送邮箱; ② 设置相应的发送请求位为1 ,即TXBxCON bits.TXREQ=1(x=1,2,3); ③ 若CAN 总线允许发送, 则启动最高优先级信息的发送; ④ 若发送成功,则TXREQ 被清零,TXBxIF 被置1,如果中断被使能, 则会产生中断; ⑤ 若信息发送失败,则TXREQ 保持为1 ,并置位相应的状态标志。
2.3 信息的接收 PIC18F458 有2 个具有多重接收滤波器的完全接收缓冲器和1 个单独信息组合的缓冲器。接收邮箱初始化时,要设置其标识符及相关的屏蔽寄存器、接收优先级等。 MAB 寄存器接收所有来自总线的下一条信息,RXB0 和RXB1 则接收来自协议驱动的完整信息。MAB 接收所有信息, 但是只有满足过滤条件的信息才被传送到RXBx 中。 3 软件设计 下面的程序例程实现的是发送缓冲器0 向接收缓冲器0 发送数据的自测试模式, 其中接收采用中断方式,发送采用查询方式。该程序实现了PIC18F458 单片机CAN 模块的最小程序, 经过适当修改即可用于实际工程程序中, 并在实现工程中验证了它的正确性。 #include <p18f458.h> int CAN_FLAG; voidinitcan(){ TRISB=(TRISB|0X08)&0XFB; CANCON=0X80; while(CANSTAT&0X80==0)continue; BRGCON1=0X01; BRGCON2=0X90; BRGCON3=0X42; TXB0CON=0X03; TXB0SIDH=0XFF; TXB0SIDL=0XE0; TXB0DLC=0X08; TXB0D0=0X00; TXB0D1=0X01; TXB0D2=0X02; TXB0D3=0X03; TXB0D4=0X04; TXB0D5=0X05; TXB0D6=0X06; TXB0D7=0X07; RXB0SIDH=0XFF; RXB0SIDL=0XE0; RXB0CON=0X20; RXB0DLC=0X08; RXB0D0=0X00; RXB0D1=0X00; RXB0D2=0X00; RXB0D3=0X00; RXB0D4=0X00; RXB0D5=0X00; RXB0D6=0X00; RXB0D7=0X00; RXF0SIDH=0XFF; RXF0SIDL=0XE0; RXM0SIDH=0X00; RXM0SIDL=0X00; CANCON=0X40; while(CANSTAT&0X40==0)continue; PIR3=0X00; PIE3=0X01; IPR3=0X01; } #pragma interrupt can_isr #pragma code low_ISR=0x18 void low_ISR() { _asm gotocan_isr _endasm } #pragma code voidcan_isr() { if(PIR3bits.RXB0IF==1)CAN_FLAG=1; PIR3bits.RXB0IF=0; RXB0CONbits.RXFUL=0; } main(){ INTCON=0x00; initcan(); W D T C O N = 0 ; INTCON=0xc0; while(1) { TXB0CONbits.TXREQ=1; while(PIR3bits.TXB0IF!=1)continue; while(CAN_FLAG==0)continue; CAN_FLAG=0; TXB0CONbits.TXREQ=0; TXB0D0=RXB0D0+1; TXB0D1=RXB0D1+1; TXB0D2=RXB0D2+1; TXB0D3=RXB0D3+1; TXB0D4=RXB0D4+1; TXB0D5=RXB0D5+1; TXB0D6=RXB0D6+1; TXB0D7=RXB0D7+1; } }
|