CAN通讯程序C语言,基于51单片机的CAN通讯协议C语言程序

//-----------------------函数声明,变量定义--------------------------------------------------------

#include

sbit  int0 = P3^2;

//-----------------------定义寻址的基址--------------------------------------------------------

#define base_Adr 0x00

//-----------------------定义总线定时寄存器的值--------------------------------------------------------

#define SJA_BTR0 0x00                                  //该值需要用户根据实际需要的波特率进行计算

#define SJA_BTR1 0x16                                  //具体计算见文章说明

//-----------------------设置接收报文类型(标示符)--------------------------------------------------------

//该值需要用户根据实际需要重新配置

#define SJA_ACR 0x00                                  //验收代码寄存器的值

#define SJA_AMR 0x16                                  //验收屏蔽寄存器的值

//-----------------------设置输出始终类型--------------------------------------------------------

//该值需要用户根据实际需要重新配置

#define SJA_OCR 0x00                                  //输出控制寄存器的值

#define SJA_CDR 0x16                                  //始终分频寄存器的值

//-----------------------设置SJA中断,1为开中断--------------------------------------------------------

#define SJA_OIE 0                                    //溢出中断

#define SJA_EIE 0                                    //错误中断

#define SJA_TIE 0                                    //发送中断

#define SJA_RIE 0                                    //接收中断

//-----------------------定义地址指针,指向基址--------------------------------------------------------

unsigned char xdata *SJA_base_Adr = base_Adr;

//-----------------------定义硬件故障标志位--------------------------------------------------------

bit bdata    connect_OK=0;                            //connect_OK=1设备连接正常

//connect_OK=0设备连接故障

//-----------------------定义硬件故障标志位--------------------------------------------------------

bit bdata    SJA_workmode=1;                          //SJA_workmode=1SJA工作在工作模式

//SJA_workmode=0工作在复位模式

//-----------------------定义SJA1000读写缓冲区的数据结构--------------------------------------------------------

struct BASICCAN_BUFstruct{

unsigned char FrameID_H;

unsigned char FrameLENTH ;

unsigned char FrameKIND  ;

unsigned char FrameID_L3 ;

unsigned char Frame_Data[8];

}BASICCAN_FRAME,receive_BUF,send_BUF;

//BASICCAN_BUFstruct send_BUF;

//------------------------------------------------------------------------------------------------------

// 函数类别 SJA1000基本操作

// 函数名称 CANREG_write

// 入口函数 SJAREG_ADR,setting

// 出口函数 无

// 函数功能 写SJA1000的寄存器

//------------------------------------------------------------------------------------------------------

void CANREG_write(unsigned char SJAREG_ADR, unsigned char setting)

{

*(SJA_base_Adr+SJAREG_ADR)=setting;

}

//------------------------------------------------------------------------------------------------------

// 函数类别 SJA1000基本操作

// 函数名称 CANREG_write

// 入口函数 SJAREG_ADR

// 出口函数 SJAREG_data

// 函数功能 读SJA1000的寄存器

//------------------------------------------------------------------------------------------------------

unsigned char CANREG_read(unsigned char SJAREG_ADR)

{

unsigned char SJAREG_data;

SJAREG_data=*(SJA_base_Adr+SJAREG_ADR);

return(SJAREG_data);

}

//------------------------------------------------------------------------------------------------------

// 函数类别   SJA1000基本操作

// 函数名称   SJAconnect_judge

// 入口函数   无

// 出口函数   无

// 全局变量   connect_OK

// 操作寄存器 测试寄存器(地址09)

// 函数功能   判断SJA1000与控制器连接是否正常

//------------------------------------------------------------------------------------------------------

void SJAconnect_judge(void)

{

CANREG_write(0x09,0xAA);                //写AA到测试寄存器(地址09)

if(CANREG_read(0x09)==0xAA)

{

connect_OK=1;                         //连接正常

}

else

{

connect_OK=0;                         //连接故障

}

}

//------------------------------------------------------------------------------------------------------

// 函数类别   SJA1000基本操作

