向Mifare Classic 1k NFC标签写入和读取数据

时间:2018-04-17 13:13:08

标签: android authentication permissions nfc mifare

我正在尝试在Mifare Classic 1k NFC标签上读取和写入数据。

由于this app

,我找到了卡的密钥和访问条件

<uses-permission android:name="android.permission.NFC" />出现在我的清单中。

这是我的代码:

Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);

if(tag != null) {
    Log.i("hey", Arrays.toString(tag.getTechList()));
    MifareClassic mfc = MifareClassic.get(tag) ;

    try {
        mfc.connect();
        boolean authA = mfc.authenticateSectorWithKeyA(2, MifareClassic.KEY_NFC_FORUM) ;
        boolean authB = mfc.authenticateSectorWithKeyB(2, MifareClassic.KEY_DEFAULT) ;
        Log.i("hey", "authA : " + authA) ;
        Log.i("hey", "authB : " + authB) ;

        if (authB && authA) {
            byte[] bWrite = new byte[16];
            byte[] hello = "hello".getBytes(StandardCharsets.US_ASCII);
            System.arraycopy(hello, 0, bWrite, 0, hello.length);
            mfc.writeBlock(0, bWrite);
            Log.i("hey", "write : " + Arrays.toString(bWrite));

            byte[] bRead = mfc.readBlock(0);
            String str = new String(bRead, StandardCharsets.US_ASCII);
            Log.i("hey", "read bytes : " + Arrays.toString(bRead));
            Log.i("hey", "read string : " + str);
            Toast.makeText(this, "read : " + str, Toast.LENGTH_SHORT).show();
            Log.i("hey", "expected : " + new String(bWrite, StandardCharsets.US_ASCII));
        }



        mfc.close();

    } catch (IOException e) {
        Toast.makeText(this, "Error", Toast.LENGTH_SHORT).show();
        e.printStackTrace();
        Log.i("hey", "Error") ;
    }


}

当我尝试这样写作和阅读时,我读到的并不是我写的。

这是logcat:

I/hey: [android.nfc.tech.NfcA, android.nfc.tech.MifareClassic, android.nfc.tech.NdefFormatable]
I/hey: authA : true
    authB : true
I/hey: write : [104, 101, 108, 108, 111, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
I/hey: read byte : [-78, 0, 0, 0, 0, 0, -1, 7, -128, 105, -1, -1, -1, -1, -1, -1]
    read string : �������������i������
I/hey: expected : hello����������������������

我试过不同的字符集,但它没有改变任何东西。

我还尝试只通过注释写入部分并将身份验证密钥A更改为MifareClassic.KEY_MIFARE_APPLICATION_DIRECTORY来读取扇区0,这次我得到了这个Logcat:

I/hey: [android.nfc.tech.NfcA, android.nfc.tech.MifareClassic, android.nfc.tech.NdefFormatable] 
I/hey: authA : true
        authB : true 
I/hey: read bytes : [-123, -121, 0, 16, 18, 8, 4, 0, 1, -64, 62, -70, 74, 124, 78, 29]
        read string : �������>�J|N

有关如何修复它以正确编写和显示文本的任何想法?

1 个答案:

答案 0 :(得分:3)

首先对第2部分进行身份验证:

boolean authA = mfc.authenticateSectorWithKeyA(2, MifareClassic.KEY_NFC_FORUM);
boolean authB = mfc.authenticateSectorWithKeyB(2, MifareClassic.KEY_DEFAULT);

然后,您尝试编写并读取块0:

mfc.writeBlock(0, bWrite);
byte[] bRead = mfc.readBlock(0);

这有几个问题:

  1. 使用密钥A和密钥B进行身份验证没有意义。只有最后一次身份验证才能确定标记的身份验证状态。由于所有扇区似乎都可以使用密钥B写入,因此您可以安全地使用第二行(仅mfc.authenticateSectorWithKeyB())。

  2. 您对扇区2进行身份验证,扇区2由块8,9,10和11组成。写入和读取块0在该身份验证状态下没有意义。请注意,读/写操作通常使用全局块编号。扇区仅用作身份验证/访问控制目的的逻辑单元。您可以使用mfc.sectorToBlock()mfc.getBlockCountInSector()轻松计算行业编号中的块编号。

  3. 块0(扇区0)是一个特殊块,即制造商块。它通常包含在生产过程中刻录到标签中的UID,SAK和制造商信息。该块是只读的,无法更改。因此,写入该块不会产生任何影响。