原创 modbus协议--51端程序的实现

2008-6-5 12:14 6751 11 43 分类: 工业电子
RTU需要一个定时器来判断3.5个流逝时间。
#define ENABLE    1
#define DISABLE    0
#define TRUE    1
#define FAULT    0
#define RECEIVE_EN    0
#define TRANSFER_EN    1
#define MAX_RXBUF  0x20

extern unsigned char emissivity;
extern unsigned char tx_count,txbuf[15];
extern unsigned char rx_count,rxbuf[15];
extern unsigned char tx_number,rx_number;
extern bit rx_ok;
unsigned char rx_temp;
void InitTimer1()            //针对标准8051
{
    TMOD=(TMOD|0xf0)&0x1f;    //将T1设为16位定时器
    TF1=0;
    TH1=0x62; //设T1位3.5位的接收时间35bit/9600bit/s=3.646ms
    TL1=0x80;//晶振为11.0592MHz,T=65535-3.646ms*11.0592MHz/12=0xf2df
            //0x6280是22.1184M下LPC9XX下的值。
    ET1=1;                                    //允许T1中断
    TR1=1;                                    //T1开始计数
}

void timer1() interrupt 3 using 2 //定时器中断
{
    TH1=0x62;    //3.646ms interrupt
    TL1=0x80;
    if(rx_count>=5)    //超时后,若接收缓冲区有数则判断为收到一帧
    {
        rx_ok=TRUE;
    }
}

void scomm() interrupt 4 using 3    //modbus RTU模式
{
    if(TI)
    {
        TI = 0;
        if(tx_count < tx_number)    //是否发送结束
        {
            SBUF = txbuf[tx_count];
        }
        tx_count++;
    }
    if(RI)
    {
        rx_temp=SBUF;
        if(rx_ok==FAULT)    //已接收到一帧数据,在未处理之前收到的数舍弃
        {
            if(rx_count
                rxbuf[rx_count]=rx_temp;
            rx_count++;
        }
        TH1=0x62;        //timer1 reset,count again
        TL1=0x80;
        RI=0;
    }
}
在主循环中判断标志rx_ok来执行帧处理。
if(rx_ok)
        {
            ParseFrame();
            KB0=1;
            REN=0;
            tx_count=0;
            TI=1;       //启动发送响应帧
            rx_count=0;
            rx_ok=0;
        }
WORD MAKEWORD(a, b)
{
    int_byte itemp;
    itemp.items.high=a;   
    itemp.items.low=b;
    return (itemp.item);   
}
// 解析帧并发送响应帧 (在帧完整的前提下调用)
bit ParseFrame()
{
    unsigned char byAddr ;    // 地址
    unsigned char byFunCode ;    // 功能代码
    int_byte wCRC;

   
    wCRC.item = MAKEWORD(rxbuf[rx_count-1], rxbuf[rx_count-2]);
    if(wCRC.item != CRC(rxbuf, rx_count-2))    // 判断校验是否正确
    return FALSE;

    // 正式解析
    byAddr = rxbuf[0];    // 地址
    byFunCode = rxbuf[1];    // 功能代码

    // 如果地址不对
    if( (byAddr != m_byAddress) && (byAddr != 0) )
        return FALSE;

    if(byAddr == m_byAddress)
    {
        AddSendByte(m_byAddress) ;    // 地址
        switch( byFunCode )
        {
        case 3:            // 读保持寄存器
            Fun3(3);
            break;
        ....// 添加命令散转
        ......
        default:
            ErroRespond(1);
            return FALSE;
            break;
        }
    }   

    wCRC.item = CRC(txbuf,tx_number);

    AddSendByte(wCRC.items.low);
    AddSendByte(wCRC.items.high);
    return TRUE;
}
// 根据接收帧模式发送相应,模式的数据
BOOL AddSendByte(const BYTE byData)
{
    txbuf[tx_number]=byData;
    tx_number++;
    if(tx_number>30)return FALSE;
    return TRUE;
}

// 异常响应            描述        响应解释
//   01              无效功能    变送器不允许执行收到的功能
//   02              无效地址    数据栏中的地址是不允许的
//   03              无效数据    数据栏中的数据是不允许的
//   06              忙        收到的消息没错,但从机正在执行一个长的程序命令
bit ErroRespond(const unsigned char byErroCode)
{
//    printf("\nErroRespond%02X \n", byErroCode);
    if( !AddSendByte(rxbuf[1] | 0x80) )
        return FALSE;
    return AddSendByte(byErroCode);   
}
//***CRC Calculation for MODBUS Protocol for VC++***//
//数组snd为地址等传输字节,num为字节数//
unsigned int CRC(unsigned char *snd, unsigned char num)
{
    unsigned char i, j;
    unsigned int c,crc=0xFFFF;
    for(i = 0; i < num; i ++)
    {
        c = snd & 0x00FF;
        crc ^= c;
        for(j = 0;j < 8; j ++)
        {
            if (crc & 0x0001)
            {
                crc>>=1;
                crc^=0xA001;
            }
            else crc>>=1;
        }
    }   
    return(crc);
}



)>

