java - 使用低级 NFC 通信方法将 NDEF 数据写入 NTAG216 标签

标签 java android nfc mifare ndef

我已经实现了通过低级通信方法(遵循 NTAG212 Mifare Ultralight with Authentication 和 NTAG216 的数据表)与 NTAG216 交互的代码。

到目前为止我取得的成就:

  • 如果未设置或新标签/空白标签,请在 NTAG216 上设置密码写保护。
  • 如果已设置密码,则使用 PWD_AUTH 并比较 PACK 进行身份验证。
  • 读取数据。
  • 写入/覆盖数据。

到目前为止我没能做到的:

  • 检测我写入其他应用程序标签的 NDEF 消息。换句话说,我可以使用 writePage() 方法写入标签,也可以使用 readPage() 读取标签。但是,在编写页面时,我将 NdefMessage 转换为字节数组,既可以读取也可以写入。但是,在其他应用程序中未检测到此 NDEF 消息。

我需要做什么才能检测到我从其他应用程序写入的 NDEF 消息?

最佳答案

NTAG216 是 NFC Forum Type 2 标签。因此,您必须关注 NFC 论坛 Type 2 Tag Operation specification向此类 NFC 标签写入数据时。

因此,您需要遵循一些规则才能使标签作为 NDEF 标签(类型 2 标签)被发现:

首先,需要配置能力容器(位于 block 3)。

  • 字节 0 必须设置为“魔法”值 0xE1
  • 字节 1 必须设置为 0x10 以指示映射版本 1.0。
  • 字节 2 必须设置为 0x6D 以指示 NTAG216 的内存大小。
  • 字节 3 可以设置为 0x00 以指示对 NDEF 数据的读/写访问或设置为 0x0F 以指示只读访问(请注意,这些是对 NDEF 数据的权限应用层)。

所以你可以把能力容器写成:

byte[] response = nfc.transceive(new byte[] {
    (byte)0xA2, // WRITE
    (byte)3,    // block address
    (byte)0xE1, (byte)0x10, (byte)0x6D, (byte)0x00
});

NTAG216 已附带正确配置的功能容器,因此无需手动执行此操作。另请注意, block 3 是一次性可编程的,这意味着位只能设置为 1,而不能再次清零。因此,如果您确实已经用不同的值覆盖了功能容器,那么该标签很可能不再用作 NDEF 标签。

数据必须写入从 block 4 开始的数据 block 。NDEF 消息必须包装到 NDEF 消息 TLV(标签长度值)结构中。此 TLV 的标记是 0x03。长度可以是单字节格式(对于长度在 0 到 254 字节之间的 NDEF 消息)或三字节格式(对于长度为 255 或更多字节的 NDEF 消息)。此 TLV block 的数据是实际的 NDEF 消息(您可以从 ndefMessage.toByteArray() 获取)。

例如,对于 NDEF 消息 D1 01 0D 55 01 65 78 61 6D 70 6C 65 2E 63 6F 6D 2F(这是一个 URI 记录,URL 为“http://www. example.com/"),你会得到以下 TLV 结构:

03 11 D1010D55016578616D706C652E636F6D2F

If you have alonger NDEF message (e.g. one with 259 bytes), you would use the three-byte length format:

03 FF0103 D101FF5501...

Further, you should mark the end of the data on the tag with a Terminator TLV (tag 0xFE, no length and data fields):

FE

You could then write this data to the tag as:

byte[] ndefMessage = new byte[] {
    (byte)0xD1, (byte)0x01, (byte)0x0D, (byte)0x55, (byte)0x01, (byte)0x65, (byte)0x78, (byte)0x61, (byte)0x6D, (byte)0x70, (byte)0x6C, (byte)0x65, (byte)0x2E, (byte)0x63, (byte)0x6F, (byte)0x6D, (byte)0x2F
};

// wrap into TLV structure
byte[] tlvEncodedData = null;
if (ndefMessage.length < 255) {
    tlvEncodedData = new byte[ndefMessage.length + 3];
    tlvEncodedData[0] = (byte)0x03;  // NDEF TLV tag
    tlvEncodedData[1] = (byte)(ndefMessage.length & 0x0FF);  // NDEF TLV length (1 byte)
    System.arraycopy(ndefMessage, 0, tlvEncodedData, 2, ndefMessage.length);
    tlvEncodedData[2 + ndefMessage.length] = (byte)0xFE;  // Terminator TLV tag
} else {
    tlvEncodedData = new byte[ndefMessage.length + 5];
    tlvEncodedData[0] = (byte)0x03;  // NDEF TLV tag
    tlvEncodedData[1] = (byte)0xFF;  // NDEF TLV length (3 byte, marker)
    tlvEncodedData[2] = (byte)((ndefMessage.length >>> 8) & 0x0FF);  // NDEF TLV length (3 byte, hi)
    tlvEncodedData[3] = (byte)(ndefMessage.length & 0x0FF);          // NDEF TLV length (3 byte, lo)
    System.arraycopy(ndefMessage, 0, tlvEncodedData, 4, ndefMessage.length);
    tlvEncodedData[4 + ndefMessage.length] = (byte)0xFE;  // Terminator TLV tag
}

// fill up with zeros to block boundary:
tlvEncodedData = Arrays.copyOf(tlvEncodedData, (tlvEncodedData.length / 4 + 1) * 4);
for (int i = 0; i < tlvEncodedData.length; i += 4) {
    byte[] command = new byte[] {
        (byte)0xA2, // WRITE
        (byte)((4 + i / 4) & 0x0FF), // block address
        0, 0, 0, 0
    };
    System.arraycopy(tlvEncodedData, i, command, 2, 4);
    byte[] response = nfc.transceive(command);
}

最后,请注意,如果您在 NDEF 数据区域设置了读取密码,则不能将标签用作 NDEF 标签,因为 NFC 论坛 2 类标签操作规范要求标签可自由读取。

关于java - 使用低级 NFC 通信方法将 NDEF 数据写入 NTAG216 标签,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42105626/

相关文章:

android - 从电子邮件启动 Android Intent

android - 可以调用 Vibrator.vibrate() 的最短持续时间是多少,设备仍会振动?

windows-phone-8 - 如何使用 Windows Phone 8 NFC 应用程序将 APDU 命令发送到智能卡

nfc - 为什么有些NFC天线有两个发射器: TX1 and TX2?

java - 当前 Android 时间(纳秒)

java - Android:具有多个元素的Arraylist

java - JPanel 上的 JLabel 定位

java - 使用 maven 依赖项从 ant 运行项目 java 类

Android Studio 0.6.1 和 Facebook SDK 3.15.0 著名错误。怎么解决

android - 如何为基于读卡器主机的卡仿真获取 AID