如何以编程方式检测NFC芯片的内存大小/页码

时间:2015-02-20 11:01:14

标签: tags nfc rfid mifare fingerprinting

除了探测特定页码并检查它们是否存在之外,以编程方式检测NFC芯片(MIFARE Ultralight或NTAGxxx)的内存大小或页数的最简单方法是什么?

ATR中是否有特定字节返回或存储在芯片内存的受保护区域中的某个字节,告知它是特定的NTAGxxx芯片,还是页数还是字节数?

3 个答案:

答案 0 :(得分:3)

通常,您只能通过广泛的指纹识别和使用读取命令进行探测来确定这一点。

  1. 指纹识别:这意味着您需要拥有一个特定于制造商特征的数据库(例如,从数据表中收集,而且 - 更不经常的是 - 来自经验测试,因为此类信息不是(公开?或者根本没有?)记录了大多数标签),你可以检查你的标签。

  2. 探测:

    • 从第0页开始
    • 阅读页面
    • 成功时:将读取地址增加4页并重新开始
    • 失败时:将读取地址递减1并尝试再次读取,直到找到可读地址
    • 在部门结束时:从下一个部门开始

    这样你就可以找到最后一个可读的页面地址,它不一定是标签内存的“真正”结尾(例如,最后可能有受保护的页面)。

  3. 如果您的Ultralight和NTAG标签是EV1标签(即它们实现了Ultralight EV1命令集),则公开GET_VERSION命令,您可以使用该命令检索存储大小信息。版本信息结合所有现有标签指纹的数据库(您可以从Ultraight / NTAG数据表中获取该信息)将允许您可靠地 1 确定标签类型,从而确定内存大小

    1 )除了作为“NTAG”出售的一些假冒芯片模仿真实NTAG芯片的参数但似乎有内存大小,缺乏命令支持等不符合他们模仿参数的芯片。

    对于遵循NFC论坛类型2标签操作规范的NFC标签,您还可以依赖编码到功能容器中的标签内存大小。但是,此内存大小不一定是物理内存大小。

答案 1 :(得分:0)

我认为,从已接受的答案中,以下代码是有意义的,

boxed

getMaxTransceiveLength()用于检索可以通过transceive(byte [])发送的最大字节数。 收发命令的前3个字节分别用于标志,命令和页码。

我们需要检查以下代码是否正常工作?

int numberOfPages = 0;
int tagType = ulTag.getType();
if(tagType == MifareUltralight.TYPE_ULTRALIGHT) {
    numberOfPages = 16;
} else if(tagType == MifareUltralight.TYPE_ULTRALIGHT_C) {
    numberOfPages = 47;
}

答案 2 :(得分:0)

对于MIFARE ULTRALIGHT,您可以通过这种方式检查版本。

public static final int OFFSET_128_BYTE =  0x025;
public static final int OFFSET_48_BYTE  =  0x010;

public static int checkType(MifareUltralight mrfid){
    int cfgOffset = -1;
    byte[] response = null;
    byte[] writeResponse = null;
    try {
        response = mrfid.transceive(new byte[]{
                (byte) 0x60 // GET_VERSION
        });
    } catch (TagLostException tle) {
        Utility.log("setPasswordAndEnable GET_VERSION, TagLostException ... ");
        return cfgOffset;
    } catch (IOException ioe) {
        Utility.log("setPasswordAndEnable GET_VERSION, IOException ... ");
        return cfgOffset;
    }
    if ((response != null) && (response.length >= 8)) {
        // success
        if ((response[0] == (byte) 0x00) && (response[1] == (byte) 0x04)) {
            // tag is from NXP
            if (response[2] == (byte) 0x03) {
                // MIFARE Ultralight
                if ((response[4] == (byte) 0x01) && (response[5] == (byte) 0x00)) {
                    // MIFARE Ultralight EV1 (V0)
                    switch (response[6]) {
                        case (byte) 0x0B:
                            // MF0UL11
                            cfgOffset = OFFSET_48_BYTE;
                            break;
                        case (byte) 0x0E:
                            // MF0UL21
                            cfgOffset = OFFSET_128_BYTE;
                            break;

                        default:
                            // unknown
                            break;
                    }
                }
            }
        }
    }
    return cfgOffset;
}