PARTNER CONTENT

文章评论32条评论)

登录后参与讨论

用户6383 2010-5-15 16:53

P0对应bitData[0] P1对应bitData[1] 请问: bitData[0] = 0X xx; bitData[1] = ??

用户280159 2010-4-13 11:35

我对消息帧接收那一块还不太清楚,你这里只接收了5次就清0了... 我看有些资料上都有7-8次的.. 还有你这里只对地址和功能代码进行了处理..并没有处理数据位啊..能加我QQ指导一下吗?156469218

用户541817 2009-6-22 15:19

过来光顾你了..不错.看看

用户1448330 2009-6-13 00:24

学习了!^_^ 这个毕竟是楼主的劳动成果,没有义务放出源程序,全部取决于他的意愿啊……

用户189824 2009-2-14 16:28

楼主是不是把主函数没有写出来啊

用户189824 2009-2-14 15:43

上面是不是把函数给除去了

用户189824 2009-2-14 15:37

为什么程序无法编译成功啊 m_byAddress好像是没有定义啊

用户1404696 2009-2-11 08:17

CRC16算法,实现的种类有多种,结果是一样的。 我已经加了QQ,我的QQ名字是azmao.

用户194804 2009-2-6 07:14

能加下你的QQ吗?想向你请教。我的QQ:361188726

用户194804 2009-2-6 07:04

是不是每一种crc算法得出的数据都不同,要求上下位要统一crc算法?如果是这样,modbus岂不是很混乱?
相关推荐阅读
用户1404696 2009-06-12 15:28
MODBUS协议在TI DSP TMS320F22335上的实现(一)
唉,做的仪表越来越复杂了,需要使用DSP了,开始一点一点将单片机的东西移植到DSP上面了。先说TMS320F28335的串口吧。有三个串口,SCIA,SCIB和SCIC,GPIO的管脚对应如下SCIA...
用户1404696 2009-02-25 10:56
仪表防爆技术
 n年前做的,拿出来晒晒。 一:防爆的基本概念 二:电气设备的防爆形式 三:防爆标志及铭牌 四:防爆设计 五:防爆技术的应用 ...
用户1404696 2008-10-28 10:22
ST7920汉字点阵LCD的接口程序(直接接方式)
//总线扩展方式连接12832点阵汉字液晶//A0--D/I//A1--R/W//E连接地址分配的0xD000#include <absacc.h>#include <REG52.H...
用户1404696 2008-10-28 10:11
ST7920汉字点阵LCD的接口程序(间接方式)
/***********************************************************            ST7920控制器间接方式驱动程序           ...
用户1404696 2008-10-21 08:46
一个经典的按键扫描程序
按键是比较复杂。可以用状态机表示。 每10mS执行一次键盘扫描任务 0、无键,若有键进入状态1 1、第一次检到有键。若有键进入状态2,同时将键值送缓冲区。 若无键进入状态0。 2、若有键则计时,按键...
用户1404696 2008-10-11 16:09
生产测井仪器--有线遥传短接
   该仪器采用有线多路遥传技术实现单芯电缆传输。可以配DDL-Ⅲ、DDL-Ⅴ及所有与DDL-Ⅲ相兼容的地面系统。遥传短节(WTC)原理遥传短节核心由两片单片机组成,主要向井下仪器各参数发送地址,并接...
我要评论
32
11
关闭 站长推荐上一条 /3 下一条