ARM Programming: ใช้งาน CRC module ของ STM32

Spread the love

microcontroller ตระกูล STM32 หลายๆ ตัวจะมีฮาร์ดแวร์สำหรับช่วยคำนวณค่า CRC มาด้วย (อยากรู้ว่า CRC คืออะไรดูที่นี่ Wikipedia Cyclic_redundancy_check)
แต่ความสามารถของ CRC module ของ microcontroller แต่ละตัวจะมีไม่เท่ากัน ในรุ่นเล็กๆ อย่าง STM32F030 จะคำนวณได้เฉพาะ CRC32 เท่านั้น ไม่สามารถเปลี่ยนให้คำนวณค่า CRC แบบอื่นได้ ในขณะที่รุ่นใหญ่กว่าเช่น STM32F072 และ F3 F4 จะสามารถเปลี่ยนให้คำนวณค่า CRC แบบอื่นได้

เริ่มใช้ CRC module อย่างไร

ถ้าเขียนโปรแกรมที่ใช้ HAL ก็ใช้ STM32CubeMX สร้าง code ขึ้นมาก่อนได้

เมื่อเปิด code ใน ide แล้วให้ดูใน MX_CRC_INIT() ส่วนนี้จะเป็นการตั้งค่าสำหรับ CRC module รายละเอียดของการตั้งค่าสามารถดูได้จากเอกสาร UM1785 User Manual Description of STM32F0 HAL and low-layer drivers UM1725 User Manual Description of STM32F4 HAL and LL drivers หรือเอกสารแบบเดียวกันสำหรับ STM32 ตระกูลอื่นๆ

ด้านล่างนี้เป็นตัวอย่างสำหรับการคำนวณ CRC16 สำหรับ protocol Modbus

#define MODBUS_CRC_POLY 0x8005

// คำนวณ CRC
uwCRCValue = HAL_CRC_Calculate(&hcrc, (uint32_t *)&commsBuffer, 0x09);

static void MX_CRC_Init(void)
{
  hcrc.Instance = CRC;
  hcrc.Init.DefaultPolynomialUse = DEFAULT_POLYNOMIAL_DISABLE;
  hcrc.Init.DefaultInitValueUse = DEFAULT_INIT_VALUE_DISABLE;
  hcrc.Init.InputDataInversionMode = CRC_INPUTDATA_INVERSION_BYTE;
  hcrc.Init.OutputDataInversionMode = CRC_OUTPUTDATA_INVERSION_ENABLE;
  hcrc.InputDataFormat = CRC_INPUTDATA_FORMAT_BYTES;
    
  hcrc.Init.GeneratingPolynomial    = MODBUS_CRC_POLY;
  hcrc.Init.CRCLength = CRC_POLYLENGTH_16B;
  hcrc.Init.InitValue = 0xFFFF;
    
  if (HAL_CRC_Init(&hcrc) != HAL_OK)
  {
    Error_Handler();
  }
}

การตั้งค่าสำหรับ CRC แบบอื่นนั้นหาได้จากเอกสารของ CRC นั้น หรือที่ผมใช้ก็มาจากเอกสารของ crcmod ที่เป็น library สำหรับคำนวณ CRC สำหรับ Python

CRC algorithm สำหรับ modbus

NamePolynomialReversed?Init valueXOR outCheck
CRC-16 modbus0x18005True0xFFFF0x00000x4B37

เวลานำไปตั้งค่าในโปรแกรม polynomial ตัด 1 ข้างหน้าออก เหลือเพียง 0x8005 ส่วน reversed ก็ enable input/output inversion กำหนด initial value เป็น 0xFFFF ส่วน check คือค่า CRC เมื่อ Input เป็น “123456789” หรือ {0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38,0x39}

ปรียบเทียบกับการคำนวณด้วย software
กรณีที่ต้องการใช้ software คำนวณ CRC16(modbus) เมื่อจำเป็นต้องใช้งานกับ mcu ที่รองรับเฉพาะ CRC32 ที่ https://community.st.com/thread/15006 มีตัวอย่างการคำนวณ CRC หลายๆ แบบอยู่ ซึ่งสำหรับ CRC16 modbus ก็มีอยู่ 3 วิธี เขาเรียกว่าแบบ Slow Quick และ Fast แบบ slow เป็นการคำนวณทีละบิต แบบ Quick จะใช้ lookup table ขนาด 4 บิต ส่วนแบบ fast จะใช้ lookup table ขนาด 8 บิต

เปรียบเทียบความเร็ว ในการคำนวณ modbus CRC ขนาดของ data คือ 9 ไบต์ จำนวน 100,000 ครั้ง บนบอร์ด STM32F072B-Discovery

 1/1000 s
hardware CRC622
fast629
quick759
slow2475

แนะนำ ติชม หรือพูดคุย เชิญที่ twitter ครับ @vorasilp หรือ email vorasilp_at_gmail.com


Spread the love

Add a Comment

Your email address will not be published. Required fields are marked *

This site uses Akismet to reduce spam. Learn how your comment data is processed.