// 函数名称   setting_SJA_resetmode

// 入口函数   无

// 出口函数   无

// 全局变量   SJA_workmode

// 操作寄存器 控制寄存器(地址00)

// 函数功能   设置SJA工作在复位模式

//------------------------------------------------------------------------------------------------------

void setting_SJA_resetmode(void)

{

unsigned char CONTROL_REGdata;

CONTROL_REGdata=CANREG_read(0x00);

CONTROL_REGdata=CONTROL_REGdata|0x01;

CANREG_write(0x00,CONTROL_REGdata);

if((CANREG_read(0x00)&0x01)==1)

{

SJA_workmode=0;                        //置复位模式成功

}

else

{

SJA_workmode=1;                         //置复位模式失败

}

}

//------------------------------------------------------------------------------------------------------

// 函数类别   SJA1000基本操作

// 函数名称   setting_SJA_resetmode

// 入口函数   无

// 出口函数   无

// 全局变量   SJA_workmode

// 操作寄存器 控制寄存器(地址00)

// 函数功能   设置SJA工作在正常工作模式

//------------------------------------------------------------------------------------------------------

void setting_SJA_workingmode(void)

{

unsigned char CONTROL_REGdata;

CONTROL_REGdata=CANREG_read(0x00);

CONTROL_REGdata=CONTROL_REGdata&0xFE;

CANREG_write(0x00,CONTROL_REGdata);

if((CANREG_read(0x00)&0x01)==0)

{

SJA_workmode=1;                        //置工作模式成功

}

else

{

SJA_workmode=0;                         //置工作模式失败

}

}

//------------------------------------------------------------------------------------------------------

// 函数类别   SJA1000基本操作

// 函数名称   setting_SJA_rate

// 入口函数   SJA_BTR0,SJA_BTR1

// 出口函数   setting_success

// 操作寄存器 总线定时寄存器BTR1(地址07)和BTR0(地址06)

// 函数功能   设置SJA波特率

// 特殊要求   只能在复位工作模式下设置

//------------------------------------------------------------------------------------------------------

bit setting_SJA_rate(void)

{

bit setting_success;

while(SJA_workmode)

{

setting_SJA_resetmode();                   //设置SJA工作在复位模式

}

CANREG_write(0x06,SJA_BTR0);

CANREG_write(0x07,SJA_BTR1);

if((CANREG_read(0x06)==SJA_BTR0)&(CANREG_read(0x07)==SJA_BTR1))

{

setting_success=1;                        //波特率设置成功

}

else

{

setting_success=0;                         //波特率设置失败

}

return(setting_success);

}

//------------------------------------------------------------------------------------------------------

// 函数类别   SJA1000基本操作

// 函数名称   setting_SJA_dataselect

// 入口函数   SJA_ACR,SJA_AMR

// 出口函数   setting_success

// 操作寄存器 验收代码寄存器ACR(地址04)和验收屏蔽寄存器AMR(地址05)

// 函数功能   设置SJA接收数据类型

// 特殊要求   只能在复位工作模式下设置

//------------------------------------------------------------------------------------------------------

bit setting_SJA_dataselect(void)

{

bit setting_success;

while(SJA_workmode)

{

setting_SJA_resetmode();                   //设置SJA工作在复位模式

}

CANREG_write(0x04,SJA_ACR);

CANREG_write(0x05,SJA_AMR);

if((CANREG_read(0x04)==SJA_ACR)&(CANREG_read(0x05)==SJA_AMR))

{

setting_success=1;                        //滤波器设置成功

}

else

{

setting_success=0;                         //滤波器设置失败

}

return(setting_success);

}

//------------------------------------------------------------------------------------------------------

// 函数类别   SJA1000基本操作

// 函数名称   setting_SJA_CLK

// 入口函数   SJA_OCR,SJA_CDR

// 出口函数   setting_success

// 操作寄存器 输出控制寄存器OCR(地址08)和时钟分频寄存器CDR(地址31)

// 函数功能   设置SJA输出始终类型

