CRC 计算实例
例如在银联境外二维码测试平台上生成一个EMVco的商家二维码
000201010211153125000344000203441000000000000065204597253033445802HK5913Test Merchant6002HK626001200000000000000000000005200000000000000000000007080000001063045855
000201010211153125000344000203441000000000000065204597253033445802HK5913Test Merchant6002HK62600120000000000000000000000520000000000000000000000708000000106304 5855
可以分析出tag 63的值为6304 5855,参与CRC计算的字符串为:
000201010211153125000344000203441000000000000065204597253033445802HK5913Test Merchant6002HK62600120000000000000000000000520000000000000000000000708000000106304
1.规范中对tag63中的CRC相关描述
《EMVCo-Merchant-Presented-QR-Specification-v1-1》,下载地址https://www.emvco.com/emv-technologies/qrcodes/
2.CRC相关知识
CRC16常见几个标准的算法及C语言实现_C/C++_leumber的专栏-CSDN博客
https://blog.csdn.net/leumber/article/details/54311811
3. CRC计算
TLV
也就是 tag-length-value 这样的数据结构格式,tag表示显示的值是的意义,length表示value的字节长度,value则是tag具体对应的值
比如:6304c8b2
tag: 63
length: 04
value: c8b2
一个最小单位的tlv数据就是这样组成的,除了直接 tlv
,还可以使用 tltlv
这样的嵌套结构
Emvco Merchant
tlv
字符串拼接成一个代表 Emvco 的字符串,包含了根据公司需要的一些特定的数据内容,Emv官方文档 对某些特定的 tag
做了规定。
CRC16
Emvco Merchant 使用 CRC16 进行验证字符串,在最后面添加 6304 然后计算出对应的 CRC16值添加到最后面就组成了一个完整的 Emvco Merchant 支付字符串
计算方式采用 CCITT_FALSE 方式进行计算,在线计算CRC16
步骤:
- 字符串转 16 进制字符串
- 字符串转字节码数组
- 通过位移计算 CRC16 字符串
通过字节数组计算 CRC16 (kotlin)
private fun computeCRC16_CCITT_FALSE(bytes: ByteArray): Int {
// initial value
var crc = 0xffff
// polynial value
val polynomial = 0x1021
for (index in bytes.indices) {
val b = bytes[index]
for (i in 0..7) {
val bit = b.toInt() shr 7 - i and 1 == 1
val c15 = crc shr 15 and 1 == 1
crc = crc shl 1
if (c15 xor bit)
crc = crc xor polynomial
}
}
crc = crc and 0xffff
return crc
}