2017-04-12 13:45:33 +02:00
|
|
|
#if defined(HAVE_CONFIG_H)
|
2017-06-29 12:25:53 +02:00
|
|
|
#include "config.h"
|
2017-04-12 13:45:33 +02:00
|
|
|
#endif
|
2015-05-13 02:06:55 +02:00
|
|
|
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <err.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#if defined(HAVE_SYS_ENDIAN_H)
|
2017-06-29 12:25:53 +02:00
|
|
|
#include <sys/endian.h>
|
2015-05-13 02:06:55 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(HAVE_ENDIAN_H)
|
2017-06-29 12:25:53 +02:00
|
|
|
#include <endian.h>
|
2015-05-13 02:06:55 +02:00
|
|
|
#endif
|
|
|
|
|
2017-03-04 02:33:08 +01:00
|
|
|
#if defined(HAVE_COREFOUNDATION_COREFOUNDATION_H)
|
2017-06-29 12:25:53 +02:00
|
|
|
#include <CoreFoundation/CoreFoundation.h>
|
2017-03-04 02:33:08 +01:00
|
|
|
#endif
|
|
|
|
|
2015-05-13 02:06:55 +02:00
|
|
|
#include <nfc/nfc.h>
|
|
|
|
|
|
|
|
#include <freefare.h>
|
2017-03-04 02:33:08 +01:00
|
|
|
#include "../libfreefare/freefare_internal.h"
|
2015-05-13 02:06:55 +02:00
|
|
|
|
|
|
|
#define NDEF_BUFFER_SIZE 512
|
|
|
|
|
|
|
|
void
|
2017-06-27 13:58:31 +02:00
|
|
|
usage(char *progname)
|
2015-05-13 02:06:55 +02:00
|
|
|
{
|
2017-06-27 13:58:31 +02:00
|
|
|
fprintf(stderr, "usage: %s [options]\n", progname);
|
|
|
|
fprintf(stderr, "\nAvailable options:\n");
|
|
|
|
fprintf(stderr, " -o FILE Write NDEF message to FILE\n");
|
2015-05-13 02:06:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2017-06-27 13:58:31 +02:00
|
|
|
main(int argc, char *argv[])
|
2015-05-13 02:06:55 +02:00
|
|
|
{
|
|
|
|
int error = EXIT_SUCCESS;
|
|
|
|
nfc_device *device = NULL;
|
|
|
|
FreefareTag *tags = NULL;
|
|
|
|
|
|
|
|
int ch;
|
|
|
|
char *ndef_file = NULL;
|
2017-06-27 13:58:31 +02:00
|
|
|
while ((ch = getopt(argc, argv, "o:")) != -1) {
|
2015-05-13 02:06:55 +02:00
|
|
|
switch (ch) {
|
|
|
|
case 'o':
|
|
|
|
ndef_file = optarg;
|
2015-05-13 19:52:33 +02:00
|
|
|
break;
|
2015-05-13 02:06:55 +02:00
|
|
|
case '?':
|
2017-06-27 13:58:31 +02:00
|
|
|
usage(argv[0]);
|
|
|
|
exit(EXIT_FAILURE);
|
2015-05-13 02:06:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
nfc_connstring devices[8];
|
|
|
|
|
|
|
|
size_t device_count;
|
|
|
|
|
|
|
|
nfc_context *context;
|
2017-06-27 13:58:31 +02:00
|
|
|
nfc_init(&context);
|
2015-05-13 02:06:55 +02:00
|
|
|
if (context == NULL)
|
2017-06-27 13:58:31 +02:00
|
|
|
errx(EXIT_FAILURE, "Unable to init libnfc (malloc)");
|
2015-05-13 02:06:55 +02:00
|
|
|
|
2017-06-27 13:58:31 +02:00
|
|
|
device_count = nfc_list_devices(context, devices, 8);
|
2015-05-13 02:06:55 +02:00
|
|
|
if (device_count <= 0)
|
2017-06-27 13:58:31 +02:00
|
|
|
errx(EXIT_FAILURE, "No NFC device found.");
|
2015-05-13 02:06:55 +02:00
|
|
|
|
|
|
|
for (size_t d = 0; d < device_count; d++) {
|
2017-06-27 13:58:31 +02:00
|
|
|
device = nfc_open(context, devices[d]);
|
2015-05-13 02:06:55 +02:00
|
|
|
if (!device) {
|
2017-06-27 13:58:31 +02:00
|
|
|
warnx("nfc_open() failed.");
|
2015-05-13 02:06:55 +02:00
|
|
|
error = EXIT_FAILURE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-06-27 13:58:31 +02:00
|
|
|
tags = freefare_get_tags(device);
|
2015-05-13 02:06:55 +02:00
|
|
|
if (!tags) {
|
2017-06-27 13:58:31 +02:00
|
|
|
nfc_close(device);
|
|
|
|
errx(EXIT_FAILURE, "Error listing FeliCa tag.");
|
2015-05-13 02:06:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; (!error) && tags[i]; i++) {
|
|
|
|
int block = 1;
|
|
|
|
uint8_t ndef_message[NDEF_BUFFER_SIZE];
|
|
|
|
int ndef_space_left = NDEF_BUFFER_SIZE;
|
|
|
|
uint8_t *p = ndef_message;
|
|
|
|
ssize_t s;
|
|
|
|
|
|
|
|
// FIXME Instead of reading as much as we can, we should read until end of NDEF record (if any is found).
|
2017-06-27 13:58:31 +02:00
|
|
|
while ((ndef_space_left >= 16) && (s = felica_read(tags[i], FELICA_SC_RO, block++, p, 16)) > 0) {
|
2015-05-13 02:06:55 +02:00
|
|
|
p += s;
|
|
|
|
ndef_space_left -= s;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t ndef_message_length = 0;
|
|
|
|
|
|
|
|
if ((ndef_message[0] & 0x80) == 0x80) {
|
|
|
|
uint8_t *ndef_record;
|
|
|
|
do {
|
|
|
|
ndef_record = ndef_message + ndef_message_length;
|
|
|
|
|
|
|
|
size_t ndef_record_length = 1 + 1 + ndef_record[1];
|
|
|
|
size_t payload_length;
|
|
|
|
size_t payload_length_length;
|
|
|
|
|
|
|
|
if (ndef_record[0] & 0x10) {
|
|
|
|
/* Short record */
|
|
|
|
payload_length = ndef_record[2];
|
|
|
|
payload_length_length = 1;
|
|
|
|
} else {
|
2017-06-27 13:58:31 +02:00
|
|
|
payload_length = be32toh(*(uint32_t *)(ndef_record + 2));
|
2015-05-13 02:06:55 +02:00
|
|
|
payload_length_length = 4;
|
|
|
|
}
|
|
|
|
ndef_record_length += payload_length_length;
|
|
|
|
ndef_record_length += payload_length;
|
|
|
|
|
|
|
|
if (ndef_record[0] & 0x08) {
|
|
|
|
ndef_record_length += 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
ndef_message_length += ndef_record_length;
|
|
|
|
if (ndef_message_length > NDEF_BUFFER_SIZE)
|
2017-06-27 13:58:31 +02:00
|
|
|
errx(EXIT_FAILURE, "NDEF message truncated");
|
2015-05-13 02:06:55 +02:00
|
|
|
|
|
|
|
} while ((ndef_record[0] & 0x40) != 0x40);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ndef_message_length == 0)
|
2017-06-27 13:58:31 +02:00
|
|
|
errx(EXIT_FAILURE, "No NDEF message found");
|
2015-05-13 02:06:55 +02:00
|
|
|
|
|
|
|
FILE *f;
|
|
|
|
if (ndef_file)
|
2017-06-27 13:58:31 +02:00
|
|
|
f = fopen(ndef_file, "w");
|
2015-05-13 02:06:55 +02:00
|
|
|
else
|
|
|
|
f = stdout;
|
|
|
|
|
2017-06-27 13:58:31 +02:00
|
|
|
if (fwrite(ndef_message, ndef_message_length, 1, f) != 1)
|
|
|
|
err(EXIT_FAILURE, "Can't write NDEF message");
|
2015-05-13 02:06:55 +02:00
|
|
|
|
2017-06-27 13:58:31 +02:00
|
|
|
fclose(f);
|
2015-05-13 02:06:55 +02:00
|
|
|
}
|
|
|
|
|
2017-06-27 13:58:31 +02:00
|
|
|
freefare_free_tags(tags);
|
|
|
|
nfc_close(device);
|
2015-05-13 02:06:55 +02:00
|
|
|
}
|
|
|
|
exit(EXIT_SUCCESS);
|
|
|
|
}
|