xref: /openbmc/linux/drivers/platform/chrome/cros_ec_proto.c (revision 0db00e5d86dc793aab9722ad3728d99166eb7d96)
1b861297cSEnric Balletbo i Serra // SPDX-License-Identifier: GPL-2.0
2b861297cSEnric Balletbo i Serra // ChromeOS EC communication protocol helper functions
3b861297cSEnric Balletbo i Serra //
4b861297cSEnric Balletbo i Serra // Copyright (C) 2015 Google, Inc
5062476f2SJavier Martinez Canillas 
6062476f2SJavier Martinez Canillas #include <linux/delay.h>
7062476f2SJavier Martinez Canillas #include <linux/device.h>
8062476f2SJavier Martinez Canillas #include <linux/module.h>
9840d9f13SEnric Balletbo i Serra #include <linux/platform_data/cros_ec_commands.h>
10840d9f13SEnric Balletbo i Serra #include <linux/platform_data/cros_ec_proto.h>
11062476f2SJavier Martinez Canillas #include <linux/slab.h>
126f1d912bSVic Yang #include <asm/unaligned.h>
13062476f2SJavier Martinez Canillas 
1458a2109fSRaul E Rangel #include "cros_ec_trace.h"
1558a2109fSRaul E Rangel 
16062476f2SJavier Martinez Canillas #define EC_COMMAND_RETRIES	50
17062476f2SJavier Martinez Canillas 
180d080459SGuenter Roeck static const int cros_ec_error_map[] = {
190d080459SGuenter Roeck 	[EC_RES_INVALID_COMMAND] = -EOPNOTSUPP,
200d080459SGuenter Roeck 	[EC_RES_ERROR] = -EIO,
210d080459SGuenter Roeck 	[EC_RES_INVALID_PARAM] = -EINVAL,
220d080459SGuenter Roeck 	[EC_RES_ACCESS_DENIED] = -EACCES,
230d080459SGuenter Roeck 	[EC_RES_INVALID_RESPONSE] = -EPROTO,
240d080459SGuenter Roeck 	[EC_RES_INVALID_VERSION] = -ENOPROTOOPT,
250d080459SGuenter Roeck 	[EC_RES_INVALID_CHECKSUM] = -EBADMSG,
260d080459SGuenter Roeck 	[EC_RES_IN_PROGRESS] = -EINPROGRESS,
270d080459SGuenter Roeck 	[EC_RES_UNAVAILABLE] = -ENODATA,
280d080459SGuenter Roeck 	[EC_RES_TIMEOUT] = -ETIMEDOUT,
290d080459SGuenter Roeck 	[EC_RES_OVERFLOW] = -EOVERFLOW,
300d080459SGuenter Roeck 	[EC_RES_INVALID_HEADER] = -EBADR,
310d080459SGuenter Roeck 	[EC_RES_REQUEST_TRUNCATED] = -EBADR,
320d080459SGuenter Roeck 	[EC_RES_RESPONSE_TOO_BIG] = -EFBIG,
330d080459SGuenter Roeck 	[EC_RES_BUS_ERROR] = -EFAULT,
340d080459SGuenter Roeck 	[EC_RES_BUSY] = -EBUSY,
350d080459SGuenter Roeck 	[EC_RES_INVALID_HEADER_VERSION] = -EBADMSG,
360d080459SGuenter Roeck 	[EC_RES_INVALID_HEADER_CRC] = -EBADMSG,
370d080459SGuenter Roeck 	[EC_RES_INVALID_DATA_CRC] = -EBADMSG,
380d080459SGuenter Roeck 	[EC_RES_DUP_UNAVAILABLE] = -ENODATA,
390d080459SGuenter Roeck };
400d080459SGuenter Roeck 
cros_ec_map_error(uint32_t result)410d080459SGuenter Roeck static int cros_ec_map_error(uint32_t result)
420d080459SGuenter Roeck {
430d080459SGuenter Roeck 	int ret = 0;
440d080459SGuenter Roeck 
450d080459SGuenter Roeck 	if (result != EC_RES_SUCCESS) {
460d080459SGuenter Roeck 		if (result < ARRAY_SIZE(cros_ec_error_map) && cros_ec_error_map[result])
470d080459SGuenter Roeck 			ret = cros_ec_error_map[result];
480d080459SGuenter Roeck 		else
490d080459SGuenter Roeck 			ret = -EPROTO;
500d080459SGuenter Roeck 	}
510d080459SGuenter Roeck 
520d080459SGuenter Roeck 	return ret;
530d080459SGuenter Roeck }
540d080459SGuenter Roeck 
prepare_tx(struct cros_ec_device * ec_dev,struct cros_ec_command * msg)5523a34e3aSTzung-Bi Shih static int prepare_tx(struct cros_ec_device *ec_dev,
56062476f2SJavier Martinez Canillas 		      struct cros_ec_command *msg)
57062476f2SJavier Martinez Canillas {
582c7589afSStephen Barber 	struct ec_host_request *request;
592c7589afSStephen Barber 	u8 *out;
602c7589afSStephen Barber 	int i;
612c7589afSStephen Barber 	u8 csum = 0;
62062476f2SJavier Martinez Canillas 
63c2dcb1b0STzung-Bi Shih 	if (msg->outsize + sizeof(*request) > ec_dev->dout_size)
64c2dcb1b0STzung-Bi Shih 		return -EINVAL;
652c7589afSStephen Barber 
66062476f2SJavier Martinez Canillas 	out = ec_dev->dout;
672c7589afSStephen Barber 	request = (struct ec_host_request *)out;
682c7589afSStephen Barber 	request->struct_version = EC_HOST_REQUEST_VERSION;
692c7589afSStephen Barber 	request->checksum = 0;
702c7589afSStephen Barber 	request->command = msg->command;
712c7589afSStephen Barber 	request->command_version = msg->version;
722c7589afSStephen Barber 	request->reserved = 0;
732c7589afSStephen Barber 	request->data_len = msg->outsize;
742c7589afSStephen Barber 
752c7589afSStephen Barber 	for (i = 0; i < sizeof(*request); i++)
762c7589afSStephen Barber 		csum += out[i];
772c7589afSStephen Barber 
782c7589afSStephen Barber 	/* Copy data and update checksum */
792c7589afSStephen Barber 	memcpy(out + sizeof(*request), msg->data, msg->outsize);
80062476f2SJavier Martinez Canillas 	for (i = 0; i < msg->outsize; i++)
812c7589afSStephen Barber 		csum += msg->data[i];
82062476f2SJavier Martinez Canillas 
832c7589afSStephen Barber 	request->checksum = -csum;
84062476f2SJavier Martinez Canillas 
852c7589afSStephen Barber 	return sizeof(*request) + msg->outsize;
86062476f2SJavier Martinez Canillas }
87062476f2SJavier Martinez Canillas 
prepare_tx_legacy(struct cros_ec_device * ec_dev,struct cros_ec_command * msg)8823a34e3aSTzung-Bi Shih static int prepare_tx_legacy(struct cros_ec_device *ec_dev,
8923a34e3aSTzung-Bi Shih 			     struct cros_ec_command *msg)
9023a34e3aSTzung-Bi Shih {
9123a34e3aSTzung-Bi Shih 	u8 *out;
9223a34e3aSTzung-Bi Shih 	u8 csum;
9323a34e3aSTzung-Bi Shih 	int i;
9423a34e3aSTzung-Bi Shih 
9523a34e3aSTzung-Bi Shih 	if (msg->outsize > EC_PROTO2_MAX_PARAM_SIZE)
9623a34e3aSTzung-Bi Shih 		return -EINVAL;
9723a34e3aSTzung-Bi Shih 
9823a34e3aSTzung-Bi Shih 	out = ec_dev->dout;
9923a34e3aSTzung-Bi Shih 	out[0] = EC_CMD_VERSION0 + msg->version;
10023a34e3aSTzung-Bi Shih 	out[1] = msg->command;
10123a34e3aSTzung-Bi Shih 	out[2] = msg->outsize;
10223a34e3aSTzung-Bi Shih 	csum = out[0] + out[1] + out[2];
10323a34e3aSTzung-Bi Shih 	for (i = 0; i < msg->outsize; i++)
10423a34e3aSTzung-Bi Shih 		csum += out[EC_MSG_TX_HEADER_BYTES + i] = msg->data[i];
10523a34e3aSTzung-Bi Shih 	out[EC_MSG_TX_HEADER_BYTES + msg->outsize] = csum;
10623a34e3aSTzung-Bi Shih 
10723a34e3aSTzung-Bi Shih 	return EC_MSG_TX_PROTO_BYTES + msg->outsize;
10823a34e3aSTzung-Bi Shih }
10923a34e3aSTzung-Bi Shih 
cros_ec_xfer_command(struct cros_ec_device * ec_dev,struct cros_ec_command * msg)110810be30dSTzung-Bi Shih static int cros_ec_xfer_command(struct cros_ec_device *ec_dev, struct cros_ec_command *msg)
111062476f2SJavier Martinez Canillas {
112062476f2SJavier Martinez Canillas 	int ret;
113d48b8c58SShawn Nematbakhsh 	int (*xfer_fxn)(struct cros_ec_device *ec, struct cros_ec_command *msg);
114062476f2SJavier Martinez Canillas 
1152c7589afSStephen Barber 	if (ec_dev->proto_version > 2)
116d48b8c58SShawn Nematbakhsh 		xfer_fxn = ec_dev->pkt_xfer;
1172c7589afSStephen Barber 	else
118d48b8c58SShawn Nematbakhsh 		xfer_fxn = ec_dev->cmd_xfer;
1192c7589afSStephen Barber 
12094d4e7afSEnrico Granata 	if (!xfer_fxn) {
12194d4e7afSEnrico Granata 		/*
12294d4e7afSEnrico Granata 		 * This error can happen if a communication error happened and
12394d4e7afSEnrico Granata 		 * the EC is trying to use protocol v2, on an underlying
12494d4e7afSEnrico Granata 		 * communication mechanism that does not support v2.
12594d4e7afSEnrico Granata 		 */
126810be30dSTzung-Bi Shih 		dev_err_once(ec_dev->dev, "missing EC transfer API, cannot send command\n");
12794d4e7afSEnrico Granata 		return -EIO;
12894d4e7afSEnrico Granata 	}
12994d4e7afSEnrico Granata 
130205c9326SRaul E Rangel 	trace_cros_ec_request_start(msg);
131d48b8c58SShawn Nematbakhsh 	ret = (*xfer_fxn)(ec_dev, msg);
132205c9326SRaul E Rangel 	trace_cros_ec_request_done(msg, ret);
133810be30dSTzung-Bi Shih 
134810be30dSTzung-Bi Shih 	return ret;
135810be30dSTzung-Bi Shih }
136810be30dSTzung-Bi Shih 
cros_ec_wait_until_complete(struct cros_ec_device * ec_dev,uint32_t * result)1370aad9affSTzung-Bi Shih static int cros_ec_wait_until_complete(struct cros_ec_device *ec_dev, uint32_t *result)
1380aad9affSTzung-Bi Shih {
1390aad9affSTzung-Bi Shih 	struct {
1400aad9affSTzung-Bi Shih 		struct cros_ec_command msg;
1410aad9affSTzung-Bi Shih 		struct ec_response_get_comms_status status;
1420aad9affSTzung-Bi Shih 	} __packed buf;
1430aad9affSTzung-Bi Shih 	struct cros_ec_command *msg = &buf.msg;
1440aad9affSTzung-Bi Shih 	struct ec_response_get_comms_status *status = &buf.status;
1450aad9affSTzung-Bi Shih 	int ret = 0, i;
1460aad9affSTzung-Bi Shih 
1470aad9affSTzung-Bi Shih 	msg->version = 0;
1480aad9affSTzung-Bi Shih 	msg->command = EC_CMD_GET_COMMS_STATUS;
1490aad9affSTzung-Bi Shih 	msg->insize = sizeof(*status);
1500aad9affSTzung-Bi Shih 	msg->outsize = 0;
1510aad9affSTzung-Bi Shih 
1520aad9affSTzung-Bi Shih 	/* Query the EC's status until it's no longer busy or we encounter an error. */
1530aad9affSTzung-Bi Shih 	for (i = 0; i < EC_COMMAND_RETRIES; ++i) {
1540aad9affSTzung-Bi Shih 		usleep_range(10000, 11000);
1550aad9affSTzung-Bi Shih 
1560aad9affSTzung-Bi Shih 		ret = cros_ec_xfer_command(ec_dev, msg);
1570aad9affSTzung-Bi Shih 		if (ret == -EAGAIN)
1580aad9affSTzung-Bi Shih 			continue;
1590aad9affSTzung-Bi Shih 		if (ret < 0)
1600aad9affSTzung-Bi Shih 			return ret;
1610aad9affSTzung-Bi Shih 
1620aad9affSTzung-Bi Shih 		*result = msg->result;
1630aad9affSTzung-Bi Shih 		if (msg->result != EC_RES_SUCCESS)
1640aad9affSTzung-Bi Shih 			return ret;
1650aad9affSTzung-Bi Shih 
1663e1c715eSTzung-Bi Shih 		if (ret == 0) {
1673e1c715eSTzung-Bi Shih 			ret = -EPROTO;
1683e1c715eSTzung-Bi Shih 			break;
1693e1c715eSTzung-Bi Shih 		}
1703e1c715eSTzung-Bi Shih 
1710aad9affSTzung-Bi Shih 		if (!(status->flags & EC_COMMS_STATUS_PROCESSING))
1720aad9affSTzung-Bi Shih 			return ret;
1730aad9affSTzung-Bi Shih 	}
1740aad9affSTzung-Bi Shih 
1757f95d2b6STzung-Bi Shih 	if (i >= EC_COMMAND_RETRIES)
1767f95d2b6STzung-Bi Shih 		ret = -EAGAIN;
1777f95d2b6STzung-Bi Shih 
1780aad9affSTzung-Bi Shih 	return ret;
1790aad9affSTzung-Bi Shih }
1800aad9affSTzung-Bi Shih 
cros_ec_send_command(struct cros_ec_device * ec_dev,struct cros_ec_command * msg)181810be30dSTzung-Bi Shih static int cros_ec_send_command(struct cros_ec_device *ec_dev, struct cros_ec_command *msg)
182810be30dSTzung-Bi Shih {
183810be30dSTzung-Bi Shih 	int ret = cros_ec_xfer_command(ec_dev, msg);
184810be30dSTzung-Bi Shih 
1850aad9affSTzung-Bi Shih 	if (msg->result == EC_RES_IN_PROGRESS)
1860aad9affSTzung-Bi Shih 		ret = cros_ec_wait_until_complete(ec_dev, &msg->result);
1872c7589afSStephen Barber 
1882c7589afSStephen Barber 	return ret;
1892c7589afSStephen Barber }
1902c7589afSStephen Barber 
191c9b46568SGwendal Grignou /**
192c9b46568SGwendal Grignou  * cros_ec_prepare_tx() - Prepare an outgoing message in the output buffer.
193c9b46568SGwendal Grignou  * @ec_dev: Device to register.
194c9b46568SGwendal Grignou  * @msg: Message to write.
195c9b46568SGwendal Grignou  *
19623a34e3aSTzung-Bi Shih  * This is used by all ChromeOS EC drivers to prepare the outgoing message
19723a34e3aSTzung-Bi Shih  * according to different protocol versions.
198c9b46568SGwendal Grignou  *
19971d3ae7fSTzung-Bi Shih  * Return: number of prepared bytes on success or negative error code.
200c9b46568SGwendal Grignou  */
cros_ec_prepare_tx(struct cros_ec_device * ec_dev,struct cros_ec_command * msg)2012c7589afSStephen Barber int cros_ec_prepare_tx(struct cros_ec_device *ec_dev,
2022c7589afSStephen Barber 		       struct cros_ec_command *msg)
2032c7589afSStephen Barber {
2042c7589afSStephen Barber 	if (ec_dev->proto_version > 2)
20523a34e3aSTzung-Bi Shih 		return prepare_tx(ec_dev, msg);
2062c7589afSStephen Barber 
20723a34e3aSTzung-Bi Shih 	return prepare_tx_legacy(ec_dev, msg);
2082c7589afSStephen Barber }
2092c7589afSStephen Barber EXPORT_SYMBOL(cros_ec_prepare_tx);
2102c7589afSStephen Barber 
211c9b46568SGwendal Grignou /**
212c9b46568SGwendal Grignou  * cros_ec_check_result() - Check ec_msg->result.
213c9b46568SGwendal Grignou  * @ec_dev: EC device.
214c9b46568SGwendal Grignou  * @msg: Message to check.
215c9b46568SGwendal Grignou  *
216c9b46568SGwendal Grignou  * This is used by ChromeOS EC drivers to check the ec_msg->result for
21797b11dd6STzung-Bi Shih  * EC_RES_IN_PROGRESS and to warn about them.
218c9b46568SGwendal Grignou  *
21997b11dd6STzung-Bi Shih  * The function should not check for furthermore error codes.  Otherwise,
22097b11dd6STzung-Bi Shih  * it would break the ABI.
22197b11dd6STzung-Bi Shih  *
22297b11dd6STzung-Bi Shih  * Return: -EAGAIN if ec_msg->result == EC_RES_IN_PROGRESS.  Otherwise, 0.
223c9b46568SGwendal Grignou  */
cros_ec_check_result(struct cros_ec_device * ec_dev,struct cros_ec_command * msg)2242c7589afSStephen Barber int cros_ec_check_result(struct cros_ec_device *ec_dev,
2252c7589afSStephen Barber 			 struct cros_ec_command *msg)
2262c7589afSStephen Barber {
2272c7589afSStephen Barber 	switch (msg->result) {
2282c7589afSStephen Barber 	case EC_RES_SUCCESS:
2292c7589afSStephen Barber 		return 0;
2302c7589afSStephen Barber 	case EC_RES_IN_PROGRESS:
2312c7589afSStephen Barber 		dev_dbg(ec_dev->dev, "command 0x%02x in progress\n",
2322c7589afSStephen Barber 			msg->command);
2332c7589afSStephen Barber 		return -EAGAIN;
2342c7589afSStephen Barber 	default:
2352c7589afSStephen Barber 		dev_dbg(ec_dev->dev, "command 0x%02x returned %d\n",
2362c7589afSStephen Barber 			msg->command, msg->result);
2372c7589afSStephen Barber 		return 0;
2382c7589afSStephen Barber 	}
2392c7589afSStephen Barber }
2402c7589afSStephen Barber EXPORT_SYMBOL(cros_ec_check_result);
2412c7589afSStephen Barber 
24229d99b96SShawn Nematbakhsh /*
24329d99b96SShawn Nematbakhsh  * cros_ec_get_host_event_wake_mask
24429d99b96SShawn Nematbakhsh  *
24529d99b96SShawn Nematbakhsh  * Get the mask of host events that cause wake from suspend.
24629d99b96SShawn Nematbakhsh  *
24729d99b96SShawn Nematbakhsh  * @ec_dev: EC device to call
24829d99b96SShawn Nematbakhsh  * @msg: message structure to use
249d65da5f9STzung-Bi Shih  * @mask: result when function returns 0.
25029d99b96SShawn Nematbakhsh  *
25129d99b96SShawn Nematbakhsh  * LOCKING:
25229d99b96SShawn Nematbakhsh  * the caller has ec_dev->lock mutex, or the caller knows there is
25329d99b96SShawn Nematbakhsh  * no other command in progress.
25429d99b96SShawn Nematbakhsh  */
cros_ec_get_host_event_wake_mask(struct cros_ec_device * ec_dev,uint32_t * mask)255b4d0836eSTzung-Bi Shih static int cros_ec_get_host_event_wake_mask(struct cros_ec_device *ec_dev, uint32_t *mask)
25629d99b96SShawn Nematbakhsh {
257b4d0836eSTzung-Bi Shih 	struct cros_ec_command *msg;
25829d99b96SShawn Nematbakhsh 	struct ec_response_host_event_mask *r;
25993bea2faSTzung-Bi Shih 	int ret, mapped;
26029d99b96SShawn Nematbakhsh 
261b4d0836eSTzung-Bi Shih 	msg = kzalloc(sizeof(*msg) + sizeof(*r), GFP_KERNEL);
262b4d0836eSTzung-Bi Shih 	if (!msg)
263b4d0836eSTzung-Bi Shih 		return -ENOMEM;
264b4d0836eSTzung-Bi Shih 
26529d99b96SShawn Nematbakhsh 	msg->command = EC_CMD_HOST_EVENT_GET_WAKE_MASK;
26629d99b96SShawn Nematbakhsh 	msg->insize = sizeof(*r);
26729d99b96SShawn Nematbakhsh 
268d311664bSTzung-Bi Shih 	ret = cros_ec_send_command(ec_dev, msg);
269cfed691bSTzung-Bi Shih 	if (ret < 0)
270cfed691bSTzung-Bi Shih 		goto exit;
271cfed691bSTzung-Bi Shih 
27293bea2faSTzung-Bi Shih 	mapped = cros_ec_map_error(msg->result);
273b4d0836eSTzung-Bi Shih 	if (mapped) {
274b4d0836eSTzung-Bi Shih 		ret = mapped;
275b4d0836eSTzung-Bi Shih 		goto exit;
276b4d0836eSTzung-Bi Shih 	}
277cfed691bSTzung-Bi Shih 
278cfed691bSTzung-Bi Shih 	if (ret == 0) {
279cfed691bSTzung-Bi Shih 		ret = -EPROTO;
280cfed691bSTzung-Bi Shih 		goto exit;
281fc8cacf3SBrian Norris 	}
282cfed691bSTzung-Bi Shih 
28329d99b96SShawn Nematbakhsh 	r = (struct ec_response_host_event_mask *)msg->data;
28429d99b96SShawn Nematbakhsh 	*mask = r->mask;
285d65da5f9STzung-Bi Shih 	ret = 0;
286b4d0836eSTzung-Bi Shih exit:
287b4d0836eSTzung-Bi Shih 	kfree(msg);
28829d99b96SShawn Nematbakhsh 	return ret;
28929d99b96SShawn Nematbakhsh }
29029d99b96SShawn Nematbakhsh 
cros_ec_get_proto_info(struct cros_ec_device * ec_dev,int devidx)291b4d0836eSTzung-Bi Shih static int cros_ec_get_proto_info(struct cros_ec_device *ec_dev, int devidx)
2922c7589afSStephen Barber {
293b4d0836eSTzung-Bi Shih 	struct cros_ec_command *msg;
294b4d0836eSTzung-Bi Shih 	struct ec_response_get_protocol_info *info;
295b4d0836eSTzung-Bi Shih 	int ret, mapped;
2962c7589afSStephen Barber 
297b4d0836eSTzung-Bi Shih 	ec_dev->proto_version = 3;
298b4d0836eSTzung-Bi Shih 	if (devidx > 0)
299b4d0836eSTzung-Bi Shih 		ec_dev->max_passthru = 0;
300b4d0836eSTzung-Bi Shih 
301b4d0836eSTzung-Bi Shih 	msg = kzalloc(sizeof(*msg) + sizeof(*info), GFP_KERNEL);
302b4d0836eSTzung-Bi Shih 	if (!msg)
303b4d0836eSTzung-Bi Shih 		return -ENOMEM;
304b4d0836eSTzung-Bi Shih 
3052c7589afSStephen Barber 	msg->command = EC_CMD_PASSTHRU_OFFSET(devidx) | EC_CMD_GET_PROTOCOL_INFO;
306b4d0836eSTzung-Bi Shih 	msg->insize = sizeof(*info);
3072c7589afSStephen Barber 
308d311664bSTzung-Bi Shih 	ret = cros_ec_send_command(ec_dev, msg);
3093abc16afSPatryk Duda 	/*
3103abc16afSPatryk Duda 	 * Send command once again when timeout occurred.
3113abc16afSPatryk Duda 	 * Fingerprint MCU (FPMCU) is restarted during system boot which
3123abc16afSPatryk Duda 	 * introduces small window in which FPMCU won't respond for any
3133abc16afSPatryk Duda 	 * messages sent by kernel. There is no need to wait before next
3143abc16afSPatryk Duda 	 * attempt because we waited at least EC_MSG_DEADLINE_MS.
3153abc16afSPatryk Duda 	 */
3163abc16afSPatryk Duda 	if (ret == -ETIMEDOUT)
317d311664bSTzung-Bi Shih 		ret = cros_ec_send_command(ec_dev, msg);
3182c7589afSStephen Barber 
3192c7589afSStephen Barber 	if (ret < 0) {
3202c7589afSStephen Barber 		dev_dbg(ec_dev->dev,
3212c7589afSStephen Barber 			"failed to check for EC[%d] protocol version: %d\n",
3222c7589afSStephen Barber 			devidx, ret);
323b4d0836eSTzung-Bi Shih 		goto exit;
3242c7589afSStephen Barber 	}
3252c7589afSStephen Barber 
326b4d0836eSTzung-Bi Shih 	mapped = cros_ec_map_error(msg->result);
327b4d0836eSTzung-Bi Shih 	if (mapped) {
328b4d0836eSTzung-Bi Shih 		ret = mapped;
329b4d0836eSTzung-Bi Shih 		goto exit;
330b4d0836eSTzung-Bi Shih 	}
3312c7589afSStephen Barber 
332878c36f6STzung-Bi Shih 	if (ret == 0) {
333878c36f6STzung-Bi Shih 		ret = -EPROTO;
334878c36f6STzung-Bi Shih 		goto exit;
335878c36f6STzung-Bi Shih 	}
336878c36f6STzung-Bi Shih 
337b4d0836eSTzung-Bi Shih 	info = (struct ec_response_get_protocol_info *)msg->data;
338b4d0836eSTzung-Bi Shih 
339b4d0836eSTzung-Bi Shih 	switch (devidx) {
340b4d0836eSTzung-Bi Shih 	case CROS_EC_DEV_EC_INDEX:
341b4d0836eSTzung-Bi Shih 		ec_dev->max_request = info->max_request_packet_size -
342b4d0836eSTzung-Bi Shih 						sizeof(struct ec_host_request);
343b4d0836eSTzung-Bi Shih 		ec_dev->max_response = info->max_response_packet_size -
344b4d0836eSTzung-Bi Shih 						sizeof(struct ec_host_response);
345b4d0836eSTzung-Bi Shih 		ec_dev->proto_version = min(EC_HOST_REQUEST_VERSION,
346b4d0836eSTzung-Bi Shih 					    fls(info->protocol_versions) - 1);
347b4d0836eSTzung-Bi Shih 		ec_dev->din_size = info->max_response_packet_size + EC_MAX_RESPONSE_OVERHEAD;
348b4d0836eSTzung-Bi Shih 		ec_dev->dout_size = info->max_request_packet_size + EC_MAX_REQUEST_OVERHEAD;
349b4d0836eSTzung-Bi Shih 
350b4d0836eSTzung-Bi Shih 		dev_dbg(ec_dev->dev, "using proto v%u\n", ec_dev->proto_version);
351b4d0836eSTzung-Bi Shih 		break;
352b4d0836eSTzung-Bi Shih 	case CROS_EC_DEV_PD_INDEX:
353b4d0836eSTzung-Bi Shih 		ec_dev->max_passthru = info->max_request_packet_size -
354b4d0836eSTzung-Bi Shih 						sizeof(struct ec_host_request);
355b4d0836eSTzung-Bi Shih 
356b4d0836eSTzung-Bi Shih 		dev_dbg(ec_dev->dev, "found PD chip\n");
357b4d0836eSTzung-Bi Shih 		break;
358b4d0836eSTzung-Bi Shih 	default:
359203b2affSColin Ian King 		dev_dbg(ec_dev->dev, "unknown passthru index: %d\n", devidx);
360b4d0836eSTzung-Bi Shih 		break;
361b4d0836eSTzung-Bi Shih 	}
362b4d0836eSTzung-Bi Shih 
363b4d0836eSTzung-Bi Shih 	ret = 0;
364b4d0836eSTzung-Bi Shih exit:
365b4d0836eSTzung-Bi Shih 	kfree(msg);
366b4d0836eSTzung-Bi Shih 	return ret;
3672c7589afSStephen Barber }
3682c7589afSStephen Barber 
cros_ec_get_proto_info_legacy(struct cros_ec_device * ec_dev)369a88f7966STzung-Bi Shih static int cros_ec_get_proto_info_legacy(struct cros_ec_device *ec_dev)
3702c7589afSStephen Barber {
3712c7589afSStephen Barber 	struct cros_ec_command *msg;
372a88f7966STzung-Bi Shih 	struct ec_params_hello *params;
373a88f7966STzung-Bi Shih 	struct ec_response_hello *response;
374d394ab5cSTzung-Bi Shih 	int ret, mapped;
3752c7589afSStephen Barber 
376a88f7966STzung-Bi Shih 	ec_dev->proto_version = 2;
377a88f7966STzung-Bi Shih 
378a88f7966STzung-Bi Shih 	msg = kzalloc(sizeof(*msg) + max(sizeof(*params), sizeof(*response)), GFP_KERNEL);
3792c7589afSStephen Barber 	if (!msg)
3802c7589afSStephen Barber 		return -ENOMEM;
3812c7589afSStephen Barber 
3822c7589afSStephen Barber 	msg->command = EC_CMD_HELLO;
383a88f7966STzung-Bi Shih 	msg->insize = sizeof(*response);
384a88f7966STzung-Bi Shih 	msg->outsize = sizeof(*params);
3852c7589afSStephen Barber 
386a88f7966STzung-Bi Shih 	params = (struct ec_params_hello *)msg->data;
387a88f7966STzung-Bi Shih 	params->in_data = 0xa0b0c0d0;
3882c7589afSStephen Barber 
389d311664bSTzung-Bi Shih 	ret = cros_ec_send_command(ec_dev, msg);
3902c7589afSStephen Barber 	if (ret < 0) {
391a88f7966STzung-Bi Shih 		dev_dbg(ec_dev->dev, "EC failed to respond to v2 hello: %d\n", ret);
3922c7589afSStephen Barber 		goto exit;
393a88f7966STzung-Bi Shih 	}
394a88f7966STzung-Bi Shih 
395d394ab5cSTzung-Bi Shih 	mapped = cros_ec_map_error(msg->result);
396d394ab5cSTzung-Bi Shih 	if (mapped) {
397d394ab5cSTzung-Bi Shih 		ret = mapped;
398a88f7966STzung-Bi Shih 		dev_err(ec_dev->dev, "EC responded to v2 hello with error: %d\n", msg->result);
3992c7589afSStephen Barber 		goto exit;
400a88f7966STzung-Bi Shih 	}
401a88f7966STzung-Bi Shih 
402d394ab5cSTzung-Bi Shih 	if (ret == 0) {
403d394ab5cSTzung-Bi Shih 		ret = -EPROTO;
404d394ab5cSTzung-Bi Shih 		goto exit;
405d394ab5cSTzung-Bi Shih 	}
406d394ab5cSTzung-Bi Shih 
407a88f7966STzung-Bi Shih 	response = (struct ec_response_hello *)msg->data;
408a88f7966STzung-Bi Shih 	if (response->out_data != 0xa1b2c3d4) {
4092c7589afSStephen Barber 		dev_err(ec_dev->dev,
4102c7589afSStephen Barber 			"EC responded to v2 hello with bad result: %u\n",
411a88f7966STzung-Bi Shih 			response->out_data);
4122c7589afSStephen Barber 		ret = -EBADMSG;
4132c7589afSStephen Barber 		goto exit;
4142c7589afSStephen Barber 	}
4152c7589afSStephen Barber 
416a88f7966STzung-Bi Shih 	ec_dev->max_request = EC_PROTO2_MAX_PARAM_SIZE;
417a88f7966STzung-Bi Shih 	ec_dev->max_response = EC_PROTO2_MAX_PARAM_SIZE;
418a88f7966STzung-Bi Shih 	ec_dev->max_passthru = 0;
419a88f7966STzung-Bi Shih 	ec_dev->pkt_xfer = NULL;
420a88f7966STzung-Bi Shih 	ec_dev->din_size = EC_PROTO2_MSG_BYTES;
421a88f7966STzung-Bi Shih 	ec_dev->dout_size = EC_PROTO2_MSG_BYTES;
4222c7589afSStephen Barber 
423a88f7966STzung-Bi Shih 	dev_dbg(ec_dev->dev, "falling back to proto v2\n");
424a88f7966STzung-Bi Shih 	ret = 0;
425062476f2SJavier Martinez Canillas exit:
4262c7589afSStephen Barber 	kfree(msg);
4272c7589afSStephen Barber 	return ret;
4282c7589afSStephen Barber }
4292c7589afSStephen Barber 
430d4da97e5SGwendal Grignou /*
431d4da97e5SGwendal Grignou  * cros_ec_get_host_command_version_mask
432d4da97e5SGwendal Grignou  *
433d4da97e5SGwendal Grignou  * Get the version mask of a given command.
434d4da97e5SGwendal Grignou  *
435d4da97e5SGwendal Grignou  * @ec_dev: EC device to call
436d4da97e5SGwendal Grignou  * @msg: message structure to use
437d4da97e5SGwendal Grignou  * @cmd: command to get the version of.
438d4da97e5SGwendal Grignou  * @mask: result when function returns 0.
439d4da97e5SGwendal Grignou  *
440d4da97e5SGwendal Grignou  * @return 0 on success, error code otherwise
441d4da97e5SGwendal Grignou  *
442d4da97e5SGwendal Grignou  * LOCKING:
443d4da97e5SGwendal Grignou  * the caller has ec_dev->lock mutex or the caller knows there is
444d4da97e5SGwendal Grignou  * no other command in progress.
445d4da97e5SGwendal Grignou  */
cros_ec_get_host_command_version_mask(struct cros_ec_device * ec_dev,u16 cmd,u32 * mask)446ec513489STzung-Bi Shih static int cros_ec_get_host_command_version_mask(struct cros_ec_device *ec_dev, u16 cmd, u32 *mask)
4476f1d912bSVic Yang {
4486f1d912bSVic Yang 	struct ec_params_get_cmd_versions *pver;
4496f1d912bSVic Yang 	struct ec_response_get_cmd_versions *rver;
4506f1d912bSVic Yang 	struct cros_ec_command *msg;
451ec513489STzung-Bi Shih 	int ret, mapped;
4526f1d912bSVic Yang 
4536f1d912bSVic Yang 	msg = kmalloc(sizeof(*msg) + max(sizeof(*rver), sizeof(*pver)),
4546f1d912bSVic Yang 		      GFP_KERNEL);
4556f1d912bSVic Yang 	if (!msg)
4566f1d912bSVic Yang 		return -ENOMEM;
4576f1d912bSVic Yang 
4586f1d912bSVic Yang 	msg->version = 0;
4596f1d912bSVic Yang 	msg->command = EC_CMD_GET_CMD_VERSIONS;
4606f1d912bSVic Yang 	msg->insize = sizeof(*rver);
4616f1d912bSVic Yang 	msg->outsize = sizeof(*pver);
4626f1d912bSVic Yang 
4636f1d912bSVic Yang 	pver = (struct ec_params_get_cmd_versions *)msg->data;
4646f1d912bSVic Yang 	pver->cmd = cmd;
4656f1d912bSVic Yang 
466d311664bSTzung-Bi Shih 	ret = cros_ec_send_command(ec_dev, msg);
467ec513489STzung-Bi Shih 	if (ret < 0)
468ec513489STzung-Bi Shih 		goto exit;
469ec513489STzung-Bi Shih 
470ec513489STzung-Bi Shih 	mapped = cros_ec_map_error(msg->result);
471ec513489STzung-Bi Shih 	if (mapped) {
472ec513489STzung-Bi Shih 		ret = mapped;
473ec513489STzung-Bi Shih 		goto exit;
474ec513489STzung-Bi Shih 	}
475ec513489STzung-Bi Shih 
476aac29b04STzung-Bi Shih 	if (ret == 0) {
477aac29b04STzung-Bi Shih 		ret = -EPROTO;
478aac29b04STzung-Bi Shih 		goto exit;
479aac29b04STzung-Bi Shih 	}
480aac29b04STzung-Bi Shih 
4816f1d912bSVic Yang 	rver = (struct ec_response_get_cmd_versions *)msg->data;
4826f1d912bSVic Yang 	*mask = rver->version_mask;
483f91183aaSTzung-Bi Shih 	ret = 0;
484ec513489STzung-Bi Shih exit:
4856f1d912bSVic Yang 	kfree(msg);
4866f1d912bSVic Yang 	return ret;
4876f1d912bSVic Yang }
4886f1d912bSVic Yang 
489c9b46568SGwendal Grignou /**
490c9b46568SGwendal Grignou  * cros_ec_query_all() -  Query the protocol version supported by the
491c9b46568SGwendal Grignou  *         ChromeOS EC.
492c9b46568SGwendal Grignou  * @ec_dev: Device to register.
493c9b46568SGwendal Grignou  *
494c9b46568SGwendal Grignou  * Return: 0 on success or negative error code.
495c9b46568SGwendal Grignou  */
cros_ec_query_all(struct cros_ec_device * ec_dev)4962c7589afSStephen Barber int cros_ec_query_all(struct cros_ec_device *ec_dev)
4972c7589afSStephen Barber {
4982c7589afSStephen Barber 	struct device *dev = ec_dev->dev;
499f91183aaSTzung-Bi Shih 	u32 ver_mask;
5002c7589afSStephen Barber 	int ret;
5012c7589afSStephen Barber 
5022c7589afSStephen Barber 	/* First try sending with proto v3. */
503b4d0836eSTzung-Bi Shih 	if (!cros_ec_get_proto_info(ec_dev, CROS_EC_DEV_EC_INDEX)) {
504b4d0836eSTzung-Bi Shih 		/* Check for PD. */
505b4d0836eSTzung-Bi Shih 		cros_ec_get_proto_info(ec_dev, CROS_EC_DEV_PD_INDEX);
5062c7589afSStephen Barber 	} else {
5072c7589afSStephen Barber 		/* Try querying with a v2 hello message. */
508a88f7966STzung-Bi Shih 		ret = cros_ec_get_proto_info_legacy(ec_dev);
509a88f7966STzung-Bi Shih 		if (ret) {
5102c7589afSStephen Barber 			/*
5112c7589afSStephen Barber 			 * It's possible for a test to occur too early when
5122c7589afSStephen Barber 			 * the EC isn't listening. If this happens, we'll
5132c7589afSStephen Barber 			 * test later when the first command is run.
5142c7589afSStephen Barber 			 */
5152c7589afSStephen Barber 			ec_dev->proto_version = EC_PROTO_VERSION_UNKNOWN;
5162c7589afSStephen Barber 			dev_dbg(ec_dev->dev, "EC query failed: %d\n", ret);
517a88f7966STzung-Bi Shih 			return ret;
5182c7589afSStephen Barber 		}
5192c7589afSStephen Barber 	}
5202c7589afSStephen Barber 
5212c7589afSStephen Barber 	devm_kfree(dev, ec_dev->din);
5222c7589afSStephen Barber 	devm_kfree(dev, ec_dev->dout);
5232c7589afSStephen Barber 
5242c7589afSStephen Barber 	ec_dev->din = devm_kzalloc(dev, ec_dev->din_size, GFP_KERNEL);
5252c7589afSStephen Barber 	if (!ec_dev->din) {
5262c7589afSStephen Barber 		ret = -ENOMEM;
5272c7589afSStephen Barber 		goto exit;
5282c7589afSStephen Barber 	}
5292c7589afSStephen Barber 
5302c7589afSStephen Barber 	ec_dev->dout = devm_kzalloc(dev, ec_dev->dout_size, GFP_KERNEL);
5312c7589afSStephen Barber 	if (!ec_dev->dout) {
5322c7589afSStephen Barber 		devm_kfree(dev, ec_dev->din);
5332c7589afSStephen Barber 		ret = -ENOMEM;
5342c7589afSStephen Barber 		goto exit;
5352c7589afSStephen Barber 	}
5362c7589afSStephen Barber 
5376f1d912bSVic Yang 	/* Probe if MKBP event is supported */
538f91183aaSTzung-Bi Shih 	ret = cros_ec_get_host_command_version_mask(ec_dev, EC_CMD_GET_NEXT_EVENT, &ver_mask);
539b36f0643STzung-Bi Shih 	if (ret < 0 || ver_mask == 0) {
5406f1d912bSVic Yang 		ec_dev->mkbp_event_supported = 0;
541b36f0643STzung-Bi Shih 	} else {
5423300fdd6SEnrico Granata 		ec_dev->mkbp_event_supported = fls(ver_mask);
5433300fdd6SEnrico Granata 
544b36f0643STzung-Bi Shih 		dev_dbg(ec_dev->dev, "MKBP support version %u\n", ec_dev->mkbp_event_supported - 1);
545b36f0643STzung-Bi Shih 	}
5466f1d912bSVic Yang 
5477235560aSEvan Green 	/* Probe if host sleep v1 is supported for S0ix failure detection. */
548f91183aaSTzung-Bi Shih 	ret = cros_ec_get_host_command_version_mask(ec_dev, EC_CMD_HOST_SLEEP_EVENT, &ver_mask);
549f91183aaSTzung-Bi Shih 	ec_dev->host_sleep_v1 = (ret == 0 && (ver_mask & EC_VER_MASK(1)));
5507235560aSEvan Green 
551c214e564SBrian Norris 	/* Get host event wake mask. */
552b4d0836eSTzung-Bi Shih 	ret = cros_ec_get_host_event_wake_mask(ec_dev, &ec_dev->host_event_wake_mask);
553c214e564SBrian Norris 	if (ret < 0) {
554c214e564SBrian Norris 		/*
555c214e564SBrian Norris 		 * If the EC doesn't support EC_CMD_HOST_EVENT_GET_WAKE_MASK,
556c214e564SBrian Norris 		 * use a reasonable default. Note that we ignore various
557c214e564SBrian Norris 		 * battery, AC status, and power-state events, because (a)
558c214e564SBrian Norris 		 * those can be quite common (e.g., when sitting at full
559c214e564SBrian Norris 		 * charge, on AC) and (b) these are not actionable wake events;
560c214e564SBrian Norris 		 * if anything, we'd like to continue suspending (to save
561c214e564SBrian Norris 		 * power), not wake up.
562c214e564SBrian Norris 		 */
563c214e564SBrian Norris 		ec_dev->host_event_wake_mask = U32_MAX &
564852405d8SEvan Benn 			~(EC_HOST_EVENT_MASK(EC_HOST_EVENT_LID_CLOSED) |
565852405d8SEvan Benn 			  EC_HOST_EVENT_MASK(EC_HOST_EVENT_AC_DISCONNECTED) |
5660944ea07SEvan Benn 			  EC_HOST_EVENT_MASK(EC_HOST_EVENT_BATTERY_LOW) |
5670944ea07SEvan Benn 			  EC_HOST_EVENT_MASK(EC_HOST_EVENT_BATTERY_CRITICAL) |
568852405d8SEvan Benn 			  EC_HOST_EVENT_MASK(EC_HOST_EVENT_BATTERY) |
5690944ea07SEvan Benn 			  EC_HOST_EVENT_MASK(EC_HOST_EVENT_PD_MCU) |
5700944ea07SEvan Benn 			  EC_HOST_EVENT_MASK(EC_HOST_EVENT_BATTERY_STATUS));
571fc8cacf3SBrian Norris 		/*
572fc8cacf3SBrian Norris 		 * Old ECs may not support this command. Complain about all
573fc8cacf3SBrian Norris 		 * other errors.
574fc8cacf3SBrian Norris 		 */
575fc8cacf3SBrian Norris 		if (ret != -EOPNOTSUPP)
576fc8cacf3SBrian Norris 			dev_err(ec_dev->dev,
577fc8cacf3SBrian Norris 				"failed to retrieve wake mask: %d\n", ret);
578c214e564SBrian Norris 	}
57929d99b96SShawn Nematbakhsh 
580a27b8f31SThierry Escande 	ret = 0;
581a27b8f31SThierry Escande 
5822c7589afSStephen Barber exit:
5832c7589afSStephen Barber 	return ret;
5842c7589afSStephen Barber }
5852c7589afSStephen Barber EXPORT_SYMBOL(cros_ec_query_all);
5862c7589afSStephen Barber 
587c9b46568SGwendal Grignou /**
58857b888caSGuenter Roeck  * cros_ec_cmd_xfer() - Send a command to the ChromeOS EC.
589c9b46568SGwendal Grignou  * @ec_dev: EC device.
590c9b46568SGwendal Grignou  * @msg: Message to write.
591c9b46568SGwendal Grignou  *
59257b888caSGuenter Roeck  * Call this to send a command to the ChromeOS EC. This should be used instead
59357b888caSGuenter Roeck  * of calling the EC's cmd_xfer() callback directly. This function does not
59457b888caSGuenter Roeck  * convert EC command execution error codes to Linux error codes. Most
59557b888caSGuenter Roeck  * in-kernel users will want to use cros_ec_cmd_xfer_status() instead since
59657b888caSGuenter Roeck  * that function implements the conversion.
597c9b46568SGwendal Grignou  *
5986b194ee9SPrashant Malani  * Return:
59957b888caSGuenter Roeck  * >0 - EC command was executed successfully. The return value is the number
60057b888caSGuenter Roeck  *      of bytes returned by the EC (excluding the header).
60157b888caSGuenter Roeck  * =0 - EC communication was successful. EC command execution results are
60257b888caSGuenter Roeck  *      reported in msg->result. The result will be EC_RES_SUCCESS if the
60357b888caSGuenter Roeck  *      command was executed successfully or report an EC command execution
60457b888caSGuenter Roeck  *      error.
60557b888caSGuenter Roeck  * <0 - EC communication error. Return value is the Linux error code.
606c9b46568SGwendal Grignou  */
cros_ec_cmd_xfer(struct cros_ec_device * ec_dev,struct cros_ec_command * msg)60757b888caSGuenter Roeck int cros_ec_cmd_xfer(struct cros_ec_device *ec_dev, struct cros_ec_command *msg)
6082c7589afSStephen Barber {
60957b888caSGuenter Roeck 	int ret;
6102c7589afSStephen Barber 
6112c7589afSStephen Barber 	mutex_lock(&ec_dev->lock);
6122c7589afSStephen Barber 	if (ec_dev->proto_version == EC_PROTO_VERSION_UNKNOWN) {
6132c7589afSStephen Barber 		ret = cros_ec_query_all(ec_dev);
6142c7589afSStephen Barber 		if (ret) {
6152c7589afSStephen Barber 			dev_err(ec_dev->dev,
6162c7589afSStephen Barber 				"EC version unknown and query failed; aborting command\n");
6172c7589afSStephen Barber 			mutex_unlock(&ec_dev->lock);
6182c7589afSStephen Barber 			return ret;
6192c7589afSStephen Barber 		}
6202c7589afSStephen Barber 	}
6212c7589afSStephen Barber 
6222c7589afSStephen Barber 	if (msg->insize > ec_dev->max_response) {
6232c7589afSStephen Barber 		dev_dbg(ec_dev->dev, "clamping message receive buffer\n");
6242c7589afSStephen Barber 		msg->insize = ec_dev->max_response;
6252c7589afSStephen Barber 	}
6262c7589afSStephen Barber 
6273db0c9e5STzung-Bi Shih 	if (msg->command < EC_CMD_PASSTHRU_OFFSET(CROS_EC_DEV_PD_INDEX)) {
6282c7589afSStephen Barber 		if (msg->outsize > ec_dev->max_request) {
6292c7589afSStephen Barber 			dev_err(ec_dev->dev,
6302c7589afSStephen Barber 				"request of size %u is too big (max: %u)\n",
6312c7589afSStephen Barber 				msg->outsize,
6322c7589afSStephen Barber 				ec_dev->max_request);
6332c7589afSStephen Barber 			mutex_unlock(&ec_dev->lock);
6342c7589afSStephen Barber 			return -EMSGSIZE;
6352c7589afSStephen Barber 		}
6362c7589afSStephen Barber 	} else {
6372c7589afSStephen Barber 		if (msg->outsize > ec_dev->max_passthru) {
6382c7589afSStephen Barber 			dev_err(ec_dev->dev,
6392c7589afSStephen Barber 				"passthru rq of size %u is too big (max: %u)\n",
6402c7589afSStephen Barber 				msg->outsize,
6412c7589afSStephen Barber 				ec_dev->max_passthru);
6422c7589afSStephen Barber 			mutex_unlock(&ec_dev->lock);
6432c7589afSStephen Barber 			return -EMSGSIZE;
6442c7589afSStephen Barber 		}
6452c7589afSStephen Barber 	}
6466b194ee9SPrashant Malani 
647d311664bSTzung-Bi Shih 	ret = cros_ec_send_command(ec_dev, msg);
648062476f2SJavier Martinez Canillas 	mutex_unlock(&ec_dev->lock);
649062476f2SJavier Martinez Canillas 
65057b888caSGuenter Roeck 	return ret;
65157b888caSGuenter Roeck }
65257b888caSGuenter Roeck EXPORT_SYMBOL(cros_ec_cmd_xfer);
65357b888caSGuenter Roeck 
65457b888caSGuenter Roeck /**
65557b888caSGuenter Roeck  * cros_ec_cmd_xfer_status() - Send a command to the ChromeOS EC.
65657b888caSGuenter Roeck  * @ec_dev: EC device.
65757b888caSGuenter Roeck  * @msg: Message to write.
65857b888caSGuenter Roeck  *
65957b888caSGuenter Roeck  * Call this to send a command to the ChromeOS EC. This should be used instead of calling the EC's
66057b888caSGuenter Roeck  * cmd_xfer() callback directly. It returns success status only if both the command was transmitted
66157b888caSGuenter Roeck  * successfully and the EC replied with success status.
66257b888caSGuenter Roeck  *
66357b888caSGuenter Roeck  * Return:
66457b888caSGuenter Roeck  * >=0 - The number of bytes transferred.
66557b888caSGuenter Roeck  * <0 - Linux error code
66657b888caSGuenter Roeck  */
cros_ec_cmd_xfer_status(struct cros_ec_device * ec_dev,struct cros_ec_command * msg)66757b888caSGuenter Roeck int cros_ec_cmd_xfer_status(struct cros_ec_device *ec_dev,
66857b888caSGuenter Roeck 			    struct cros_ec_command *msg)
66957b888caSGuenter Roeck {
67057b888caSGuenter Roeck 	int ret, mapped;
67157b888caSGuenter Roeck 
67257b888caSGuenter Roeck 	ret = cros_ec_cmd_xfer(ec_dev, msg);
67357b888caSGuenter Roeck 	if (ret < 0)
67457b888caSGuenter Roeck 		return ret;
67557b888caSGuenter Roeck 
6760d080459SGuenter Roeck 	mapped = cros_ec_map_error(msg->result);
6770d080459SGuenter Roeck 	if (mapped) {
6780d080459SGuenter Roeck 		dev_dbg(ec_dev->dev, "Command result (err: %d [%d])\n",
6790d080459SGuenter Roeck 			msg->result, mapped);
6800d080459SGuenter Roeck 		ret = mapped;
6810d080459SGuenter Roeck 	}
6826b194ee9SPrashant Malani 
6839798ac6dSTomeu Vizoso 	return ret;
6849798ac6dSTomeu Vizoso }
6859798ac6dSTomeu Vizoso EXPORT_SYMBOL(cros_ec_cmd_xfer_status);
6866f1d912bSVic Yang 
get_next_event_xfer(struct cros_ec_device * ec_dev,struct cros_ec_command * msg,struct ec_response_get_next_event_v1 * event,int version,uint32_t size)68757e94c8bSNeil Armstrong static int get_next_event_xfer(struct cros_ec_device *ec_dev,
68857e94c8bSNeil Armstrong 			       struct cros_ec_command *msg,
6893300fdd6SEnrico Granata 			       struct ec_response_get_next_event_v1 *event,
69057e94c8bSNeil Armstrong 			       int version, uint32_t size)
69157e94c8bSNeil Armstrong {
69257e94c8bSNeil Armstrong 	int ret;
69357e94c8bSNeil Armstrong 
69457e94c8bSNeil Armstrong 	msg->version = version;
69557e94c8bSNeil Armstrong 	msg->command = EC_CMD_GET_NEXT_EVENT;
69657e94c8bSNeil Armstrong 	msg->insize = size;
69757e94c8bSNeil Armstrong 	msg->outsize = 0;
69857e94c8bSNeil Armstrong 
69964b02e54SPrashant Malani 	ret = cros_ec_cmd_xfer_status(ec_dev, msg);
70057e94c8bSNeil Armstrong 	if (ret > 0) {
70157e94c8bSNeil Armstrong 		ec_dev->event_size = ret - 1;
7023300fdd6SEnrico Granata 		ec_dev->event_data = *event;
70357e94c8bSNeil Armstrong 	}
70457e94c8bSNeil Armstrong 
70557e94c8bSNeil Armstrong 	return ret;
70657e94c8bSNeil Armstrong }
70757e94c8bSNeil Armstrong 
get_next_event(struct cros_ec_device * ec_dev)7086f1d912bSVic Yang static int get_next_event(struct cros_ec_device *ec_dev)
7096f1d912bSVic Yang {
7103300fdd6SEnrico Granata 	struct {
7113300fdd6SEnrico Granata 		struct cros_ec_command msg;
7123300fdd6SEnrico Granata 		struct ec_response_get_next_event_v1 event;
7133300fdd6SEnrico Granata 	} __packed buf;
7143300fdd6SEnrico Granata 	struct cros_ec_command *msg = &buf.msg;
7153300fdd6SEnrico Granata 	struct ec_response_get_next_event_v1 *event = &buf.event;
7163300fdd6SEnrico Granata 	const int cmd_version = ec_dev->mkbp_event_supported - 1;
7176f1d912bSVic Yang 
7183300fdd6SEnrico Granata 	memset(msg, 0, sizeof(*msg));
719a9eb186eSJoseph Lo 	if (ec_dev->suspended) {
720a9eb186eSJoseph Lo 		dev_dbg(ec_dev->dev, "Device suspended.\n");
721a9eb186eSJoseph Lo 		return -EHOSTDOWN;
722a9eb186eSJoseph Lo 	}
723a9eb186eSJoseph Lo 
7243300fdd6SEnrico Granata 	if (cmd_version == 0)
7253300fdd6SEnrico Granata 		return get_next_event_xfer(ec_dev, msg, event, 0,
72657e94c8bSNeil Armstrong 				  sizeof(struct ec_response_get_next_event));
72757e94c8bSNeil Armstrong 
7283300fdd6SEnrico Granata 	return get_next_event_xfer(ec_dev, msg, event, cmd_version,
7293300fdd6SEnrico Granata 				sizeof(struct ec_response_get_next_event_v1));
7306f1d912bSVic Yang }
7316f1d912bSVic Yang 
get_keyboard_state_event(struct cros_ec_device * ec_dev)7326f1d912bSVic Yang static int get_keyboard_state_event(struct cros_ec_device *ec_dev)
7336f1d912bSVic Yang {
7346f1d912bSVic Yang 	u8 buffer[sizeof(struct cros_ec_command) +
7356f1d912bSVic Yang 		  sizeof(ec_dev->event_data.data)];
7366f1d912bSVic Yang 	struct cros_ec_command *msg = (struct cros_ec_command *)&buffer;
7376f1d912bSVic Yang 
7386f1d912bSVic Yang 	msg->version = 0;
7396f1d912bSVic Yang 	msg->command = EC_CMD_MKBP_STATE;
7406f1d912bSVic Yang 	msg->insize = sizeof(ec_dev->event_data.data);
7416f1d912bSVic Yang 	msg->outsize = 0;
7426f1d912bSVic Yang 
74364b02e54SPrashant Malani 	ec_dev->event_size = cros_ec_cmd_xfer_status(ec_dev, msg);
7446f1d912bSVic Yang 	ec_dev->event_data.event_type = EC_MKBP_EVENT_KEY_MATRIX;
7456f1d912bSVic Yang 	memcpy(&ec_dev->event_data.data, msg->data,
7466f1d912bSVic Yang 	       sizeof(ec_dev->event_data.data));
7476f1d912bSVic Yang 
7486f1d912bSVic Yang 	return ec_dev->event_size;
7496f1d912bSVic Yang }
7506f1d912bSVic Yang 
751c9b46568SGwendal Grignou /**
752c9b46568SGwendal Grignou  * cros_ec_get_next_event() - Fetch next event from the ChromeOS EC.
753c9b46568SGwendal Grignou  * @ec_dev: Device to fetch event from.
754c9b46568SGwendal Grignou  * @wake_event: Pointer to a bool set to true upon return if the event might be
755c9b46568SGwendal Grignou  *              treated as a wake event. Ignored if null.
7563300fdd6SEnrico Granata  * @has_more_events: Pointer to bool set to true if more than one event is
7573300fdd6SEnrico Granata  *              pending.
7583300fdd6SEnrico Granata  *              Some EC will set this flag to indicate cros_ec_get_next_event()
7593300fdd6SEnrico Granata  *              can be called multiple times in a row.
7603300fdd6SEnrico Granata  *              It is an optimization to prevent issuing a EC command for
7613300fdd6SEnrico Granata  *              nothing or wait for another interrupt from the EC to process
7623300fdd6SEnrico Granata  *              the next message.
7633300fdd6SEnrico Granata  *              Ignored if null.
764c9b46568SGwendal Grignou  *
765c9b46568SGwendal Grignou  * Return: negative error code on errors; 0 for no data; or else number of
766c9b46568SGwendal Grignou  * bytes received (i.e., an event was retrieved successfully). Event types are
767c9b46568SGwendal Grignou  * written out to @ec_dev->event_data.event_type on success.
768c9b46568SGwendal Grignou  */
cros_ec_get_next_event(struct cros_ec_device * ec_dev,bool * wake_event,bool * has_more_events)7693300fdd6SEnrico Granata int cros_ec_get_next_event(struct cros_ec_device *ec_dev,
7703300fdd6SEnrico Granata 			   bool *wake_event,
7713300fdd6SEnrico Granata 			   bool *has_more_events)
7726f1d912bSVic Yang {
7736ad16b78SBrian Norris 	u8 event_type;
77429d99b96SShawn Nematbakhsh 	u32 host_event;
77529d99b96SShawn Nematbakhsh 	int ret;
776f74c7557SPatryk Duda 	u32 ver_mask;
77729d99b96SShawn Nematbakhsh 
7783300fdd6SEnrico Granata 	/*
7793300fdd6SEnrico Granata 	 * Default value for wake_event.
7803300fdd6SEnrico Granata 	 * Wake up on keyboard event, wake up for spurious interrupt or link
7813300fdd6SEnrico Granata 	 * error to the EC.
7823300fdd6SEnrico Granata 	 */
78329d99b96SShawn Nematbakhsh 	if (wake_event)
78429d99b96SShawn Nematbakhsh 		*wake_event = true;
78529d99b96SShawn Nematbakhsh 
7863300fdd6SEnrico Granata 	/*
7873300fdd6SEnrico Granata 	 * Default value for has_more_events.
7883300fdd6SEnrico Granata 	 * EC will raise another interrupt if AP does not process all events
7893300fdd6SEnrico Granata 	 * anyway.
7903300fdd6SEnrico Granata 	 */
7913300fdd6SEnrico Granata 	if (has_more_events)
7923300fdd6SEnrico Granata 		*has_more_events = false;
7933300fdd6SEnrico Granata 
7943300fdd6SEnrico Granata 	if (!ec_dev->mkbp_event_supported)
7953300fdd6SEnrico Granata 		return get_keyboard_state_event(ec_dev);
79629d99b96SShawn Nematbakhsh 
79729d99b96SShawn Nematbakhsh 	ret = get_next_event(ec_dev);
798f74c7557SPatryk Duda 	/*
799f74c7557SPatryk Duda 	 * -ENOPROTOOPT is returned when EC returns EC_RES_INVALID_VERSION.
800f74c7557SPatryk Duda 	 * This can occur when EC based device (e.g. Fingerprint MCU) jumps to
801f74c7557SPatryk Duda 	 * the RO image which doesn't support newer version of the command. In
802f74c7557SPatryk Duda 	 * this case we will attempt to update maximum supported version of the
803f74c7557SPatryk Duda 	 * EC_CMD_GET_NEXT_EVENT.
804f74c7557SPatryk Duda 	 */
805f74c7557SPatryk Duda 	if (ret == -ENOPROTOOPT) {
806f74c7557SPatryk Duda 		dev_dbg(ec_dev->dev,
807f74c7557SPatryk Duda 			"GET_NEXT_EVENT returned invalid version error.\n");
808*b8774d31SPatryk Duda 		mutex_lock(&ec_dev->lock);
809f74c7557SPatryk Duda 		ret = cros_ec_get_host_command_version_mask(ec_dev,
810f74c7557SPatryk Duda 							EC_CMD_GET_NEXT_EVENT,
811f74c7557SPatryk Duda 							&ver_mask);
812*b8774d31SPatryk Duda 		mutex_unlock(&ec_dev->lock);
813f74c7557SPatryk Duda 		if (ret < 0 || ver_mask == 0)
814f74c7557SPatryk Duda 			/*
815f74c7557SPatryk Duda 			 * Do not change the MKBP supported version if we can't
816f74c7557SPatryk Duda 			 * obtain supported version correctly. Please note that
817f74c7557SPatryk Duda 			 * calling EC_CMD_GET_NEXT_EVENT returned
818f74c7557SPatryk Duda 			 * EC_RES_INVALID_VERSION which means that the command
819f74c7557SPatryk Duda 			 * is present.
820f74c7557SPatryk Duda 			 */
821f74c7557SPatryk Duda 			return -ENOPROTOOPT;
822f74c7557SPatryk Duda 
823f74c7557SPatryk Duda 		ec_dev->mkbp_event_supported = fls(ver_mask);
824f74c7557SPatryk Duda 		dev_dbg(ec_dev->dev, "MKBP support version changed to %u\n",
825f74c7557SPatryk Duda 			ec_dev->mkbp_event_supported - 1);
826f74c7557SPatryk Duda 
827f74c7557SPatryk Duda 		/* Try to get next event with new MKBP support version set. */
828f74c7557SPatryk Duda 		ret = get_next_event(ec_dev);
829f74c7557SPatryk Duda 	}
830f74c7557SPatryk Duda 
831475b0873SBrian Norris 	if (ret <= 0)
83229d99b96SShawn Nematbakhsh 		return ret;
83329d99b96SShawn Nematbakhsh 
8343300fdd6SEnrico Granata 	if (has_more_events)
8353300fdd6SEnrico Granata 		*has_more_events = ec_dev->event_data.event_type &
8363300fdd6SEnrico Granata 			EC_MKBP_HAS_MORE_EVENTS;
8373300fdd6SEnrico Granata 	ec_dev->event_data.event_type &= EC_MKBP_EVENT_TYPE_MASK;
8383300fdd6SEnrico Granata 
83929d99b96SShawn Nematbakhsh 	if (wake_event) {
8406ad16b78SBrian Norris 		event_type = ec_dev->event_data.event_type;
84129d99b96SShawn Nematbakhsh 		host_event = cros_ec_get_host_event(ec_dev);
84229d99b96SShawn Nematbakhsh 
8436ad16b78SBrian Norris 		/*
8446ad16b78SBrian Norris 		 * Sensor events need to be parsed by the sensor sub-device.
8456ad16b78SBrian Norris 		 * Defer them, and don't report the wakeup here.
8466ad16b78SBrian Norris 		 */
847853c1a78SStephen Boyd 		if (event_type == EC_MKBP_EVENT_SENSOR_FIFO) {
848853c1a78SStephen Boyd 			*wake_event = false;
849853c1a78SStephen Boyd 		} else if (host_event) {
850853c1a78SStephen Boyd 			/* rtc_update_irq() already handles wakeup events. */
851853c1a78SStephen Boyd 			if (host_event & EC_HOST_EVENT_MASK(EC_HOST_EVENT_RTC))
8526ad16b78SBrian Norris 				*wake_event = false;
8536ad16b78SBrian Norris 			/* Masked host-events should not count as wake events. */
854853c1a78SStephen Boyd 			if (!(host_event & ec_dev->host_event_wake_mask))
8556ad16b78SBrian Norris 				*wake_event = false;
85629d99b96SShawn Nematbakhsh 		}
857853c1a78SStephen Boyd 	}
85829d99b96SShawn Nematbakhsh 
85929d99b96SShawn Nematbakhsh 	return ret;
8606f1d912bSVic Yang }
8616f1d912bSVic Yang EXPORT_SYMBOL(cros_ec_get_next_event);
86268c35ea2SGwendal Grignou 
863c9b46568SGwendal Grignou /**
864c9b46568SGwendal Grignou  * cros_ec_get_host_event() - Return a mask of event set by the ChromeOS EC.
865c9b46568SGwendal Grignou  * @ec_dev: Device to fetch event from.
866c9b46568SGwendal Grignou  *
867c9b46568SGwendal Grignou  * When MKBP is supported, when the EC raises an interrupt, we collect the
868c9b46568SGwendal Grignou  * events raised and call the functions in the ec notifier. This function
869c9b46568SGwendal Grignou  * is a helper to know which events are raised.
870c9b46568SGwendal Grignou  *
871c9b46568SGwendal Grignou  * Return: 0 on error or non-zero bitmask of one or more EC_HOST_EVENT_*.
872c9b46568SGwendal Grignou  */
cros_ec_get_host_event(struct cros_ec_device * ec_dev)87368c35ea2SGwendal Grignou u32 cros_ec_get_host_event(struct cros_ec_device *ec_dev)
87468c35ea2SGwendal Grignou {
87568c35ea2SGwendal Grignou 	u32 host_event;
87668c35ea2SGwendal Grignou 
87720a264c9STzung-Bi Shih 	if (!ec_dev->mkbp_event_supported)
87820a264c9STzung-Bi Shih 		return 0;
87968c35ea2SGwendal Grignou 
88068c35ea2SGwendal Grignou 	if (ec_dev->event_data.event_type != EC_MKBP_EVENT_HOST_EVENT)
88168c35ea2SGwendal Grignou 		return 0;
88268c35ea2SGwendal Grignou 
88368c35ea2SGwendal Grignou 	if (ec_dev->event_size != sizeof(host_event)) {
88468c35ea2SGwendal Grignou 		dev_warn(ec_dev->dev, "Invalid host event size\n");
88568c35ea2SGwendal Grignou 		return 0;
88668c35ea2SGwendal Grignou 	}
88768c35ea2SGwendal Grignou 
88868c35ea2SGwendal Grignou 	host_event = get_unaligned_le32(&ec_dev->event_data.data.host_event);
88968c35ea2SGwendal Grignou 
89068c35ea2SGwendal Grignou 	return host_event;
89168c35ea2SGwendal Grignou }
89268c35ea2SGwendal Grignou EXPORT_SYMBOL(cros_ec_get_host_event);
893a16b2e28SGwendal Grignou 
894a16b2e28SGwendal Grignou /**
895a16b2e28SGwendal Grignou  * cros_ec_check_features() - Test for the presence of EC features
896a16b2e28SGwendal Grignou  *
897a16b2e28SGwendal Grignou  * @ec: EC device, does not have to be connected directly to the AP,
898a16b2e28SGwendal Grignou  *      can be daisy chained through another device.
899a16b2e28SGwendal Grignou  * @feature: One of ec_feature_code bit.
900a16b2e28SGwendal Grignou  *
901a16b2e28SGwendal Grignou  * Call this function to test whether the ChromeOS EC supports a feature.
902a16b2e28SGwendal Grignou  *
903d50497c4SPrashant Malani  * Return: true if supported, false if not (or if an error was encountered).
904a16b2e28SGwendal Grignou  */
cros_ec_check_features(struct cros_ec_dev * ec,int feature)905d50497c4SPrashant Malani bool cros_ec_check_features(struct cros_ec_dev *ec, int feature)
906a16b2e28SGwendal Grignou {
9077ff22787SPrashant Malani 	struct ec_response_get_features *features = &ec->features;
908a16b2e28SGwendal Grignou 	int ret;
909a16b2e28SGwendal Grignou 
9107ff22787SPrashant Malani 	if (features->flags[0] == -1U && features->flags[1] == -1U) {
911a16b2e28SGwendal Grignou 		/* features bitmap not read yet */
912b1d288d9SPrashant Malani 		ret = cros_ec_cmd(ec->ec_dev, 0, EC_CMD_GET_FEATURES + ec->cmd_offset,
913297d34e7SPrashant Malani 				  NULL, 0, features, sizeof(*features));
914a16b2e28SGwendal Grignou 		if (ret < 0) {
915297d34e7SPrashant Malani 			dev_warn(ec->dev, "cannot get EC features: %d\n", ret);
9167ff22787SPrashant Malani 			memset(features, 0, sizeof(*features));
917a16b2e28SGwendal Grignou 		}
918a16b2e28SGwendal Grignou 
919a16b2e28SGwendal Grignou 		dev_dbg(ec->dev, "EC features %08x %08x\n",
9207ff22787SPrashant Malani 			features->flags[0], features->flags[1]);
921a16b2e28SGwendal Grignou 	}
922a16b2e28SGwendal Grignou 
9237ff22787SPrashant Malani 	return !!(features->flags[feature / 32] & EC_FEATURE_MASK_0(feature));
924a16b2e28SGwendal Grignou }
925a16b2e28SGwendal Grignou EXPORT_SYMBOL_GPL(cros_ec_check_features);
926a16b2e28SGwendal Grignou 
927a16b2e28SGwendal Grignou /**
928a16b2e28SGwendal Grignou  * cros_ec_get_sensor_count() - Return the number of MEMS sensors supported.
929a16b2e28SGwendal Grignou  *
930a16b2e28SGwendal Grignou  * @ec: EC device, does not have to be connected directly to the AP,
931a16b2e28SGwendal Grignou  *      can be daisy chained through another device.
932a16b2e28SGwendal Grignou  * Return: < 0 in case of error.
933a16b2e28SGwendal Grignou  */
cros_ec_get_sensor_count(struct cros_ec_dev * ec)934a16b2e28SGwendal Grignou int cros_ec_get_sensor_count(struct cros_ec_dev *ec)
935a16b2e28SGwendal Grignou {
936a16b2e28SGwendal Grignou 	/*
937a16b2e28SGwendal Grignou 	 * Issue a command to get the number of sensor reported.
938a16b2e28SGwendal Grignou 	 * If not supported, check for legacy mode.
939a16b2e28SGwendal Grignou 	 */
940a16b2e28SGwendal Grignou 	int ret, sensor_count;
941a16b2e28SGwendal Grignou 	struct ec_params_motion_sense *params;
942a16b2e28SGwendal Grignou 	struct ec_response_motion_sense *resp;
943a16b2e28SGwendal Grignou 	struct cros_ec_command *msg;
944a16b2e28SGwendal Grignou 	struct cros_ec_device *ec_dev = ec->ec_dev;
945a16b2e28SGwendal Grignou 	u8 status;
946a16b2e28SGwendal Grignou 
947a16b2e28SGwendal Grignou 	msg = kzalloc(sizeof(*msg) + max(sizeof(*params), sizeof(*resp)),
948a16b2e28SGwendal Grignou 		      GFP_KERNEL);
949a16b2e28SGwendal Grignou 	if (!msg)
950a16b2e28SGwendal Grignou 		return -ENOMEM;
951a16b2e28SGwendal Grignou 
952a16b2e28SGwendal Grignou 	msg->version = 1;
953a16b2e28SGwendal Grignou 	msg->command = EC_CMD_MOTION_SENSE_CMD + ec->cmd_offset;
954a16b2e28SGwendal Grignou 	msg->outsize = sizeof(*params);
955a16b2e28SGwendal Grignou 	msg->insize = sizeof(*resp);
956a16b2e28SGwendal Grignou 
957a16b2e28SGwendal Grignou 	params = (struct ec_params_motion_sense *)msg->data;
958a16b2e28SGwendal Grignou 	params->cmd = MOTIONSENSE_CMD_DUMP;
959a16b2e28SGwendal Grignou 
96064b02e54SPrashant Malani 	ret = cros_ec_cmd_xfer_status(ec->ec_dev, msg);
961a16b2e28SGwendal Grignou 	if (ret < 0) {
962a16b2e28SGwendal Grignou 		sensor_count = ret;
963a16b2e28SGwendal Grignou 	} else {
964a16b2e28SGwendal Grignou 		resp = (struct ec_response_motion_sense *)msg->data;
965a16b2e28SGwendal Grignou 		sensor_count = resp->dump.sensor_count;
966a16b2e28SGwendal Grignou 	}
967a16b2e28SGwendal Grignou 	kfree(msg);
968a16b2e28SGwendal Grignou 
969a16b2e28SGwendal Grignou 	/*
970a16b2e28SGwendal Grignou 	 * Check legacy mode: Let's find out if sensors are accessible
971a16b2e28SGwendal Grignou 	 * via LPC interface.
972a16b2e28SGwendal Grignou 	 */
97364b02e54SPrashant Malani 	if (sensor_count < 0 && ec->cmd_offset == 0 && ec_dev->cmd_readmem) {
974a16b2e28SGwendal Grignou 		ret = ec_dev->cmd_readmem(ec_dev, EC_MEMMAP_ACC_STATUS,
975a16b2e28SGwendal Grignou 				1, &status);
976a16b2e28SGwendal Grignou 		if (ret >= 0 &&
977a16b2e28SGwendal Grignou 		    (status & EC_MEMMAP_ACC_STATUS_PRESENCE_BIT)) {
978a16b2e28SGwendal Grignou 			/*
979a16b2e28SGwendal Grignou 			 * We have 2 sensors, one in the lid, one in the base.
980a16b2e28SGwendal Grignou 			 */
981a16b2e28SGwendal Grignou 			sensor_count = 2;
982a16b2e28SGwendal Grignou 		} else {
983a16b2e28SGwendal Grignou 			/*
984a16b2e28SGwendal Grignou 			 * EC uses LPC interface and no sensors are presented.
985a16b2e28SGwendal Grignou 			 */
986a16b2e28SGwendal Grignou 			sensor_count = 0;
987a16b2e28SGwendal Grignou 		}
988a16b2e28SGwendal Grignou 	}
989a16b2e28SGwendal Grignou 	return sensor_count;
990a16b2e28SGwendal Grignou }
991a16b2e28SGwendal Grignou EXPORT_SYMBOL_GPL(cros_ec_get_sensor_count);
9927101c839SPrashant Malani 
9937101c839SPrashant Malani /**
994b1d288d9SPrashant Malani  * cros_ec_cmd - Send a command to the EC.
9957101c839SPrashant Malani  *
9967101c839SPrashant Malani  * @ec_dev: EC device
9974f140639SPrashant Malani  * @version: EC command version
9987101c839SPrashant Malani  * @command: EC command
9997101c839SPrashant Malani  * @outdata: EC command output data
10007101c839SPrashant Malani  * @outsize: Size of outdata
10017101c839SPrashant Malani  * @indata: EC command input data
10027101c839SPrashant Malani  * @insize: Size of indata
10037101c839SPrashant Malani  *
10047101c839SPrashant Malani  * Return: >= 0 on success, negative error number on failure.
10057101c839SPrashant Malani  */
cros_ec_cmd(struct cros_ec_device * ec_dev,unsigned int version,int command,void * outdata,size_t outsize,void * indata,size_t insize)1006b1d288d9SPrashant Malani int cros_ec_cmd(struct cros_ec_device *ec_dev,
10074f140639SPrashant Malani 		unsigned int version,
10087101c839SPrashant Malani 		int command,
10095d122256SPrashant Malani 		void *outdata,
1010f87e15fbSPrashant Malani 		size_t outsize,
10115d122256SPrashant Malani 		void *indata,
1012f87e15fbSPrashant Malani 		size_t insize)
10137101c839SPrashant Malani {
10147101c839SPrashant Malani 	struct cros_ec_command *msg;
10157101c839SPrashant Malani 	int ret;
10167101c839SPrashant Malani 
10177101c839SPrashant Malani 	msg = kzalloc(sizeof(*msg) + max(insize, outsize), GFP_KERNEL);
10187101c839SPrashant Malani 	if (!msg)
10197101c839SPrashant Malani 		return -ENOMEM;
10207101c839SPrashant Malani 
10214f140639SPrashant Malani 	msg->version = version;
10227101c839SPrashant Malani 	msg->command = command;
10237101c839SPrashant Malani 	msg->outsize = outsize;
10247101c839SPrashant Malani 	msg->insize = insize;
10257101c839SPrashant Malani 
10267101c839SPrashant Malani 	if (outsize)
10277101c839SPrashant Malani 		memcpy(msg->data, outdata, outsize);
10287101c839SPrashant Malani 
10297101c839SPrashant Malani 	ret = cros_ec_cmd_xfer_status(ec_dev, msg);
10307101c839SPrashant Malani 	if (ret < 0)
10317101c839SPrashant Malani 		goto error;
10327101c839SPrashant Malani 
10337101c839SPrashant Malani 	if (insize)
10347101c839SPrashant Malani 		memcpy(indata, msg->data, insize);
10357101c839SPrashant Malani error:
10367101c839SPrashant Malani 	kfree(msg);
10377101c839SPrashant Malani 	return ret;
10387101c839SPrashant Malani }
1039b1d288d9SPrashant Malani EXPORT_SYMBOL_GPL(cros_ec_cmd);
1040