OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 304|回复: 8

这个CRC7循环冗余校验码老是算不对

[复制链接]

72

主题

179

帖子

0

精华

高级会员

Rank: 4

积分
559
金钱
559
注册时间
2014-10-15
在线时间
132 小时
发表于 2024-9-2 10:18:04 | 显示全部楼层 |阅读模式
1金钱

我找到一个STM32L4R5ZIT6的关于CRC的例程,


代码如下

  1. #include "main.h"

  2. /** @addtogroup STM32L4xx_HAL_Examples
  3.   * @{
  4.   */

  5. /** @addtogroup CRC_Bytes_Stream_7bit_CRC
  6.   * @{
  7.   */

  8. /* Private typedef -----------------------------------------------------------*/
  9. /* Private define ------------------------------------------------------------*/
  10. #define BUFFER_SIZE_5  5  /* CRC7_DATA8_TEST5[] is 5-byte long   */
  11. #define BUFFER_SIZE_17 17 /* CRC7_DATA8_TEST17[] is 17-byte long */
  12. #define BUFFER_SIZE_1  1  /* CRC7_DATA8_TEST1[] is 1-byte long   */
  13. #define BUFFER_SIZE_2  2  /* CRC7_DATA8_TEST2[] is 2-byte long   */

  14. /* User-defined polynomial */
  15. #define CRC_POLYNOMIAL_7B  0x65  /* X^7 + X^6 + X^5 + X^2 + 1,
  16.                                    used in Train Communication Network, IEC 60870-5[17] */

  17. /* Private macro -------------------------------------------------------------*/
  18. /* Private variables ---------------------------------------------------------*/
  19. /* CRC handler declaration */
  20. CRC_HandleTypeDef   CrcHandle;

  21. /* Used for storing CRC Value */
  22. __IO uint32_t uwCRCValue = 0;

  23. /* Bytes buffers that will consecutively yield CRCs */
  24. static const uint8_t CRC7_DATA8_TEST5[5]   = {0x12,0x34,0xBA,0x71,0xAD};
  25. static const uint8_t CRC7_DATA8_TEST17[17] = {0x12,0x34,0xBA,0x71,0xAD,
  26.                                               0x11,0x56,0xDC,0x88,0x1B,
  27.                                               0xEE,0x4D,0x82, 0x93,0xA6,
  28.                                               0x7F,0xC3};
  29. static const uint8_t CRC7_DATA8_TEST1[1]   = {0x19};                                                
  30. static const uint8_t CRC7_DATA8_TEST2[2]   = {0xAB,0xCD};

  31.       

  32. /* Expected CRC Values */
  33. /* The 7 LSB bits are the 7-bit long CRC */
  34. uint32_t uwExpectedCRCValue_1 = 0x00000057;    /* First byte stream CRC  */
  35. uint32_t uwExpectedCRCValue_2 = 0x0000006E;    /* Second byte stream CRC */
  36. uint32_t uwExpectedCRCValue_3 = 0x0000004B;    /* Third byte stream CRC  */
  37. uint32_t uwExpectedCRCValue_4 = 0x00000026;    /* Fourth byte stream CRC */

  38. /* Private function prototypes -----------------------------------------------*/
  39. void SystemClock_Config(void);
  40. static void Error_Handler(void);

  41. /* Private functions ---------------------------------------------------------*/

  42. /**
  43.   * [url=home.php?mod=space&uid=159083]@brief[/url] Main program
  44.   * [url=home.php?mod=space&uid=271674]@param[/url]  None
  45.   * @retval None
  46.   */
  47. int main(void)
  48. {

  49.   /* STM32L4xx HAL library initialization:
  50.        - Configure the Flash prefetch
  51.        - Systick timer is configured by default as source of time base, but user
  52.          can eventually implement his proper time base source (a general purpose
  53.          timer for example or other time source), keeping in mind that Time base
  54.          duration should be kept 1ms since PPP_TIMEOUT_VALUEs are defined and
  55.          handled in milliseconds basis.
  56.        - Set NVIC Group Priority to 4
  57.        - Low Level Initialization
  58.      */
  59.   HAL_Init();
  60.   
  61.   /* Configure the system clock to 120 MHz */
  62.   SystemClock_Config();

  63.   /* Configure LED1 and LED3 */
  64.   BSP_LED_Init(LED1);
  65.   BSP_LED_Init(LED3);


  66.   /****************************************************************************/
  67.   /*                                                                          */
  68.   /*                     CRC peripheral initialization                        */
  69.   /*                                                                          */   
  70.   /****************************************************************************/
  71.    
  72.   CrcHandle.Instance = CRC;

  73.   /* The default polynomial is not used. The one to be used must be defined
  74.      in CrcHandle.Init.GeneratingPolynomial */  
  75.   CrcHandle.Init.DefaultPolynomialUse    = DEFAULT_POLYNOMIAL_DISABLE;
  76.   
  77.   /* Set the value of the generating polynomial.
  78.     The one used in that example is the 7-bit long CRC generating
  79.     polynomial X^7 + X^6 + X^5 + X^2 + 1 */
  80.   CrcHandle.Init.GeneratingPolynomial    = CRC_POLYNOMIAL_7B;
  81.   
  82.   /* The user-defined generating polynomial yields a 7-bit long CRC */
  83.   CrcHandle.Init.CRCLength               = CRC_POLYLENGTH_7B;

  84.   /* The default init value is used */
  85.   CrcHandle.Init.DefaultInitValueUse     = DEFAULT_INIT_VALUE_ENABLE;

  86.   /* The input data are not inverted */
  87.   CrcHandle.Init.InputDataInversionMode  = CRC_INPUTDATA_INVERSION_NONE;

  88.   /* The output data are not inverted */
  89.   CrcHandle.Init.OutputDataInversionMode = CRC_OUTPUTDATA_INVERSION_DISABLE;

  90.   /* The input data are bytes (8-bit long data) */
  91.   CrcHandle.InputDataFormat              = CRC_INPUTDATA_FORMAT_BYTES;

  92.   /* De-initialize the CRC peripheral */
  93.   if (HAL_CRC_DeInit(&CrcHandle) != HAL_OK)
  94.   {
  95.     /* Initialization Error */
  96.     Error_Handler();
  97.   }  

  98.   /* Then, initialize the CRC handle */
  99.   if (HAL_CRC_Init(&CrcHandle) != HAL_OK)
  100.   {
  101.     /* Initialization Error */
  102.     Error_Handler();
  103.   }


  104.   /****************************************************************************/
  105.   /*                                                                          */
  106.   /*         CRC computation of a first bytes stream                          */
  107.   /*                                                                          */   
  108.   /****************************************************************************/

  109.   /* The 7-bit long CRC of a 5-byte buffer is computed. After peripheral initialization,
  110.      the CRC calculator is initialized with the default value that is 0x7F for
  111.      a 7-bit CRC.
  112.    
  113.     The computed CRC is stored in uint32_t uwCRCValue. The 7-bit long CRC is made of
  114.     uwCRCValue 7 LSB bits. */

  115.   uwCRCValue = HAL_CRC_Accumulate(&CrcHandle, (uint32_t *)&CRC7_DATA8_TEST5, BUFFER_SIZE_5);

  116.   /* Compare the CRC value to the expected one */
  117.   if (uwCRCValue != uwExpectedCRCValue_1)
  118.   {
  119.     /* Wrong CRC value: Turn LED3 on */
  120.     Error_Handler();
  121.   }

  122.   
  123.   /****************************************************************************/
  124.   /*                                                                          */
  125.   /*         CRC computation of a second bytes stream                         */
  126.   /*                                                                          */   
  127.   /****************************************************************************/

  128.   /* The 7-bit long CRC of a 17-byte buffer is computed. The CRC calculator
  129.     is not re-initialized, instead the previously computed CRC is used
  130.     as initial value. */

  131.   uwCRCValue = HAL_CRC_Accumulate(&CrcHandle, (uint32_t *)&CRC7_DATA8_TEST17, BUFFER_SIZE_17);

  132.   /* Compare the CRC value to the expected one */
  133.   if (uwCRCValue != uwExpectedCRCValue_2)
  134.   {
  135.     /* Wrong CRC value: Turn LED3 on */
  136.     Error_Handler();
  137.   }


  138.   /****************************************************************************/
  139.   /*                                                                          */
  140.   /*         CRC computation of a single byte                                 */
  141.   /*                                                                          */   
  142.   /****************************************************************************/

  143.   /* The 7-bit long CRC of a 1-byte buffer is computed. The CRC calculator
  144.     is not re-initialized, instead the previously computed CRC is used
  145.     as initial value. */

  146.   uwCRCValue = HAL_CRC_Accumulate(&CrcHandle, (uint32_t *)&CRC7_DATA8_TEST1, BUFFER_SIZE_1);

  147.   /* Compare the CRC value to the expected one */
  148.   if (uwCRCValue != uwExpectedCRCValue_3)
  149.   {
  150.     /* Wrong CRC value: Turn LED3 on */
  151.     Error_Handler();
  152.   }


  153.   /****************************************************************************/
  154.   /*                                                                          */
  155.   /*         CRC computation of the last bytes stream                         */
  156.   /*                                                                          */   
  157.   /****************************************************************************/

  158.   /* The 7-bit long CRC of a 2-byte buffer is computed. The CRC calculator
  159.     is re-initialized with the default value that is 0x7F for a 7-bit CRC.
  160.     This is done with a call to HAL_CRC_Calculate() instead of
  161.     HAL_CRC_Accumulate(). */

  162.   uwCRCValue = HAL_CRC_Calculate(&CrcHandle, (uint32_t *)&CRC7_DATA8_TEST2, BUFFER_SIZE_2);

  163.   /* Compare the CRC value to the expected one */
  164.   if (uwCRCValue != uwExpectedCRCValue_4)
  165.   {
  166.     /* Wrong CRC value: Turn LED3 on */
  167.     Error_Handler();
  168.   }
  169.   else
  170.   {
  171.     /* Right CRC value: Turn LED1 on */
  172.     BSP_LED_On(LED1);
  173.   }  


  174.   /* Infinite loop */
  175.   while (1)
  176.   {
  177.   }
  178. }
