当前位置:网站首页 > 编程语言 > 正文

max31855程序(max232程序)



开发环境:

MDK:Keil 5.30

开发板:GD32F207I-EVAL

MCU:GD32F207IK

1 CRC的校验原理

__循环冗余校验(CRC)计算单元是根据固定的生成多项式得到任一32位全字的CRC计算结果。__在其他的应用中, CRC技术主要应用于核实数据传输或者数据存储的正确性和完整性。标准EN/IEC 60335-1即提供了一种核实闪存存储器完整性的方法。 CRC计算单元可以在程序运行时计算出软件的标识,之后与在连接时生成的参考标识比较,然后存放在指定的存储器空间。那么首先来看看CRC校验原理。

1.1基本原理

CRC检验原理实际上就是在一个p位二进制数据序列之后附加一个r位二进制检验码(序列),从而构成一个总长为n=p+r位的二进制序列;附加在数据序列之后的这个检验码与数据序列的内容之间存在着某种特定的关系。如果因干扰等原因使数据序列中的某一位或某些位发生错误,这种特定关系就会被破坏。因此,通过检查这一关系,就可以实现对数据正确性的检验。

  • 几个基本概念

1、帧检验序列FCS(Frame Check Sequence):为了进行差错检验而添加的冗余码。

2、多项式模2运行:实际上是按位异或(Exclusive OR)运算,即相同为0,相异为1,也就是不考虑进位、借位的二进制加减运算。如: + = 0。

3、生成多项式(generator polynomial):当进行CRC检验时,发送方与接收方需要事先约定一个除数,即生成多项式,一般记作G(x)。生成多项式的最高位与最低位必须是1。常用的CRC码的生成多项式有:

1684499395931j27tvjj3xu

每一个生成多项式都可以与一个代码相对应,如CRC8对应代码:。

设信息字段为K位,校验字段为R位,则码字长度为N(N=K+R)。设双方事先约定了一个R次多项式g(x),则CRC码:

V(x)=A(x)g(x)=xRm(x)+r(x)

其中: m(x)为K次信息多项式, r(x)为R-1次校验多项式。

这里r(x)对应的代码即为冗余码,加在原信息字段后即形成CRC码。

r(x)的计算方法为:在K位信息字段的后面添加R个0,再除以g(x)对应的代码序列,得到的余数即为r(x)对应的代码(应为R-1位;若不足,而在高位补0)。

计算示例:

设需要发送的信息为M = ,产生多项式对应的代码为P = ,R=5。在M后加5个0,然后对P做模2除法运算,得余数r(x)对应的代码:01110。故实际需要发送的数据是1110。

1684499396491ftvj0w9tjh

当接收方收到数据后,用收到的数据对P(事先约定的)进行模2除法,若余数为0,则认为数据传输无差错;若余数不为0,则认为数据传输出现了错误,由于不知道错误发生在什么地方,因而不能进行自动纠正,一般的做法是丢弃接收的数据。

【注】几点说明:

1、CRC是一种常用的检错码,并不能用于自动纠错。

2、只要经过严格的挑选,并使用位数足够多的除数 P,那么出现检测不到的差错的概率就很小很小。

3、仅用循环冗余检验 CRC 差错检测技术只能做到无差错接受(只是非常近似的认为是无差错的),并不能保证可靠传输。

所有的GD32芯片都内置了一个硬件的CRC计算模块,可以很方便地应用到需要进行通信的程序中,这个CRC计算模块使用常见的、在以太网中使用的计算多项式:

写成16进制就是:0x04C11DB7

使用这个内置CRC模块的方法非常简单,既首先复位CRC模块(设置CRC_CR=0x01),这个操作把CRC计算的余数初始化为0xFFFFFFFF;然后把要计算的数据按每32位分割为一组数据字,并逐个地把这组数据字写入CRC_DR寄存器(既下图中的绿色框),写完所有的数据字后,就可以从CRC_DR寄存器(既下图中的兰色框)读出计算的结果。

16844993970914bmxqr214p

下面是用C语言描述的这个计算模块的算法,大家可以把它放在通信的另一端,对通信的正确性进行验证:

DWORD dwPolynomial = 0x04c11db7;

DWORD cal_crc(DWORD *ptr, int len)

