Emvco 的格式分為兩種 Consumer 和 Merchant
兩種格式都使用了 TLV(tag-length-value) 這樣的數(shù)據(jù)結(jié)構(gòu)組成
TLV
也就是 tag-length-value 這樣的數(shù)據(jù)結(jié)構(gòu)格式,tag表示顯示的值是的意義,length表示value的字節(jié)長度,value則是tag具體對應(yīng)的值
比如:6304c8b2
tag: 63
length: 04
value: c8b2
一個最小單位的tlv數(shù)據(jù)就是這樣組成的,除了直接 tlv,還可以使用 tltlv 這樣的嵌套結(jié)構(gòu)
Emvco Merchant
tlv 字符串拼接成一個代表 Emvco 的字符串,包含了根據(jù)公司需要的一些特定的數(shù)據(jù)內(nèi)容,Emv官方文檔 對某些特定的 tag 做了規(guī)定。
CRC16
Emvco Merchant 使用 CRC16 進(jìn)行驗(yàn)證字符串,在最后面添加 6304 然后計(jì)算出對應(yīng)的 CRC16值添加到最后面就組成了一個完整的 Emvco Merchant 支付字符串
計(jì)算方式采用 CCITT_FALSE 方式進(jìn)行計(jì)算,在線計(jì)算CRC16
步驟:
- 字符串轉(zhuǎn) 16 進(jìn)制字符串
- 字符串轉(zhuǎn)字節(jié)碼數(shù)組
- 通過位移計(jì)算 CRC16 字符串
通過字節(jié)數(shù)組計(jì)算 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
}