NFC: Forward NFC_EVT_TRANSACTION to user space
authorChristophe Ricard <christophe.ricard@gmail.com>
Sun, 1 Feb 2015 21:26:16 +0000 (22:26 +0100)
committerSamuel Ortiz <sameo@linux.intel.com>
Mon, 2 Feb 2015 20:50:40 +0000 (21:50 +0100)
NFC_EVT_TRANSACTION is sent through netlink in order for a
specific application running on a secure element to notify
userspace of an event. Typically the secure element application
counterpart on the host could interpret that event and act
upon it.

Forwarded information contains:
- SE host generating the event
- Application IDentifier doing the operation
- Applications parameters

Signed-off-by: Christophe Ricard <christophe-h.ricard@st.com>
Signed-off-by: Samuel Ortiz <sameo@linux.intel.com>
include/net/nfc/nfc.h
include/uapi/linux/nfc.h
net/nfc/core.c
net/nfc/netlink.c
net/nfc/nfc.h

index 12adb817c27a3f157f351915e38ecc131fd52e82..73190e65d5c13aa7217cce910203f4bd5f63d385 100644 (file)
@@ -135,6 +135,31 @@ struct nfc_se {
        u16 state;
 };
 
+/**
+ * nfc_evt_transaction - A struct for NFC secure element event transaction.
+ *
+ * @aid: The application identifier triggering the event
+ *
+ * @aid_len: The application identifier length [5:16]
+ *
+ * @params: The application parameters transmitted during the transaction
+ *
+ * @params_len: The applications parameters length [0:255]
+ *
+ */
+#define NFC_MIN_AID_LENGTH     5
+#define        NFC_MAX_AID_LENGTH      16
+#define NFC_MAX_PARAMS_LENGTH  255
+
+#define NFC_EVT_TRANSACTION_AID_TAG    0x81
+#define NFC_EVT_TRANSACTION_PARAMS_TAG 0x82
+struct nfc_evt_transaction {
+       u32 aid_len;
+       u8 aid[NFC_MAX_AID_LENGTH];
+       u8 params_len;
+       u8 params[NFC_MAX_PARAMS_LENGTH];
+} __packed;
+
 struct nfc_genl_data {
        u32 poll_req_portid;
        struct mutex genl_data_mutex;
@@ -262,6 +287,8 @@ int nfc_tm_data_received(struct nfc_dev *dev, struct sk_buff *skb);
 
 void nfc_driver_failure(struct nfc_dev *dev, int err);
 
+int nfc_se_transaction(struct nfc_dev *dev, u8 se_idx,
+                      struct nfc_evt_transaction *evt_transaction);
 int nfc_add_se(struct nfc_dev *dev, u32 se_idx, u16 type);
 int nfc_remove_se(struct nfc_dev *dev, u32 se_idx);
 struct nfc_se *nfc_find_se(struct nfc_dev *dev, u32 se_idx);
index 8119255feae4b44df8bac40415f4681a9ccf3c97..c1e2e63cf9b57873810f68056932589a9cde3ca2 100644 (file)
@@ -183,6 +183,7 @@ enum nfc_attrs {
        NFC_ATTR_SE_APDU,
        NFC_ATTR_TARGET_ISO15693_DSFID,
        NFC_ATTR_TARGET_ISO15693_UID,
+       NFC_ATTR_SE_PARAMS,
 /* private: internal use only */
        __NFC_ATTR_AFTER_LAST
 };
index 7f1b6351755cfe197ac96f0ef5a6831b44242474..cff3f1614ad4557ff178804acb55bacbad328149 100644 (file)
@@ -932,6 +932,27 @@ int nfc_remove_se(struct nfc_dev *dev, u32 se_idx)
 }
 EXPORT_SYMBOL(nfc_remove_se);
 
+int nfc_se_transaction(struct nfc_dev *dev, u8 se_idx,
+                      struct nfc_evt_transaction *evt_transaction)
+{
+       int rc;
+
+       pr_debug("transaction: %x\n", se_idx);
+
+       device_lock(&dev->dev);
+
+       if (!evt_transaction) {
+               rc = -EPROTO;
+               goto out;
+       }
+
+       rc = nfc_genl_se_transaction(dev, se_idx, evt_transaction);
+out:
+       device_unlock(&dev->dev);
+       return rc;
+}
+EXPORT_SYMBOL(nfc_se_transaction);
+
 static void nfc_release(struct device *d)
 {
        struct nfc_dev *dev = to_nfc_dev(d);
index be387e6219a0f52d96e03192317ba88777ec6346..14a2d11581da7ededf0eff4ca09293a277853b2d 100644 (file)
@@ -497,6 +497,53 @@ free_msg:
        return -EMSGSIZE;
 }
 
+int nfc_genl_se_transaction(struct nfc_dev *dev, u8 se_idx,
+                           struct nfc_evt_transaction *evt_transaction)
+{
+       struct nfc_se *se;
+       struct sk_buff *msg;
+       void *hdr;
+
+       msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
+       if (!msg)
+               return -ENOMEM;
+
+       hdr = genlmsg_put(msg, 0, 0, &nfc_genl_family, 0,
+                         NFC_EVENT_SE_TRANSACTION);
+       if (!hdr)
+               goto free_msg;
+
+       se = nfc_find_se(dev, se_idx);
+       if (!se)
+               goto free_msg;
+
+       if (nla_put_u32(msg, NFC_ATTR_DEVICE_INDEX, dev->idx) ||
+           nla_put_u32(msg, NFC_ATTR_SE_INDEX, se_idx) ||
+           nla_put_u8(msg, NFC_ATTR_SE_TYPE, se->type) ||
+           nla_put(msg, NFC_ATTR_SE_AID, evt_transaction->aid_len,
+                   evt_transaction->aid) ||
+           nla_put(msg, NFC_ATTR_SE_PARAMS, evt_transaction->params_len,
+                   evt_transaction->params))
+               goto nla_put_failure;
+
+       /* evt_transaction is no more used */
+       devm_kfree(&dev->dev, evt_transaction);
+
+       genlmsg_end(msg, hdr);
+
+       genlmsg_multicast(&nfc_genl_family, msg, 0, 0, GFP_KERNEL);
+
+       return 0;
+
+nla_put_failure:
+       genlmsg_cancel(msg, hdr);
+free_msg:
+       /* evt_transaction is no more used */
+       devm_kfree(&dev->dev, evt_transaction);
+       nlmsg_free(msg);
+       return -EMSGSIZE;
+}
+
 static int nfc_genl_send_device(struct sk_buff *msg, struct nfc_dev *dev,
                                u32 portid, u32 seq,
                                struct netlink_callback *cb,
index 88d60064890e3a9ca93db4c88fdcf840e077fabe..a8ce80b47720a6425c857a660895418b1c00d84f 100644 (file)
@@ -100,6 +100,8 @@ int nfc_genl_llc_send_sdres(struct nfc_dev *dev, struct hlist_head *sdres_list);
 
 int nfc_genl_se_added(struct nfc_dev *dev, u32 se_idx, u16 type);
 int nfc_genl_se_removed(struct nfc_dev *dev, u32 se_idx);
+int nfc_genl_se_transaction(struct nfc_dev *dev, u8 se_idx,
+                           struct nfc_evt_transaction *evt_transaction);
 
 struct nfc_dev *nfc_get_device(unsigned int idx);