纵向冗余校验LRC CRC

http://www.cnblogs.com/zhangzhifeng/p/5579688.html
http://wenku.baidu.com/link?url=4uAVtSk0zxJYht7y51o7_e2Aa5mjykJogkeQDtBkFZQawkjjtCoLnhCAV7xwxEiNcuSn6OFAHLUZc_CisDO9isefhhY4keOyygFQD3Jodh_

#include<stdio.h>   
typedef unsigned char   uchar;
typedef unsigned int    uint;
uchar result;   
uchar data[6];//={0x03,0x10,0x0b,0xe8, //0xff,0x00};

void main()
{
    int LRC(uchar*data);
    int a;
    int b;

    printf("输入要发送的数据。\n");
    scanf("%x%x%x%x%x%x",&data[0],&data[1],&data[2],&data[3],&data[4], &data[5]);
    
    a=LRC(data);
    printf("校验码为:0x%x\n",a);
    printf("输入接收到的数据\n");   
    scanf("%x%x%x%x%x%x%x",&data[0],&data[1],&data[2],&data[3],&data[4],&data[5],&b);

    if (b!=a)
    {
        printf("传送失败\n");
    }
    else
    {
        printf("传送成功\n");
    }
    
    getchar();
    getchar();
}



    1、LRC校验
    LRC域是一个包含一个8位二进制值的字节。LRC值由传输设备来计算并放到消息帧中,接收设备在接收消息的过程中计算LRC,并将它和接收到消息中LRC域中的值比较,如果两值不等,说明有错误。  LRC校验比较简单,它在ASCII协议中使用,检测了消息域中除开始的冒号及结束的回车换行号外的内容。
    它仅仅是把每一个需要传输的数据按字节叠加后取反加1即可。下面是它对应的代码:  
BYTE GetCheckCode(const char * pSendBuf, int nEnd)//获得校验码  
{  
    BYTE byLrc = 0;
    char pBuf[4];
    int nData = 0;  
    for(i=1; i<end; i+=2) //i初始为1,避开“开始标记”冒号  
    {  
        //每两个需要发送的ASCII码转化为一个十六进制数  
        pBuf [0] = pSendBuf [i];
        pBuf [1] = pSendBuf [i+1];  
        pBuf [2] = '\0';  
        sscanf(pBuf,"%x",& nData);  
        byLrc += nData;  
    }
   
    byLrc = ~ byLrc;  
    byLrc ++;  return byLrc;  
}


    2、CRC校验
    CRC域是两个字节,包含一16位的二进制值。它由传输设备计算后加入到消息中。接收设备重新计算收
    到消息的CRC,并与接收到的CRC域中的值比较,如果两值不同,则有误。 
CRC是先调入一值是全“1”的16位寄存器,然后调用一过程将消息中连续的8位字节各当前寄存器中的值
    进行处理。仅每个字符中的8Bit数据对CRC有效,起始位和停止位以及奇偶校验位均无效。  CRC产生过程中,每个8位字符都单独和寄存器内容相或(OR),结果向最低有效位方向移动,最高有效位以0填充。LSB被提取出来检测,如果LSB为1,寄存器单独和预置的值或一下,如果LSB为0,则不进行。整个过程要重复8次。在最后一位(第8位)完成后,下一个8位字节又单独和寄存器的当前值
     相或。最终寄存器中的值,是消息中所有的字节都执行之后的CRC值。  CRC添加到消息中时,低字节先加入,然后高字节。下面是它对应的代码: 
WORD GetCheckCode(const char * pSendBuf, int nEnd)//获得校验码  
{  
    WORD wCrc = WORD(0xFFFF);

    for(int i=0; i<nEnd; i++) 
    {  
        wCrc ^= WORD(BYTE(pSendBuf[i]));  
        for(int j=0; j<8; j++)  
        {  
            if(wCrc & 1)
            {  
                wCrc >>= 1;
                wCrc ^= 0xA001;
            }
            else
            { 
                wCrc >>= 1;
            }
        }
    }
    
    return wCrc;  
}


    对于一条RTU协议的命令可以简单的通过以下的步骤转化为ASCII协议的命令: 
    1、 把命令的CRC校验去掉,并且计算出LRC校验取代。 
    2、 把生成的命令串的每一个字节转化成对应的两个字节的ASCII码,比如0x03转化成0x30,0x33(0的ASCII码和3的ASCII码)。 
    3、 在命令的开头加上起始标记“:”,它的ASCII码为0x3A。 
    4、 在命令的尾部加上结束标记CR,LF(0xD,0xA),此处的CR,LF表示回车和换行的ASCII码。
 

掌握两种协议的编程方法,剩下的就是C语言的问题了。



ddr123456 2009-02-18 13:44
请问哪位高手知道LRC校验码计算方法,程序如何写(西门子S7-200与台达VFD-B变频器通信)?

ljc2008
2009-02-18 15:08
2.LRC校验码
这是MODBUS通讯协议ASCII方式的校验方法,LRC校验码不能直接用指令求出,但可编制程序自动算出,
算法:将参与校验的数据求和,取其低8位的补码为校验码
例:求数据 01H,03H,21H,02H,00H,02H 之求和校验码。

求和:01H + 03H + 21H + 02H + 00H + 02H = 29H 求补码有二种方法: ① 求反加1  ②  用FFH相减    我们下面用求反加一来做
                 0  0  1  0  1  0  0  1
                    求反: 1  1  0  1  0  1  1  0                    

                    加 1: 0  0  0  0  0  0  0  1                                    

                             1  1  0  1  0  1  1  1                                         

                                  D             7

LRC校验码为:D7H

猜你喜欢

转载自jacky-dai.iteye.com/blog/2312842