Since Adafruit lacks documentation about how the PN532 NFC/RFID Controller Shield for Arduino works, I decided I would documentate the installation and code of the Shield. There are a lot of functions the shield has, but I will explain how the code works that can read a NFC tag. The Shield is compatible with most of the Arduino's, but I am using an Arduino Leonardo.
Before you can use the shield, you need to make sure to solder the shield. If you don't know how to solder, I recommend you to find someone to do it for you, since it's quite a precise job. Make sure every pin is only connected to the corresponding hole in the shield, and doesn't touch any other pin.
If you are using the Arduino Leonardo (like I am), then you need to connect to make some adjustments to the shield. As you can see, the (2) pin is connected to the IRQ connector on the shield. This causes a problem, since the (2) is used for another task. You need to cut the connector between (2) and IRQ, and solder a wire between another pin and IRQ. The closest pin that isn't used is (6). If you did this, you need to make some adjustments to the code as well. I'll come to that in a second.
You can download the Adafruit library from here: https://github.com/adafruit/Adafruit-PN532. You need to copy-paste the examples file into your Arduino library folder to acces it from the Arduino text editor. From there, you can acces it by going to File -> Examples -> Adafruit PN523 -> ntag2xx_read.
When you first open the file, you will see something like this:
#define PN532_MOSI (3)
#define PN532_SS (4)
#define PN532_MISO (5)
#define PN532_IRQ (2)
#define PN532_RESET (3)
As you can see, the library itself defines two functions to the same pin. If you cut the connector between (2) and IRQ like I said at the note, you need to fill in the pin you soldered the IRQ with. When you did that, it should look something like this:
When you use a software or a hardware SPI connection (don't know what it is? Me neither. Luckely you will soon find out which one you're using) you need to change something in the code. You will find this line of code:
Adafruit_PN532 nfc(PN532_SCK, PN532_MISO, PN532_MOSI, PN532_SS);
If you upload the code to your Arduino, and it works, you do not need to change anything. If it doesn't, then you need to commend the previous line out, and uncomment the following line of code:
Adafruit_PN532 nfc(PN532_SS);
There is a third option. If you use an I2C connection (don't know what this is either) then you need to commend out both lines, and uncomment this one:
Adafruit_PN532 nfc(PN532_IRQ, PN532_RESET);
Again, if you don't know which one to use, just try and see if it works.
In the void setup, you will see the following:
void setup(void) {
#ifndef ESP8266
while (!Serial); // for Leonardo/Micro/Zero
#endif
Serial.begin(115200);
Serial.println("Hello!");
nfc.begin();
uint32_t versiondata = nfc.getFirmwareVersion();
if (! versiondata) {
Serial.print("Didn't find PN53x board");
while (1); // halt
}
// Got ok data, print it out!
Serial.print("Found chip PN5"); Serial.println((versiondata>>24) & 0xFF, HEX);
Serial.print("Firmware ver. "); Serial.print((versiondata>>16) & 0xFF, DEC);
Serial.print('.'); Serial.println((versiondata>>8) & 0xFF, DEC);
// configure board to read RFID tags
nfc.SAMConfig();
Serial.println("Waiting for an ISO14443A Card ...");
}
Here, the connection to the Arduino and the shield is established. When it made connection to the shield, it do the following:
Serial.println("Hello!");
Then begins the shield part. First, it checks the firmware version of the Shield. If it doesnt get a firmware version. It will print out that he couldn't find a PN532 board.
if (! versiondata) {
Serial.print("Didn't find PN53x board");
When it did find a board. It will print that it found a board, and will print the version data, and that it waits for a specific kind of card.
Serial.print("Found chip PN5"); Serial.println((versiondata>>24) & 0xFF, HEX);
Serial.print("Firmware ver. "); Serial.print((versiondata>>16) & 0xFF, DEC);
Serial.print('.'); Serial.println((versiondata>>8) & 0xFF, DEC);
// configure board to read RFID tags
nfc.SAMConfig();
Serial.println("Waiting for an ISO14443A Card ...");
}
If you upload the card, and do get the "Waiting for an ISO14443A Card...", you succesfully installed the shield. If not, read the above text carefully.
As you can see, there is a whole lot of code left in the file. Most of it is irrelevant if you just want to read out a NFC tag. I will only explain the part that does matter.
uint8_t success;
uint8_t uid[] = { 0, 0, 0, 0, 0, 0, 0 };
uint8_t uidLength;
You can see it makes 3 variables of the type uint8_t. The first is succes, which explains itself. The second is uid[]. When the NFC tag is scanned, the value of the tag is stored in this variable. The third is uidLength. Basically, there are more than one type of NFC tag, and the difference between them is the length of their values. In uidLength is stored the amouth of values inside the tag. The uid[] is the most important thing if you want to do something with the response of the card, since it stores the values of the NFC tag. Each NFC tag is different, so you can execute some code if the value of the NFC tag is a specific code.
if (success) {
// Display some basic information about the card
Serial.println("Found an ISO14443A card");
Serial.print(" UID Length: ");Serial.print(uidLength, DEC);Serial.println(" bytes");
Serial.print(" UID Value: ");
nfc.PrintHex(uid, uidLength);
Serial.println("");
This code basically tells the Arduino to print the variables that it declared above. It prints the UID length and the UID value. The UID value is the main thing you want to look at if you want to do something with the value of the NFC tag.
if (uidLength == 7)
{
uint8_t data[32];
Serial.println("Seems to be an NTAG2xx tag (7 byte UID)");
Like I said before, there are multiple kinds of NFC tags. The Adafruit shield only reads the tags with 7 values. In the code above the code checks if the tag has 7 values, and then executes the next lines of code.
These are basically all the things you need when you want to read a NFC tag. The rest of the code doesn't really matter. It is vital for the code to work, but it's not necessary to know what the code does. If you want to do something with a specific value, you can simply execute some code by making an if statement. An example of this could be:
if (uid == "value") {
"do something";
}
// At "value" and "do something" you should fill in the value and the thing you want the code to do (duh).
I hope i explained everything clear.