粉丝分享 | 图说CRC原理应用及STM32硬件CRC外设 : D9 o# X$ T3 c2 j) |+ E b' M# [3 X: r* C1 ]/ }5 ? 在嵌入式产品应用中,常常需要应对系统数据在存储或者传输过程中的完整性问题。0 J7 x( X+ }$ x7 z / k$ b: O& {, V) T( I( d4 B 所谓完整性是指数据在其生命周期中的准确性和一致性。这些数据可能存储在EEPROM/FLASH里,或者基于通信协议进行传输,它们有可能因为外界干扰或者程序错误,甚至系统入侵而导致被破坏。如果这些数据在使用前不做校验,产品功能可能失效。在一些特定领域,严重时可能会危及用户财产甚至生命安全。/ o% v% C- D" l, ~4 _ 5 _' ?- w# O R/ E/ c1 P 本文就来聊聊使用较为广泛的循环冗余校验技术,以及在STM32中的一些具体使用体会。( X$ i* c3 d1 u- V - a5 Z( j( G& I: g 所谓循环冗余校验(CRC:Cyclic Redundancy Check)是一种错误检测算法,通常在通信协议中或存储设备中用于检测原始数据的意外变动。可以简单理解成对有用数据按照一定的算法进行计算后,提取出一个特征值,并附加在有用数据后。在应用中将有用数据按照特定的算法提取特征值与预先存储的特征值进行比对,如相等则校验通过,反之校验失败,从而识别出数据是否异常。 * b S& g9 y% }5 l: k" U; C$ J S$ t 为何要校验数据完整性(Data Integrity)? / c T# {" R: i数据在存储以及传输的过程中可能发生异动。以数据通信应用场景为例,常见的错误大致有两种失效模式:# ]& ?+ N, g* K0 b: Y @ * S! i( A) N/ K" D3 } - Z* M" @0 K! I& w 单个位错误(Single Bit Error):仅仅某一个数据位出现错误,如图: % u4 q) c+ V' R " s" H0 T# w9 h, @; b( d0 @$ b4 x 存储系统中的数据也是一样。一般来说,系统在上电运行时会从物理存储介质装载系统参数,比如一些校准数据。如果由于介质的某些位被破坏,或者软件bug导致数据被误操作了,而没有数据完整性检测,这样的数据直接被应用于系统控制,一样会造成安全隐患。 8 `+ e0 q2 |& F% T- A- z2 [ 所以,对于数据完整性检测的重要性不言而喻。常见的数据完整性算法有很多种,比如简单的异或校验、CRC循环冗余校验、FEC前向纠错算法等等。而循环冗余校验在嵌入式系统中应用非常广泛,在通信协议制定、数据存储、压缩解压算法等都有广泛的应用。 ( i( W$ t" z. k$ ~ 0 z* x- W* y) p/ O 循环冗余校验使用二进制除法作为算法原理,具有强大的错误检测机制。对于二进制除法使用少量的硬件逻辑电路就可实现。至于软件代码实现,有查表法和移位计算两种思路及策略。查表法以空间换时间,移位计算法以时间换空间。& C/ }: p( E: S) a 何为循环冗余校验? 4 z) X1 Y2 b8 `5 j) C$ P0 E+ m8 z2 I# X6 V P 循环冗余校验的核心数学算法原理基于循环码,在不增加原始数据的信息基础上扩展了信息,以极小的存储代价存储其冗余特征。该算法是W. Wesley Peterson 于1961年发明的。 " Z3 Z% s: F6 u+ R; \/ J" @1 o ·根据CRC算法计算出m位冗余码,即根据该CRC校验多项式结合CRC算法从前面有效数据中提取出特征冗余码,这就是冗余的真实含义。1 z5 U# V% D2 L7 E" a2 [ ·实际传输或者存储的就是n+m位二进制数据。# v3 H3 l5 |* i; | 4 O9 m. `% Q( P3 c 这里引出一个概念:多项式,在CRC校验算法中多项式可做如下理解及表示: 其基本的算法处理过程示意如下: 其实CRC的本质就是二进制多项式除法求取冗余码的计算过程,无论软件的查表法、移位计算法,还是纯硬件逻辑电路实现,本质都是一样的。对于数字逻辑电路利用移位计算则更具优势,因为几乎不占用CPU时间。8 E- n' B5 b, ?4 P* N 常见的CRC校验多项式 # y! a9 B/ ^, q3 _常见的CRC校验多项式算子有哪些? 4 H1 B; r7 f2 W0 D0 c; X 不同的校验多项式,除了复杂度有差异外,从应用角度看有什么差异呢?从应用角度看主要体现在错误诊断率。不妨看看CRC-16以及CRC-CCITT的错误检测效果:7 u3 B ?* E3 j& g+ M R . B/ q4 i, t( Y" R% A$ Z 可完全检测出单bit及双bit错误4 V6 Z' l+ H4 P& |3 n! ~ : W" W: U, j$ X. Z1 v 0 ] C$ p5 w, [" c Z { 奇数个位错误 能检测出16位长度及小于16的突发错误 . l1 t% M& O, Y 能以99.997%的概率检测出长度为17位及以上的错误 选择不同的校验多项式算子,其位错误诊断成功率是不一样的,当然其计算开销也不一样。我们来查查权威的IEC标准看看。下图截自《IEC61508-7》。 7 O) q8 ] N" w$ q+ N 注:IEC61508是国际电工委员会功能安全标准(Functional safety of electrical/electronic/programmable electronicsafety-related systems)。( V7 f4 r% c, W1 ~ 1 \8 n1 ^: K9 y+ x* z 技术发展至今,已有大量不同的校验多项式生成器被各行各业使用。下面是来自wikipedia截图,供大家参考:0 T2 A8 @8 t9 W( \: Y STM32的CRC硬件外设 % }) o6 \; A9 W }如下图,STM32内置了一个CRC-32硬件计算单元,实现了一个固定多项式0x4C11DB7(16进制表示),可应用于以太网报文校验码计算。2 h8 |! T; h8 U* D" D! G EWARM 从 v5.5 版本之后开始支持 STM32 芯片的 CRC计算。计算整个 FLASH的 CRC 校验值并保存在 FLASH末尾的过程,可以在 IAR 中完成。通过配置EWARM 的 CRC 计算参数,自动对整个 FLASH 空间进行 CRC 计算,并将计算结果放到 内部FLASH空间 的末尾。9 e3 A# ^2 o" u$ S# U4 F2 [- { R8 R9 r & A( [# w& b! @6 h1 | 3 Z7 N6 Y. s5 ^0 g 或许你会问,这有什么应用价值呢?不妨以基于MCU程序的升级为例。在代码升级过程中,如果不对bootloader升级接口传入的二进制程序文件做校验,就无法及时发现升级过程中发生的代码错误。相反,如果原始代码添加了校验码,升级程序在接受到升级文件后做校验计算,并与待升级文件末尾的校验码进行比对,如果不匹配则放弃升级,这样就不至于将无效的甚至有安全隐患的代码写进芯片。8 }& t/ p" R2 k# c, i 4 M: K$ p8 ^4 j a' y8 Q + M1 `9 z0 X0 ~ 修改 Link 文件,指定 checksum 在FLASH 中的存储位置,在 Link 文件中增加下面语句。& |$ R3 o, a' _
& H h: D( F7 h7 v" y3 h; _9 | 配置 Checksum 页面的参数1 h1 U N/ L$ a" y! a o 2 a( x! Z$ {3 \. n% T3 e 9 |) S" }4 @7 L- Y4 Q. b IAR Checksum 页说明(v6.4 及以上)6 |* A/ Y. r& w. h% r 7 L3 W4 p4 \3 F6 c 红线圈出的部分:定义了FLASH 中需要计算 CRC 的范围和空闲字节填充值。 checksum 计算参数的设定部分: Checksum size :选择 checksum 的大小(字节数)0 s. ?$ }$ ~ g Alignment:指定 checksum 的对齐方式。不填的话默认 2 字节对齐。 Algorithm:选择 checksum 的算法8 G) M3 k! a: ~. e# N) R4 b9 W Complement:是否需要进行补码计算。选择“As is”就是不进行补码计算。& B! F* d' K( o; N. V4 L ) U7 w4 I2 D( X R4 X7 U* D - R# a$ f# H; h% @, P' P8 H Bit order:位输出的顺序。MSB first,每个字节的高位在前。LSB first,每个字节的低位在前。$ `5 h- r$ ?% p3 w 4 K, K' i( |& x: c+ V; M Reverse byte order within word:对于输入数据,在一个字内反转各个字节的顺序。 + X4 Z1 ?5 w0 \; ?5 B7 { 8 y* O/ r2 A: N- R: h7 R Initial value:checksum 计算的初始化值 Checksum unit size :选择进行迭代的单元大小,按 8-bit,16-bit 还是 32-bit 进行迭代。 * C% U2 H: n# j* K+ }5 h ! L" w. p6 f: i) \6 U STM32 CRC 外设使用默认配置时 IAR 的配置* l- h6 K7 E2 F STM32CRC 外设的配置: ·POLY= 0x4C11DB7(CRC32) 3 W( e# X: _$ { \. S3 e# A. g ·Initial_Crc = 0Xffffffff 3 j; ?! ~! n: t1 f# L; E ·输入/输出数据不反转 ·输入数据:0x08000000~0x0801FFFB。(最后 4 个字节用来放计算出的 CRC 值) 4 M; V @0 R7 d7 k* N' _ 在实验的过程发现, ”Alignment ”似乎对计算出的 CRC 值没有影响。但“Reverse byte order within word ”与“Checksumunit size ”这两项的配置有一定关系。如果后者选择 32-bit,则不能勾选前者;反之如果后者选择 8-bit,则一定要勾选上“ Reverse byte order within word ”。也可以参照下图进行设置: 代码怎么写? % Y5 u. N) O8 tI7 c! G$ S; _! G3 [4 i; B" C5 ? 如前文描述,这个应用可以用于对Flash中数据进行校验,参考代码如下:7 R0 r) R3 C- W; S7 b$ }
小结 + N, ^. s) w7 U, P1 ^" m对于CRC应用,还可以根据多项式算子编写纯软件方案,网上有很多现成的代码。其基本思路无外乎查表法以及移位计算法。差异在于一个牺牲存储空间以换取计算效率,一个牺牲计算时间而节省存储空间,至于如何选择,则根据所设计的系统综合考虑,一般根据应用场景来定。. V. R) y9 J: H) P2 d4 Y7 {) u9 b 将块数据利用CRC算法计算出冗余码,有的文章、标准称这个冗余码为签名。实际应用时计算有效数据所得校验码与预存校验码进行比较,相等则校验通过,反之则失败。当然,也可以将原数据与所存校验码一起传入校验算法,所得结果为0则校验通过,反之失败。 对于数据通信,一般会在报文的尾部添加有效数据的校验码,再由接收方校验收到报文的数据完整性。 ~" z5 b8 [% }9 y! {- u , ?2 r4 m9 i7 A- `3 b# z |
谢谢分享~~ |
谢谢分享~~ |
好资料,就是看得有点懵圈 |
谢谢分享!! |
【银杏科技ARM+FPGA双核心应用】STM32H7系列10——ADC
【银杏科技ARM+FPGA双核心应用】STM32H7系列57——MDK_FLM
【STM32图书分享之九】—《STM32F 32位ARM微控制器应用设计与实践》
无刷直流电机控制应用+基于STM8S系列单片机---电子书
STM32 USB的程序,包含固件、驱动和测试用的应用程序
工业以太网总线ETHERCAT驱动程序设计及应用(扫描版)
【实战经验】STM32 DFSDM测量温度应用
【实战经验】基于STM32 I2S的音频应用开发介绍
【中文文档】AN4112_使用STM32F05xx模拟比较器的六个应用案例
[连载]STM32F103ZET6 uCGUI3.90a控件应用教程(第五节 复选框和单选按钮)