Chinaunix首页 | 论坛 | 博客
  • 博客访问: 1600363
  • 博文数量: 354
  • 博客积分: 8137
  • 博客等级: 中将
  • 技术积分: 5137
  • 用 户 组: 普通用户
  • 注册时间: 2009-11-26 15:40
文章分类

全部博文(354)

文章存档

2010年(300)

2009年(54)

分类: 嵌入式

2010-09-15 13:49:11

/*************************************************
*功能描述: CEPARK CAN开发板-CAN to RS232
*说明:将两块CAN总线开发板CANH---CANH,CANL---CANL相连
* 通过串口调试软件发送数据到单片机中,数码管显示相应的数值,并通过CAN发送到另一块CAN开发板上。
* 如发送a,显示10.a即为16进制中的10.
*版本:v2.00
*声明:程序来自网络,欢迎网友交流改进
* 活跃论坛,共同进步
*作者:协成电子    、hnrain
*时间:2010/09/14
*邮箱:hnrain1004@gmail.com    ,HQL19982003@163.com ,qq:87675298
*网站:bbs.cepark.com ;写程序如果有问题,可以在这里讨论
*我们的口号:让不懂can的学习者,也能轻松玩转can

*推荐教材:
饶运涛,邹继军《现场总线 CAN 原理与应用技术》.北京:北京航空航天大学出版社 (这本书,看着不错)
周立功《增强型80c51单片机速成与实战》还有周立功写的一些can通信程序算法很不错,值得学习
本程序,融入了那么一点点点点点,他们的编程思想,哈哈....仅供学习用
****************************************************/

#include <reg52.h>
#include <intrins.h>
#include "sjapelican.h"
#include "config.h"


void INT0_Data(void) interrupt 0
{//INT0按键为计数按键

    EA = 0;
    Txd_data++; //存储计数结果,并为待发送的数据

    Peli_TXD();
    EA = 1;
}

void Peli_RXD(void) interrupt 2
{//接收数据函数,在中断服务程序中调用


    uint8 data Status;
    EA = 0;//关CPU中断


    Status = SJA_IR;
    if(Status & RI_BIT)
    {//IR.0 = 1 接收中断

        RX_buffer[0] = SJA_RBSR0;
        RX_buffer[1] = SJA_RBSR1;
        RX_buffer[2] = SJA_RBSR2;
        RX_buffer[3] = SJA_RBSR3;
        RX_buffer[4] = SJA_RBSR4;
        RX_buffer[5] = SJA_RBSR5;
        RX_buffer[6] = SJA_RBSR6;
        RX_buffer[7] = SJA_RBSR7;
        RX_buffer[8] = SJA_RBSR8;
        RX_buffer[9] = SJA_RBSR9;
        RX_buffer[10] = SJA_RBSR10;
        RX_buffer[11] = SJA_RBSR11;
        RX_buffer[12] = SJA_RBSR12;

        SJA_CMR = RRB_BIT;
        Status = SJA_ALC;//释放仲裁随时捕捉寄存器

        Status = SJA_ECC;//释放错误代码捕捉寄存器

        RS232_TXD();
    }
    SJA_IER = RIE_BIT;// .0=1--接收中断使能;


    Rxd_data = RX_buffer[5];

    EA = 1;//打开CPU中断

}

void RS232_RXD(void) interrupt 4    //接收数据函数,在中断服务程序中调用

{
    EA = 0;//关闭所有中断

    ES = 0;//关闭串口

    if(RI==1)
    {
        RI = 0;
        Txd_data = SBUF;
        Peli_TXD();//232接受过来的数据发到can总线上去;

    }
    _nop_();
    ES = 1;//打开串口

    EA = 1;//打开中断

}

void MCU_Init(void)
{
    //CPU初始化

    SJA_RST = 0;//SJA1000复位有效

    mDelay(10);    //延时

    SJA_RST = 1;//CAN总线复位管脚,复位无效

    SJA_CS = 0;//CAN总线片选有效

    EX1 = 1;//外部中断1使能;CAN总线接收中断

    IT1 = 0;//CAN总线接收中断,低电平触发

    IT0 = 1;//外部中断0负边沿触发

    EX0 = 1;//打开外部中断0

    EA = 1; //打开总中断

}

void RS232_Init(void)
{
   SCON=0x50; //串口工作方式1,允许接收

   PCON=0x80; //波特率翻倍

   //TCON=0;

   TMOD=0x20; //T1用于串口波特率控制

   TL1=0xfa; //初始化T1,波特率为9600(晶振11.0592M)

   TH1=0xfa;
   TR1=1; //开定时器

   ES = 1; //打开串口中断

}
void main(void)
{
    MCU_Init();
    RS232_Init();
    Peli_Init();
 // mDelay(1);

    while(1)
    {
        LED_Disp_Seg7();

    }

}
//SJA1000 的初始化

