你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

【经验分享】STM32:Modbus-RTU通讯协议——CRC校验

[复制链接]
STMCU小助手 发布时间:2022-5-3 15:45
循环冗余校验(Cyclic Redundancy Check,CRC)是一种根据网络数据包或计算机文件等数据产生简短固定位数校验码的一种信道编码技术,主要用来检测或校验数据传输或者保存后可能出现的错误。
$ [8 p7 m1 q+ V4 B5 y* N
) m2 S1 k# _& m8 S/ [+ q6 d一、16位CRC校验计算方法
/ p  P2 T0 N2 M; ^: V1、 预置1个16位的寄存器为十六进制FFFF(全1),此寄存器为CRC寄存器。4 l2 r; `, `* x1 u" ^+ O2 a  {

, @! W# b; P# b+ w7 `2、 把第一个8位二进制数据(即通讯信息帧的第一个字节)与16位的CRC寄存器的低八位相异或,吧结果存放于CRC寄存器。
) ?% n5 b* M; H& i% v( z7 A
. M( H2 F3 v3 q7 W! I) m+ K- l3、 把CRC寄存器的内容右移一位(朝低位)用0填补最高位,并检测右移后的移出位。
6 j4 K3 b" Y' T& N+ @/ Y! u; b. e: p6 u# b: y* G  P0 g6 t
4、 如果移出位为零,则重复第三步(再次右移一位);如果移出位为1,CRC寄存器与多项式A001进行异或。
4 a8 ?/ p+ ]! K, R- a/ S: X  Y, H8 W% }
5、 重复步骤3和4,直到右移8次,这样整个8位数据全部进行了处理。# q% }! u. J7 }+ Z) X$ K6 ?" I

5 e1 U" _/ E( k" U- c6、 重复步骤2和5,进行通讯信息帧下一个字节的处理。9 e) K; f5 \* {/ v; b/ Z" J# X# A
" W$ ~% I* l2 Y9 I/ V. q
7、 将该通讯信息帧所有字节按上述步骤计算完成后,得到的16位CRC寄存器的高、低字节进行交换。
1 G' J7 f9 ^9 t5 n( r' J* h3 t/ s  I- z6 k! g; ]
8、 最后得到的CRC寄存器内容即为:CRC校验码。* T$ C* L" ~$ a. B0 k4 L, T
+ f& g0 p4 ]6 ]  D# S
二、16位CRC校验程序代码
' \5 M, ]3 \; v( k( Q5 N5 p5 H1. 计算法

. Q$ t7 O3 S* B# `! [具体程序如下:) G2 |6 H9 \7 {7 m8 I