复制代码

多项式是0x65,没说初始值,有四组数据,

得到的CRC校验值分别为

uint32_t uwExpectedCRCValue_1 = 0x00000057;    /* First byte stream CRC  */

uint32_t uwExpectedCRCValue_2 = 0x0000006E;    /* Second byte stream CRC */

uint32_t uwExpectedCRCValue_3 = 0x0000004B;    /* Third byte stream CRC  */

uint32_t uwExpectedCRCValue_4 = 0x00000026;    /* Fourth byte stream CRC */

我不知道这个校验码是怎么得来的?网上下了好多CRC计算器,算的值都不对。

请问高手,谁知道是怎么计算的?谢谢!


正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

4

主题

881

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
4195
金钱
4195
注册时间
2019-9-4
在线时间
881 小时
发表于 2024-9-2 10:52:58 | 显示全部楼层
CRC(循环冗余校验)是一种常用的用于检测数据传输或存储中错误的技术。在STM32微控制器中,CRC计算可以通过硬件CRC模块来完成。在你提供的代码中,使用了STM32的HAL库函数来计算CRC值。

首先,让我们来理解一下代码中的几个关键点:

1. 多项式:CRC计算中使用的多项式是0x65,这表示多项式是x^7 + x^6 + x^5 + x^2 + 1。

