289 lines
8 KiB
C
289 lines
8 KiB
C
/*-
|
|
* 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 <nfc/nfc-types.h>
|
|
|
|
// 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
|
|
typedef struct {
|
|
mifareul_block amb[11];
|
|
} mifareul_ev1_mf0ul21_tag;
|
|
|
|
// NOT really UL but so similar we can re-use this code
|
|
// 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_
|