Android NFC近場通信03----讀寫MifareClassic卡

                                       Android NFC近場通信02----讀寫MifareClassic卡

一.MifareClassic卡 相關

一般來說,給予MifareClassic的射頻卡,一般內存大小有3種:

1K: 16個分區(sector),每個分區4個塊(block),每個塊(block) 16個byte數據

2K: 32個分區,每個分區4個塊(block),每個塊(block) 16個byte數據

4K:64個分區,每個分區4個塊(block),每個塊(block) 16個byte數據

對於所有基於MifareClassic的卡來說,每個區最後一個塊叫Trailer,16個byte, 主要來存放讀寫該區的key,可以有A,B兩個KEY,每個key長6byte,默認的key一般是FF 或 0,最後一個塊的內存結構如下:

Block 0  Data 16bytes
Block 1  Data 16 bytes
Block 2  Data 16 bytes
Block 3  Trailer 16 bytes
Trailer:
Key A: 6 bytes
Access Conditions: 4 bytes
Key B: 6 bytes

M1卡分爲16個扇區,每個扇區由4塊(塊0、塊1、塊2、塊3)組成,(我們也將16個扇區的64個塊按絕對地址編號爲0~63,)存貯結構如右表所示


0

0

數據塊

0

1

數據塊

1

2

數據塊

2

3

密碼A  存取控制    密碼B

數據塊

3

1

0

數據塊

4

1

數據塊

5

2

數據塊

6

3

密碼A  存取控制    密碼B

數據塊

7


 

15

0

數據塊

60

1

數據塊

61

2

數據塊

62

3

密碼A  存取控制    密碼B

數據塊

63


0扇區的0(即絕對地址0塊),它用於存放廠商代碼,已經固化,不可更改。

每個扇區的塊0、塊1、塊2爲數據塊,可用於存貯數據。

每個扇區的塊3爲控制塊,包括了密碼A、存取控制、密碼B。具體結構如下:

A1A2 A3 A4 A5               FF 07 8069               B0 B1 B2 B3 B4 B5

密碼A6字節)          存取控制(4字節)          密碼B6字節)  

每個扇區的密碼和存取控制都是獨立的,可以根據實際需要設定各自的密碼及存取控制;

存取控制爲4個字節,共32位,扇區中的每個塊(包括數據塊和控制塊)的存取條件是由密碼和存取控制共同決定的。
工作原理:

讀寫器向M1卡發一組固定頻率的電磁波,卡片內有一個LC串聯諧振電路,其頻率與訊寫器發射的頻率相同,在電磁波的激勵下,LC諧振電路產生共振,從而使電容內有了電荷,在這個電容的另一端,接有一個單向導通的電子泵,將電容內的電荷送到另一個電容內儲存,當所積累的電荷達到2V時,此電容可做爲電源爲其它電路提供工作電壓,將卡內數據發射出去或接取讀寫器的數據。

二.讀取數據 

//tag 就是在上一篇中onNewIntent中獲取的tag
MifareClassic mc = MifareClassic.get(tag);
        short startAddress = 0;
        short endAddress = 5;

        byte[] data = new byte[(endAddress - startAddress + 1 ) * ByteCountPerBlock];
        
        try {            
            mc.connect();for (short i = startAddress; i <= endAddress; i++ ,time++) {
                boolean auth = false;
                short sectorAddress = getSectorAddress(i);
                auth = mc.authenticateSectorWithKeyA(sectorAddress, MifareClassic.KEY_DEFAULT);
                if (auth){
                    
                    //the last block of the sector is used for KeyA and KeyB cannot be overwritted
                    short readAddress = (short)(sectorAddress == 0 ? i : i + sectorAddress);
                    
                    byte[] response = mc.readBlock(readAddress);
                    CombineByteArray(data, response, time * ByteCountPerBlock);
                }
                else{
                    throw new NfcException(NfcErrorCode.TemporaryError,
                            "Authorization Error.");
                }
            }

            mc.close();
            
        }
        catch (NfcException ne) {
            throw ne;
        }
        catch (IOException e) {
            throw new NfcException(NfcErrorCode.TemporaryError,
                    "Get response, what it is not successfully.", e);
        }
        finally
        {
            try {
                mc.close();
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }
三.寫數據 

//tag 就是在上一篇中onNewIntent中獲取的tag
MifareClassic mc = MifareClassic.get(tag);

            try {
                mc.connect();
                boolean auth = false;
                short sectorAddress = 0
                auth = mc.authenticateSectorWithKeyA(sectorAddress,
                        MifareClassic.KEY_DEFAULT);
                if (auth) {
                    //the last block of the sector is used for KeyA and KeyB cannot be overwritted
                    
                    mc.writeBlock(readAddress, dataTemp);

                    mc.close();
                } 
            }finally
            {
                try {
                    mc.close();
                } catch (IOException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
            }
        }

附:NFC 論壇 http://www.nfcchina.org/forum.php

發佈了82 篇原創文章 · 獲贊 76 · 訪問量 23萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章