2. 初始值:代码中使用了默认的初始值,这是通过CrcHandle.Init.DefaultInitValueUse = DEFAULT_INIT_VALUE_ENABLE;来设置的。对于7位CRC,这个默认值通常是0x7F。

3. 数据格式:输入数据格式是字节(8位),这是通过CrcHandle.InputDataFormat = CRC_INPUTDATA_FORMAT_BYTES;来设置的。

4. 计算函数:使用了HAL_CRC_Accumulate函数来连续计算多个数据块的CRC,以及HAL_CRC_Calculate函数来计算单个数据块的CRC。

5. 预期的CRC值:代码中提供了四组预期的CRC值,这些值是预先计算好的,用于验证代码计算的正确性。

要计算CRC值,你需要按照以下步骤进行:

1. 初始化CRC模块:设置多项式、初始值、数据格式等。

2. 计算CRC:将数据块输入到CRC模块中,进行计算。

3. 获取结果:从CRC模块中读取计算结果。

在你的情况下,如果你使用在线CRC计算器,你需要确保:

- 使用相同的多项式(0x65)。
- 使用相同的初始值(通常是0x7F)。
- 确保数据格式正确(8位字节)。
- 如果计算器支持,确保输入数据的反转和输出数据的反转设置与代码中一致。

如果你在计算时没有得到预期的结果,可能的原因包括:

