2010-10-04 14:46:03 +02:00
|
|
|
/*-
|
|
|
|
* Public platform independent Near Field Communication (NFC) library
|
|
|
|
*
|
|
|
|
* Copyright (C) 2010, Romuald Conty
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU Lesser General Public License as published by the
|
|
|
|
* Free Software Foundation, either version 3 of the License, or (at your
|
|
|
|
* option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
* more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2010-10-06 22:31:36 +02:00
|
|
|
* @file nfc-emulate-tag.c
|
2010-10-14 13:53:27 +02:00
|
|
|
* @brief Emulates a simple tag
|
2010-10-04 14:46:03 +02:00
|
|
|
*/
|
|
|
|
|
2010-10-08 21:24:54 +02:00
|
|
|
// Note that depending on the device (initiator) you'll use against, this
|
2010-10-08 16:45:48 +02:00
|
|
|
// emulator it might work or not. Some readers are very strict on responses
|
2010-10-06 22:31:36 +02:00
|
|
|
// timings, e.g. a Nokia NFC and will drop communication too soon for us.
|
|
|
|
|
2010-10-04 14:46:03 +02:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include "config.h"
|
|
|
|
#endif // HAVE_CONFIG_H
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <string.h>
|
2010-10-12 17:51:57 +02:00
|
|
|
#include <signal.h>
|
2010-10-04 14:46:03 +02:00
|
|
|
|
|
|
|
#include <nfc/nfc.h>
|
|
|
|
|
|
|
|
#include <nfc/nfc-messages.h>
|
|
|
|
#include "nfc-utils.h"
|
|
|
|
|
|
|
|
#define MAX_FRAME_LEN 264
|
|
|
|
|
|
|
|
static byte_t abtRx[MAX_FRAME_LEN];
|
2010-10-12 16:56:42 +02:00
|
|
|
static size_t szRx;
|
2010-10-04 14:46:03 +02:00
|
|
|
static nfc_device_t *pnd;
|
|
|
|
static bool quiet_output = false;
|
2010-10-06 22:31:36 +02:00
|
|
|
static bool init_mfc_auth = false;
|
2010-10-04 14:46:03 +02:00
|
|
|
|
2010-10-12 17:51:57 +02:00
|
|
|
void
|
|
|
|
intr_hdlr (void)
|
|
|
|
{
|
|
|
|
printf ("\nQuitting...\n");
|
|
|
|
if (pnd != NULL) {
|
|
|
|
nfc_disconnect(pnd);
|
|
|
|
}
|
|
|
|
exit (EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
2010-10-04 14:46:03 +02:00
|
|
|
bool
|
|
|
|
target_io( const nfc_target_t nt, const byte_t * pbtInput, const size_t szInput, byte_t * pbtOutput, size_t *pszOutput )
|
|
|
|
{
|
|
|
|
bool loop = true;
|
|
|
|
*pszOutput = 0;
|
|
|
|
|
|
|
|
// Show transmitted command
|
|
|
|
if (!quiet_output) {
|
|
|
|
printf (" In: ");
|
|
|
|
print_hex (pbtInput, szInput);
|
|
|
|
}
|
|
|
|
if(szInput) {
|
|
|
|
switch(pbtInput[0]) {
|
2010-10-06 22:31:36 +02:00
|
|
|
case 0x60: // Mifare authA
|
|
|
|
case 0x61: // Mifare authB
|
|
|
|
// Let's give back a very random nonce...
|
|
|
|
*pszOutput = 2;
|
|
|
|
pbtOutput[0] = 0x12;
|
|
|
|
pbtOutput[1] = 0x34;
|
|
|
|
// Next commands will be without CRC
|
|
|
|
init_mfc_auth = true;
|
|
|
|
break;
|
2010-10-04 14:46:03 +02:00
|
|
|
case 0xe0: // RATS
|
|
|
|
// Send ATS
|
|
|
|
*pszOutput = nt.nti.nai.szAtsLen + 1;
|
|
|
|
pbtOutput[0] = nt.nti.nai.szAtsLen + 1; // ISO14443-4 says that ATS contains ATS_Lenght as first byte
|
|
|
|
if(nt.nti.nai.szAtsLen) {
|
|
|
|
memcpy(pbtOutput+1, nt.nti.nai.abtAts, nt.nti.nai.szAtsLen);
|
|
|
|
}
|
|
|
|
break;
|
2010-10-04 21:17:40 +02:00
|
|
|
case 0xc2: // S-block DESELECT
|
|
|
|
if (!quiet_output) {
|
|
|
|
printf("Target released me. Bye!\n");
|
|
|
|
}
|
|
|
|
loop = false;
|
|
|
|
break;
|
2010-10-04 14:46:03 +02:00
|
|
|
default: // Unknown
|
|
|
|
if (!quiet_output) {
|
|
|
|
printf("Unknown frame, emulated target abort.\n");
|
|
|
|
}
|
|
|
|
loop = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Show transmitted command
|
2010-10-04 21:17:40 +02:00
|
|
|
if ((!quiet_output) && *pszOutput) {
|
2010-10-04 14:46:03 +02:00
|
|
|
printf (" Out: ");
|
|
|
|
print_hex (pbtOutput, *pszOutput);
|
|
|
|
}
|
|
|
|
return loop;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
nfc_target_emulate_tag(nfc_device_t* pnd, const nfc_target_t nt)
|
|
|
|
{
|
|
|
|
size_t szTx;
|
|
|
|
byte_t abtTx[MAX_FRAME_LEN];
|
|
|
|
bool loop = true;
|
|
|
|
|
2010-10-13 13:35:28 +02:00
|
|
|
if (!nfc_target_init (pnd, NTM_PASSIVE_ONLY, nt, abtRx, &szRx)) {
|
2010-10-04 21:17:40 +02:00
|
|
|
nfc_perror (pnd, "nfc_target_init");
|
2010-10-04 14:46:03 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
while ( loop ) {
|
2010-10-12 16:56:42 +02:00
|
|
|
loop = target_io( nt, abtRx, szRx, abtTx, &szTx );
|
2010-10-04 14:46:03 +02:00
|
|
|
if (szTx) {
|
2010-10-04 21:17:40 +02:00
|
|
|
if (!nfc_target_send_bytes(pnd, abtTx, szTx)) {
|
|
|
|
nfc_perror (pnd, "nfc_target_send_bytes");
|
|
|
|
return false;
|
2010-10-04 14:46:03 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( loop ) {
|
2010-10-06 22:31:36 +02:00
|
|
|
if ( init_mfc_auth ) {
|
|
|
|
nfc_configure (pnd, NDO_HANDLE_CRC, false);
|
|
|
|
init_mfc_auth = false;
|
|
|
|
}
|
2010-10-12 16:56:42 +02:00
|
|
|
if (!nfc_target_receive_bytes(pnd, abtRx, &szRx)) {
|
2010-10-04 21:17:40 +02:00
|
|
|
nfc_perror (pnd, "nfc_target_receive_bytes");
|
|
|
|
return false;
|
2010-10-04 14:46:03 +02:00
|
|
|
}
|
2010-10-06 22:31:36 +02:00
|
|
|
}
|
2010-10-04 14:46:03 +02:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
main (int argc, char *argv[])
|
|
|
|
{
|
|
|
|
const char *acLibnfcVersion;
|
|
|
|
|
2010-10-12 17:51:57 +02:00
|
|
|
#ifdef WIN32
|
|
|
|
signal (SIGINT, (void (__cdecl *) (int)) intr_hdlr);
|
|
|
|
#else
|
|
|
|
signal (SIGINT, (void (*)()) intr_hdlr);
|
|
|
|
#endif
|
|
|
|
|
2010-10-04 14:46:03 +02:00
|
|
|
// Try to open the NFC reader
|
|
|
|
pnd = nfc_connect (NULL);
|
|
|
|
|
|
|
|
// Display libnfc version
|
|
|
|
acLibnfcVersion = nfc_version ();
|
|
|
|
printf ("%s use libnfc %s\n", argv[0], acLibnfcVersion);
|
|
|
|
|
|
|
|
if (pnd == NULL) {
|
|
|
|
ERR("Unable to connect to NFC device");
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf ("Connected to NFC device: %s\n", pnd->acName);
|
2010-10-12 11:44:39 +02:00
|
|
|
|
2010-10-06 22:31:36 +02:00
|
|
|
// Example of a Mifare Classic Mini
|
|
|
|
// Note that crypto1 is not implemented in this example
|
|
|
|
nfc_target_t nt = {
|
2010-10-13 21:17:51 +02:00
|
|
|
.nm.nmt = NMT_ISO14443A,
|
|
|
|
.nm.nbr = NBR_UNDEFINED,
|
2010-10-06 22:31:36 +02:00
|
|
|
.nti.nai.abtAtqa = { 0x00, 0x04 },
|
|
|
|
.nti.nai.abtUid = { 0x08, 0xab, 0xcd, 0xef },
|
|
|
|
.nti.nai.btSak = 0x09,
|
|
|
|
.nti.nai.szUidLen = 4,
|
|
|
|
.nti.nai.szAtsLen = 0,
|
|
|
|
};
|
2010-10-12 11:44:39 +02:00
|
|
|
/*
|
|
|
|
// Example of a FeliCa
|
|
|
|
nfc_target_t nt = {
|
2010-10-13 21:17:51 +02:00
|
|
|
.nm.nmt = NMT_FELICA,
|
|
|
|
.nm.nbr = NBR_UNDEFINED,
|
2010-10-12 11:44:39 +02:00
|
|
|
.nti.nfi.abtId = { 0x01, 0xFE, 0x56, 0x78, 0x9A, 0xBC, 0xDE, 0xFF },
|
|
|
|
.nti.nfi.abtPad = { 0x12, 0x34, 0x56, 0x78, 0x9A, 0xBC, 0xDE, 0xFF },
|
|
|
|
.nti.nfi.abtSysCode = { 0xFF, 0xFF },
|
|
|
|
};
|
|
|
|
*/
|
2010-10-07 14:59:12 +02:00
|
|
|
/*
|
2010-10-06 22:31:36 +02:00
|
|
|
// Example of a ISO14443-4 (DESfire)
|
2010-10-04 14:46:03 +02:00
|
|
|
nfc_target_t nt = {
|
2010-10-13 21:17:51 +02:00
|
|
|
.nm.nmt = NMT_ISO14443A,
|
|
|
|
.nm.nbr = NBR_UNDEFINED,
|
2010-10-04 14:46:03 +02:00
|
|
|
.nti.nai.abtAtqa = { 0x03, 0x44 },
|
|
|
|
.nti.nai.abtUid = { 0x08, 0xab, 0xcd, 0xef },
|
|
|
|
.nti.nai.btSak = 0x20,
|
|
|
|
.nti.nai.szUidLen = 4,
|
|
|
|
.nti.nai.abtAts = { 0x75, 0x77, 0x81, 0x02, 0x80 },
|
|
|
|
.nti.nai.szAtsLen = 5,
|
|
|
|
};
|
2010-10-07 14:59:12 +02:00
|
|
|
*/
|
2010-10-04 14:46:03 +02:00
|
|
|
|
|
|
|
printf ("%s will emulate this ISO14443-A tag:\n", argv[0]);
|
|
|
|
print_nfc_iso14443a_info( nt.nti.nai );
|
|
|
|
|
|
|
|
printf ("NFC device (configured as target) is now emulating the tag, please touch it with a second NFC device (initiator)\n");
|
|
|
|
if (!nfc_target_emulate_tag (pnd, nt)) {
|
|
|
|
nfc_perror (pnd, "nfc_target_emulate_tag");
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
nfc_disconnect(pnd);
|
|
|
|
exit (EXIT_SUCCESS);
|
|
|
|
}
|
|
|
|
|