APIs and example programs related to Dial RFID (NFC).
#include "M5Dial.h" void setup() { auto cfg = M5.config(); M5Dial.begin(cfg, false, true); // encoder, RFID Serial.begin(115200); } void loop() { // PICC: Proximity Integrated Circuit Card if (M5Dial.Rfid.PICC_IsNewCardPresent() && M5Dial.Rfid.PICC_ReadCardSerial()) { M5Dial.Display.clear(); uint8_t piccType = M5Dial.Rfid.PICC_GetType(M5Dial.Rfid.uid.sak); Serial.print(F("PICC type: ")); Serial.println(M5Dial.Rfid.PICC_GetTypeName(piccType)); // Check if the tag / card is of type MIFARE Classic if (piccType != MFRC522::PICC_TYPE_MIFARE_MINI && piccType != MFRC522::PICC_TYPE_MIFARE_1K && piccType != MFRC522::PICC_TYPE_MIFARE_4K) { Serial.println(F("This tag / card is not of type MIFARE Classic.\n")); delay(500); return; } // Output the stored UID data for (byte i = 0; i < M5Dial.Rfid.uid.size; i++) { Serial.printf("%02X ", M5Dial.Rfid.uid.uidByte[i]); } Serial.println("\n"); delay(500); } }
#include "M5Dial.h" MFRC522::MIFARE_Key key; void setup() { auto cfg = M5.config(); M5Dial.begin(cfg, false, true); // encoder, RFID Serial.begin(115200); // Prepare the key (used both as key A and as key B) with FFFFFFFFFFFFh, // which is the default at chip delivery from the factory. for (byte i = 0; i < 6; i++) { key.keyByte[i] = 0xFF; } } // Helper routine to dump a byte array as hex values to Serial. void dump_byte_array(byte *buffer, byte bufferSize) { for (byte i = 0; i < bufferSize; i++) { Serial.print(buffer[i] < 0x10 ? " 0" : " "); Serial.print(buffer[i], HEX); } } void loop() { M5Dial.update(); // PICC: Proximity Integrated Circuit Card if (M5Dial.Rfid.PICC_IsNewCardPresent() && M5Dial.Rfid.PICC_ReadCardSerial()) { M5Dial.Display.clear(); uint8_t piccType = M5Dial.Rfid.PICC_GetType(M5Dial.Rfid.uid.sak); Serial.print(F("PICC type: ")); Serial.println(M5Dial.Rfid.PICC_GetTypeName(piccType)); // Check if the tag / card is of type MIFARE Classic if (piccType != MFRC522::PICC_TYPE_MIFARE_MINI && piccType != MFRC522::PICC_TYPE_MIFARE_1K && piccType != MFRC522::PICC_TYPE_MIFARE_4K) { Serial.println(F("This tag / card is not of type MIFARE Classic.\n")); delay(500); return; } // Output the stored UID data String uid = ""; for (byte i = 0; i < M5Dial.Rfid.uid.size; i++) { Serial.printf("%02X ", M5Dial.Rfid.uid.uidByte[i]); uid += String(M5Dial.Rfid.uid.uidByte[i], HEX); } Serial.println("\n"); // M5Dial.Rfid.PICC_DumpToSerial(&(M5Dial.Rfid.uid)); // Serial.println("\n"); // In this example, we use the second sector (sector #1) including blocks #4, #5, #6, #7 byte sector = 1; byte blockAddr = 4; byte trailerBlock = 7; byte dataBlock[] = { 0x01, 0x02, 0x03, 0x04, // 1, 2, 3, 4, 0x05, 0x06, 0x07, 0x08, // 5, 6, 7, 8, 0x09, 0x0a, 0x0b, 0x0c, // 9, 10, 11, 12, 0x0d, 0x0e, 0x0f, 0xff // 13, 14, 15, 255 }; MFRC522::StatusCode status; byte buffer[18]; byte size = sizeof(buffer); // Authenticate using key A Serial.println(F("Authenticating using key A...")); status = (MFRC522::StatusCode)M5Dial.Rfid.PCD_Authenticate(MFRC522::PICC_CMD_MF_AUTH_KEY_A, trailerBlock, &key, &(M5Dial.Rfid.uid)); if (status != MFRC522::STATUS_OK) { Serial.print(F("PCD_Authenticate() failed: ")); Serial.println(M5Dial.Rfid.GetStatusCodeName(status)); return; } // Show the whole sector as it currently is Serial.println(F("Current data in sector: ")); M5Dial.Rfid.PICC_DumpMifareClassicSectorToSerial(&(M5Dial.Rfid.uid), &key, sector); Serial.println(""); // Read data from the block Serial.print(F("Reading data from block #")); Serial.print(blockAddr); Serial.println(F(" ...")); status = (MFRC522::StatusCode)M5Dial.Rfid.MIFARE_Read(blockAddr, buffer, &size); if (status != MFRC522::STATUS_OK) { Serial.print(F("MIFARE_Read() failed: ")); Serial.println(M5Dial.Rfid.GetStatusCodeName(status)); } Serial.print(F("Data in block #")); Serial.print(blockAddr); Serial.println(F(": ")); dump_byte_array(buffer, 16); Serial.println("\n"); // Authenticate using key B Serial.println(F("Authenticating again using key B...")); status = (MFRC522::StatusCode)M5Dial.Rfid.PCD_Authenticate(MFRC522::PICC_CMD_MF_AUTH_KEY_B, trailerBlock, &key, &(M5Dial.Rfid.uid)); if (status != MFRC522::STATUS_OK) { Serial.print(F("PCD_Authenticate() failed: ")); Serial.println(M5Dial.Rfid.GetStatusCodeName(status)); return; } // Write data to the block Serial.print(F("Writing data into block #")); Serial.print(blockAddr); Serial.println(F(" ...")); dump_byte_array(dataBlock, 16); Serial.println(); status = (MFRC522::StatusCode)M5Dial.Rfid.MIFARE_Write(blockAddr, dataBlock, 16); if (status != MFRC522::STATUS_OK) { Serial.print(F("MIFARE_Write() failed: ")); Serial.println(M5Dial.Rfid.GetStatusCodeName(status)); } Serial.println(""); // Read data from the block again, now it should be what we have written Serial.print(F("Reading data from block #")); Serial.print(blockAddr); Serial.println(F(" ...")); status = (MFRC522::StatusCode)M5Dial.Rfid.MIFARE_Read(blockAddr, buffer, &size); if (status != MFRC522::STATUS_OK) { Serial.print(F("MIFARE_Read() failed: ")); Serial.println(M5Dial.Rfid.GetStatusCodeName(status)); } Serial.print(F("Data in block #")); Serial.print(blockAddr); Serial.println(F(": ")); dump_byte_array(buffer, 16); Serial.println("\n"); // Check if the data in block is what we have written, by counting the number of bytes that are equal Serial.println(F("Checking result...")); byte count = 0; for (byte i = 0; i < 16; i++) { // Compare buffer (what we've read) with dataBlock (what we've written) if (buffer[i] == dataBlock[i]) { count++; } } Serial.print(F("Number of bytes that match = ")); Serial.println(count); if (count == 16) { Serial.println(F("Success :-)")); } else { Serial.println(F("Failure, no match :-(")); Serial.println(F(" perhaps the write didn't work properly...")); } Serial.println(); // Dump the sector data Serial.println(F("Current data in sector: ")); M5Dial.Rfid.PICC_DumpMifareClassicSectorToSerial(&(M5Dial.Rfid.uid), &key, sector); Serial.println(""); // Halt PICC M5Dial.Rfid.PICC_HaltA(); // Stop encryption on PCD M5Dial.Rfid.PCD_StopCrypto1(); Serial.println("===================="); } }This program reads the data from Block 4 in Sector 1 after completing Key A authentication, and then modifies the data of Block 4 in Sector 1 after completing Key B authentication. During this process, the tag / card must remain close to the Dial. The program output is as follows:
Below are usage notes for the common APIs. A typical flow for reading / writing a MIFARE card is:
Operation return status codes
enum StatusCode { STATUS_OK = 1, // Success STATUS_ERROR = 2, // Error in communication STATUS_COLLISION = 3, // Collision detected STATUS_TIMEOUT = 4, // Timeout in communication STATUS_NO_ROOM = 5, // The buffer is not big enough STATUS_INTERNAL_ERROR = 6, // Internal error in the code. Should not happen ;-) STATUS_INVALID = 7, // Invalid argument STATUS_CRC_WRONG = 8, // The CRC_A does not match STATUS_MIFARE_NACK = 9 // The MIFARE PICC responded with NAK };Function Prototype:
void begin(); Description:
When calling M5Dial.begin(), you can set the parameter enableRFID to true to initialize it together.
M5Dial.begin(m5::M5Unified::config_t cfg, bool enableEncoder, bool enableRFID); Input Parameters:
Return Value:
Function Prototype:
bool PICC_IsNewCardPresent(); Description:
IDLE state. Cards in the HALT state will be ignored.Input Parameters:
Return Value:
Function Prototype:
bool PICC_ReadCardSerial(); Description:
Uid uid;. Before performing the read operation, you must call PICC_IsNewCardPresent(), PICC_RequestA(), or PICC_WakeupA() to ensure a card is detected.for (byte i = 0; i < M5Dial.Rfid.uid.size; i++) { Serial.printf("%02X ", M5Dial.Rfid.uid.uidByte[i]); }Input Parameters:
Return Value:
Function Prototype:
uint8_t PICC_RequestA(uint8_t *bufferATQA, uint8_t *bufferSize); Description:
Input Parameters:
Return Value:
Function Prototype:
uint8_t PICC_WakeupA(uint8_t *bufferATQA, uint8_t *bufferSize); Description:
Input Parameters:
Return Value:
Function Prototype:
uint8_t PICC_Select(Uid *uid, uint8_t validBits = 0); Description:
Input Parameters:
Return Value:
Function Prototype:
uint8_t PICC_HaltA(); Description:
Input Parameters:
Return Value:
Function Prototype:
uint8_t PCD_Authenticate(uint8_t command, uint8_t blockAddr, MIFARE_Key *key, Uid *uid); Description:
PCD_StopCrypto1(); otherwise, new communication cannot be started.Input Parameters:
FFFFFFFFFFFFReturn Value:
Function Prototype:
void PCD_StopCrypto1(); Description:
PCD_StopCrypto1(); otherwise, new communication cannot be started.Input Parameters:
Return Value:
Function Prototype:
uint8_t MIFARE_Read(uint8_t blockAddr, uint8_t *buffer, uint8_t *bufferSize); Description:
Input Parameters:
Return Value:
Function Prototype:
uint8_t MIFARE_Write(uint8_t blockAddr, uint8_t *buffer, uint8_t bufferSize); Description:
Input Parameters:
Return Value:
MFRC522 library as its driver. For more related APIs, see MFRC522 Source Code.