blob: d5ce9737b0589dd1863d9d04bb5e62696f83c05d [file] [log] [blame] [edit]
/*
*
* neard - Near Field Communication manager
*
* Copyright (C) 2011 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 <stdlib.h>
#include <errno.h>
#include <string.h>
#include <glib.h>
#include <gdbus.h>
#include "near.h"
#define TYPE3_IDM_LEN 8
#define TYPE3_ATTR_BLOCK_SIZE 16
struct near_tag {
char *path;
uint32_t adapter_idx;
uint32_t target_idx;
uint32_t protocol;
uint32_t type;
enum near_tag_sub_type sub_type;
enum near_tag_memory_layout layout;
bool readonly;
uint8_t nfcid[NFC_MAX_NFCID1_LEN];
uint8_t nfcid_len;
uint8_t iso15693_dsfid;
uint8_t iso15693_uid[NFC_MAX_ISO15693_UID_LEN];
size_t data_length;
uint8_t *data;
uint32_t next_record;
GList *records;
bool blank;
/* Tag specific structures */
struct {
uint8_t IDm[TYPE3_IDM_LEN];
uint8_t attr[TYPE3_ATTR_BLOCK_SIZE];
uint8_t ic_type;
} t3;
struct {
uint16_t max_ndef_size;
uint16_t c_apdu_max_size;
uint16_t r_apdu_max_size;
uint16_t file_id;
} t4;
struct {
uint8_t blk_size;
uint8_t num_blks;
} t5;
DBusMessage *write_msg; /* Pending write message */
struct near_ndef_message *write_ndef;
};
static DBusConnection *connection = NULL;
static GHashTable *tag_hash;
static GSList *driver_list = NULL;
struct near_tag *near_tag_get_tag(uint32_t adapter_idx, uint32_t target_idx)
{
struct near_tag *tag;
char *path;
path = g_strdup_printf("%s/nfc%d/tag%d", NFC_PATH,
adapter_idx, target_idx);
if (!path)
return NULL;
tag = g_hash_table_lookup(tag_hash, path);
g_free(path);
/* TODO refcount */
return tag;
}
static void append_records(DBusMessageIter *iter, void *user_data)
{
struct near_tag *tag = user_data;
GList *list;
DBG("");
for (list = tag->records; list; list = list->next) {
struct near_ndef_record *record = list->data;
char *path;
path = __near_ndef_record_get_path(record);
if (!path)
continue;
dbus_message_iter_append_basic(iter, DBUS_TYPE_OBJECT_PATH,
&path);
}
}
static const char *type_string(struct near_tag *tag)
{
const char *type;
DBG("type 0x%x", tag->type);
switch (tag->type) {
case NFC_PROTO_JEWEL:
type = "Type 1";
break;
case NFC_PROTO_MIFARE:
type = "Type 2";
break;
case NFC_PROTO_FELICA:
type = "Type 3";
break;
case NFC_PROTO_ISO14443:
case NFC_PROTO_ISO14443_B:
type = "Type 4";
break;
case NFC_PROTO_ISO15693:
type = "Type 5";
break;
default:
type = NULL;
near_error("Unknown tag type 0x%x", tag->type);
break;
}
return type;
}
static const char *protocol_string(struct near_tag *tag)
{
const char *protocol;
DBG("protocol 0x%x", tag->protocol);
switch (tag->protocol) {
case NFC_PROTO_FELICA_MASK:
protocol = "Felica";
break;
case NFC_PROTO_MIFARE_MASK:
protocol = "MIFARE";
break;
case NFC_PROTO_JEWEL_MASK:
protocol = "Jewel";
break;
case NFC_PROTO_ISO14443_MASK:
case NFC_PROTO_ISO14443_B_MASK:
protocol = "ISO-DEP";
break;
case NFC_PROTO_ISO15693_MASK:
protocol = "ISO-15693";
break;
default:
near_error("Unknown tag protocol 0x%x", tag->protocol);
protocol = NULL;
}
return protocol;
}
static gboolean property_get_type(const GDBusPropertyTable *property,
DBusMessageIter *iter, void *user_data)
{
struct near_tag *tag = user_data;
const char *type;
type = type_string(tag);
if (!type)
return FALSE;
dbus_message_iter_append_basic(iter, DBUS_TYPE_STRING, &type);
return TRUE;
}
static gboolean property_get_protocol(const GDBusPropertyTable *property,
DBusMessageIter *iter, void *user_data)
{
struct near_tag *tag = user_data;
const char *protocol;
protocol = protocol_string(tag);
if (!protocol)
return FALSE;
dbus_message_iter_append_basic(iter, DBUS_TYPE_STRING, &protocol);
return TRUE;
}
static gboolean property_get_readonly(const GDBusPropertyTable *property,
DBusMessageIter *iter, void *user_data)
{
struct near_tag *tag = user_data;
dbus_bool_t readonly;
readonly = tag->readonly;
dbus_message_iter_append_basic(iter, DBUS_TYPE_BOOLEAN, &readonly);
return TRUE;
}
static gboolean property_get_adapter(const GDBusPropertyTable *property,
DBusMessageIter *iter, void *user_data)
{
struct near_tag *tag = user_data;
struct near_adapter *adapter;
const char *path;
adapter = __near_adapter_get(tag->adapter_idx);
if (!adapter)
return FALSE;
path = __near_adapter_get_path(adapter);
if (!path)
return FALSE;
dbus_message_iter_append_basic(iter, DBUS_TYPE_OBJECT_PATH, &path);
return TRUE;
}
static void tag_read_cb(uint32_t adapter_idx, uint32_t target_idx, int status)
{
struct near_tag *tag;
tag = near_tag_get_tag(adapter_idx, target_idx);
if (!tag)
return;
if (tag->write_msg) {
dbus_message_unref(tag->write_msg);
tag->write_msg = NULL;
}
__near_adapter_start_check_presence(adapter_idx, target_idx);
}
static void write_cb(uint32_t adapter_idx, uint32_t target_idx, int status)
{
struct near_tag *tag;
DBusConnection *conn;
DBusMessage *reply;
DBG("Write status %d", status);
tag = near_tag_get_tag(adapter_idx, target_idx);
if (!tag)
return;
conn = near_dbus_get_connection();
if (!conn)
goto out;
if (status != 0) {
if (tag->write_msg) {
reply = __near_error_failed(tag->write_msg, -status);
if (reply)
g_dbus_send_message(conn, reply);
}
} else {
g_dbus_send_reply(conn, tag->write_msg, DBUS_TYPE_INVALID);
}
near_ndef_records_free(tag->records);
tag->records = NULL;
g_free(tag->data);
tag->data = NULL;
if (status == 0) {
/*
* If writing succeeded,
* check presence will be restored after reading
*/
__near_tag_read(tag, tag_read_cb);
return;
}
out:
if (tag->write_msg) {
dbus_message_unref(tag->write_msg);
tag->write_msg = NULL;
}
__near_adapter_start_check_presence(tag->adapter_idx, tag->target_idx);
}
static void format_cb(uint32_t adapter_idx, uint32_t target_idx, int status)
{
struct near_tag *tag;
int err;
DBG("format status %d", status);
tag = near_tag_get_tag(adapter_idx, target_idx);
if (!tag)
return;
if (!tag->write_msg)
return;
if (status == 0) {
err = __near_tag_write(tag, tag->write_ndef,
write_cb);
if (err < 0)
goto error;
} else {
err = status;
goto error;
}
return;
error:
write_cb(tag->adapter_idx, tag->target_idx, err);
}
static DBusMessage *write_ndef(DBusConnection *conn,
DBusMessage *msg, void *data)
{
struct near_tag *tag = data;
struct near_ndef_message *ndef, *ndef_with_header = NULL;
int tlv_len_size, err;
DBG("conn %p", conn);
if (tag->readonly) {
DBG("Read only tag");
return __near_error_permission_denied(msg);
}
if (tag->write_msg)
return __near_error_in_progress(msg);
ndef = __ndef_build_from_message(msg);
if (!ndef)
return __near_error_failed(msg, EINVAL);
tag->write_msg = dbus_message_ref(msg);
/* Add NDEF header information depends upon tag type */
switch (tag->type) {
case NFC_PROTO_JEWEL:
case NFC_PROTO_MIFARE:
case NFC_PROTO_ISO15693:
if (ndef->length < 0xff)
tlv_len_size = 3;
else
tlv_len_size = 5;
ndef_with_header = g_try_malloc0(sizeof(
struct near_ndef_message));
if (!ndef_with_header)
goto fail;
ndef_with_header->offset = 0;
ndef_with_header->length = ndef->length + tlv_len_size;
ndef_with_header->data =
g_try_malloc0(ndef->length + tlv_len_size);
if (!ndef_with_header->data)
goto fail;
ndef_with_header->data[0] = TLV_NDEF;
if (ndef->length < 0xff) {
ndef_with_header->data[1] = ndef->length;
} else {
ndef_with_header->data[1] = 0xff;
ndef_with_header->data[2] =
(uint8_t)(ndef->length >> 8);
ndef_with_header->data[3] = (uint8_t)(ndef->length);
}
memcpy(ndef_with_header->data + tlv_len_size - 1, ndef->data,
ndef->length);
ndef_with_header->data[ndef->length + tlv_len_size - 1] =
TLV_END;
break;
case NFC_PROTO_FELICA:
ndef_with_header = g_try_malloc0(sizeof(
struct near_ndef_message));
if (!ndef_with_header)
goto fail;
ndef_with_header->offset = 0;
ndef_with_header->length = ndef->length;
ndef_with_header->data = g_try_malloc0(
ndef_with_header->length);
if (!ndef_with_header->data)
goto fail;
memcpy(ndef_with_header->data, ndef->data, ndef->length);
break;
case NFC_PROTO_ISO14443:
case NFC_PROTO_ISO14443_B:
ndef_with_header = g_try_malloc0(sizeof(
struct near_ndef_message));
if (!ndef_with_header)
goto fail;
ndef_with_header->offset = 0;
ndef_with_header->length = ndef->length + 2;
ndef_with_header->data = g_try_malloc0(ndef->length + 2);
if (!ndef_with_header->data)
goto fail;
ndef_with_header->data[0] = (uint8_t)(ndef->length >> 8);
ndef_with_header->data[1] = (uint8_t)(ndef->length);
memcpy(ndef_with_header->data + 2, ndef->data, ndef->length);
break;
default:
g_free(ndef->data);
g_free(ndef);
return __near_error_failed(msg, EOPNOTSUPP);
}
g_free(ndef->data);
g_free(ndef);
tag->write_ndef = ndef_with_header;
err = __near_tag_write(tag, ndef_with_header, write_cb);
if (err < 0)
return __near_error_failed(msg, -err);
return g_dbus_create_reply(msg, DBUS_TYPE_INVALID);
fail:
dbus_message_unref(tag->write_msg);
tag->write_msg = NULL;
return __near_error_failed(msg, ENOMEM);
}
static const GDBusMethodTable tag_methods[] = {
{ GDBUS_ASYNC_METHOD("Write", GDBUS_ARGS({"attributes", "a{sv}"}),
NULL, write_ndef) },
{ },
};
static const GDBusPropertyTable tag_properties[] = {
{ "Type", "s", property_get_type },
{ "Protocol", "s", property_get_protocol },
{ "ReadOnly", "b", property_get_readonly },
{ "Adapter", "o", property_get_adapter },
{ }
};
void __near_tag_append_records(struct near_tag *tag, DBusMessageIter *iter)
{
GList *list;
for (list = tag->records; list; list = list->next) {
struct near_ndef_record *record = list->data;
char *path;
path = __near_ndef_record_get_path(record);
if (!path)
continue;
dbus_message_iter_append_basic(iter, DBUS_TYPE_OBJECT_PATH,
&path);
}
}
#define NFC_TAG_A (NFC_PROTO_ISO14443_MASK | NFC_PROTO_NFC_DEP_MASK | \
NFC_PROTO_JEWEL_MASK | NFC_PROTO_MIFARE_MASK)
#define NFC_TAG_A_TYPE2 0x00
#define NFC_TAG_A_TYPE4 0x01
#define NFC_TAG_A_NFC_DEP 0x02
#define NFC_TAG_A_TYPE4_DEP 0x03
#define NFC_TAG_A_SENS_RES_SSD_JEWEL 0x00
#define NFC_TAG_A_SENS_RES_PLATCONF_JEWEL 0x0c
#define NFC_TAG_A_SEL_PROT(sel_res) (((sel_res) & 0x60) >> 5)
#define NFC_TAG_A_SEL_CASCADE(sel_res) (((sel_res) & 0x04) >> 2)
#define NFC_TAG_A_SENS_RES_SSD(sens_res) ((sens_res) & 0x001f)
#define NFC_TAG_A_SENS_RES_PLATCONF(sens_res) (((sens_res) & 0x0f00) >> 8)
static enum near_tag_sub_type get_tag_type2_sub_type(uint8_t sel_res)
{
switch (sel_res) {
case 0x00:
return NEAR_TAG_NFC_T2_MIFARE_ULTRALIGHT;
case 0x08:
return NEAR_TAG_NFC_T2_MIFARE_CLASSIC_1K;
case 0x09:
return NEAR_TAG_NFC_T2_MIFARE_MINI;
case 0x18:
return NEAR_TAG_NFC_T2_MIFARE_CLASSIC_4K;
case 0x20:
return NEAR_TAG_NFC_T2_MIFARE_DESFIRE;
case 0x28:
return NEAR_TAG_NFC_T2_JCOP30;
case 0x38:
return NEAR_TAG_NFC_T2_MIFARE_4K_EMUL;
case 0x88:
return NEAR_TAG_NFC_T2_MIFARE_1K_INFINEON;
case 0x98:
return NEAR_TAG_NFC_T2_MPCOS;
}
return NEAR_TAG_NFC_SUBTYPE_UNKNOWN;
}
static void set_tag_type(struct near_tag *tag,
uint16_t sens_res, uint8_t sel_res)
{
uint8_t platconf, ssd, proto;
DBG("protocol 0x%x sens_res 0x%x sel_res 0x%x", tag->protocol,
sens_res, sel_res);
switch (tag->protocol) {
case NFC_PROTO_JEWEL_MASK:
platconf = NFC_TAG_A_SENS_RES_PLATCONF(sens_res);
ssd = NFC_TAG_A_SENS_RES_SSD(sens_res);
DBG("Jewel");
if ((ssd == NFC_TAG_A_SENS_RES_SSD_JEWEL) &&
(platconf == NFC_TAG_A_SENS_RES_PLATCONF_JEWEL))
tag->type = NFC_PROTO_JEWEL;
break;
case NFC_PROTO_MIFARE_MASK:
case NFC_PROTO_ISO14443_MASK:
proto = NFC_TAG_A_SEL_PROT(sel_res);
DBG("proto 0x%x", proto);
switch (proto) {
case NFC_TAG_A_TYPE2:
tag->type = NFC_PROTO_MIFARE;
tag->sub_type = get_tag_type2_sub_type(sel_res);
break;
case NFC_TAG_A_TYPE4:
tag->type = NFC_PROTO_ISO14443;
break;
case NFC_TAG_A_TYPE4_DEP:
tag->type = NFC_PROTO_NFC_DEP;
break;
}
break;
case NFC_PROTO_FELICA_MASK:
tag->type = NFC_PROTO_FELICA;
break;
case NFC_PROTO_ISO14443_B_MASK:
tag->type = NFC_PROTO_ISO14443_B;
break;
case NFC_PROTO_ISO15693_MASK:
tag->type = NFC_PROTO_ISO15693;
break;
default:
tag->type = NFC_PROTO_MAX;
break;
}
DBG("tag type 0x%x", tag->type);
}
static int tag_initialize(struct near_tag *tag,
uint32_t adapter_idx, uint32_t target_idx,
uint32_t protocols,
uint16_t sens_res, uint8_t sel_res,
uint8_t *nfcid, uint8_t nfcid_len,
uint8_t iso15693_dsfid,
uint8_t iso15693_uid_len, uint8_t *iso15693_uid)
{
DBG("");
tag->path = g_strdup_printf("%s/nfc%d/tag%d", NFC_PATH,
adapter_idx, target_idx);
if (!tag->path)
return -ENOMEM;
tag->adapter_idx = adapter_idx;
tag->target_idx = target_idx;
tag->protocol = protocols;
tag->next_record = 0;
tag->readonly = false;
if (nfcid_len && nfcid_len <= NFC_MAX_NFCID1_LEN) {
tag->nfcid_len = nfcid_len;
memcpy(tag->nfcid, nfcid, nfcid_len);
} else if (iso15693_uid_len) {
tag->iso15693_dsfid = iso15693_dsfid;
memcpy(tag->iso15693_uid, iso15693_uid, iso15693_uid_len);
}
set_tag_type(tag, sens_res, sel_res);
return 0;
}
struct near_tag *__near_tag_add(uint32_t adapter_idx, uint32_t target_idx,
uint32_t protocols,
uint16_t sens_res, uint8_t sel_res,
uint8_t *nfcid, uint8_t nfcid_len,
uint8_t iso15693_dsfid,
uint8_t iso15693_uid_len, uint8_t *iso15693_uid)
{
struct near_tag *tag;
char *path;
tag = near_tag_get_tag(adapter_idx, target_idx);
if (tag)
return NULL;
tag = g_try_malloc0(sizeof(struct near_tag));
if (!tag)
return NULL;
if (tag_initialize(tag, adapter_idx, target_idx,
protocols,
sens_res, sel_res,
nfcid, nfcid_len,
iso15693_dsfid,
iso15693_uid_len, iso15693_uid) < 0) {
g_free(tag);
return NULL;
}
path = g_strdup(tag->path);
if (!path) {
g_free(tag);
return NULL;
}
g_hash_table_insert(tag_hash, path, tag);
DBG("connection %p", connection);
g_dbus_register_interface(connection, tag->path,
NFC_TAG_INTERFACE,
tag_methods, NULL,
tag_properties, tag, NULL);
return tag;
}
void __near_tag_remove(struct near_tag *tag)
{
char *path = tag->path;
DBG("path %s", tag->path);
g_hash_table_remove(tag_hash, path);
}
const char *__near_tag_get_path(struct near_tag *tag)
{
return tag->path;
}
uint32_t __near_tag_get_type(struct near_tag *tag)
{
return tag->type;
}
enum near_tag_sub_type near_tag_get_subtype(uint32_t adapter_idx,
uint32_t target_idx)
{
struct near_tag *tag;
tag = near_tag_get_tag(adapter_idx, target_idx);
if (!tag)
return NEAR_TAG_NFC_SUBTYPE_UNKNOWN;
return tag->sub_type;
}
uint8_t *near_tag_get_nfcid(uint32_t adapter_idx, uint32_t target_idx,
uint8_t *nfcid_len)
{
struct near_tag *tag;
uint8_t *nfcid;
tag = near_tag_get_tag(adapter_idx, target_idx);
if (!tag)
goto fail;
nfcid = g_try_malloc0(tag->nfcid_len);
if (!nfcid)
goto fail;
memcpy(nfcid, tag->nfcid, tag->nfcid_len);
*nfcid_len = tag->nfcid_len;
return nfcid;
fail:
*nfcid_len = 0;
return NULL;
}
int near_tag_set_nfcid(uint32_t adapter_idx, uint32_t target_idx,
uint8_t *nfcid, size_t nfcid_len)
{
struct near_tag *tag;
DBG("NFCID len %zd", nfcid_len);
tag = near_tag_get_tag(adapter_idx, target_idx);
if (!tag)
return -ENODEV;
if (tag->nfcid_len > 0)
return -EALREADY;
if (nfcid_len > NFC_MAX_NFCID1_LEN)
return -EINVAL;
memcpy(tag->nfcid, nfcid, nfcid_len);
tag->nfcid_len = nfcid_len;
return 0;
}
uint8_t *near_tag_get_iso15693_dsfid(uint32_t adapter_idx, uint32_t target_idx)
{
struct near_tag *tag;
uint8_t *iso15693_dsfid;
tag = near_tag_get_tag(adapter_idx, target_idx);
if (!tag)
goto fail;
iso15693_dsfid = g_try_malloc0(NFC_MAX_ISO15693_DSFID_LEN);
if (!iso15693_dsfid)
goto fail;
memcpy(iso15693_dsfid, &tag->iso15693_dsfid,
NFC_MAX_ISO15693_DSFID_LEN);
return iso15693_dsfid;
fail:
return NULL;
}
uint8_t *near_tag_get_iso15693_uid(uint32_t adapter_idx, uint32_t target_idx)
{
struct near_tag *tag;
uint8_t *iso15693_uid;
tag = near_tag_get_tag(adapter_idx, target_idx);
if (!tag)
goto fail;
iso15693_uid = g_try_malloc0(NFC_MAX_ISO15693_UID_LEN);
if (!iso15693_uid)
goto fail;
memcpy(iso15693_uid, tag->iso15693_uid, NFC_MAX_ISO15693_UID_LEN);
return iso15693_uid;
fail:
return NULL;
}
int near_tag_add_data(uint32_t adapter_idx, uint32_t target_idx,
uint8_t *data, size_t data_length)
{
struct near_tag *tag;
tag = near_tag_get_tag(adapter_idx, target_idx);
if (!tag)
return -ENODEV;
tag->data_length = data_length;
tag->data = g_try_malloc0(data_length);
if (!tag->data)
return -ENOMEM;
if (data)
memcpy(tag->data, data, data_length);
return 0;
}
int near_tag_add_records(struct near_tag *tag, GList *records,
near_tag_io_cb cb, int status)
{
GList *list;
struct near_ndef_record *record;
char *path;
DBG("records %p", records);
for (list = records; list; list = list->next) {
record = list->data;
path = g_strdup_printf("%s/nfc%d/tag%d/record%d",
NFC_PATH, tag->adapter_idx,
tag->target_idx, tag->next_record);
if (!path)
continue;
__near_ndef_record_register(record, path);
tag->next_record++;
tag->records = g_list_append(tag->records, record);
}
__near_agent_ndef_parse_records(tag->records);
near_dbus_property_changed_array(tag->path,
NFC_TAG_INTERFACE, "Records",
DBUS_TYPE_OBJECT_PATH, append_records,
tag);
if (cb)
cb(tag->adapter_idx, tag->target_idx, status);
g_list_free(records);
return 0;
}
void near_tag_set_ro(struct near_tag *tag, bool readonly)
{
tag->readonly = readonly;
}
void near_tag_set_blank(struct near_tag *tag, bool blank)
{
tag->blank = blank;
}
bool near_tag_get_blank(struct near_tag *tag)
{
return tag->blank;
}
uint8_t *near_tag_get_data(struct near_tag *tag, size_t *data_length)
{
if (!data_length)
return NULL;
*data_length = tag->data_length;
return tag->data;
}
size_t near_tag_get_data_length(struct near_tag *tag)
{
return tag->data_length;
}
uint32_t near_tag_get_adapter_idx(struct near_tag *tag)
{
return tag->adapter_idx;
}
uint32_t near_tag_get_target_idx(struct near_tag *tag)
{
return tag->target_idx;
}
enum near_tag_memory_layout near_tag_get_memory_layout(struct near_tag *tag)
{
if (!tag)
return NEAR_TAG_MEMORY_UNKNOWN;
return tag->layout;
}
void near_tag_set_memory_layout(struct near_tag *tag,
enum near_tag_memory_layout layout)
{
if (!tag)
return;
tag->layout = layout;
}
void near_tag_set_max_ndef_size(struct near_tag *tag, uint16_t size)
{
if (!tag)
return;
tag->t4.max_ndef_size = size;
}
uint16_t near_tag_get_max_ndef_size(struct near_tag *tag)
{
if (!tag)
return 0;
return tag->t4.max_ndef_size;
}
void near_tag_set_c_apdu_max_size(struct near_tag *tag, uint16_t size)
{
if (!tag)
return;
tag->t4.c_apdu_max_size = size;
}
uint16_t near_tag_get_c_apdu_max_size(struct near_tag *tag)
{
if (!tag)
return 0;
return tag->t4.c_apdu_max_size;
}
void near_tag_set_r_apdu_max_size(struct near_tag *tag, uint16_t size)
{
if (!tag)
return;
tag->t4.r_apdu_max_size = size;
}
uint16_t near_tag_get_r_apdu_max_size(struct near_tag *tag)
{
if (!tag)
return 0;
return tag->t4.r_apdu_max_size;
}
void near_tag_set_file_id(struct near_tag *tag, uint16_t file_id)
{
if (!tag)
return;
tag->t4.file_id = file_id;
}
uint16_t near_tag_get_file_id(struct near_tag *tag)
{
if (!tag)
return 0;
return tag->t4.file_id;
}
void near_tag_set_idm(struct near_tag *tag, uint8_t *idm, uint8_t len)
{
if (!tag || len > TYPE3_IDM_LEN)
return;
memset(tag->t3.IDm, 0, TYPE3_IDM_LEN);
memcpy(tag->t3.IDm, idm, len);
}
uint8_t *near_tag_get_idm(struct near_tag *tag, uint8_t *len)
{
if (!tag || !len)
return NULL;
*len = TYPE3_IDM_LEN;
return tag->t3.IDm;
}
void near_tag_set_attr_block(struct near_tag *tag, uint8_t *attr, uint8_t len)
{
if (!tag || len > TYPE3_ATTR_BLOCK_SIZE)
return;
memset(tag->t3.attr, 0, TYPE3_ATTR_BLOCK_SIZE);
memcpy(tag->t3.attr, attr, len);
}
uint8_t *near_tag_get_attr_block(struct near_tag *tag, uint8_t *len)
{
if (!tag || !len)
return NULL;
*len = TYPE3_ATTR_BLOCK_SIZE;
return tag->t3.attr;
}
void near_tag_set_ic_type(struct near_tag *tag, uint8_t ic_type)
{
if (!tag)
return;
tag->t3.ic_type = ic_type;
}
uint8_t near_tag_get_ic_type(struct near_tag *tag)
{
if (!tag)
return 0;
return tag->t3.ic_type;
}
uint8_t near_tag_get_blk_size(struct near_tag *tag)
{
return tag->t5.blk_size;
}
void near_tag_set_blk_size(struct near_tag *tag, uint8_t blk_size)
{
tag->t5.blk_size = blk_size;
}
uint8_t near_tag_get_num_blks(struct near_tag *tag)
{
return tag->t5.num_blks;
}
void near_tag_set_num_blks(struct near_tag *tag, uint8_t num_blks)
{
tag->t5.num_blks = num_blks;
}
static gint cmp_prio(gconstpointer a, gconstpointer b)
{
const struct near_tag_driver *driver1 = a;
const struct near_tag_driver *driver2 = b;
return driver2->priority - driver1->priority;
}
int near_tag_driver_register(struct near_tag_driver *driver)
{
DBG("");
if (!driver->read)
return -EINVAL;
driver_list = g_slist_insert_sorted(driver_list, driver, cmp_prio);
return 0;
}
void near_tag_driver_unregister(struct near_tag_driver *driver)
{
DBG("");
driver_list = g_slist_remove(driver_list, driver);
}
int __near_tag_read(struct near_tag *tag, near_tag_io_cb cb)
{
GSList *list;
DBG("type 0x%x", tag->type);
/* Stop check presence while reading */
__near_adapter_stop_check_presence(tag->adapter_idx, tag->target_idx);
for (list = driver_list; list; list = list->next) {
struct near_tag_driver *driver = list->data;
DBG("driver type 0x%x", driver->type);
if (driver->type == tag->type)
return driver->read(tag->adapter_idx, tag->target_idx,
cb);
}
return 0;
}
int __near_tag_write(struct near_tag *tag,
struct near_ndef_message *ndef,
near_tag_io_cb cb)
{
GSList *list;
int err;
DBG("type 0x%x", tag->type);
for (list = driver_list; list; list = list->next) {
struct near_tag_driver *driver = list->data;
DBG("driver type 0x%x", driver->type);
if (driver->type == tag->type) {
/* Stop check presence while writing */
__near_adapter_stop_check_presence(tag->adapter_idx,
tag->target_idx);
if (tag->blank && driver->format) {
DBG("Blank tag detected, formatting");
err = driver->format(tag->adapter_idx,
tag->target_idx, format_cb);
} else {
err = driver->write(tag->adapter_idx,
tag->target_idx, ndef,
cb);
}
break;
}
}
if (!list)
err = -EOPNOTSUPP;
if (err < 0)
__near_adapter_start_check_presence(tag->adapter_idx,
tag->target_idx);
return err;
}
int __near_tag_check_presence(struct near_tag *tag, near_tag_io_cb cb)
{
GSList *list;
DBG("type 0x%x", tag->type);
for (list = driver_list; list; list = list->next) {
struct near_tag_driver *driver = list->data;
DBG("driver type 0x%x", driver->type);
if (driver->type == tag->type) {
if (!driver->check_presence)
continue;
return driver->check_presence(tag->adapter_idx, tag->target_idx, cb);
}
}
return -EOPNOTSUPP;
}
static void free_tag(gpointer data)
{
struct near_tag *tag = data;
DBG("tag %p", tag);
near_ndef_records_free(tag->records);
g_dbus_unregister_interface(connection, tag->path,
NFC_TAG_INTERFACE);
g_free(tag->path);
g_free(tag->data);
g_free(tag);
}
int __near_tag_init(void)
{
DBG("");
connection = near_dbus_get_connection();
tag_hash = g_hash_table_new_full(g_str_hash, g_str_equal,
g_free, free_tag);
return 0;
}
void __near_tag_cleanup(void)
{
DBG("");
g_hash_table_destroy(tag_hash);
tag_hash = NULL;
}