{

DWORD xbit;

DWORD data;

DWORD CRC = 0xFFFFFFFF; // init

while (len--) {

xbit = 1 << 31;

data = *ptr++;

for (int bits = 0; bits < 32; bits++) {

if (CRC & 0x) {

CRC <<= 1;

CRC ^= dwPolynomial;

}

else

CRC <<= 1;

if (data & xbit)

CRC ^= dwPolynomial;

xbit >>= 1;

}

}

return CRC;

}

有几点需要说明:

1)上述算法中变量CRC,在每次循环结束包含了计算的余数,它始终是向左移位(既从最低位向最高位移动),溢出的数据位被丢弃。

2)输入的数据始终是以32位为单位,如果原始数据少于32位,需要在低位补0,当然也可以高位补0。

3)假定输入的DWORD数组中每个分量是按小端存储。

4)输入数据是按照最高位最先计算,最低位最后计算的顺序进行。

例如:

如果输入0x,内存中按字节存放的顺序是:0x41, 0x42, 0x43, 0x44。计算的结果是:0xCF534AE1

如果输入0x,内存中按字节存放的顺序是:0x44, 0x43, 0x42, 0x41。计算的结果是:0xABCF9A63

  • 数据寄存器(CRC_DRTA)

1684499397348se5a386mmo

CRC_DATA用于接收待计算的新数据,直接将其写入即可。刚写入的数据不能被读出来,因为读取该寄存器得到的是上次CRC计算的结果。

  • 独立数据寄存器(CRC_FDATA)

16844993976384vdyqj2v0m

注:此寄存器不参与CRC计算,可以存放任何数据。

  • 控制寄存器(CRC_CTL)

1684499397900f57ynzubzt

CRC_CTL用来复位CRC_DATA寄存器,设置其值为0xFFFFFFFF,然后该位被硬件自动清零。该位对CRC_FDATA寄存器没有影响。

4 CRC具体代码实现

代码很简单。

brief main function

param[in] none

param[out] none

retval none

*/

int main(void)

{

//systick init

sysTick_init();

//usart init 8-N-1

com_init(COM1, , 0, 1);

printf('CRC Test ');

/* Enable CRC clock */

rcu_periph_clock_enable(RCU_CRC);

/* Compute the CRC of 'DataBuffer' */

CRCValue = crc_block_data_calculate((uint32_t *)DataBuffer, BUFFER_SIZE);

printf(' 32-bit CRC check code : 0x%X ', CRCValue);

while(1)

{

delay_ms(1000);

}

}

就使用了crc_block_data_calculate()函数,传入一个要计算的数据和大小,就得到了计算的CRC值。

将编译好的程序下载到板子中,通过串口助手可以看到如下现象。

168449939821845ledyjncw

然后使用CRC计算工具来计算。

168449939857966nz9r3480

可以看到和软件计算的一致。

值得注意的是,STM32的硬件CRC的结果异或值是0x00000000。

【注】关于CRC的更多内容可以自行查阅相关资料,笔者这里推荐一篇文章A PAINLESS GUIDE TO CRC ERROR DETECTION ALGORITHMS,感兴趣的朋友自己去看看吧。

到此这篇max31855程序(max232程序)的文章就介绍到这了,更多相关内容请继续浏览下面的相关推荐文章,希望大家都能在编程的领域有一番成就!

版权声明


相关文章:

  • 贴吧缺少动态库什么意思(贴吧缺少动态库什么意思呀)2025-04-08 12:00:08
  • 访问不了github网站(为什么访问不了github)2025-04-08 12:00:08
  • 书旗小说看目录怎么那么难点开呢(书旗小说看目录怎么那么难点开呢知乎)2025-04-08 12:00:08
  • 画一画,圈一圈,填一填(圈一圈画一画填一填二年级)2025-04-08 12:00:08
  • 密码仓库(仓库里的密码情)2025-04-08 12:00:08
  • 国内改ip(国内改ip免费版)2025-04-08 12:00:08
  • 操作系统大题及答案(操作系统大题解析)2025-04-08 12:00:08
  • 合并数组的方法(数组合并数组)2025-04-08 12:00:08
  • 做网页的代码(制作网页的代码)2025-04-08 12:00:08
  • 打印机共享修复补丁错误(win10共享打印机709错误补丁)2025-04-08 12:00:08
  • 全屏图片