1,读取
我们需要外接一个NFC Reader让Windows可以读取NFC卡片的内容。
因为特殊原因,我们选择了Sony rc-s380 NFC Reader。相关介绍
我们需要下载并安装NFC Port Software,以便我们可以顺利使用上述NFC Reader。
安装之后,我们需要去Github获取一个NFC Port Software的.Net封装包:tijins/NfcLib
解压并运行nfc_lib_sample之后我们可以看到这样一个WindowFrom程序。
因为现在要读取的卡类型试MIFARE,所以我们勾选正确的CheckBox,其他设置保持不变,依次点击上述三个橘色方框按钮。
在红色箭头所指的方向就能看到第一个block的十六进制字符串的数据。
为了读取卡片所有block/page的信息,我们将btRead_Click的代码稍稍修改为:
private void btRead_Click(object sender, EventArgs e)
{
byte block = (byte)nudBlock.Value;
try
{
byte[] data = null;
data = new byte[NfcLib.MF_BLOCK_LENGTH];
if (card is Mifare)
{
StringBuilder readSB = new StringBuilder();
for(byte i=;i < ; i++)//我现在的卡包含45个blocks/pages
{
((Mifare)card).Read(i, data, );
readSB.AppendLine(Utility.ByteToHex(data, , data.Length));
}
tbRead.Text = readSB.ToString();
}
}
catch (Exception ex)
{
MessageBox.Show(ex.Message);
}
}
有个小遗憾就是,如果NFC Reader没有连接的时候启动app会抛出异常,而如果app已经启动再拔出然后插入NFC Reader又会无法再读卡除非重启app或者重新初始化。这个需要有点改进。
2,解析
根据上述步骤,我们得到了某张卡的信息如下:
C9 0A C9 5A 1D E1
0A C9 5A 1D E1 A0 0C
1D E1 A0 0C 0F D1
E1 A0 0C 0F D1 0B
A0 0C 0F D1 0B 6E
0F D1 0B 6E
0B 6E FE
6E FE
FE
FE (以下省略...)
然后,我们对它做一点移位:
04 C9 02 47 0A C9 5A 84 1D 48 00 00 E1 10 12 00
0A C9 5A 84 1D 48 00 00 E1 10 12 00 01 03 A0 0C
1D 48 00 00 E1 10 12 00 01 03 A0 0C 34 03 0F D1
E1 10 12 00 01 03 A0 0C 34 03 0F D1 01 0B 54 02
01 03 A0 0C 34 03 0F D1 01 0B 54 02 65 6E 32 30
34 03 0F D1 01 0B 54 02 65 6E 32 30 31 38 31 30
01 0B 54 02 65 6E 32 30 31 38 31 30 32 35 FE 00
65 6E 32 30 31 38 31 30 32 35 FE 00 00 00 00 00
31 38 31 30 32 35 FE 00 00 00 00 00 00 00 00 00
32 35 FE 00 00 00 00 00 00 00 00 00 00 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
可以看出,每一行只有第一列是我们需要的tag信息。
C9
0A C9 5A
1D
E1 00 A0 0C
0F D1
0B
6E FE
前面的四行是卡的UID以及制造商等信息。我们只需要关心下半段的数据。
(Tag: Lock Control TLV)
(Length: bytes)
A0 0C (Value: Information on position and function of lock bytes)
(Tag: NDEF Message TLV)
0F (Length: bytes)
D1 0B 6E
FE (Tag: Terminator TLV; has no length field)
所以,我们得到了NDEF message:D1 01 0B 54 02 65 6E 32 30 31 38 31 30 32 35
NDEF message:
D1 (Header byte of record )
(Type length: byte)
0B (Payload length: bytes)
(Type: "T")
6E (Payload field)
我们再分析最为关键的payload field,如下:
The payload field:
(Status byte: Text is UTF- encoded, Language code has a length of bytes)
6E (Language code: "en")
(Text: "")
可以用在线工具来验证一下 “32 30 31 38 31 30 32 35 ”
最后,上述步骤,用写C#的实现如下(不保证所有Mifare卡适用):
private string GetNfcTag(List<byte[]> cardContentList)
{
List<byte> cardBytes = new List<byte>();
foreach(byte[] rowCardContent in cardContentList)
{
cardBytes.AddRange(rowCardContent.Take());
} byte[] cardUid = cardBytes.Take().ToArray();
string cardUidStr = Utility.ByteToHex(cardUid, , cardUid.Length); byte[] cardMaker = cardBytes.Skip().Take().ToArray();
string cardMakerStr = Utility.ByteToHex(cardMaker, , cardMaker.Length); byte[] memoryBytes = cardBytes.Skip().ToArray(); byte lockControlByte = memoryBytes[];
byte lockByteCount = memoryBytes[];//how many bytes are the lock bytes
byte[] lockBytes = memoryBytes.Skip().Take(lockByteCount).ToArray(); int nedfLengthByteIndex = + lockByteCount + + ;
byte ndefLength = memoryBytes[nedfLengthByteIndex]; if (ndefLength == )//Tag empty
return string.Empty; byte[] ndefBytes = memoryBytes.Skip(nedfLengthByteIndex + ).Take(ndefLength).ToArray(); byte payloadHeader = ndefBytes[];
byte payloadTypeLength = ndefBytes[];
byte payloadLength = ndefBytes[];
byte[] payloadType = ndefBytes.Skip().Take(payloadTypeLength).ToArray(); byte[] payloadBytes = ndefBytes.Skip( + payloadType.Length).Take(payloadLength).ToArray(); byte languageCodeLength = payloadBytes[];
byte[] languageCode = payloadBytes.Skip().Take(languageCodeLength).ToArray();
byte[] tag = payloadBytes.Skip( + languageCode.Length).Take(payloadLength - languageCode.Length - ).ToArray(); string tagHex= Utility.ByteToHex(tag , , tag .Length);
string tagStr = Encoding.UTF8.GetString(tag);
return tagStr;
}
参考链接:
How to read binary blocks of mifare card?