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

发布者:reaper2009最新更新时间:2017-02-04 来源: eefocus关键字:51单片机  CAN通讯协议 手机看文章 扫描二维码
随时随地手机看文章

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

#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函数中没有任何函数调用,自己可以根据需要进行调用。


关键字:51单片机  CAN通讯协议 引用地址:基于51单片机的CAN通讯协议C语言程序

上一篇:WPF与51单片机之间的串口通信
下一篇:51单片机实现4位数以内的加减法

推荐阅读最新更新时间:2024-03-16 15:32

51单片机总结--C语言之预处理
编译预处理器是C语言编译器的一个重要组成部分。很好的利用C语言的预处理命令可以增强代码的可读性,灵活性,和易于修改等特点,便于程序的结构化。 预处理命令由符号 # 开头,包括宏定义,文件包含,条件处理三个部分。 其中条件编译我还没有用过,所以就详细介绍一下宏定义和文件包含。 一.宏定义 宏定义命令为#define,它的作用就是实现用一个简单易读的字符串来代替 另一个字符串。增加程序的可读性,和维护性。 宏定义分为不带参数的宏定义,和带参数的宏定义。 不带参的宏定义: 一般格式:#define 标识符 常量表达式 例如用一个字符代替一个常数 #define PI 3.1415926 当程序中
[单片机]
51单片机产生pwm 占空比10khz频率 Proteus仿真程序
仿真原理图如下 单片机源程序如下: #include reg51.h #define uchar unsigned char #define uint unsigned int sbit pwm=P1^1; uchar num=0; main() { TMOD=0x02; TH0=TL0=206; ET0=1; TR0=1; EA=1; while(1); } void init() interrupt 1 { num++; if(num =6) pwm=1; else pwm=0; if(num==10)
[单片机]
<font color='red'>51单片机</font>产生pwm 占空比10khz频率 Proteus仿真程序
51单片机做一个计时器
同样我们使用的仍然是STC的8051单片机,这段代码是用来做一个计时器,通过数码管实时显示出来,因为板子上数码管有6个,所以可以设置小时,分钟,秒三个单位,在Proteus上也是可以模拟的,计时结束可以用蜂鸣器,或者继电器接口连接 一个闹铃,闹铃用电磁式的就可以,当然简单来说蜂鸣器就是一个不错的选择。 程序中没有设置小时单位,想要加上的小伙伴可以用显示分钟,秒的方法同样显示出来,但添加后要注意延时的时间,时间设置不当可能会出现显示不清楚,数字抖动,或者未选中的数码管也隐隐发光(“鬼影”)。具体大小可以自己调试着来,也是很简单的。下边是源代码: #include reg52.h #define uint unsigne
[单片机]
51单片机】普通I/O口模拟SPI口C语言程序
89C51系列单片机都不带SPI口,所在在这种情况下,我们可以模拟SPI口来现实我们要的功能,程序如下: //-----------------------函数声明,变量定义------------ #include #include sbit SCK=P1^0; // 将p1.0口模拟时钟输出 sbit MOSI=P1^1; // 将p1.1口模拟主机输出 sbit MISO=P1^2; // 将p1.1口模拟主机输入 sbit SS1=P1^3; // 将p1.1口模拟片选 #define delayNOP(); {_nop_();_nop_();_nop_();_nop_();}; //-------------------
[单片机]
51单片机实现光源的跟踪
AD采集两路电压,并比较,控制电机转动方向,并将数据传至上位机 西安理工大学--自动化与信息工程学院 邹艺良 编写(2014.08.25) 邮箱:262276047@qq.com //******************************* //西安理工大学--自动化与信息工程学院--邹艺良 //******************************* #include #include #include void ADC_init(); int get_ADC_result(char ch); unsigned int show_ADC_result_ch0(); unsigned in
[单片机]
51单片机+PWM控制渐变七彩灯C51程序
一、硬件介绍: (采用5050LED 2W) RGB三色LED控制引脚分别为单片机P1.2 、 P1.1 、 P1.0。LED正极接主电源(24V)正极,负极接驱动3颗三极管的集电极,单片机控制脚分别接3颗NPN三极管,三极管发射极接地,而单片机的供电是来自三端稳压器7805,祥细原理如下: 二、实物图片: 三、软件部分: 1、/*原理: 先亮红灯(保持一会儿)----红绿过度(绿加1、红减1循环240次)------ 绿灯亮起(保持一会儿)----绿兰过度(兰加1、绿减1循环240次) -----兰灯亮起(保持)-----兰白过度(绿加1、红加1循环240次、兰不变) ---白红过度,技术支持网站: http
[单片机]
<font color='red'>51单片机</font>+PWM控制渐变七彩灯C51程序
Keil C51单片机集成开发环境编程与调试教程
同 VC 之类的通用 C 语言集成开发环境(IDE)一样,Keil 也采用“工程” (Project)的方式管理源代码及相关文件,这种管理方式为由多个源代码文件组 成的大型程序开发提供了方便。不管是最简单的 C51程序,还是复杂的多文件 程序都需要以下步骤: 1)先建立新的工程文件; 2)在工程中新建源代码文件,或是将已经存在的源代码文件加入工程; 3)编译; 4)调试,修正错误再编译; 5)将生成的二进制文件*.hex 烧入单片机。 本教程重点介绍上述前 4 个步骤。 二、Keil中新建工程的步骤 1. 单击菜单“Project——New uVision Project……” 出现新建工程对话框: 在此对话框中选择存
[单片机]
Keil C<font color='red'>51单片机</font>集成开发环境编程与调试教程
51单片机定时器和计数器
8051单片机内部有两个定时/计数器T0及T1,具有定时和计数两种功能。T0及T1在计数过程中不需要CPU参与,也不影响CPU的其他工作。当计数溢出后,定时/计数器给出中断信号,申请CPU停止当前的工作,去处理预先设定的中断事件。 一、T1内部结构 定时器工作模式:对内部时钟信号计数。由于时钟频率是定值,所以可根据计数值计算出定时时间。 计数器工作模式:是对加在T1(P3.5)引脚上的外部脉冲进行计数。 二、计数功能 计数器用于统计从TO(P3.4)和Tl(P3.5)两个引脚输入脉冲的负跳变数量。负跳变是指前一个机器周期采样为高电平,后一个机器周期为低电平。每输入一个脉冲负跳变,计数器加1。 输入脉冲的高电平与低电平至少应保持
[单片机]
<font color='red'>51单片机</font>定时器和计数器
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved