blob: 8cfdc8f2cb9d0c7ea9a3596b2008bc535889464d [file] [log] [blame] [edit]
/*
* neard - Near Field Communication manager
*
* Copyright (C) 2013 Intel Corporation. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*
* 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 General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <glib.h>
#include <glib/gprintf.h>
#include "src/near.h"
#include "include/ndef.h"
enum record_type {
RECORD_TYPE_WKT_SMART_POSTER = 0x01,
RECORD_TYPE_WKT_URI = 0x02,
RECORD_TYPE_WKT_TEXT = 0x03,
RECORD_TYPE_WKT_SIZE = 0x04,
RECORD_TYPE_WKT_TYPE = 0x05,
RECORD_TYPE_WKT_ACTION = 0x06,
RECORD_TYPE_WKT_HANDOVER_REQUEST = 0x07,
RECORD_TYPE_WKT_HANDOVER_SELECT = 0x08,
RECORD_TYPE_WKT_HANDOVER_CARRIER = 0x09,
RECORD_TYPE_WKT_ALTERNATIVE_CARRIER = 0x0a,
RECORD_TYPE_WKT_COLLISION_RESOLUTION = 0x0b,
RECORD_TYPE_WKT_ERROR = 0x0c,
RECORD_TYPE_MIME_TYPE = 0x0d,
RECORD_TYPE_EXT_AAR = 0x0e,
RECORD_TYPE_UNKNOWN = 0xfe,
RECORD_TYPE_ERROR = 0xff
};
struct near_ndef_record_header {
uint8_t mb;
uint8_t me;
uint8_t cf;
uint8_t sr;
uint8_t il;
uint8_t tnf;
uint8_t il_length;
uint8_t *il_field;
uint32_t payload_len;
uint32_t offset;
uint8_t type_len;
enum record_type rec_type;
char *type_name;
uint32_t header_len;
};
struct near_ndef_text_payload {
char *encoding;
char *language_code;
char *data;
};
struct near_ndef_uri_payload {
uint8_t identifier;
uint32_t field_length;
uint8_t *field;
};
struct near_ndef_sp_payload {
struct near_ndef_uri_payload *uri;
uint8_t number_of_title_records;
struct near_ndef_text_payload **title_records;
uint32_t size; /* from Size record*/
char *type; /* from Type record*/
char *action;
};
struct near_ndef_aar_payload {
char *package;
};
struct near_ndef_record {
char *path;
struct near_ndef_record_header *header;
/* specific payloads */
struct near_ndef_text_payload *text;
struct near_ndef_uri_payload *uri;
struct near_ndef_sp_payload *sp;
struct near_ndef_mime_payload *mime;
struct near_ndef_ho_payload *ho; /* handover payload */
struct near_ndef_aar_payload *aar;
char *type;
uint8_t *data;
size_t data_len;
};
struct near_ndef_ho_payload {
uint8_t version; /* version id */
uint16_t collision_record; /* collision record */
uint8_t number_of_ac_payloads; /* At least 1 ac is needed */
struct near_ndef_ac_payload **ac_payloads;
/* Optional records */
uint16_t *err_record; /* not NULL if present */
uint8_t number_of_cfg_payloads; /* extra NDEF records */
struct near_ndef_mime_payload **cfg_payloads;
};
struct near_ndef_ac_payload {
enum carrier_power_state cps; /* carrier power state */
uint8_t cdr_len; /* carrier data reference length: 0x01 */
uint8_t *cdr; /* carrier data reference */
uint8_t adata_refcount; /* auxiliary data reference count */
/* !: if adata_refcount == 0, then there's no data reference */
uint16_t **adata; /* auxiliary data reference */
};
/* http://www.intel.com URI NDEF */
static uint8_t uri[] = {0xd1, 0x1, 0xa, 0x55, 0x1, 0x69, 0x6e, 0x74,
0x65, 0x6c, 0x2e, 0x63, 0x6f, 0x6d};
/* 'hello' - UTF-8 - en-US Text NDEF */
static uint8_t text[] = {0xd1, 0x1, 0xb, 0x54, 0x5, 0x65, 0x6e, 0x2d,
0x55, 0x53, 0x68, 0x65, 0x6c, 0x6c, 0x6f};
/* Smart poster with a http://intel.com URI record */
static uint8_t single_sp[] = {0xd1, 0x2, 0xe, 0x53, 0x70, 0xd1, 0x1, 0xa,
0x55, 0x3, 0x69, 0x6e, 0x74, 0x65, 0x6c, 0x2e,
0x63, 0x6f, 0x6d};
/* Smart poster with a http://intel.com URI record and a 'Intel' title */
static uint8_t title_sp[] = {0xd1, 0x2, 0x1a, 0x53, 0x70, 0x91, 0x1, 0xa,
0x55, 0x3, 0x69, 0x6e, 0x74, 0x65, 0x6c, 0x2e,
0x63, 0x6f, 0x6d, 0x51, 0x1, 0x8, 0x54, 0x2,
0x65, 0x6e, 0x49, 0x6e, 0x74, 0x65, 0x6c};
/* AAR record with a "com.example.aar" package name */
static uint8_t aar[] = {0xd4, 0xf, 0xf, 0x61, 0x6e, 0x64, 0x72, 0x6f, 0x69,
0x64, 0x2e, 0x63, 0x6f, 0x6d, 0x3a, 0x70, 0x6b, 0x67,
0x63, 0x6f, 0x6d, 0x2e, 0x65, 0x78, 0x61, 0x6d, 0x70,
0x6c, 0x65, 0x2e, 0x61, 0x61, 0x72};
/* Sample Bluetooth Handover Select Message on an NFC Forum Tag */
static uint8_t ho_hs_bt[] = {0x91, 0x02, 0x0A, 0x48, 0x73, 0x12, 0xD1, 0x02,
0x04, 0x61, 0x63, 0x03, 0x01, 0x30, 0x00, 0x5A,
0x20, 0x1F, 0x01, 0x61, 0x70, 0x70, 0x6C, 0x69,
0x63, 0x61, 0x74, 0x69, 0x6F, 0x6E, 0x2F, 0x76,
0x6E, 0x64, 0x2E, 0x62, 0x6C, 0x75, 0x65, 0x74,
0x6F, 0x6F, 0x74, 0x68, 0x2E, 0x65, 0x70, 0x2E,
0x6F, 0x6F, 0x62, 0x30, 0x1F, 0x00, 0x03, 0x07,
0x80, 0x88, 0xbf, 0x01, 0x04, 0x0D, 0x80, 0x06,
0x04, 0x05, 0x03, 0x18, 0x11, 0x23, 0x11, 0x0B,
0x09, 0x44, 0x65, 0x79, 0x69, 0x63, 0x65, 0x4e,
0x61, 0x6d, 0x65};
static void test_ndef_free_record(struct near_ndef_record *record)
{
g_free(record->header);
g_free(record->type);
g_free(record->data);
g_free(record);
}
static void test_ndef_uri(void)
{
GList *records;
struct near_ndef_record *record;
records = near_ndef_parse_msg(uri, sizeof(uri), NULL);
g_assert(records);
g_assert(g_list_length(records) == 1);
record = (struct near_ndef_record *)(records->data);
g_assert(record->header->rec_type == RECORD_TYPE_WKT_URI);
g_assert(record->header->mb == 1);
g_assert(record->header->me == 1);
g_assert(record->uri);
g_assert(record->uri->field_length == strlen("intel.com"));
g_assert(strncmp((char *) record->uri->field, "intel.com",
record->uri->field_length) == 0);
if (g_test_verbose())
g_print("NDEF URI field: %s\n", record->uri->field);
g_free(record->uri->field);
g_free(record->uri);
test_ndef_free_record(record);
}
static void test_ndef_text(void)
{
GList *records;
struct near_ndef_record *record;
records = near_ndef_parse_msg(text, sizeof(text), NULL);
g_assert(records);
g_assert(g_list_length(records) == 1);
record = (struct near_ndef_record *)(records->data);
g_assert(record->header->rec_type == RECORD_TYPE_WKT_TEXT);
g_assert(record->header->mb == 1);
g_assert(record->header->me == 1);
g_assert(record->text);
g_assert(strcmp(record->text->data, "hello") == 0);
g_assert(strcmp(record->text->encoding, "UTF-8") == 0);
g_assert(strcmp(record->text->language_code, "en-US") == 0);
if (g_test_verbose()) {
g_print("NDEF Text data: %s\n", record->text->data);
g_print("NDEF Text Encoding: %s\n", record->text->encoding);
g_print("NDEF Text Language: %s\n",
record->text->language_code);
}
g_free(record->text->data);
g_free(record->text->encoding);
g_free(record->text->language_code);
g_free(record->text);
test_ndef_free_record(record);
}
static void test_ndef_single_sp(void)
{
GList *records;
struct near_ndef_record *record;
struct near_ndef_uri_payload *uri;
records = near_ndef_parse_msg(single_sp, sizeof(single_sp), NULL);
g_assert(records);
g_assert(g_list_length(records) == 1);
record = (struct near_ndef_record *) records->data;
g_assert(record->header->rec_type == RECORD_TYPE_WKT_SMART_POSTER);
g_assert(record->header->mb == 1);
g_assert(record->header->me == 1);
g_assert(record->sp);
g_assert(record->sp->number_of_title_records == 0);
g_assert(!record->sp->type);
g_assert(!record->sp->action);
g_assert(record->sp->size == 0);
g_assert(record->sp->uri);
uri = (struct near_ndef_uri_payload *) record->sp->uri;
g_assert(uri->field_length == strlen("intel.com"));
g_assert(strncmp((char *) uri->field, "intel.com",
uri->field_length) == 0);
if (g_test_verbose())
g_print("NDEF SP URI field: %.*s\n", uri->field_length,
(char *) uri->field);
g_free(uri->field);
g_free(uri);
g_free(record->sp);
test_ndef_free_record(record);
}
static void test_ndef_title_sp(void)
{
GList *records;
struct near_ndef_record *record;
struct near_ndef_uri_payload *uri;
struct near_ndef_text_payload *text;
records = near_ndef_parse_msg(title_sp, sizeof(title_sp), NULL);
g_assert(records);
g_assert(g_list_length(records) == 1);
record = (struct near_ndef_record *) records->data;
g_assert(record->header->rec_type == RECORD_TYPE_WKT_SMART_POSTER);
g_assert(record->header->mb == 1);
g_assert(record->header->me == 1);
g_assert(record->sp);
g_assert(record->sp->number_of_title_records == 1);
g_assert(!record->sp->type);
g_assert(!record->sp->action);
g_assert(record->sp->size == 0);
g_assert(record->sp->uri);
g_assert(record->sp->title_records[0]);
uri = (struct near_ndef_uri_payload *) record->sp->uri;
text = (struct near_ndef_text_payload *) record->sp->title_records[0];
g_assert(uri->field_length == strlen("intel.com"));
g_assert(strncmp((char *) uri->field, "intel.com",
uri->field_length) == 0);
if (g_test_verbose())
g_print("NDEF SP URI field: %.*s\n", uri->field_length,
(char *) uri->field);
g_assert(strcmp(text->data, "Intel") == 0);
g_assert(strcmp(text->encoding, "UTF-8") == 0);
g_assert(strcmp(text->language_code, "en") == 0);
if (g_test_verbose()) {
g_print("NDEF SP Title data: %s\n", text->data);
g_print("NDEF SP Title Encoding: %s\n", text->encoding);
g_print("NDEF SP Title Language: %s\n", text->language_code);
}
g_free(uri->field);
g_free(uri);
g_free(text->data);
g_free(text->encoding);
g_free(text->language_code);
g_free(text);
g_free(record->sp);
test_ndef_free_record(record);
}
static void test_ndef_aar(void)
{
GList *records;
struct near_ndef_record *record;
records = near_ndef_parse_msg(aar, sizeof(aar), NULL);
g_assert(records);
g_assert(g_list_length(records) == 1);
record = (struct near_ndef_record *)(records->data);
g_assert(record->header->rec_type == RECORD_TYPE_EXT_AAR);
g_assert(record->header->mb == 1);
g_assert(record->header->me == 1);
g_assert(record->aar);
g_assert(record->aar->package);
g_assert(strcmp((char *) record->aar->package, "com.example.aar") == 0);
if (g_test_verbose())
g_print("NDEF AAR package: %s\n", record->aar->package);
g_free(record->aar->package);
g_free(record->aar);
test_ndef_free_record(record);
}
static void test_ndef_ho_hs_bt(void)
{
GList *records;
struct near_ndef_record *record;
struct near_ndef_ho_payload *ho;
struct near_ndef_ac_payload *ac;
/* TODO
* It is not possible to run HO code as unit test now as this code does
* both parsing and acting (eg. trying to access HO agent) in same run.
* Don't enable this test until it is fixed.
*/
return;
records = near_ndef_parse_msg(ho_hs_bt, sizeof(ho_hs_bt), NULL);
g_assert(records);
g_assert(g_list_length(records) == 2);
record = records->data;
ho = record->ho;
g_assert(ho->number_of_ac_payloads == 1);
ac = ho->ac_payloads[0];
g_assert(ac->cdr_len == 1);
g_assert(memcmp(ac->cdr, "0", ac->cdr_len) == 0);
records = g_list_next(records);
record = records->data;
g_assert(strcmp(record->type, BT_MIME_STRING_2_1) == 0);
}
int main(int argc, char **argv)
{
g_test_init(&argc, &argv, NULL);
g_test_add_func("/testNDEF-parse/Test URI NDEF", test_ndef_uri);
g_test_add_func("/testNDEF-parse/Test Text NDEF", test_ndef_text);
g_test_add_func("/testNDEF-parse/Test Single record SmartPoster NDEF",
test_ndef_single_sp);
g_test_add_func("/testNDEF-parse/Test Title record SmartPoster NDEF",
test_ndef_title_sp);
g_test_add_func("/testNDEF-parse/Android Application Record NDEF",
test_ndef_aar);
g_test_add_func("/testNDEF-parse/Test Handover Select NDEF",
test_ndef_ho_hs_bt);
return g_test_run();
}