/*- * Free/Libre Near Field Communication (NFC) library * * Libnfc historical contributors: * Copyright (C) 2009 Roel Verdult * Copyright (C) 2009-2013 Romuald Conty * Copyright (C) 2010-2012 Romain Tartière * Copyright (C) 2010-2013 Philippe Teuwen * Copyright (C) 2012-2013 Ludovic Rousseau * See AUTHORS file for a more comprehensive list of contributors. * Additional contributors of this file: * Copyright (C) 2017-2018 Adam Laurie * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * 1) Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * 2 )Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * * Note that this license only applies on the examples, NFC library itself is under LGPL * */ /** * @file mifare.h * @brief provide samples structs and functions to manipulate MIFARE Classic and Ultralight tags using libnfc */ #ifndef _LIBNFC_MIFARE_H_ # define _LIBNFC_MIFARE_H_ # include // Compiler directive, set struct alignment to 1 uint8_t for compatibility # pragma pack(1) typedef enum { MC_AUTH_A = 0x60, MC_AUTH_B = 0x61, MC_READ = 0x30, MC_WRITE = 0xA0, MC_TRANSFER = 0xB0, MC_DECREMENT = 0xC0, MC_INCREMENT = 0xC1, MC_STORE = 0xC2 } mifare_cmd; // MIFARE command params struct mifare_param_auth { uint8_t abtKey[6]; uint8_t abtAuthUid[4]; }; struct mifare_param_data { uint8_t abtData[16]; }; struct mifare_param_value { uint8_t abtValue[4]; }; struct mifare_param_trailer { uint8_t abtKeyA[6]; uint8_t abtAccessBits[4]; uint8_t abtKeyB[6]; }; typedef union { struct mifare_param_auth mpa; struct mifare_param_data mpd; struct mifare_param_value mpv; struct mifare_param_trailer mpt; } mifare_param; // Reset struct alignment to default # pragma pack() bool nfc_initiator_mifare_cmd(nfc_device *pnd, const mifare_cmd mc, const uint8_t ui8Block, mifare_param *pmp); // Compiler directive, set struct alignment to 1 uint8_t for compatibility # pragma pack(1) // MIFARE Classic typedef struct { uint8_t abtUID[4]; // beware for 7bytes UID it goes over next fields uint8_t btBCC; uint8_t btSAK; // beware it's not always exactly SAK uint8_t abtATQA[2]; uint8_t abtManufacturer[8]; } mifare_classic_block_manufacturer; typedef struct { uint8_t abtData[16]; } mifare_classic_block_data; typedef struct { uint8_t abtKeyA[6]; uint8_t abtAccessBits[4]; uint8_t abtKeyB[6]; } mifare_classic_block_trailer; typedef union { mifare_classic_block_manufacturer mbm; mifare_classic_block_data mbd; mifare_classic_block_trailer mbt; } mifare_classic_block; typedef struct { mifare_classic_block amb[256]; } mifare_classic_tag; // MIFARE Ultralight typedef struct { uint8_t sn0[3]; uint8_t btBCC0; uint8_t sn1[4]; uint8_t btBCC1; uint8_t internal; uint8_t lock[2]; uint8_t otp[4]; } mifareul_block_manufacturer; // MIFARE Ultralight EV1 MF0UL11 Config Pages typedef struct { uint8_t mod; uint8_t rfui1[2]; uint8_t auth0; uint8_t access; uint8_t vctid; uint8_t rfui2[2]; uint8_t pwd[4]; uint8_t pack[2]; uint8_t rfui3[2]; } mifareul_block_config11; // MIFARE Ultralight EV1 MF0UL21 ConfigA Pages typedef struct { uint8_t lock[3]; uint8_t rfui0; uint8_t mod; uint8_t rfui1[2]; uint8_t auth0; uint8_t access; uint8_t vctid; uint8_t rfui2[2]; uint8_t pwd[4]; } mifareul_block_config21A; // MIFARE Ultralight EV1 MF0UL21 ConfigB Pages typedef struct { uint8_t pack[2]; uint8_t rfui3[2]; uint8_t dummy[12]; } mifareul_block_config21B; // MIFARE NTAG21[3/5/6] Manufacturer Pages typedef struct { uint8_t sn0[4]; } mifarentag_block_manuf21356A; typedef struct { uint8_t sn1[4]; } mifarentag_block_manuf21356B; typedef struct { uint8_t sn2; uint8_t internal; uint8_t lock[2]; } mifarentag_block_manuf21356C; typedef struct { uint8_t cc[4]; } mifarentag_block_manuf21356D; // MIFARE NTAG21[3/5/6] Config Pages typedef struct { uint8_t dynlock[3]; uint8_t rfui0; } mifarentag_block_config21356A; typedef struct { uint8_t cfg0[4]; } mifarentag_block_config21356B; typedef struct { uint8_t cfg1[4]; } mifarentag_block_config21356C; typedef struct { uint8_t pwd[4]; } mifarentag_block_config21356D; typedef struct { uint8_t pack[2]; uint8_t rfui1[2]; } mifarentag_block_config21356E; typedef struct { uint8_t abtData[16]; } mifareul_block_data; typedef struct { uint8_t abtData[4]; } mifarentag_block_data; typedef union { mifarentag_block_manuf21356A mbm21356a; mifarentag_block_manuf21356B mbm21356b; mifarentag_block_manuf21356C mbm21356c; mifarentag_block_manuf21356D mbm21356d; mifarentag_block_data mbd; mifarentag_block_config21356A mbc21356a; mifarentag_block_config21356B mbc21356b; mifarentag_block_config21356C mbc21356c; mifarentag_block_config21356D mbc21356d; mifarentag_block_config21356E mbc21356e; } mifarentag_block; typedef union { mifareul_block_manufacturer mbm; mifareul_block_data mbd; mifareul_block_config11 mbc11; mifareul_block_config21A mbc21a; mifareul_block_config21B mbc21b; } mifareul_block; // standard UL tag - 1 manuf block + 3 user blocks typedef struct { mifareul_block amb[4]; } mifareul_tag; // UL EV1 MF0UL11 tag - 1 manuf block + 3 user blocks + 1 config block typedef struct { mifareul_block amb[5]; } mifareul_ev1_mf0ul11_tag; // UL EV1 MF0UL21 tag - 1 manuf block + 8 user blocks + 1/4 lock block + 1 config block // (note that tag is actually 3 bytes smaller due to 1/4 block, so don't rely on this for sizing!) typedef struct { mifareul_block amb[11]; } mifareul_ev1_mf0ul21_tag; // NTAG is a range of NXP tags some of which are essentially Ultralights so we can support them here // if Edwin van Andel doesn't distract us... // https://www.nxp.com/docs/en/data-sheet/NTAG213_215_216.pdf // NTAG213 EEPROM: 180 bytes, organized in 45 pages of 4 byte per page. // 26 bytes reserved for manufacturer and configuration data // 34 bits used for the read-only locking mechanism // 4 bytes available as capability container // 144 bytes user programmable read/write memory typedef struct { mifarentag_block amb[45]; } mifarentag_213_tag; // NTAG215 EEPROM: 540 bytes, organized in 135 pages of 4 byte per page. // 26 bytes reserved for manufacturer and configuration data // 28 bits used for the read-only locking mechanism // 4 bytes available as capability container // 504 bytes user programmable read/write memory typedef struct { mifarentag_block amb[135]; } mifarentag_215_tag; // NTAG216 EEPROM: 924 bytes, organized in 231 pages of 4 byte per page. // 26 bytes reserved for manufacturer and configuration data // 37 bits used for the read-only locking mechanism // 4 bytes available as capability container // 888 bytes user programmable read/write memory typedef struct { mifarentag_block amb[231]; } mifarentag_216_tag; // dummy max size with all structures in it for reading, rounded up to a multiple of 16 bytes typedef union { mifareul_block ul[58]; mifarentag_block nt[232]; } maxtag; // Reset struct alignment to default # pragma pack() #endif // _LIBNFC_MIFARE_H_