| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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>
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | #include <linux/sched.h>
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | #include <linux/module.h>
 | 
					
						
							|  |  |  | #include <linux/moduleparam.h>
 | 
					
						
							|  |  |  | #include <linux/device.h>
 | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:46 +03:00
										 |  |  | #include <linux/slab.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | #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
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:49 +03:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * struct mei_nfc_dev - NFC mei device | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  |  * | 
					
						
							|  |  |  |  * @cl: NFC host client | 
					
						
							|  |  |  |  * @cl_info: NFC info host client | 
					
						
							|  |  |  |  * @init_work: perform connection to the info client | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:50 +03:00
										 |  |  |  * @send_wq: send completion wait queue | 
					
						
							| 
									
										
										
										
											2014-01-08 22:31:46 +02:00
										 |  |  |  * @fw_ivn: NFC Interface Version Number | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  |  * @vendor_id: NFC manufacturer ID | 
					
						
							|  |  |  |  * @radio_type: NFC radio type | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:50 +03:00
										 |  |  |  * @bus_name: bus name | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @req_id:  message counter | 
					
						
							|  |  |  |  * @recv_req_id: reception message counter | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | struct mei_nfc_dev { | 
					
						
							|  |  |  | 	struct mei_cl *cl; | 
					
						
							|  |  |  | 	struct mei_cl *cl_info; | 
					
						
							|  |  |  | 	struct work_struct init_work; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 	wait_queue_head_t send_wq; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 	u8 fw_ivn; | 
					
						
							|  |  |  | 	u8 vendor_id; | 
					
						
							|  |  |  | 	u8 radio_type; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:30 +02:00
										 |  |  | 	char *bus_name; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	u16 req_id; | 
					
						
							|  |  |  | 	u16 recv_req_id; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* 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); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:30 +02:00
										 |  |  | /* 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
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | static void mei_nfc_free(struct mei_nfc_dev *ndev) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-11-05 18:18:52 +02:00
										 |  |  | 	if (!ndev) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 	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); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-06-10 10:10:26 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-05 18:18:52 +02:00
										 |  |  | 	kfree(ndev); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:30 +02:00
										 |  |  | 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: | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 			dev_err(dev->dev, "Unknown radio type 0x%x\n", | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:30 +02:00
										 |  |  | 				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: | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 			dev_err(dev->dev, "Unknown radio type 0x%x\n", | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:30 +02:00
										 |  |  | 				ndev->radio_type); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return -EINVAL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	default: | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 		dev_err(dev->dev, "Unknown vendor ID 0x%x\n", | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:30 +02:00
										 |  |  | 			ndev->vendor_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 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) { | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 		dev_err(dev->dev, "Could not send connect cmd\n"); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bytes_recv = __mei_cl_recv(cl, (u8 *)reply, connect_resp_length); | 
					
						
							|  |  |  | 	if (bytes_recv < 0) { | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 		dev_err(dev->dev, "Could not read connect response\n"); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 		ret = bytes_recv; | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 	dev_info(dev->dev, "IVN 0x%x Vendor ID 0x%x\n", | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 		 connect_resp->fw_ivn, connect_resp->vendor_id); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 	dev_info(dev->dev, "ME FW %d.%d.%d.%d\n", | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 		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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 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) { | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 		dev_err(dev->dev, "Could not send IF version cmd\n"); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 		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)) { | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 		dev_err(dev->dev, "Could not read IF version\n"); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 		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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | static int mei_nfc_enable(struct mei_cl_device *cldev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct mei_device *dev; | 
					
						
							| 
									
										
										
										
											2014-11-05 18:18:52 +02:00
										 |  |  | 	struct mei_nfc_dev *ndev; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-05 18:18:52 +02:00
										 |  |  | 	ndev = (struct mei_nfc_dev *)cldev->priv_data; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 	dev = ndev->cl->dev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ret = mei_nfc_connect(ndev); | 
					
						
							|  |  |  | 	if (ret < 0) { | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 		dev_err(dev->dev, "Could not connect to NFC"); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 		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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-12 18:07:57 +03:00
										 |  |  | 	err = -ENOMEM; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 	mei_buf = kzalloc(length + MEI_NFC_HEADER_SIZE, GFP_KERNEL); | 
					
						
							|  |  |  | 	if (!mei_buf) | 
					
						
							| 
									
										
										
										
											2014-08-12 18:07:57 +03:00
										 |  |  | 		goto out; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	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) | 
					
						
							| 
									
										
										
										
											2014-08-12 18:07:57 +03:00
										 |  |  | 		goto out; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!wait_event_interruptible_timeout(ndev->send_wq, | 
					
						
							|  |  |  | 				ndev->recv_req_id == ndev->req_id, HZ)) { | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 		dev_err(dev->dev, "NFC MEI command timeout\n"); | 
					
						
							| 
									
										
										
										
											2014-02-19 17:35:49 +02:00
										 |  |  | 		err = -ETIME; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		ndev->req_id++; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-08-12 18:07:57 +03:00
										 |  |  | out: | 
					
						
							|  |  |  | 	kfree(mei_buf); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 	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, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | static void mei_nfc_init(struct work_struct *work) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct mei_device *dev; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:30 +02:00
										 |  |  | 	struct mei_cl_device *cldev; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 	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); | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 		dev_err(dev->dev, "Could not connect to the NFC INFO ME client"); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mutex_unlock(&dev->device_lock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (mei_nfc_if_version(ndev) < 0) { | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 		dev_err(dev->dev, "Could not get the NFC interface version"); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 	dev_info(dev->dev, "NFC MEI VERSION: IVN 0x%x Vendor ID 0x%x Type 0x%x\n", | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 		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); | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 		dev_err(dev->dev, "Could not disconnect the NFC INFO ME client"); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mutex_unlock(&dev->device_lock); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:30 +02:00
										 |  |  | 	if (mei_nfc_build_bus_name(ndev) < 0) { | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 		dev_err(dev->dev, "Could not build the bus ID name\n"); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:30 +02:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 	cldev = mei_cl_add_device(dev, mei_nfc_guid, ndev->bus_name, &nfc_ops); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:30 +02:00
										 |  |  | 	if (!cldev) { | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 		dev_err(dev->dev, "Could not add the NFC device to the MEI bus\n"); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:30 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cldev->priv_data = ndev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 	return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | err: | 
					
						
							| 
									
										
										
										
											2014-01-08 20:57:44 +02:00
										 |  |  | 	mutex_lock(&dev->device_lock); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 	mei_nfc_free(ndev); | 
					
						
							| 
									
										
										
										
											2014-01-08 20:57:44 +02:00
										 |  |  | 	mutex_unlock(&dev->device_lock); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int mei_nfc_host_init(struct mei_device *dev) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-11-05 18:18:52 +02:00
										 |  |  | 	struct mei_nfc_dev *ndev; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 	struct mei_cl *cl_info, *cl = NULL; | 
					
						
							| 
									
										
										
										
											2014-08-24 12:08:55 +03:00
										 |  |  | 	struct mei_me_client *me_cl; | 
					
						
							|  |  |  | 	int ret; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-05 18:18:52 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* in case of internal reset bail out
 | 
					
						
							|  |  |  | 	 * as the device is already setup | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	cl = mei_cl_bus_find_cl_by_uuid(dev, mei_nfc_guid); | 
					
						
							|  |  |  | 	if (cl) | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-05 18:18:52 +02:00
										 |  |  | 	ndev = kzalloc(sizeof(struct mei_nfc_dev), GFP_KERNEL); | 
					
						
							|  |  |  | 	if (!ndev) { | 
					
						
							|  |  |  | 		ret = -ENOMEM; | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-21 22:05:38 +03:00
										 |  |  | 	ndev->cl_info = mei_cl_allocate(dev); | 
					
						
							|  |  |  | 	ndev->cl = mei_cl_allocate(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cl = ndev->cl; | 
					
						
							|  |  |  | 	cl_info = ndev->cl_info; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!cl || !cl_info) { | 
					
						
							|  |  |  | 		ret = -ENOMEM; | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* check for valid client id */ | 
					
						
							| 
									
										
										
										
											2014-08-24 12:08:55 +03:00
										 |  |  | 	me_cl = mei_me_cl_by_uuid(dev, &mei_nfc_info_guid); | 
					
						
							|  |  |  | 	if (!me_cl) { | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 		dev_info(dev->dev, "nfc: failed to find the client\n"); | 
					
						
							| 
									
										
										
										
											2014-02-19 17:35:49 +02:00
										 |  |  | 		ret = -ENOTTY; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-24 12:08:55 +03:00
										 |  |  | 	cl_info->me_client_id = me_cl->client_id; | 
					
						
							| 
									
										
										
										
											2014-08-21 14:29:15 +03:00
										 |  |  | 	cl_info->cl_uuid = me_cl->props.protocol_name; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ret = mei_cl_link(cl_info, MEI_HOST_CLIENT_ID_ANY); | 
					
						
							|  |  |  | 	if (ret) | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_add_tail(&cl_info->device_link, &dev->device_list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* check for valid client id */ | 
					
						
							| 
									
										
										
										
											2014-08-24 12:08:55 +03:00
										 |  |  | 	me_cl = mei_me_cl_by_uuid(dev, &mei_nfc_guid); | 
					
						
							|  |  |  | 	if (!me_cl) { | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:42 +03:00
										 |  |  | 		dev_info(dev->dev, "nfc: failed to find the client\n"); | 
					
						
							| 
									
										
										
										
											2014-02-19 17:35:49 +02:00
										 |  |  | 		ret = -ENOTTY; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 		goto err; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-24 12:08:55 +03:00
										 |  |  | 	cl->me_client_id = me_cl->client_id; | 
					
						
							| 
									
										
										
										
											2014-08-21 14:29:15 +03:00
										 |  |  | 	cl->cl_uuid = me_cl->props.protocol_name; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ret = mei_cl_link(cl, MEI_HOST_CLIENT_ID_ANY); | 
					
						
							|  |  |  | 	if (ret) | 
					
						
							|  |  |  | 		goto err; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_add_tail(&cl->device_link, &dev->device_list); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 	ndev->req_id = 1; | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	INIT_WORK(&ndev->init_work, mei_nfc_init); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:31 +02:00
										 |  |  | 	init_waitqueue_head(&ndev->send_wq); | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | 	schedule_work(&ndev->init_work); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | err: | 
					
						
							|  |  |  | 	mei_nfc_free(ndev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-11 13:26:06 +02:00
										 |  |  | void mei_nfc_host_exit(struct mei_device *dev) | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:29 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-11-05 18:18:52 +02:00
										 |  |  | 	struct mei_nfc_dev *ndev; | 
					
						
							|  |  |  | 	struct mei_cl *cl; | 
					
						
							|  |  |  | 	struct mei_cl_device *cldev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cl = mei_cl_bus_find_cl_by_uuid(dev, mei_nfc_guid); | 
					
						
							|  |  |  | 	if (!cl) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cldev = cl->device; | 
					
						
							|  |  |  | 	if (!cldev) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2014-09-29 16:31:37 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-05 18:18:52 +02:00
										 |  |  | 	ndev = (struct mei_nfc_dev *)cldev->priv_data; | 
					
						
							|  |  |  | 	if (ndev) | 
					
						
							|  |  |  | 		cancel_work_sync(&ndev->init_work); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cldev->priv_data = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mutex_lock(&dev->device_lock); | 
					
						
							|  |  |  | 	/* Need to remove the device here
 | 
					
						
							|  |  |  | 	 * since mei_nfc_free will unlink the clients | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	mei_cl_remove_device(cldev); | 
					
						
							|  |  |  | 	mei_nfc_free(ndev); | 
					
						
							|  |  |  | 	mutex_unlock(&dev->device_lock); | 
					
						
							| 
									
										
										
										
											2014-02-17 15:13:19 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2013-11-11 13:26:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-11 03:03:30 +02:00
										 |  |  | 
 |