- 使用了错误的多项式。
- 使用了错误的初始值。
- 数据格式设置不正确。
- 输入或输出数据反转设置不正确。

你可以尝试手动计算第一个数据块的CRC值,以验证你的计算器设置是否正确。例如,对于第一个数据块{0x12, 0x34, 0xBA, 0x71, 0xAD},你可以按照以下步骤进行:

1. 将数据块的每个字节依次输入到CRC计算器中。
2. 使用多项式0x65和初始值0x7F。
3. 确保数据格式为8位字节。
4. 计算结果应该是0x57(只取最低7位)。

如果你的计算器设置正确,你应该能够得到与uwExpectedCRCValue_1相同的结果。如果结果仍然不正确,你可能需要检查计算器的设置或尝试使用不同的计算器。
回复

使用道具 举报

72

主题

179

帖子

0

精华

高级会员

Rank: 4

积分
559
金钱
559
注册时间
2014-10-15
在线时间
132 小时
 楼主| 发表于 2024-9-2 11:15:16 | 显示全部楼层
谢谢!
我的代码:
#include <stdio.h>
#include <stdint.h>

typedef unsigned char uint8_t;
typedef unsigned short uint16_t;




uint8_t crc7_mmc(uint8_t *data, uint16_t length)
{
    uint8_t i;
    uint8_t crc = 0x7f;        // Initial value
    while(length--)
    {
        crc ^= *data++;        // crc ^= *data; data++;
        for ( i = 0; i < 8; i++ )
        {
            if ( crc & 0x80 )
                crc = (crc << 1) ^ 0x65;        // 0x12 = 0x09<<(8-7)
            else
                crc <<= 1;
        }
    }
    return crc ;
}


int main(void){
    uint8_t a=0x19;
    uint8_t b;
    b=crc7_mmc(&a,1);
    printf("%x\n",b);
    return 0;
}
你看哪里有问题?计算出结果位0x24.
回复

使用道具 举报

72

主题

179

帖子

0

精华

高级会员

Rank: 4

积分
559
金钱
559
注册时间
2014-10-15
在线时间
132 小时
 楼主| 发表于 2024-9-2 12:50:15 | 显示全部楼层
  1. uint8_t checkCRC7(uint8_t *data, uint32_t len){
  2.     uint8_t crc = 0xfe;
  3.     uint8_t crcP = 0x65;
  4.     uint8_t i, j;
  5.    
  6.     for ( i = 0; i < len; i++){
  7.         crc ^= data[i];
  8.         for (j = 0; j < 8; j++){
  9.             if ( crc &0x80){
  10.                 crc ^= 0x65;
  11.             }
  12.             crc = crc <<1;
  13.         }
  14.     }       
  15.    
  16.         crc = crc >> 1;
  17.     return crc;   
  18. }
复制代码
这段代码计算第一组和第四组数据都是对的
第二组和第三组不对
回复

使用道具 举报

72

主题

179

帖子

0

精华

高级会员

Rank: 4

