bus-fixup.c 6.8 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
/*
 *
 * 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>
S
Samuel Ortiz 已提交
18
#include <linux/sched.h>
19 20 21
#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/device.h>
22
#include <linux/slab.h>
23
#include <linux/uuid.h>
24

25 26 27 28 29
#include <linux/mei_cl_bus.h>

#include "mei_dev.h"
#include "client.h"

30 31 32
#define MEI_UUID_NFC_INFO UUID_LE(0xd2de1625, 0x382d, 0x417d, \
			0x48, 0xa4, 0xef, 0xab, 0xba, 0x8a, 0x12, 0x06)

33 34 35 36 37
static const uuid_le mei_nfc_info_guid = MEI_UUID_NFC_INFO;

#define MEI_UUID_NFC_HCI UUID_LE(0x0bb17a78, 0x2a8e, 0x4c50, \
			0x94, 0xd4, 0x50, 0x26, 0x67, 0x23, 0x77, 0x5c)

38 39
#define MEI_UUID_ANY NULL_UUID_LE

40 41 42 43 44 45 46 47 48 49 50
/**
 * number_of_connections - determine whether an client be on the bus
 *    according number of connections
 *    We support only clients:
 *       1. with single connection
 *       2. and fixed clients (max_number_of_connections == 0)
 *
 * @cldev: me clients device
 */
static void number_of_connections(struct mei_cl_device *cldev)
{
51
	dev_dbg(&cldev->dev, "running hook %s\n", __func__);
52 53 54 55 56

	if (cldev->me_cl->props.max_number_of_connections > 1)
		cldev->do_match = 0;
}

57 58 59 60 61 62 63
/**
 * blacklist - blacklist a client from the bus
 *
 * @cldev: me clients device
 */
static void blacklist(struct mei_cl_device *cldev)
{
64 65
	dev_dbg(&cldev->dev, "running hook %s\n", __func__);

66 67 68
	cldev->do_match = 0;
}

69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103
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;


#define MEI_NFC_CMD_MAINTENANCE 0x00
#define MEI_NFC_SUBCMD_IF_VERSION 0x01

104 105 106 107 108 109 110 111
/* 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

112 113 114 115 116 117 118 119 120 121
/**
 * mei_nfc_if_version - get NFC interface version
 *
 * @cl: host client (nfc info)
 * @ver: NFC interface version to be filled in
 *
 * Return: 0 on success; < 0 otherwise
 */
static int mei_nfc_if_version(struct mei_cl *cl,
			      struct mei_nfc_if_version *ver)
