diff options
Diffstat (limited to 'drivers/misc/mei/nfc.c')
| -rw-r--r-- | drivers/misc/mei/nfc.c | 554 | 
1 files changed, 554 insertions, 0 deletions
diff --git a/drivers/misc/mei/nfc.c b/drivers/misc/mei/nfc.c new file mode 100644 index 00000000000..3adf8a70f26 --- /dev/null +++ b/drivers/misc/mei/nfc.c @@ -0,0 +1,554 @@ +/* + * + * Intel Management Engine Interface (Intel MEI) Linux driver + * Copyright (c) 2003-2013, Intel Corporation. + * + * This program is free software; you can redistribute it and/or modify it + * under the terms and conditions of the GNU General Public License, + * version 2, as published by the Free Software Foundation. + * + * This program is distributed in the hope 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. + * + */ + +#include <linux/kernel.h> +#include <linux/sched.h> +#include <linux/module.h> +#include <linux/moduleparam.h> +#include <linux/device.h> +#include <linux/pci.h> +#include <linux/mei_cl_bus.h> + +#include "mei_dev.h" +#include "client.h" + +struct mei_nfc_cmd { +	u8 command; +	u8 status; +	u16 req_id; +	u32 reserved; +	u16 data_size; +	u8 sub_command; +	u8 data[]; +} __packed; + +struct mei_nfc_reply { +	u8 command; +	u8 status; +	u16 req_id; +	u32 reserved; +	u16 data_size; +	u8 sub_command; +	u8 reply_status; +	u8 data[]; +} __packed; + +struct mei_nfc_if_version { +	u8 radio_version_sw[3]; +	u8 reserved[3]; +	u8 radio_version_hw[3]; +	u8 i2c_addr; +	u8 fw_ivn; +	u8 vendor_id; +	u8 radio_type; +} __packed; + +struct mei_nfc_connect { +	u8 fw_ivn; +	u8 vendor_id; +} __packed; + +struct mei_nfc_connect_resp { +	u8 fw_ivn; +	u8 vendor_id; +	u16 me_major; +	u16 me_minor; +	u16 me_hotfix; +	u16 me_build; +} __packed; + +struct mei_nfc_hci_hdr { +	u8 cmd; +	u8 status; +	u16 req_id; +	u32 reserved; +	u16 data_size; +} __packed; + +#define MEI_NFC_CMD_MAINTENANCE 0x00 +#define MEI_NFC_CMD_HCI_SEND 0x01 +#define MEI_NFC_CMD_HCI_RECV 0x02 + +#define MEI_NFC_SUBCMD_CONNECT    0x00 +#define MEI_NFC_SUBCMD_IF_VERSION 0x01 + +#define MEI_NFC_HEADER_SIZE 10 + +/** mei_nfc_dev - NFC mei device + * + * @cl: NFC host client + * @cl_info: NFC info host client + * @init_work: perform connection to the info client + * @fw_ivn: NFC Intervace Version Number + * @vendor_id: NFC manufacturer ID + * @radio_type: NFC radio type + */ +struct mei_nfc_dev { +	struct mei_cl *cl; +	struct mei_cl *cl_info; +	struct work_struct init_work; +	wait_queue_head_t send_wq; +	u8 fw_ivn; +	u8 vendor_id; +	u8 radio_type; +	char *bus_name; + +	u16 req_id; +	u16 recv_req_id; +}; + +static struct mei_nfc_dev nfc_dev; + +/* UUIDs for NFC F/W clients */ +const uuid_le mei_nfc_guid = UUID_LE(0x0bb17a78, 0x2a8e, 0x4c50, +				     0x94, 0xd4, 0x50, 0x26, +				     0x67, 0x23, 0x77, 0x5c); + +static const uuid_le mei_nfc_info_guid = UUID_LE(0xd2de1625, 0x382d, 0x417d, +					0x48, 0xa4, 0xef, 0xab, +					0xba, 0x8a, 0x12, 0x06); + +/* Vendors */ +#define MEI_NFC_VENDOR_INSIDE 0x00 +#define MEI_NFC_VENDOR_NXP    0x01 + +/* Radio types */ +#define MEI_NFC_VENDOR_INSIDE_UREAD 0x00 +#define MEI_NFC_VENDOR_NXP_PN544    0x01 + +static void mei_nfc_free(struct mei_nfc_dev *ndev) +{ +	if (ndev->cl) { +		list_del(&ndev->cl->device_link); +		mei_cl_unlink(ndev->cl); +		kfree(ndev->cl); +	} + +	if (ndev->cl_info) { +		list_del(&ndev->cl_info->device_link); +		mei_cl_unlink(ndev->cl_info); +		kfree(ndev->cl_info); +	} +} + +static int mei_nfc_build_bus_name(struct mei_nfc_dev *ndev) +{ +	struct mei_device *dev; + +	if (!ndev->cl) +		return -ENODEV; + +	dev = ndev->cl->dev; + +	switch (ndev->vendor_id) { +	case MEI_NFC_VENDOR_INSIDE: +		switch (ndev->radio_type) { +		case MEI_NFC_VENDOR_INSIDE_UREAD: +			ndev->bus_name = "microread"; +			return 0; + +		default: +			dev_err(&dev->pdev->dev, "Unknow radio type 0x%x\n", +				ndev->radio_type); + +			return -EINVAL; +		} + +	case MEI_NFC_VENDOR_NXP: +		switch (ndev->radio_type) { +		case MEI_NFC_VENDOR_NXP_PN544: +			ndev->bus_name = "pn544"; +			return 0; +		default: +			dev_err(&dev->pdev->dev, "Unknow radio type 0x%x\n", +				ndev->radio_type); + +			return -EINVAL; +		} + +	default: +		dev_err(&dev->pdev->dev, "Unknow vendor ID 0x%x\n", +			ndev->vendor_id); + +		return -EINVAL; +	} + +	return 0; +} + +static int mei_nfc_connect(struct mei_nfc_dev *ndev) +{ +	struct mei_device *dev; +	struct mei_cl *cl; +	struct mei_nfc_cmd *cmd, *reply; +	struct mei_nfc_connect *connect; +	struct mei_nfc_connect_resp *connect_resp; +	size_t connect_length, connect_resp_length; +	int bytes_recv, ret; + +	cl = ndev->cl; +	dev = cl->dev; + +	connect_length = sizeof(struct mei_nfc_cmd) + +			sizeof(struct mei_nfc_connect); + +	connect_resp_length = sizeof(struct mei_nfc_cmd) + +			sizeof(struct mei_nfc_connect_resp); + +	cmd = kzalloc(connect_length, GFP_KERNEL); +	if (!cmd) +		return -ENOMEM; +	connect = (struct mei_nfc_connect *)cmd->data; + +	reply = kzalloc(connect_resp_length, GFP_KERNEL); +	if (!reply) { +		kfree(cmd); +		return -ENOMEM; +	} + +	connect_resp = (struct mei_nfc_connect_resp *)reply->data; + +	cmd->command = MEI_NFC_CMD_MAINTENANCE; +	cmd->data_size = 3; +	cmd->sub_command = MEI_NFC_SUBCMD_CONNECT; +	connect->fw_ivn = ndev->fw_ivn; +	connect->vendor_id = ndev->vendor_id; + +	ret = __mei_cl_send(cl, (u8 *)cmd, connect_length); +	if (ret < 0) { +		dev_err(&dev->pdev->dev, "Could not send connect cmd\n"); +		goto err; +	} + +	bytes_recv = __mei_cl_recv(cl, (u8 *)reply, connect_resp_length); +	if (bytes_recv < 0) { +		dev_err(&dev->pdev->dev, "Could not read connect response\n"); +		ret = bytes_recv; +		goto err; +	} + +	dev_info(&dev->pdev->dev, "IVN 0x%x Vendor ID 0x%x\n", +		 connect_resp->fw_ivn, connect_resp->vendor_id); + +	dev_info(&dev->pdev->dev, "ME FW %d.%d.%d.%d\n", +		connect_resp->me_major, connect_resp->me_minor, +		connect_resp->me_hotfix, connect_resp->me_build); + +	ret = 0; + +err: +	kfree(reply); +	kfree(cmd); + +	return ret; +} + +static int mei_nfc_if_version(struct mei_nfc_dev *ndev) +{ +	struct mei_device *dev; +	struct mei_cl *cl; + +	struct mei_nfc_cmd cmd; +	struct mei_nfc_reply *reply = NULL; +	struct mei_nfc_if_version *version; +	size_t if_version_length; +	int bytes_recv, ret; + +	cl = ndev->cl_info; +	dev = cl->dev; + +	memset(&cmd, 0, sizeof(struct mei_nfc_cmd)); +	cmd.command = MEI_NFC_CMD_MAINTENANCE; +	cmd.data_size = 1; +	cmd.sub_command = MEI_NFC_SUBCMD_IF_VERSION; + +	ret = __mei_cl_send(cl, (u8 *)&cmd, sizeof(struct mei_nfc_cmd)); +	if (ret < 0) { +		dev_err(&dev->pdev->dev, "Could not send IF version cmd\n"); +		return ret; +	} + +	/* to be sure on the stack we alloc memory */ +	if_version_length = sizeof(struct mei_nfc_reply) + +		sizeof(struct mei_nfc_if_version); + +	reply = kzalloc(if_version_length, GFP_KERNEL); +	if (!reply) +		return -ENOMEM; + +	bytes_recv = __mei_cl_recv(cl, (u8 *)reply, if_version_length); +	if (bytes_recv < 0 || bytes_recv < sizeof(struct mei_nfc_reply)) { +		dev_err(&dev->pdev->dev, "Could not read IF version\n"); +		ret = -EIO; +		goto err; +	} + +	version = (struct mei_nfc_if_version *)reply->data; + +	ndev->fw_ivn = version->fw_ivn; +	ndev->vendor_id = version->vendor_id; +	ndev->radio_type = version->radio_type; + +err: +	kfree(reply); +	return ret; +} + +static int mei_nfc_enable(struct mei_cl_device *cldev) +{ +	struct mei_device *dev; +	struct mei_nfc_dev *ndev = &nfc_dev; +	int ret; + +	dev = ndev->cl->dev; + +	ret = mei_nfc_connect(ndev); +	if (ret < 0) { +		dev_err(&dev->pdev->dev, "Could not connect to NFC"); +		return ret; +	} + +	return 0; +} + +static int mei_nfc_disable(struct mei_cl_device *cldev) +{ +	return 0; +} + +static int mei_nfc_send(struct mei_cl_device *cldev, u8 *buf, size_t length) +{ +	struct mei_device *dev; +	struct mei_nfc_dev *ndev; +	struct mei_nfc_hci_hdr *hdr; +	u8 *mei_buf; +	int err; + +	ndev = (struct mei_nfc_dev *) cldev->priv_data; +	dev = ndev->cl->dev; + +	mei_buf = kzalloc(length + MEI_NFC_HEADER_SIZE, GFP_KERNEL); +	if (!mei_buf) +		return -ENOMEM; + +	hdr = (struct mei_nfc_hci_hdr *) mei_buf; +	hdr->cmd = MEI_NFC_CMD_HCI_SEND; +	hdr->status = 0; +	hdr->req_id = ndev->req_id; +	hdr->reserved = 0; +	hdr->data_size = length; + +	memcpy(mei_buf + MEI_NFC_HEADER_SIZE, buf, length); + +	err = __mei_cl_send(ndev->cl, mei_buf, length + MEI_NFC_HEADER_SIZE); +	if (err < 0) +		return err; + +	kfree(mei_buf); + +	if (!wait_event_interruptible_timeout(ndev->send_wq, +				ndev->recv_req_id == ndev->req_id, HZ)) { +		dev_err(&dev->pdev->dev, "NFC MEI command timeout\n"); +		err = -ETIMEDOUT; +	} else { +		ndev->req_id++; +	} + +	return err; +} + +static int mei_nfc_recv(struct mei_cl_device *cldev, u8 *buf, size_t length) +{ +	struct mei_nfc_dev *ndev; +	struct mei_nfc_hci_hdr *hci_hdr; +	int received_length; + +	ndev = (struct mei_nfc_dev *)cldev->priv_data; + +	received_length = __mei_cl_recv(ndev->cl, buf, length); +	if (received_length < 0) +		return received_length; + +	hci_hdr = (struct mei_nfc_hci_hdr *) buf; + +	if (hci_hdr->cmd == MEI_NFC_CMD_HCI_SEND) { +		ndev->recv_req_id = hci_hdr->req_id; +		wake_up(&ndev->send_wq); + +		return 0; +	} + +	return received_length; +} + +static struct mei_cl_ops nfc_ops = { +	.enable = mei_nfc_enable, +	.disable = mei_nfc_disable, +	.send = mei_nfc_send, +	.recv = mei_nfc_recv, +}; + +static void mei_nfc_init(struct work_struct *work) +{ +	struct mei_device *dev; +	struct mei_cl_device *cldev; +	struct mei_nfc_dev *ndev; +	struct mei_cl *cl_info; + +	ndev = container_of(work, struct mei_nfc_dev, init_work); + +	cl_info = ndev->cl_info; +	dev = cl_info->dev; + +	mutex_lock(&dev->device_lock); + +	if (mei_cl_connect(cl_info, NULL) < 0) { +		mutex_unlock(&dev->device_lock); +		dev_err(&dev->pdev->dev, +			"Could not connect to the NFC INFO ME client"); + +		goto err; +	} + +	mutex_unlock(&dev->device_lock); + +	if (mei_nfc_if_version(ndev) < 0) { +		dev_err(&dev->pdev->dev, "Could not get the NFC interfave version"); + +		goto err; +	} + +	dev_info(&dev->pdev->dev, +		"NFC MEI VERSION: IVN 0x%x Vendor ID 0x%x Type 0x%x\n", +		ndev->fw_ivn, ndev->vendor_id, ndev->radio_type); + +	mutex_lock(&dev->device_lock); + +	if (mei_cl_disconnect(cl_info) < 0) { +		mutex_unlock(&dev->device_lock); +		dev_err(&dev->pdev->dev, +			"Could not disconnect the NFC INFO ME client"); + +		goto err; +	} + +	mutex_unlock(&dev->device_lock); + +	if (mei_nfc_build_bus_name(ndev) < 0) { +		dev_err(&dev->pdev->dev, +			"Could not build the bus ID name\n"); +		return; +	} + +	cldev = mei_cl_add_device(dev, mei_nfc_guid, ndev->bus_name, &nfc_ops); +	if (!cldev) { +		dev_err(&dev->pdev->dev, +			"Could not add the NFC device to the MEI bus\n"); + +		goto err; +	} + +	cldev->priv_data = ndev; + + +	return; + +err: +	mei_nfc_free(ndev); + +	return; +} + + +int mei_nfc_host_init(struct mei_device *dev) +{ +	struct mei_nfc_dev *ndev = &nfc_dev; +	struct mei_cl *cl_info, *cl = NULL; +	int i, ret; + +	/* already initialzed */ +	if (ndev->cl_info) +		return 0; + +	cl_info = mei_cl_allocate(dev); +	cl = mei_cl_allocate(dev); + +	if (!cl || !cl_info) { +		ret = -ENOMEM; +		goto err; +	} + +	/* check for valid client id */ +	i = mei_me_cl_by_uuid(dev, &mei_nfc_info_guid); +	if (i < 0) { +		dev_info(&dev->pdev->dev, "nfc: failed to find the client\n"); +		ret = -ENOENT; +		goto err; +	} + +	cl_info->me_client_id = dev->me_clients[i].client_id; + +	ret = mei_cl_link(cl_info, MEI_HOST_CLIENT_ID_ANY); +	if (ret) +		goto err; + +	cl_info->device_uuid = mei_nfc_info_guid; + +	list_add_tail(&cl_info->device_link, &dev->device_list); + +	/* check for valid client id */ +	i = mei_me_cl_by_uuid(dev, &mei_nfc_guid); +	if (i < 0) { +		dev_info(&dev->pdev->dev, "nfc: failed to find the client\n"); +		ret = -ENOENT; +		goto err; +	} + +	cl->me_client_id = dev->me_clients[i].client_id; + +	ret = mei_cl_link(cl, MEI_HOST_CLIENT_ID_ANY); +	if (ret) +		goto err; + +	cl->device_uuid = mei_nfc_guid; + +	list_add_tail(&cl->device_link, &dev->device_list); + +	ndev->cl_info = cl_info; +	ndev->cl = cl; +	ndev->req_id = 1; + +	INIT_WORK(&ndev->init_work, mei_nfc_init); +	init_waitqueue_head(&ndev->send_wq); +	schedule_work(&ndev->init_work); + +	return 0; + +err: +	mei_nfc_free(ndev); + +	return ret; +} + +void mei_nfc_host_exit(void) +{ +	struct mei_nfc_dev *ndev = &nfc_dev; + +	if (ndev->cl && ndev->cl->device) +		mei_cl_remove_device(ndev->cl->device); + +	mei_nfc_free(ndev); +}  | 