// 特殊要求   只能在复位工作模式下设置

//------------------------------------------------------------------------------------------------------

bit setting_SJA_CLK(void)

{

bit setting_success;

while(SJA_workmode)

{

setting_SJA_resetmode();                   //设置SJA工作在复位模式

}

CANREG_write(0x08,SJA_OCR);

CANREG_write(31,SJA_CDR);

if((CANREG_read(0x08)==SJA_OCR)&(CANREG_read(31)==SJA_CDR))

{

setting_success=1;                        //滤波器设置成功

}

else

{

setting_success=0;                         //滤波器设置失败

}

return(setting_success);

}

//------------------------------------------------------------------------------------------------------

// 函数类别   SJA1000基本操作

// 函数名称   setting_SJA_interrupt

// 入口函数   SJA_OIE,SJA_EIE,SJA_TIE,SJA_RIE

// 出口函数   setting_success

// 操作寄存器 控制寄存器(00)

// 函数功能   设置SJA中断类型和中断状态

// 特殊要求   只能在复位工作模式下设置

//------------------------------------------------------------------------------------------------------

bit setting_SJA_interrupt(void)

{

bit setting_success;

unsigned char CONT_buf,temp=0;

while(SJA_workmode)

{

setting_SJA_resetmode();                   //设置SJA工作在复位模式

}

CONT_buf=CANREG_read(0x00);

temp=SJA_OIE;

temp=temp<<4;

temp=temp|SJA_EIE;

temp=temp<<3;

temp=temp|SJA_TIE;

temp=temp<<2;

temp=temp|SJA_RIE;

temp=temp<<1;

CONT_buf=(temp&0x1E)|(CONT_buf&0x01);

CANREG_write(0x00,CONT_buf);

if(CANREG_read(0x00)==CONT_buf)

{

setting_success=1;                        //滤波器设置成功

}

else

{

setting_success=0;                         //滤波器设置失败

}

return(setting_success);

}

//------------------------------------------------------------------------------------------------------

// 函数类别   SJA1000基本操作

// 函数名称   Write_SJAsendBUF

// 入口函数   无

// 出口函数   setting_success

// 操作寄存器 发送缓存器(10-19)状态寄存器02

// 函数功能   写发送缓存器

// 特殊要求   只能在工作模式下写

//------------------------------------------------------------------------------------------------------

bit Write_SJAsendBUF(void)

