CRC校验原理及代码

 参考:CRC校验原理及步骤https://blog.csdn.net/d_leo/article/details/73572373

什么是CRC校验?

CRC即循环冗余校验码:是数据通信领域中最常用的一种查错校验码,其特征是信息字段和校验字段的长度可以任意选定。循环冗余检查(CRC)是一种数据传输检错功能,对数据进行多项式计算,并将得到的结果附在帧的后面,接收设备也执行类似的算法,以保证数据传输的正确性和完整性。

 

CRC校验原理:

其根本思想就是先在要发送的帧后面附加一个数(这个就是用来校验的校验码,但要注意,这里的数也是二进制序列的,下同),生成一个新帧发送给接收端。当然,这个附加的数不是随意的,它要使所生成的新帧能与发送端和接收端共同选定的某个特定数整除(注意,这里不是直接采用二进制除法,而是采用一种称之为“模2除法”)。到达接收端后,再把接收到的新帧除以(同样采用“模2除法”)这个选定的除数。因为在发送端发送数据帧之前就已通过附加一个数,做了“去余”处理(也就已经能整除了),所以结果应该是没有余数。如果有余数,则表明该帧在传输过程中出现了差错。

模2除法:

模2除法与算术除法类似,但每一位除的结果不影响其它位,即不向上一位借位,所以实际上就是异或。在循环冗余校验码(CRC)的计算中有应用到模2除法。

例:

CRC校验码计算示例:

现假设选择的CRC生成多项式为G(X) = X4 + X3 + 1,要求出二进制序列10110011的CRC校验码。下面是具体的计算过程:

①将多项式转化为二进制序列,由G(X) = X4 + X3 + 1可知二进制一种有五位,第4位、第三位和第零位分别为1,则序列为11001

②多项式的位数位5,则在数据帧的后面加上5-1位0,数据帧变为101100110000,然后使用模2除法除以除数11001,得到余数。

③将计算出来的CRC校验码添加在原始帧的后面,真正的数据帧为101100110100,再把这个数据帧发送到接收端。

④接收端收到数据帧后,用上面选定的除数,用模2除法除去,验证余数是否为0,如果为0,则说明数据帧没有出错。

CRC校验电路原理图

 对应的并行CRC代码

`timescale 1ns / 1ps
//
// Company: 
// Engineer: 
// 
// Create Date:    14:18:49 08/14/2018 
// Design Name: 
// Module Name:    crc16-papa 
// Project Name: 
// Target Devices: 
// Tool versions: 
// Description: 
//
// Dependencies: 
//
// Revision: 
// Revision 0.01 - File Created
// Additional Comments: 
//
//
module crc16papa (reset, gclk, soc, data_in, crc_out);
input reset,gclk,soc;
input [7:0] data_in;
output reg [15:0] crc_out;
reg temp;
reg [15:0] crc_temp;
parameter delay = 1'b1;
integer i,j,k,l;
//产生crc_out
always @(posedge gclk or negedge reset)
    begin
		if (!reset)
			crc_out <= #delay 16'b0;
		else if (soc)
			crc_out <= #delay 16'b0;
		else
			crc_out <= #delay crc_temp;
	end
//产生crc_temp
always @(crc_out or data_in)
	begin
		crc_temp = crc_out;
		for (i=7;i>=0;i=i-1)
			begin
				temp = data_in[i] ^crc_temp[15];
				for (j=15; j>12; j=j-1)
					crc_temp[j] = crc_temp[j-1];
				crc_temp[12] = temp ^ crc_temp[11];
				for (k=11; k>5; k=k-1)
					crc_temp[k] = crc_temp[k-1];
				crc_temp[5] = temp ^ crc_temp[4];
				for (l=4; l>0; l=l-1)
					crc_temp[l] = crc_temp[l-1];
				crc_temp[0] = temp;
		end
	end
endmodule	

 

 

 

  • 64
    点赞
  • 328
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
CRC算法原理是一种简单的循环冗余校验(Cyclic Redundancy Check)算法,主要用于数据传输过程中的差错检测。 其基本思想是将要发送的数据看作二进制位序列,通过生成多项式进行计算,得到一个余数。将该余数附加到要发送的数据后面,接收方在接收到数据后也进行一系列的计算与检验,如果得到的余数为0,则认定数据无误。 具体的实现步骤是: 1. 定义生成多项式,通常用多项式的系数表示,例如CRC32使用的是0x04C11DB7。生成多项式选择不同,CRC算法的效果也会不同。 2. 将要发送的数据按照二进制形式进行排列,并进行位填充(通常在数据后面加上若干个0)。 3. 选择一个初始的除数,通常为全1或全0。 4. 将数据与初始的除数进行异或运算,得到余数。 5. 将余数附加到原始数据后面,并作为新的输入数据进行下一轮的计算,重复前面的步骤。 6. 最终得到的余数即为CRC校验码,将其附加到要发送的数据后面。 7. 接收方收到数据后按照相同的生成多项式和步骤进行计算,如果最终的余数为0,则认定数据无误。 在C语言中,可以通过使用位运算和循环来实现CRC算法。具体的实现代码如下: ```c #include <stdio.h> unsigned int crc32(unsigned char *data, unsigned int length) { unsigned int crc = 0xFFFFFFFF; unsigned int i, j; for (i = 0; i < length; i++) { crc ^= data[i]; for (j = 0; j < 8; j++) { if (crc & 1) crc = (crc >> 1) ^ 0xEDB88320; else crc >>= 1; } } return crc ^ 0xFFFFFFFF; } int main() { unsigned char data[] = "123456789"; unsigned int length = sizeof(data) - 1; unsigned int crc = crc32(data, length); printf("CRC32: %08X\n", crc); return 0; } ``` 以上代码实现了一个简单的CRC32算法,能够计算出输入数据的CRC校验码。具体的计算过程是将输入数据转化为二进制形式,并进行异或和位移运算,最后输出CRC校验码。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值