如何在PHP HEX中计算CRC16 CCITT?

时间:2015-05-04 16:55:14

标签: php hex crc crc16

我尝试使用PHP CRC16 CCITT函数来计算校验和。

设备向我发送了一个包含Checksum的PACKET:

  

10 00 00 00 00 00 00 00 12 51 09 08 00 18 00 04 02 14 00 0c 00 0c 02   1c 00 02 00 00 00 00 00 00 a0 77

校验和在最后:a0 77

我尝试过使用

How to calculate crc16 in php

Convert C to PHP for CRC16 Function

没有成功,CRC 16计算返回:E6 F4而不是a0 77

当我查找时,我得到了正确的Hex信息:

  

100000000000000012510908001800040214000c000c021c0002000000000000

在网站http://www.lammertbies.nl/comm/info/crc-calculation.html上,但我无法重现它。 (确保选择输入类型为HEX)

你能帮我弄清楚如何获得十六进制值字符串的crc16 CCITT

  

100000000000000012510908001800040214000c000c021c0002000000000000

我正在寻找校验和a0 77

3 个答案:

答案 0 :(得分:3)

我能够通过以下实现产生相同的校验和:

define('CRC16POLYN', 0x1021);

function CRC16Normal($buffer) {
    $result = 0xFFFF;
    if (($length = strlen($buffer)) > 0) {
        for ($offset = 0; $offset < $length; $offset++) {
            $result ^= (ord($buffer[$offset]) << 8);
            for ($bitwise = 0; $bitwise < 8; $bitwise++) {
                if (($result <<= 1) & 0x10000) $result ^= CRC16POLYN;
                $result &= 0xFFFF;
            }
        }
    }
    return $result;
}

echo dechex(CRC16Normal(hex2bin('100000000000000012510908001800040214000c000c021c0002000000000000')));

上面给出a077输出。

http://jsep.from.so

上找到的代码段

答案 1 :(得分:0)

我从Deplhi函数中发现了它是如何工作的。它获得一个字节数组,而不是十六进制字符串来完成这项工作。 这是代码:

// $commands = [0x35, 0x02, 0x02, 0x00, 0x10, 0x03];       // => 0x5ba3
$commands = [0x44, 0x02, 0x02, 0x01, 0x10, 0x03];       // => 0x55c0
var_dump(dechex(getChecksum($commands)));

function getChecksum($byteArray) {
    $polynom = 0x8408;
    $in_crc = 0x0000;
    for ($i = 0; $i < sizeof($byteArray); $i++) {
        for ($n = 0; $n < 8; $n++) {
            if((($byteArray[$i] & 0x0001) ^ $in_crc) & 0x0001) 
                $in_crc = ($in_crc >> 1) ^ $polynom;
            else 
                $in_crc = $in_crc >> 1;
            $byteArray[$i] = $byteArray[$i] >> 1;
        }
        $result = $in_crc;
    }
    return $result;
}

可以在this 在线CRC计算器上证明该解决方案。 使用的算法为 CRC-16 / KERMIT

答案 2 :(得分:0)

所以我一直在寻找如何使用多项式'1021'(十六进制)和初始值'FFFF'(十六进制)根据[ISO / IEC 13239]计算校验和。我发现此link可以计算许多类型的CRC。提交输入后,我想要字段CRC-CCITT (0xFFFF)的值,如下所示:

CRC CCITT 0xFFFF

因此,我采用了JS函数(在.js file中,函数名称为CRCFFFF)来计算CCITT,并创建了它的PHP版本。

function crcChecksum($str) {
    // The PHP version of the JS str.charCodeAt(i)
    function charCodeAt($str, $i) {
        return ord(substr($str, $i, 1));
    }

    $crc = 0xFFFF;
    $strlen = strlen($str);
    for($c = 0; $c < $strlen; $c++) {
        $crc ^= charCodeAt($str, $c) << 8;
        for($i = 0; $i < 8; $i++) {
            if($crc & 0x8000) {
                $crc = ($crc << 1) ^ 0x1021;
            } else {
                $crc = $crc << 1;
            }
        }
    }
    $hex = $crc & 0xFFFF;
    $hex = dechex($hex);
    $hex = strtoupper($hex);

    return $hex;
}

$result = crcChecksum('replace with your string here');// In my case, this gave me the desired output, which is '627B' (without '0x')

如果您需要“ 0x”,只需将其添加:

$result = '0x' . crcChecksum('replace with your string here');// result = '0x627B'