打印

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

[复制链接]
2858|17
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
azmao|  楼主 | 2008-6-5 13:31 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
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 MODBUS_T35,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=0x6280
    ET1=1;                                    //允许T1中断
    TR1=1;                                    //T1开始计数
}

void timer1() interrupt 3 using 2 //定时器中断
{
    TH1=0x62;    //3.646ms interrupt
    TL1=0x80;
    MODBUS_T35=ENABLE;
    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<MAX_RXBUF)
                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(" ErroRespond%02X  ", 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);
}

相关帖子

沙发
原野之狼| | 2008-6-5 14:07 | 只看该作者

沙发一个

使用特权

评论回复
板凳
zsk199114| | 2012-10-3 11:42 | 只看该作者
谢谢大哥     就是找这个呀呀呀呀呀

使用特权

评论回复
地板
zsk199114| | 2012-10-3 11:42 | 只看该作者
哈哈哈哈哈哈哈哈啊哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈啊哈哈

使用特权

评论回复
5
zsk199114| | 2012-10-3 11:43 | 只看该作者
无花无花无花.............

使用特权

评论回复
6
zsk199114| | 2012-10-3 11:47 | 只看该作者
哥们   你的程序不全啊
差的太多了吧

使用特权

评论回复
7
lihaolongli| | 2012-10-3 16:43 | 只看该作者
不错的说

使用特权

评论回复
8
hjs1618| | 2012-10-4 21:36 | 只看该作者
这个能用吗?

使用特权

评论回复
9
iamsailingchina| | 2012-11-2 10:38 | 只看该作者
很好啊

使用特权

评论回复
10
syd_dong| | 2012-12-31 17:21 | 只看该作者
学习下。。:D

使用特权

评论回复
11
sensorexpert| | 2012-12-31 17:37 | 只看该作者
代码写得很简练

使用特权

评论回复
12
ningling_21| | 2012-12-31 18:06 | 只看该作者
代码未完,待续...

使用特权

评论回复
13
海川先生| | 2012-12-31 20:04 | 只看该作者
楼主的名字可以理解成阿滋猫吗

使用特权

评论回复
14
yueliangruying| | 2013-4-11 15:03 | 只看该作者
本帖最后由 yueliangruying 于 2013-4-11 15:09 编辑

接收缓冲区有数则判断为收到一帧,---感觉数据不安全

使用特权

评论回复
15
kangzp| | 2013-4-13 13:37 | 只看该作者
佩服楼主,我也打算自己写一个的,憋了两周,写了一半,又下载了个freemodbus改了一个!

使用特权

评论回复
16
zhanghaiw| | 2013-4-13 15:40 | 只看该作者
楼主把协议传上啊,还有把时序图贴出来啊!

使用特权

评论回复
17
程序猿| | 2014-3-20 08:50 | 只看该作者
谢谢!太好了

使用特权

评论回复
18
coody| | 2014-3-20 11:16 | 只看该作者
MODBUS协议是一个软件层的协议,跟什么MCU无关,电脑都可以用

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

11

主题

50

帖子

0

粉丝