访问超过 2kbytes 的 NFC 标签内存
Access NFC tag memory over 2kbytes
我需要开发一个 Android 应用程序,它能够读取包含大约 8 kByte 二进制数据的 NfcV 标签。单个标记块为 8 个字节。
我写了下面的代码:
for (int i = 0; i < 256; ++i)
{
byte[] cmd = new byte[] {
0x02,
0x20,
(byte)i // Block number
};
byte[] block = nfcV.transceive(cmd);
for(int j = 0; j < 8; ++j) this.sensorData[i * 8 + j] = block[j + 1];
}
但这让我只能读取标签的前 2 kByte(256 个 8 字节块)。
如何读取全部 8 kByte?
不幸的是,我对标签的了解很少。我知道它是德州仪器制造的,Taginfo 应用程序说它与
ISO/IEC 15693-3 和 ISO/IEC 15693-2。
您正在使用 READ SINGLE BLOCK 命令(命令代码 0x20
)读取 ISO/IEC 15693 标签。该标准只为从 0 到 255 的块地址定义了 READ SINGLE BLOCK 命令。因此,由于您的标签似乎具有 8 字节的块大小,因此该命令将您限制为标签内存的前 2 KB。
读取超出此地址 space 未由 ISO/IEC 15693 标准定义。因此,这取决于您使用的标签类型,因此您应该查阅标签的用户手册。一些标签制造商通过定义协议扩展来克服此地址-space 限制(请参阅命令的请求标志字节中的 Protocol_Extension_flag)
byte[] cmd = new byte[]{
(byte)0x08, //Protocol_Extension_flag=1
(byte)0x20, //READ SINGLE BLOCK
(byte)(address & 0x00FF), (byte)((address>>8) & 0x00FF)
};
但是,这只有在您的代码支持此特定协议扩展时才有效。根据您的标签类型,标签也可能使用其他方法来寻址剩余内存。
另请注意,一些 Android 设备将无法正常使用未寻址的 ISO/IEC 15693 命令。因此,通常最好坚持使用指定的命令版本。上述命令的寻址版本(Addressed_flag 集和命令中包含的标签的 UID)如下所示:
byte tagId = nfcV.getTag().getId();
byte[] cmd = new byte[]{
(byte)0x28, //Addressed_flag=1, Protocol_Extension_flag=1
(byte)0x20, //READ SINGLE BLOCK
(byte)0x00, (byte)0x00, (byte)0x00, (byte)0x00, (byte)0x00, (byte)0x00, (byte)0x00, (byte)0x00, // placeholder for UID
(byte)(address & 0x00FF), (byte)((address>>8) & 0x00FF)
};
System.arraycopy(tagId, 0, cmd, 2, 8);
在上述两种情况下,您可以尝试 Data_rate_flag(标志字节的第二低位)和 Sub-carrier_flag(标志字节的最低位)的变体,尽管我我不确定 Android 设备将如何处理这个问题。
我需要开发一个 Android 应用程序,它能够读取包含大约 8 kByte 二进制数据的 NfcV 标签。单个标记块为 8 个字节。
我写了下面的代码:
for (int i = 0; i < 256; ++i)
{
byte[] cmd = new byte[] {
0x02,
0x20,
(byte)i // Block number
};
byte[] block = nfcV.transceive(cmd);
for(int j = 0; j < 8; ++j) this.sensorData[i * 8 + j] = block[j + 1];
}
但这让我只能读取标签的前 2 kByte(256 个 8 字节块)。
如何读取全部 8 kByte?
不幸的是,我对标签的了解很少。我知道它是德州仪器制造的,Taginfo 应用程序说它与 ISO/IEC 15693-3 和 ISO/IEC 15693-2。
您正在使用 READ SINGLE BLOCK 命令(命令代码 0x20
)读取 ISO/IEC 15693 标签。该标准只为从 0 到 255 的块地址定义了 READ SINGLE BLOCK 命令。因此,由于您的标签似乎具有 8 字节的块大小,因此该命令将您限制为标签内存的前 2 KB。
读取超出此地址 space 未由 ISO/IEC 15693 标准定义。因此,这取决于您使用的标签类型,因此您应该查阅标签的用户手册。一些标签制造商通过定义协议扩展来克服此地址-space 限制(请参阅命令的请求标志字节中的 Protocol_Extension_flag)
byte[] cmd = new byte[]{
(byte)0x08, //Protocol_Extension_flag=1
(byte)0x20, //READ SINGLE BLOCK
(byte)(address & 0x00FF), (byte)((address>>8) & 0x00FF)
};
但是,这只有在您的代码支持此特定协议扩展时才有效。根据您的标签类型,标签也可能使用其他方法来寻址剩余内存。
另请注意,一些 Android 设备将无法正常使用未寻址的 ISO/IEC 15693 命令。因此,通常最好坚持使用指定的命令版本。上述命令的寻址版本(Addressed_flag 集和命令中包含的标签的 UID)如下所示:
byte tagId = nfcV.getTag().getId();
byte[] cmd = new byte[]{
(byte)0x28, //Addressed_flag=1, Protocol_Extension_flag=1
(byte)0x20, //READ SINGLE BLOCK
(byte)0x00, (byte)0x00, (byte)0x00, (byte)0x00, (byte)0x00, (byte)0x00, (byte)0x00, (byte)0x00, // placeholder for UID
(byte)(address & 0x00FF), (byte)((address>>8) & 0x00FF)
};
System.arraycopy(tagId, 0, cmd, 2, 8);
在上述两种情况下,您可以尝试 Data_rate_flag(标志字节的第二低位)和 Sub-carrier_flag(标志字节的最低位)的变体,尽管我我不确定 Android 设备将如何处理这个问题。