{

bit setting_success=0;

unsigned char i;

while(SJA_workmode==0)

{

setting_SJA_workingmode();                   //设置SJA在工作模式

}

if((CANREG_read(0x02)&0x10)==0)

{

if((CANREG_read(0x02)&0x04)!=0)

{

CANREG_write(0x10,send_BUF.FrameID_H);

CANREG_write(0x11,(send_BUF.FrameLENTH<<4)||(send_BUF.FrameKIND<<3)||(send_BUF.FrameID_L3));

if(send_BUF.FrameKIND==0)

{for(i=0;i

CANREG_write(0x12+i,send_BUF.Frame_Data[i]);

}

setting_success=1;                        //发送寄存器写成功

}

}

return(setting_success);

}

//------------------------------------------------------------------------------------------------------

// 函数类别   SJA1000基本操作

// 函数名称   Write_SJAsendBUF

// 入口函数   无

// 出口函数   setting_success

// 操作寄存器 接收缓存器(20-29)状态寄存器02

// 函数功能   写发送缓存器

// 特殊要求   只能在工作模式下写

//------------------------------------------------------------------------------------------------------

bit read_SJAreceiveBUF(void)

{

bit setting_success=0;

unsigned char i;

while(SJA_workmode==0)

{

setting_SJA_workingmode();                   //设置SJA在工作模式

}

if((CANREG_read(0x02)&0x01)!=0)

{

if((CANREG_read(0x02)&0x10)==0)

{

receive_BUF.FrameID_H=CANREG_read(0x20);

receive_BUF.FrameLENTH=((CANREG_read(0x21)&0xF0)>>4);

receive_BUF.FrameKIND=((CANREG_read(0x21)&0x08)>>3);

receive_BUF.FrameID_L3=(CANREG_read(0x21)&0x07);

if(receive_BUF.FrameKIND==0)

{for(i=0;i

receive_BUF.Frame_Data[i]=CANREG_read(0x22+i);

}

setting_success=1;                        //接收寄存器读成功

}

}

return(setting_success);

}

//------------------------------------------------------------------------------------------------------

// 函数类别   供调用子程序

// 函数名称   SJA1000_init

// 入口函数   无

// 出口函数   无

// 操作寄存器  1)控制寄存器(地址00)

//             2)收代码寄存器ACR(地址04)

//             3)验收屏蔽寄存器AMR(地址05)

//             4)总线定时寄存器BTR0(地址06)

//             5)总线定时寄存器BTR1(地址07)

//             6)输出控制寄存器OCR(地址08)

//             7)测试寄存器(地址09)

//             8)和时钟分频寄存器CDR(地址31)

// 函数功能   SJA1000初始化设置

// 特殊要求   在复位模式进行,初始化结束进入工作状态

//------------------------------------------------------------------------------------------------------

void SJA1000_init(void)

{

while(connect_OK==0)

{

SJAconnect_judge();             //检测设备连接

}

while(SJA_workmode)

{

setting_SJA_resetmode();         //置SJA1000为复位工作模式

}

while(setting_SJA_rate()==0)

{

setting_SJA_rate();               //设置总线波特率

}

while(setting_SJA_dataselect()==0)

{

setting_SJA_dataselect();          //设置SJA接收数据的格式(标示位)

}

while(setting_SJA_CLK()==0)

{

setting_SJA_CLK();                 //设置SJA输出始终的形式

}

}

//------------------------------------------------------------------------------------------------------

// 函数类别   中断处理函数

// 函数名称   send_interrupt

// 入口函数   无

// 出口函数   无

// 操作寄存器

// 函数功能   接收中断处理函数

//------------------------------------------------------------------------------------------------------

send_interrupt()

{

}

//------------------------------------------------------------------------------------------------------

// 函数类别   发送中断处理函数

// 函数名称   receive_interrupt

// 入口函数

// 出口函数

// 操作寄存器

// 函数功能    发送中断处理函数

//------------------------------------------------------------------------------------------------------

receive_interrupt()

{

}

//------------------------------------------------------------------------------------------------------

// 函数类别   中断函数

// 函数名称   SJA_INTR

// 入口函数   无

// 出口函数   无

// 操作寄存器 中断寄存器(地址03)

// 函数功能   中断分析,判断是什么中断,调用相应的中断处理函数

//------------------------------------------------------------------------------------------------------

void SJA_INTR() interrupt 0 using 1 //CanBus接口芯片产生中断(INTR0)

{

unsigned char sta;

unsigned char temp;

EX0 = 0;

sta = CANREG_read(3);            //读中断寄存器

temp = sta & 0x20;

if(temp == 0x20)

SJA1000_init();

temp = sta & 0x04;

if(temp == 0x04)

SJA1000_init();                 //消极错误中断,错误报警中断,均导致重启

temp = sta & 0x02;

if(temp == 0x02)                  //发送中断处理

{

send_interrupt();

}

temp = sta & 0x01;

if(temp == 0x01)                   //接收中断,接收数据

{

receive_interrupt();

}

EX0 = 1;

}

main()

{

}

本程序是基于51单片机的CAN(sja1000)通信协议的操作程序,利用51单片机的中断来操作,每个函数都有详细的注释,希望能帮助到初学者,在main函数中没有任何函数调用,自己可以根据需要进行调用。

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: CAN通信是一种用于实现微控制器之间的高速数据传输的通信协议。51单片机(也称为STC单片机)可以通过CAN总线与其他设备进行通信。CAN总线是一种双线制的通信系统,由CAN发送器和CAN接收器组成。 在CAN通信中,可以使用标准帧和扩展帧进行数据传输。标准帧用于传输11位的标识符,扩展帧可以传输29位的标识符,这使得CAN通信可以支持更多的设备和更大的数据量。 为了在51单片机中实现CAN通信,首先需要连接CAN收发器和CAN控制器。然后,程序员需要编写代码来控制CAN收发器和CAN控制器之间的数据传输。这包括设置CAN收发器的通信速率、初始化CAN控制器的寄存器以及配置CAN消息的发送和接收。 一旦CAN通信的硬件和软件设置完成,51单片机可以发送和接收CAN消息。发送CAN消息时,程序员需要设置消息的标识符、数据长度以及数据内容。接收CAN消息时,程序员需要检查CAN接收缓冲区,以判断是否有新的CAN消息到达,并读取相应的数据。 通过CAN通信51单片机可以实现与其他设备的实时数据交换。这对于控制系统、汽车电子以及工业自动化等应用非常有用。CAN通信具有高可靠性、高带宽以及抗干扰能力强的特点,使得51单片机成为许多应用中的理想选择。 ### 回答2: 51单片机CAN通信是指使用51单片机通过CAN总线与其他设备进行通信的一种方式。CAN(Controller Area Network,控制器局域网络)是一种快速、可靠的串行总线通信协议,广泛应用于汽车电子、工业自动化等领域。 51单片机CAN通信主要包括以下几个方面的内容: 1. CAN总线:CAN总线是一种双线制的串行通信总线,其中包含CAN_H和CAN_L两个信号线。CAN总线具有抗干扰能力强、传输速率高、可靠性好等特点,因此在工业控制领域得到广泛应用。 2. 51单片机51单片机是一款常用的低成本、低功耗的单片机,具备对CAN通信的支持。它可以通过CAN收发器与CAN总线相连接,实现与其他CAN设备的通信。 3. CAN控制器:51单片机通过内部的CAN控制器与CAN总线连接。CAN控制器负责处理接收和发送CAN数据帧,控制数据的传输和接收。 4. 编程:对于51单片机CAN通信,需要通过编程来实现。通过设置CAN控制寄存器和CAN速率寄存器,配置CAN通信参数,如波特率、滤波器等。然后通过发送和接收函数,实现与其他设备之间的数据交换。 5. 通信协议:CAN通信协议是指在CAN总线上进行数据交换时所采用的一种规范。它定义了数据帧的格式、报文的优先级、错误检测和纠正等。根据具体应用需求,可以选择使用标准CAN协议(CAN 2.0A/B)或扩展CAN协议(CAN FD) 综上所述,51单片机CAN通信是一种使用51单片机与其他设备进行串行通信的方法。通过使用CAN总线、CAN控制器和编程配置,可以实现与其他设备之间的数据交换。这种通信方式在工业控制、汽车电子等领域具有广泛的应用前景。 ### 回答3: 51单片机CAN通信是指使用51单片机来实现CAN总线通信。 CAN(Controller Area Network)是一种广泛应用于汽车、工业控制等领域的串行通信协议。CAN总线具有高可靠性、高实时性和高容错性的特点,能够在复杂的电磁环境中稳定传输数据。 在51单片机中,可以通过使用外部的CAN控制器芯片来实现CAN通信。通过与CAN控制器芯片的连接,51单片机可以实现CAN通信的发送和接收功能。 要实现CAN通信,首先需要初始化CAN控制器芯片,在51单片机中设置CAN通信的波特率、报文格式等参数。然后可以使用相应的函数来发送和接收CAN数据。 在发送数据时,可以将要发送的数据封装成CAN数据帧,并通过相应的函数将数据发送到CAN总线上。在接收数据时,可以通过循环读取CAN控制器接收到的数据,并进行相应的处理。 通过51单片机实现CAN通信可以应用于许多领域,比如汽车电子系统中的ECU(Engine Control Unit)通信、工业自动化中的设备联网等。 总之,51单片机可以通过与CAN控制器芯片的连接来实现CAN总线通信,通过相应的函数和设置参数,可以实现CAN数据的发送和接收,为各个领域的应用提供了稳定可靠的通信手段。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值