1、概述
CRC16循环冗余校验常用在MODBUS协议中,用于校验报文的完整性。CRC16校验值为uint16_t 无符号整形2字节,在MODBUS协议中,低检验字节在前,高校验字节在后,比如校验结果crc16=0x1788,则MODBUS中的校验顺序为 ...0x88 0x17。
以下为自己写的linux C 代码,可以直接用。
2、C程序
/*******************************************************************************
_____ ___ ____ ___ _____ _ _ _
| ____|_ _| _ \|_ _| |_ _|__(_)_ __ __ _| | | |_ _ __ _
| _| | || |_) || | | |/ __| | '_ \ / _` | |_| | | | |/ _` |
| |___ | || _ < | | | |\__ \ | | | | (_| | _ | |_| | (_| |
|_____|___|_| \_\___| |_||___/_|_| |_|\__, |_| |_|\__,_|\__,_|* File Name : main.c* Description : This file provides code for crc16 caculation in linuxc.* Author : jackwang by jiawang16@foxmail.com* Date : 2018-07-15
*******************************************************************************/
/*! -------------------------------------------------------------------------- */
/*! Include headers */
#include <stdio.h>
#include <string.h>
#include <stdbool.h>/*! -------------------------------------------------------------------------- */
/*! Private function declarations */
static unsigned char Char2Int(char chr,bool *isOK);/*! convert char to int type*/
static unsigned char HexStr2Int(char *str, bool *isOK);/*!convert hexstr to int*/
/*! caculate crc16 of buff-input:arr_buff and length: len */
static unsigned short GenerateCRC16(unsigned char *arr_buff, unsigned short len);/*! -------------------------------------------------------------------------- */
/*! main function defination */
int main(int argc, char* argv[])
{/*! variable define */int ret = 0;int numByte = argc; unsigned char bccVal = 0x00;char inPutbuff[10];unsigned char databuff[1000];unsigned short buffsize = 0;unsigned short crcVal;bool isOK;int Nibb;if(argc == 1){printf("[note] no params to caculate, please input hex string,splite by space!\r\n");}else{printf("[note] input %d byte: ",numByte-1);for(int i = 1; i < numByte; i++){printf("%s ",argv[i]);}printf("\r\n");for(int i = 1; i < numByte; i++){memcpy(inPutbuff,argv[i],2);Nibb = HexStr2Int(inPutbuff,&isOK);if(isOK){databuff[i-1] = Nibb;buffsize++;}}crcVal = GenerateCRC16(databuff,buffsize);printf("[note] crc16 value: 0x%04X\r\n",crcVal);}return 0;
}/*! -------------------------------------------------------------------------- */
/*! Private function definations */
/*! ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ */
static unsigned char Char2Int(char chr,bool *isOK)
{unsigned char nibb1;if(chr >= '0' && chr <= '9'){ nibb1 = chr - '0'; *isOK = true;}else if(chr >= 'a' && chr <= 'f'){ nibb1 = chr - 'a' + 10; *isOK = true;}else if(chr >= 'A' && chr <= 'F'){ nibb1 = chr -'A' + 10; *isOK = true; }else{ printf("[error] invalid hex str input: %c \r\n",chr); *isOK = false; }return nibb1;
}/*! ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ */
static unsigned char HexStr2Int(char *str, bool *isOK)
{unsigned char nibb1,nibb2;bool isOK1,isOK2;nibb1 = Char2Int(*str, &isOK1);nibb2 = Char2Int(*(str+1),&isOK2);if(isOK1 && isOK2){*isOK = true;return nibb1*16 + nibb2;}else{*isOK = false;return 0;}
} /*! ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ */
static unsigned short GenerateCRC16(unsigned char *arr_buff, unsigned short len)
{unsigned short crc=0xFFFF;unsigned char i, j;for ( j=0; j < len;j++){crc=crc ^*arr_buff++;for ( i=0; i<8; i++) {if( ( crc&0x0001) >0){crc=crc>>1;crc=crc^ 0xa001;}else{ crc=crc>>1;}}}return crc;
}
3、编译
~$ gcc main.c -o getcrc16
4、使用
~$ ./getcrc16 01 03 00 04 00 04
~$ [note] input 6 byte: 01 03 00 04 00 04
~$ [note] crc16 value: 0xC805