void Peli_Init(void)
{
    uint8 bdata Status;
    do
    {// .0=1---reset MODRe,进入复位模式,以便设置相应的寄存器

     //防止未进入复位模式,重复写入

        SJA_MOD = RM_BIT |AFM_BIT;
    Status = SJA_MOD ;
    }
    while(!(Status & RM_BIT));

    SJA_CDR = CANMode_BIT|CLKOff_BIT;// CDR.3=1--时钟关闭, .7=0---basic CAN, .7=1---Peli CAN

    SJA_BTR0 = 0x03;
    SJA_BTR1 = 0x1c;//16M晶振,波特率125Kbps

    SJA_IER = RIE_BIT;// .0=1--接收中断使能; .1=0--关闭发送中断使能

    SJA_OCR = NormalMode|Tx0PullDn|OCPOL1_BIT|Tx1PullUp;// 配置输出控制寄存器

    SJA_CMR = RRB_BIT;//释放接收缓冲器


    SJA_ACR0 = 0x11;
    SJA_ACR1 = 0x22;
    SJA_ACR2 = 0x33;
    SJA_ACR3 = 0x44;//初始化标示码


    SJA_AMR0 = 0xff;
    SJA_AMR1 = 0xff;
    SJA_AMR2 = 0xff;
    SJA_AMR3 = 0xff;//初始化掩码


    do
    {
    SJA_MOD = AFM_BIT;
    Status = SJA_MOD;
     }
    while( Status & RM_BIT );

}//SJA1000 的初始化



void Peli_TXD( void )
{
    uint8 data Status;
//初始化标示码头信息

    TX_buffer[0] = 0x88;//.7=0扩展帧;.6=0数据帧; .3=1数据长度

    TX_buffer[1] = 0x01;//本节点地址

    TX_buffer[2] = 0x02;//

    TX_buffer[3] = 0x03;//

    TX_buffer[4] = 0x04;//


//初始化发送数据单元

    TX_buffer[5] = Txd_data;
    TX_buffer[6] = 0x22;
    TX_buffer[7] = 0x33;
    TX_buffer[8] = 0x44;//

    TX_buffer[9] = 0x55;//

    TX_buffer[10] = 0x66;//

    TX_buffer[11] = 0x77;//

    TX_buffer[12] = 0x88;//


    do
    {
        Status = SJA_SR;
    }
    while( Status & RS_BIT); //SR.4=1 正在接收,等待

    
    do
    {
        Status = SJA_SR;
    }
    while(!(Status & TCS_BIT)); //SR.3=0,发送请求未处理完,等待


    do
    {
        Status = SJA_SR;
    }
    while(!(Status & TBS_BIT)); //SR.2=0,发送缓冲器被锁。等待


    SJA_TBSR0 = TX_buffer[0];
    SJA_TBSR1 = TX_buffer[1];
    SJA_TBSR2 = TX_buffer[2];
    SJA_TBSR3 = TX_buffer[3];
    SJA_TBSR4 = TX_buffer[4];
    SJA_TBSR5 = TX_buffer[5];
    SJA_TBSR6 = TX_buffer[6];
    SJA_TBSR7 = TX_buffer[7];
    SJA_TBSR8 = TX_buffer[8];
    SJA_TBSR9 = TX_buffer[9];
    SJA_TBSR10 = TX_buffer[10];
    SJA_TBSR11 = TX_buffer[11];
    SJA_TBSR12 = TX_buffer[12];

    SJA_CMR = TR_BIT;//置位发送接收请求

}

void mDelay(uint16 mtime)
{
    for(; mtime > 0; mtime--)
    {
        uint8 j = 244;
        while(--j);
    }
}

void LED_Disp_Seg7()
{
   LedCtrl = LedCtrl | 0xf0;

   DisBuff[0] = Txd_data%10;//取个位数

   DisBuff[1] = Txd_data%100/10; //取十位数

   DisBuff[2] = Rxd_data%10;     //百位数

   DisBuff[3] = Rxd_data%100/10; //千位数

  
   LedPort = LED_Disp[DisBuff[0]];
   LedCtrl = LedCtrl & 0x7f;
   mDelay(5);
   LedCtrl = LedCtrl | 0xf0;

   LedPort = LED_Disp[DisBuff[1]];
   LedCtrl = LedCtrl & 0xbf;
   mDelay(5);
   LedCtrl = LedCtrl | 0xf0;

   LedPort = LED_Disp[DisBuff[2]];
   LedCtrl = LedCtrl & 0xdf;
   mDelay(5);
   LedCtrl = LedCtrl | 0xf0;

   LedPort = LED_Disp[DisBuff[3]];
   LedCtrl = LedCtrl & 0xef;
   mDelay(5);
   LedCtrl = LedCtrl | 0xf0;
}

void RS232_TXD(void)
{
    EA = 0;//关闭所有中断

    ES = 0;//关闭串口中断,采用查询发送方式

    TI = 1;
    while(TI)
    {
        TI = 0;//一字节发送完后清除标志位

        SBUF = Rxd_data;    //can接收到的数据发送出去

        while(!TI)
        {}//等待发送

        TI = 0;
    }
    _nop_();
    ES = 1;//打开串口中断

    EA = 1;//打开中断

}


文件:8 CAN-RS232互转.zip
大小:70KB
下载:下载

CEPARK CAN总线开发板 程序更新20100914

阅读(2091) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~