OpenEdv-开源电子网

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

再问STM32L4R5板子CRC程序的问题

[复制链接]

72

主题

179

帖子

0

精华

高级会员

Rank: 4

积分
561
金钱
561
注册时间
2014-10-15
在线时间
132 小时
发表于 2023-6-5 16:40:09 | 显示全部楼层 |阅读模式
1金钱
我有一个STM32L4R5板子,有四个CRC例程,我搞好了三个。
还有一个CRC7。
代码如下:
  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. #define CRC_POLYNOMIAL_7B  0x65  /* X^7 + X^6 + X^5 + X^2 + 1,*/
  18. /* Private macro -------------------------------------------------------------*/
  19. /* Private variables ---------------------------------------------------------*/
  20. /* CRC handler declaration */
  21. CRC_HandleTypeDef   CrcHandle;

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

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

  32.       

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

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

  42. /* Private functions ---------------------------------------------------------*/

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

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

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


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

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

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

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

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

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

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

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


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

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

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

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

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

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

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

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


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

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

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

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


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

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

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

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


  175.   /* Infinite loop */
  176.   while (1)
  177.   {
  178.   }
  179. }
复制代码
我找到一个例子,代码如下:
  1. # include <stdio.h>
  2. # include <string.h>

  3. #define uint8_t unsigned char
  4. #define uint16_t unsigned int
  5. #define uint32_t unsigned  long int

  6. #define TAB_LEN 256
  7. #define ALPHA 0x65

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

  12. const uint8_t CRC7_DATA8_TEST5[5]   = {0x12,0x34,0xBA,0x71,0xAD};
  13. const uint8_t CRC7_DATA8_TEST17[17] = {0x12,0x34,0xBA,0x71,0xAD,
  14.                                               0x11,0x56,0xDC,0x88,0x1B,
  15.                                               0xEE,0x4D,0x82, 0x93,0xA6,
  16.                                               0x7F,0xC3};
  17. const uint8_t CRC7_DATA8_TEST1[1]   = {0x19};                                                
  18. const uint8_t CRC7_DATA8_TEST2[2]   = {0xAB,0xCD};
  19. const unsigned char testdat[10] = "0123456789";
  20.       

  21. /* Expected CRC Values */
  22. /* The 7 LSB bits are the 7-bit long CRC */
  23. uint32_t uwExpectedCRCValue_1 = 0x00000057;    /* First byte stream CRC  */
  24. uint32_t uwExpectedCRCValue_2 = 0x0000006E;    /* Second byte stream CRC */
  25. uint32_t uwExpectedCRCValue_3 = 0x0000004B;    /* Third byte stream CRC  */
  26. uint32_t uwExpectedCRCValue_4 = 0x00000026;    /* Fourth byte stream CRC */


  27. unsigned char result;

  28. int table_gen8(unsigned char *buf){
  29.     unsigned int alpha = ALPHA;  //x^7+x^3+1
  30.     int i,j;
  31.     unsigned char tmp;
  32.     for(i=0;i<TAB_LEN;i++){
  33.         tmp = i;
  34.         for(j=0;j<8;j++){
  35.             if(tmp & 0x80)         
  36.                 tmp ^= alpha;
  37.             tmp <<= 1;
  38.         }      
  39.         buf[i] = tmp>>1;    /*余数为7bit,计算中用了8bit,结尾多一位0要去掉*/
  40.     }   
  41.     return 0;
  42. }

  43. uint8_t checkCRC7(uint8_t *data, uint32_t len){
  44.     uint8_t crc = 0x7F;
  45.     uint8_t crcP = 0x65;
  46.     uint8_t i, j;
  47.    
  48.     for ( i = 0; i < len; i++){
  49.         crc ^= data[i];
  50.         for (j = 0; j < 8; j++){
  51.             if ( crc &0x80){
  52.                 crc ^= 0x65;
  53.             }
  54.             crc = crc <<1;
  55.         }
  56.     }       
  57.    
  58.         crc = crc >> 1;
  59.     return crc;   
  60. }

  61. unsigned char get_crc7(unsigned char start, const unsigned char *buff, int len, unsigned char *table){
  62.     unsigned char accu = start;
  63.     //unsigned char accu = (start<<1);
  64.     unsigned int i= 0;
  65.     for (i=0;  i < len; i++) {
  66.         accu = table[(accu << 1) ^ buff[i]];
  67.         //accu = table[accu^ buff[i]];
  68.     }
  69.     return (accu);
  70. }



  71. int main(void){
  72.     unsigned char data[TAB_LEN] = {0};
  73.     int i,j;
  74.     printf("CRC7 table:\n");
  75.     table_gen8(data);
  76.    /* for(i=0;i<TAB_LEN/8;i++){
  77.         for(j=0;j<8;j++)
  78.             printf("0x%02x   ",data[i*8+j]);
  79.         printf("\n");
  80.     }
  81.     printf("\n");*/
  82.     /*Test*/
  83.    
  84.     //result = get_crc7(0x7f, CRC7_DATA8_TEST5, 5, data);
  85.     //result = get_crc7(0x7f, CRC7_DATA8_TEST17, 17, data);
  86.     //result = get_crc7(0x7f, CRC7_DATA8_TEST1, 1, data);
  87.     //result = get_crc7(0x7f, CRC7_DATA8_TEST2, 2, data);
  88.     printf("get_crc7:0x%02x\n",result);
  89.     return 0;
  90. }
复制代码
一共四组数据,第一组和第四组值是对的。而第二组和第三组值不对。
本来应该第二组是0x5E,第三组是0x4b.实测值为0x26和0x2f。
请问高手,谁知道软件如何计算出正确的值?或者我写的这个程序哪里有问题?谢谢!

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

使用道具 举报

72

主题

179

帖子

0

精华

高级会员

Rank: 4

积分
561
金钱
561
注册时间
2014-10-15
在线时间
132 小时
 楼主| 发表于 2023-6-17 11:48:28 | 显示全部楼层
我自己写了一个代码,如下:
  1. uint8_t crc7_mmc(uint8_t *data, uint16_t length)
  2. {
  3.     uint8_t i;
  4.     uint8_t crc = 0xFE;     
  5.     while(length--)
  6.     {
  7.         crc ^= *data++;        
  8.         for ( i = 0; i < 8; i++ )
  9.         {
  10.             if ( crc & 0x80 )
  11.                 //crc = (crc << 1) ^ 0x65;   
  12.                                 crc = (crc << 1) ^ 0xCA;      
  13.             else
  14.                 crc <<= 1;
  15.         }
  16.     }
  17.     return crc >> 1;
  18. }
复制代码
用这个程序算的也是第一组和第四组数据对,第二组和第三组数据不对,我无语了。
回复

使用道具 举报

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

本版积分规则



关闭

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

正点原子公众号

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

GMT+8, 2024-11-24 06:50

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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