122
{
123
	struct mei_device *bus;
124 125 126 127 128
	struct mei_nfc_cmd cmd = {
		.command = MEI_NFC_CMD_MAINTENANCE,
		.data_size = 1,
		.sub_command = MEI_NFC_SUBCMD_IF_VERSION,
	};
129 130 131 132
	struct mei_nfc_reply *reply = NULL;
	size_t if_version_length;
	int bytes_recv, ret;

133
	bus = cl->dev;
134

135
	WARN_ON(mutex_is_locked(&bus->device_lock));
136

137
	ret = __mei_cl_send(cl, (u8 *)&cmd, sizeof(struct mei_nfc_cmd), 1);
138
	if (ret < 0) {
139
		dev_err(bus->dev, "Could not send IF version cmd\n");
140 141 142 143 144 145 146 147 148 149 150
		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;

151
	ret = 0;
152 153
	bytes_recv = __mei_cl_recv(cl, (u8 *)reply, if_version_length);
	if (bytes_recv < 0 || bytes_recv < sizeof(struct mei_nfc_reply)) {
154
		dev_err(bus->dev, "Could not read IF version\n");
155 156 157 158
		ret = -EIO;
		goto err;
	}

159
	memcpy(ver, reply->data, sizeof(struct mei_nfc_if_version));
160

161 162
	dev_info(bus->dev, "NFC MEI VERSION: IVN 0x%x Vendor ID 0x%x Type 0x%x\n",
		ver->fw_ivn, ver->vendor_id, ver->radio_type);
163 164 165 166 167 168

err:
	kfree(reply);
	return ret;
}

169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191
/**
 * mei_nfc_radio_name - derive nfc radio name from the interface version
 *
 * @ver: NFC radio version
 *
 * Return: radio name string
 */
static const char *mei_nfc_radio_name(struct mei_nfc_if_version *ver)
{

	if (ver->vendor_id == MEI_NFC_VENDOR_INSIDE) {
		if (ver->radio_type == MEI_NFC_VENDOR_INSIDE_UREAD)
			return "microread";
	}

	if (ver->vendor_id == MEI_NFC_VENDOR_NXP) {
		if (ver->radio_type == MEI_NFC_VENDOR_NXP_PN544)
			return "pn544";
	}

	return NULL;
}

192 193 194 195 196 197 198 199
/**
 * mei_nfc - The nfc fixup function. The function retrieves nfc radio
 *    name and set is as device attribute so we can load
 *    the proper device driver for it
 *
 * @cldev: me client device (nfc)
 */
static void mei_nfc(struct mei_cl_device *cldev)
200
{
201
	struct mei_device *bus;
202 203 204 205 206
	struct mei_cl *cl;
	struct mei_me_client *me_cl = NULL;
	struct mei_nfc_if_version ver;
	const char *radio_name = NULL;
	int ret;
207

208
	bus = cldev->bus;
209

210
	dev_dbg(&cldev->dev, "running hook %s\n", __func__);
211

212
	mutex_lock(&bus->device_lock);
213 214 215 216 217 218 219
	/* we need to connect to INFO GUID */
	cl = mei_cl_alloc_linked(bus, MEI_HOST_CLIENT_ID_ANY);
	if (IS_ERR(cl)) {
		ret = PTR_ERR(cl);
		cl = NULL;
		dev_err(bus->dev, "nfc hook alloc failed %d\n", ret);
		goto out;
220 221
	}

222 223 224 225 226
	me_cl = mei_me_cl_by_uuid(bus, &mei_nfc_info_guid);
	if (!me_cl) {
		ret = -ENOTTY;
		dev_err(bus->dev, "Cannot find nfc info %d\n", ret);
		goto out;
227 228
	}

229 230 231 232 233
	ret = mei_cl_connect(cl, me_cl, NULL);
	if (ret < 0) {
		dev_err(&cldev->dev, "Can't connect to the NFC INFO ME ret = %d\n",
			ret);
		goto out;
234 235
	}

236
	mutex_unlock(&bus->device_lock);
237

238 239 240
	ret = mei_nfc_if_version(cl, &ver);
	if (ret)
		goto disconnect;
241

242
	radio_name = mei_nfc_radio_name(&ver);
243

244 245 246 247 248
	if (!radio_name) {
		ret = -ENOENT;
		dev_err(&cldev->dev, "Can't get the NFC interface version ret = %d\n",
			ret);
		goto disconnect;
249 250
	}

251 252
	dev_dbg(bus->dev, "nfc radio %s\n", radio_name);
	strlcpy(cldev->name, radio_name, sizeof(cldev->name));
253

254
disconnect:
255
	mutex_lock(&bus->device_lock);
256 257
	if (mei_cl_disconnect(cl) < 0)
		dev_err(bus->dev, "Can't disconnect the NFC INFO ME\n");
258

259
	mei_cl_flush_queues(cl, NULL);
260

261 262 263 264 265
out:
	mei_cl_unlink(cl);
	mutex_unlock(&bus->device_lock);
	mei_me_cl_put(me_cl);
	kfree(cl);
266

267 268
	if (ret)
		cldev->do_match = 0;
269

270
	dev_dbg(bus->dev, "end of fixup match = %d\n", cldev->do_match);
271
}
T
Tomas Winkler 已提交
272

273 274 275 276 277 278
#define MEI_FIXUP(_uuid, _hook) { _uuid, _hook }

static struct mei_fixup {

	const uuid_le uuid;
	void (*hook)(struct mei_cl_device *cldev);
279
} mei_fixups[] = {
280
	MEI_FIXUP(MEI_UUID_ANY, number_of_connections),
281
	MEI_FIXUP(MEI_UUID_NFC_INFO, blacklist),
282
	MEI_FIXUP(MEI_UUID_NFC_HCI, mei_nfc),
283
};
284 285

/**
286
 * mei_cldev_fixup - run fixup handlers
287 288 289
 *
 * @cldev: me client device
 */
290
void mei_cl_bus_dev_fixup(struct mei_cl_device *cldev)
291 292 293 294 295 296 297 298 299 300 301 302 303
{
	struct mei_fixup *f;
	const uuid_le *uuid = mei_me_cl_uuid(cldev->me_cl);
	int i;

	for (i = 0; i < ARRAY_SIZE(mei_fixups); i++) {

		f = &mei_fixups[i];
		if (uuid_le_cmp(f->uuid, MEI_UUID_ANY) == 0 ||
		    uuid_le_cmp(f->uuid, *uuid) == 0)
			f->hook(cldev);
	}
}
304