积分
559
金钱
559
注册时间
2014-10-15
在线时间
132 小时
 楼主| 发表于 2024-9-2 13:25:04 | 显示全部楼层
  1. uint32_t checkCRC71(uint8_t *data, uint32_t len){
  2.     uint32_t crc = 0x000000fe;
  3.     uint8_t crcP = 0x65;
  4.     uint8_t i, j;
  5.    
  6.     for ( i = 0; i < len; i++){
  7.         crc ^= (uint32_t)data[i];
  8.         for (j = 0; j < 32; j++){
  9.             if ( crc &0x80000000){
  10.                 crc ^= 0x65;
  11.             }
  12.             crc = crc <<1;
  13.         }
  14.     }       
  15.    
  16.         //crc = crc >> 1;
  17.     return crc;   
  18. }
复制代码
我把代码改成这样,只有第四组是对的
回复

使用道具 举报

0

主题

20

帖子

0

精华

初级会员

Rank: 2

积分
152
金钱
152
注册时间
2023-12-6
在线时间
32 小时
发表于 2024-9-3 10:39:13 | 显示全部楼层
本帖最后由 JiangYoo 于 2024-9-3 10:47 编辑

  • uint8_t checkCRC7(uint8_t *data, uint32_t len){
  •     uint8_t crc = 0xfe << 1;
  •     uint8_t crcP = 0x65 << 1;
  •     uint8_t i, j;
  •     for ( i = 0; i < len; i++){
  •         crc ^= data;
  •         for (j = 0; j < 8; j++){
  •             crc = (crc & 0x80) ? ((crc << 1) ^ crcP) : (crc << 1)
  •         }
  •     return crc;
  • }
  • // 用这个来,你上面的校验步骤都不对

回复

使用道具 举报

0

主题

20

帖子

0

精华

初级会员

Rank: 2

积分
152
金钱
152
注册时间
2023-12-6
在线时间
32 小时
发表于 2024-9-3 10:44:34 | 显示全部楼层
/**
  * @brief  CRC7/MMC.
  * @param  [in] pData 数据源地址
  * @param  [in] Size  数据大小
  * @return CRC校验结果值
  * @note   CRC参数:
  *         WITDH: 7      POLY:   X7+X3+1(0x09)
  *         INIT:  0x00   XOROUT: 0x00
  *         REFIN: FALSE  REFOUT: FALSE
  */
uint8_t CRC7_MMC(uint8_t const* pData, EMB_SIZE Size)
{
  EMB_ASSERT((pData != NULL_PTR) && (Size > 0), 0x00);

  uint8_t POLY = 0x09 << 1;

  uint8_t CRC_Value = 0x00;

  while (Size--)
  {
    CRC_Value ^= *pData++;

    for (uint8_t i = 0; i < 8; i++)
    {
      CRC_Value = (CRC_Value & 0x80) ? ((CRC_Value << 1) ^ POLY) : (CRC_Value << 1);
    }
  }

  return CRC_Value >> 1;
}

我不知道你这是哪里弄来的,CRC7-MMC在网上的多项式,初始值跟你的都不一样,都是用0x09的多项式

回复

使用道具 举报

72

主题

179

帖子

0

精华

高级会员

Rank: 4

积分
559
金钱
559
注册时间
2014-10-15
在线时间
132 小时
 楼主| 发表于 2024-9-3 13:14:51 | 显示全部楼层
JiangYoo 发表于 2024-9-3 10:44
/**
  * @brief  CRC7/MMC.
  * @param   pData 数据源地址

我修改了,因为多项式不一样
回复

使用道具 举报

72

主题

179

帖子

0

精华

高级会员

Rank: 4

积分
559
金钱
559
注册时间
2014-10-15
在线时间
132 小时
 楼主| 发表于 2024-9-3 13:15:20 | 显示全部楼层
JiangYoo 发表于 2024-9-3 10:39
  • uint8_t checkCRC7(uint8_t *data, uint32_t len){
  •     uint8_t crc = 0xfe

  • 多谢!用你的不好使
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则



    关闭

    原子哥极力推荐上一条 /2 下一条

    正点原子公众号

    QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

    GMT+8, 2024-11-22 06:37

    Powered by OpenEdv-开源电子网

    © 2001-2030 OpenEdv-开源电子网

    快速回复 返回顶部 返回列表