1 x" w3 k8 ?9 G, v' q+ c
  1. uint16_t GetModbusCRC16_Cal(uint8_t *data, uint32_t len)//Modbus-CRC校验-----计算法* F6 p8 \% S& L% N
  2. {. I& M: Z$ \( O- k8 E. S2 z
  3.         uint8_t temp;
    : V- J2 e4 N0 \, c) ^
  4.         uint16_t wcrc = 0XFFFF;//16位crc寄存器预置
    4 K0 h1 S  x! q3 \9 J
  5.         uint32_t i = 0, j = 0;//计数
    ' ^& `5 e0 c: t+ Q8 F+ I2 k
  6.         for (i = 0; i < len; i++)//循环计算每个数据9 X! H. Y; J$ @; P' t2 R
  7.         {4 y  F5 f5 g: g' i5 @, D. _2 K# [
  8.                 temp = data<i> & 0X00FF;//将八位数据与crc寄存器亦或2 k$ S# d/ n9 w- P
  9.                 wcrc ^= temp;                                                //将数据存入crc寄存器
    2 e$ \# P; d2 }# r  H- v
  10.                 for (j = 0; j < 8; j++)        //循环计算数据的0 t* |( t1 \4 u
  11.                 {
    + `# A8 D3 S) n6 R$ \
  12.                         if (wcrc & 0X0001)//判断右移出的是不是1,如果是1则与多项式进行异或。. L% j! p9 |+ t0 b
  13.                         {
    1 j5 X7 {3 X. Q# t# a
  14.                                 wcrc >>= 1;//先将数据右移一位
    6 U1 i7 N. S$ k8 o* \: a4 n3 E
  15.                                 wcrc ^= 0XA001;//与上面的多项式进行异或
    : r9 a9 U; O: R+ u5 b% f$ U
  16.                         }/ @& ~' L  u& o* |
  17.                         else//如果不是1,则直接移出
    8 G; l# P7 [2 A. n9 I6 B
  18.                         {# s+ w, c* @0 K+ o  {
  19.                                 wcrc >>= 1;//直接移出
    5 ^9 d' i$ ~& u! V3 G
  20.                         }+ ]% x. ]. K7 H8 @: }5 G
  21.                 }; ]5 v$ x  ^% p! w
  22.         }
    ; f3 J$ y4 |: C% y. T. P& Y) Q
  23. 5 p) r) h1 r' ^6 F9 }1 }
  24.   return ((wcrc << 8)|(wcrc>>8));//高低位置换  [, X8 E  ~( s
  25. }</i>
复制代码
$ B2 ]5 _% i! {! d  `
2. 查表法
" A  E' ]9 E8 P具体程序如下:1 T( x2 @; ]  X' g! k/ I2 W

9 k: g- T9 T) l. N
  1. uint16_t GetModbusCRC16_Tab(uint8_t *data, uint32_t len)//Modbus-CRC校验-----查表法
    ' y, n0 M+ L) Q
  2. {, M7 d* i2 y7 ]
  3.     unsigned int index;$ p: k6 O4 c! @6 }
  4.     unsigned char crcH = 0xFF;  //高CRC字节
    % `) @4 G* ^5 ^. p+ r$ x  @1 g
  5.     unsigned char crcL = 0xFF;  //低CRC字节: F0 h1 O1 f* m" e8 @+ ^1 J
  6.     const unsigned char TabH[] = {  //CRC高位字节值表
    / I+ {% e$ a6 U+ r+ n, {4 K; b2 N
  7.         0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,  % Z3 _. f+ a) B) l$ ]) U( R- r
  8.         0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,  * _1 R: g1 r, v7 p8 T
  9.         0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,  . B, w  _! _* s* N/ u2 q7 \
  10.         0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,  & t* d  e0 Z8 M' `6 ?
  11.         0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,  3 O3 r+ x+ m( {
  12.         0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41,  
    0 Z8 _! U; [9 ~9 l/ V
  13.         0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,  / P5 ]7 e7 e1 g9 R# T' q
  14.         0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,  ; o6 `6 q! f; {4 Q
  15.         0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,  & h/ r2 [8 g1 {0 {/ U
  16.         0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,  
    7 ?5 D2 W" q+ h' _' t  a
  17.         0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,  + H" X% g( e% i( w% j. n5 a
  18.         0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,  
    6 A; H1 [& t, O/ o' f+ ]( c
  19.         0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,  . A# M" i" q; Q$ g: B. O
  20.         0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,  / f  C5 [# p8 ~0 Z9 V
  21.         0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,  
    9 g: n2 c* l- [: w, {* b* D5 \
  22.         0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,  : B5 M1 t# G- R
  23.         0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,  $ U7 E" [  Y: J
  24.         0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,  - m' K% q8 J3 `. l  y9 e7 u* C
  25.         0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,  % `4 O7 L+ A! d% H3 A. R$ |/ X
  26.         0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,  
    & k$ P" z6 Q1 Y' r8 ~$ {! V
  27.         0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,  
    7 Y5 ?# }, Z1 A6 S
  28.         0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,    Q: I$ n4 m7 J/ p
  29.         0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,  / S$ |( r9 _/ |1 E
  30.         0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,  2 o% n5 i# D9 r% D7 M
  31.         0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,  
    ' S5 K" ~/ T5 k- [8 d6 H2 D
  32.         0x80, 0x41, 0x00, 0xC1, 0x81, 0x40  
    9 Y3 P" l1 g& p; D/ Q- |
  33.     } ;  , |# i" k$ O6 H
  34.     const unsigned char TabL[] = {  //CRC低位字节值表( Q* Q( O1 I8 i! M8 l  D, G  U
  35.         0x00, 0xC0, 0xC1, 0x01, 0xC3, 0x03, 0x02, 0xC2, 0xC6, 0x06,  ) ~1 K' F/ A- R3 Z% D" M! g
  36.         0x07, 0xC7, 0x05, 0xC5, 0xC4, 0x04, 0xCC, 0x0C, 0x0D, 0xCD,  
    1 H4 e* J. |; h8 G6 ]* N
  37.         0x0F, 0xCF, 0xCE, 0x0E, 0x0A, 0xCA, 0xCB, 0x0B, 0xC9, 0x09,  4 D' n! P  x% I: `1 x2 J$ R, G$ `
  38.         0x08, 0xC8, 0xD8, 0x18, 0x19, 0xD9, 0x1B, 0xDB, 0xDA, 0x1A,  " i( W) s7 p; n# Z
  39.         0x1E, 0xDE, 0xDF, 0x1F, 0**, 0x1D, 0x1C, 0xDC, 0x14, 0xD4,  % x( |8 O3 L% V) i5 W6 x8 d
  40.         0xD5, 0x15, 0xD7, 0x17, 0x16, 0xD6, 0xD2, 0x12, 0x13, 0xD3,  
    ( }$ S$ }1 V  w3 ~  b
  41.         0x11, 0xD1, 0xD0, 0x10, 0xF0, 0x30, 0x31, 0xF1, 0x33, 0xF3,  + {9 Z9 P" t. b/ t
  42.         0xF2, 0x32, 0x36, 0xF6, 0xF7, 0x37, 0xF5, 0x35, 0x34, 0xF4,  1 A6 M0 p( _" ?: Z: O! ]8 l: k& M
  43.         0x3C, 0xFC, 0xFD, 0x3D, 0xFF, 0x3F, 0x3E, 0xFE, 0xFA, 0x3A,  
    0 `2 n* V( H) S: s; \
  44.         0x3B, 0xFB, 0x39, 0xF9, 0xF8, 0x38, 0x28, 0xE8, 0xE9, 0x29,  5 g" ]" p" V, [, |
  45.         0xEB, 0x2B, 0x2A, 0xEA, 0xEE, 0x2E, 0x2F, 0xEF, 0x2D, 0xED,  
    # J! K& D7 M# N6 s- t
  46.         0xEC, 0x2C, 0xE4, 0x24, 0x25, 0xE5, 0x27, 0xE7, 0xE6, 0x26,  3 S) e9 a2 z5 a
  47.         0x22, 0xE2, 0xE3, 0x23, 0xE1, 0x21, 0x20, 0xE0, 0xA0, 0x60,  
    ' k" |3 t! o( [( T" ~
  48.         0x61, 0xA1, 0x63, 0xA3, 0xA2, 0x62, 0x66, 0xA6, 0xA7, 0x67,  3 k# m# p( G2 C% t8 @+ u& C
  49.         0xA5, 0x65, 0x64, 0xA4, 0x6C, 0xAC, 0xAD, 0x6D, 0xAF, 0x6F,  ) a9 v. l6 R8 a5 q
  50.         0x6E, 0xAE, 0xAA, 0x6A, 0x6B, 0xAB, 0x69, 0xA9, 0xA8, 0x68,  
    " ~$ w& S" n8 J; O' C) N+ z
  51.         0x78, 0xB8, 0xB9, 0x79, 0xBB, 0x7B, 0x7A, 0xBA, 0xBE, 0x7E,  
    5 z; a' Q, B/ T% ]
  52.         0x7F, 0xBF, 0x7D, 0xBD, 0xBC, 0x7C, 0xB4, 0x74, 0x75, 0xB5,  : E& {3 b$ C& }! ^+ L' ?
  53.         0x77, 0xB7, 0xB6, 0x76, 0x72, 0xB2, 0xB3, 0x73, 0xB1, 0x71,  
    4 f1 a' h! P5 }
  54.         0x70, 0xB0, 0x50, 0x90, 0x91, 0x51, 0x93, 0x53, 0x52, 0x92,  . C8 h) z% s: ]+ e' f# s
  55.         0x96, 0x56, 0x57, 0x97, 0x55, 0x95, 0x94, 0x54, 0x9C, 0x5C,  4 {3 P3 J4 k# a9 c$ p% l3 R
  56.         0x5D, 0x9D, 0x5F, 0x9F, 0x9E, 0x5E, 0x5A, 0x9A, 0x9B, 0x5B,    n' c  D. t4 X: W4 ]" h
  57.         0x99, 0x59, 0x58, 0x98, 0x88, 0x48, 0x49, 0x89, 0x4B, 0x8B,  
    ) U; w8 n/ K6 y' K9 q0 X: k2 o
  58.         0x8A, 0x4A, 0x4E, 0x8E, 0x8F, 0x4F, 0x8D, 0x4D, 0x4C, 0x8C,  & j0 F: o* t, X' z- O3 o
  59.         0x44, 0x84, 0x85, 0x45, 0x87, 0x47, 0x46, 0x86, 0x82, 0x42,  
    . o" M6 S6 \" J* P2 }& }0 U; @, t
  60.         0x43, 0x83, 0x41, 0x81, 0x80, 0x40  & j1 J! z! P! N; X
  61.     } ;3 R0 K) U& [! @. g
  62.     while (len--)  //计算指定长度的CRC
    , m' W/ {; }3 V8 r$ K5 t* y. P  |" v
  63.     {; g3 _! p* G* A0 O( y: b: O
  64.         index = crcL ^ *data++;& _0 q: B* Z. P8 V' m$ j
  65.         crcL = crcH        ^ TabH[ index];- w* E, j# Y- w0 ^
  66.         crcH = TabL[ index];: ]7 g" J, ^8 Z; `" J# I
  67.     }
    # [% R! _7 B: @1 C: ^4 j  f& H- O
  68.                 return crcL<<8|crcH;//高低位置换
      S( l! ~! g. \: z9 P
  69. }  
    $ P$ {! J( n* y" k- `
  70. " _* t" J- s, Y7 J9 \6 F
  71. * y. V6 a& s+ A! s. l) b

  72. # B( M1 S6 Y/ T0 E" n) g3 c7 g
复制代码

. I- \4 Q% c; w/ p8 Y7 }: }$ b: L
收藏 评论0 发布时间:2022-5-3 15:45

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版