BCC校验(异或校验),LRC校验(纵向冗余校验),累加和校验(CheckSum) 源码

#include <stdio.h>
#include <string.h>


/**
 *******************************************************************************
 * @brief   获取 BCC校验(异或校验) 函数
 * @param   [in] *buf   - 数据块指针
 * @param   [in] size 	- 数据块字节大小
 * @return  校验码
 * @note    将每个字节数据(一般是两个16进制的字符)进行异或后即得到校验码
 *          适用于大多数要求不高的数据通讯,像很多单片机系统的串口通讯。
 *******************************************************************************
 */
unsigned char GetBCC(unsigned char *buf, unsigned long  size)
{
    unsigned char check = 0; //校验字节 初始化
    while(size--) check ^= *buf++; //异或计数
    return check;
}

/**
 *******************************************************************************
 * @brief   获取 LRC校验(纵向冗余校验) 函数
 * @param   [in] *buf   - 数据块指针
 * @param   [in] size 	- 数据块字节大小
 * @return  校验码
 * @note    将每个字节数据求和结果与256求余,再按位取反加 1(256 - 余数)后即得到校验码
 *          在工业领域Modbus协议Ascii模式采用该算法
 *******************************************************************************
 */
unsigned char GetLRC(unsigned char *buf, unsigned long size)
{
    unsigned char check = 0; //校验字节 初始化
    while(size--) check += *buf++ ; //累加求余(只保留余数字节就是数据求和结果与256求余)
    return (unsigned char)((check ^ 0xFF) + 1);
}

/**
 *******************************************************************************
 * @brief   获取 累加和校验 函数
 * @param   [in] *buf   - 数据块指针
 * @param   [in] size 	- 数据块字节大小
 * @return  校验码
 * @note    校验和的数值超过十六进制的FF,也就是255,就要求其补码作为校验和。
 *******************************************************************************
 */
unsigned char GetCheckSum8(unsigned char *buf, unsigned long size)
{
    unsigned long cksum = 0; //校验字节 初始化
    while(size--) cksum += *buf++ ;
	if(cksum > 0xFF)
	{
        cksum = ~cksum;
        cksum += 1;
	}
    return (unsigned char)(cksum & 0xFF);
}

/**
 *******************************************************************************
 * @brief   获取 累加和校验 函数
 * @param   [in] *buf   - 数据块指针
 * @param   [in] size 	- 数据块字节大小
 * @return  校验码
 * @note    16位数据进行累加,奇数字节则最后补0凑成16位,再把累加和的高16位与低16位进行累加,
 *          发生进位则把进位累加上,最后取反。
 *******************************************************************************
 */
unsigned short GetCheckSum16(unsigned short *buf, unsigned long size)
{
	unsigned long cksum = 0;

	while(size > 1) //16位为单位数字相加
	{
		cksum += *buf++;
		size -= 2;
	}
	if(size) cksum += *((unsigned char *)buf); //长度是奇数

	cksum = (cksum>>16) + (cksum&0xFFFF); //高16位 与 低16位相加
	cksum += (cksum>>16); //加上进位

    return (unsigned short)(~cksum); //最后取反
}

/**
 *******************************************************************************
 * @brief   主函数
 * @param   [in] None
 * @return  None
 * @note    验证校验函数
 *******************************************************************************
 */
void main()
{
    char *str = "Hello World!";

	printf("BCC (\"%s\") = 0x%02X\n", str, GetBCC((unsigned char *)str, strlen(str)));
    printf("LRC (\"%s\") = 0x%02X\n", str, GetLRC((unsigned char *)str, strlen(str)));
    printf("CheckSum8 (\"%s\") = 0x%02X\n", str, GetCheckSum8((unsigned char *)str, strlen(str)));
    printf("CheckSum16 (\"%s\") = 0x%02X\n", str, GetCheckSum16((unsigned short *)str, strlen(str)));

	while(1);
}

运行结果:

  • 1
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Modbus协议有多种校验方式,常用的有两种:CRC校验LRC校验。 1. CRC校验 CRC校验是采用循环冗余校验的方式,可以检测数据传输过程中的错误。Modbus协议中使用的是16位CRC校验,计算方式如下: 首先,将传输数据按照8位一组进行分组,并将每组数据看作16位的二进制数。然后,将每组数据与上一个16位的CRC校验值进行异或运算,得到一个新的16位的结果。最后,将所有分组的结果再次进行异或运算,得到最终的16位CRC校验值。 以下是一个Python实现的CRC校验函数: ```python def calculate_crc(data): crc = 0xFFFF for i in range(len(data)): crc ^= data[i] for j in range(8): if (crc & 0x0001): crc >>= 1 crc ^= 0xA001 else: crc >>= 1 return crc ``` 2. LRC校验 LRC校验是采用纵向冗余校验的方式,可以检测传输数据中的一些简单错误。Modbus协议中使用的是8位LRC校验,计算方式如下: 首先,将传输数据中的每个字节相加,得到一个8位的结果。然后,将这个结果取反(按位取反),得到最终的8位LRC校验值。 以下是一个Python实现的LRC校验函数: ```python def calculate_lrc(data): lrc = 0 for i in range(len(data)): lrc += data[i] lrc = (~lrc) & 0xFF return lrc ``` 注意:在使用Modbus协议进行通信时,需要根据具体情况选择适当的校验方式。同时,在计算校验值时,需要考虑传输数据的字节顺序(大端序或小端序)。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值