832 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			832 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /*
 | |
|  * Management-Controller-to-Driver Interface
 | |
|  *
 | |
|  * Copyright 2008-2013 Solarflare Communications Inc.
 | |
|  * Copyright (C) 2022-2023, Advanced Micro Devices, Inc.
 | |
|  */
 | |
| #include <linux/delay.h>
 | |
| #include <linux/slab.h>
 | |
| #include <linux/io.h>
 | |
| #include <linux/spinlock.h>
 | |
| #include <linux/netdevice.h>
 | |
| #include <linux/etherdevice.h>
 | |
| #include <linux/ethtool.h>
 | |
| #include <linux/if_vlan.h>
 | |
| #include <linux/timer.h>
 | |
| #include <linux/list.h>
 | |
| #include <linux/pci.h>
 | |
| #include <linux/device.h>
 | |
| #include <linux/rwsem.h>
 | |
| #include <linux/vmalloc.h>
 | |
| #include <net/netevent.h>
 | |
| #include <linux/log2.h>
 | |
| #include <linux/net_tstamp.h>
 | |
| #include <linux/wait.h>
 | |
| 
 | |
| #include "bitfield.h"
 | |
| #include "mcdi.h"
 | |
| 
 | |
| static void cdx_mcdi_cancel_cmd(struct cdx_mcdi *cdx, struct cdx_mcdi_cmd *cmd);
 | |
| static void cdx_mcdi_wait_for_cleanup(struct cdx_mcdi *cdx);
 | |
| static int cdx_mcdi_rpc_async_internal(struct cdx_mcdi *cdx,
 | |
| 				       struct cdx_mcdi_cmd *cmd,
 | |
| 				       unsigned int *handle);
 | |
| static void cdx_mcdi_start_or_queue(struct cdx_mcdi_iface *mcdi,
 | |
| 				    bool allow_retry);
 | |
| static void cdx_mcdi_cmd_start_or_queue(struct cdx_mcdi_iface *mcdi,
 | |
| 					struct cdx_mcdi_cmd *cmd);
 | |
| static bool cdx_mcdi_complete_cmd(struct cdx_mcdi_iface *mcdi,
 | |
| 				  struct cdx_mcdi_cmd *cmd,
 | |
| 				  struct cdx_dword *outbuf,
 | |
| 				  int len,
 | |
| 				  struct list_head *cleanup_list);
 | |
| static void cdx_mcdi_timeout_cmd(struct cdx_mcdi_iface *mcdi,
 | |
| 				 struct cdx_mcdi_cmd *cmd,
 | |
| 				 struct list_head *cleanup_list);
 | |
| static void cdx_mcdi_cmd_work(struct work_struct *context);
 | |
| static void cdx_mcdi_mode_fail(struct cdx_mcdi *cdx, struct list_head *cleanup_list);
 | |
| static void _cdx_mcdi_display_error(struct cdx_mcdi *cdx, unsigned int cmd,
 | |
| 				    size_t inlen, int raw, int arg, int err_no);
 | |
| 
 | |
| static bool cdx_cmd_cancelled(struct cdx_mcdi_cmd *cmd)
 | |
| {
 | |
| 	return cmd->state == MCDI_STATE_RUNNING_CANCELLED;
 | |
| }
 | |
| 
 | |
| static void cdx_mcdi_cmd_release(struct kref *ref)
 | |
| {
 | |
| 	kfree(container_of(ref, struct cdx_mcdi_cmd, ref));
 | |
| }
 | |
| 
 | |
| static unsigned int cdx_mcdi_cmd_handle(struct cdx_mcdi_cmd *cmd)
 | |
| {
 | |
| 	return cmd->handle;
 | |
| }
 | |
| 
 | |
| static void _cdx_mcdi_remove_cmd(struct cdx_mcdi_iface *mcdi,
 | |
| 				 struct cdx_mcdi_cmd *cmd,
 | |
| 				 struct list_head *cleanup_list)
 | |
| {
 | |
| 	/* if cancelled, the completers have already been called */
 | |
| 	if (cdx_cmd_cancelled(cmd))
 | |
| 		return;
 | |
| 
 | |
| 	if (cmd->completer) {
 | |
| 		list_add_tail(&cmd->cleanup_list, cleanup_list);
 | |
| 		++mcdi->outstanding_cleanups;
 | |
| 		kref_get(&cmd->ref);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void cdx_mcdi_remove_cmd(struct cdx_mcdi_iface *mcdi,
 | |
| 				struct cdx_mcdi_cmd *cmd,
 | |
| 				struct list_head *cleanup_list)
 | |
| {
 | |
| 	list_del(&cmd->list);
 | |
| 	_cdx_mcdi_remove_cmd(mcdi, cmd, cleanup_list);
 | |
| 	cmd->state = MCDI_STATE_FINISHED;
 | |
| 	kref_put(&cmd->ref, cdx_mcdi_cmd_release);
 | |
| 	if (list_empty(&mcdi->cmd_list))
 | |
| 		wake_up(&mcdi->cmd_complete_wq);
 | |
| }
 | |
| 
 | |
| static unsigned long cdx_mcdi_rpc_timeout(struct cdx_mcdi *cdx, unsigned int cmd)
 | |
| {
 | |
| 	if (!cdx->mcdi_ops->mcdi_rpc_timeout)
 | |
| 		return MCDI_RPC_TIMEOUT;
 | |
| 	else
 | |
| 		return cdx->mcdi_ops->mcdi_rpc_timeout(cdx, cmd);
 | |
| }
 | |
| 
 | |
| int cdx_mcdi_init(struct cdx_mcdi *cdx)
 | |
| {
 | |
| 	struct cdx_mcdi_iface *mcdi;
 | |
| 	int rc = -ENOMEM;
 | |
| 
 | |
| 	cdx->mcdi = kzalloc(sizeof(*cdx->mcdi), GFP_KERNEL);
 | |
| 	if (!cdx->mcdi)
 | |
| 		goto fail;
 | |
| 
 | |
| 	mcdi = cdx_mcdi_if(cdx);
 | |
| 	mcdi->cdx = cdx;
 | |
| 
 | |
| 	mcdi->workqueue = alloc_ordered_workqueue("mcdi_wq", 0);
 | |
| 	if (!mcdi->workqueue)
 | |
| 		goto fail2;
 | |
| 	mutex_init(&mcdi->iface_lock);
 | |
| 	mcdi->mode = MCDI_MODE_EVENTS;
 | |
| 	INIT_LIST_HEAD(&mcdi->cmd_list);
 | |
| 	init_waitqueue_head(&mcdi->cmd_complete_wq);
 | |
| 
 | |
| 	mcdi->new_epoch = true;
 | |
| 
 | |
| 	return 0;
 | |
| fail2:
 | |
| 	kfree(cdx->mcdi);
 | |
| 	cdx->mcdi = NULL;
 | |
| fail:
 | |
| 	return rc;
 | |
| }
 | |
| 
 | |
| void cdx_mcdi_finish(struct cdx_mcdi *cdx)
 | |
| {
 | |
| 	struct cdx_mcdi_iface *mcdi;
 | |
| 
 | |
| 	mcdi = cdx_mcdi_if(cdx);
 | |
| 	if (!mcdi)
 | |
| 		return;
 | |
| 
 | |
| 	cdx_mcdi_wait_for_cleanup(cdx);
 | |
| 
 | |
| 	destroy_workqueue(mcdi->workqueue);
 | |
| 	kfree(cdx->mcdi);
 | |
| 	cdx->mcdi = NULL;
 | |
| }
 | |
| 
 | |
| static bool cdx_mcdi_flushed(struct cdx_mcdi_iface *mcdi, bool ignore_cleanups)
 | |
| {
 | |
| 	bool flushed;
 | |
| 
 | |
| 	mutex_lock(&mcdi->iface_lock);
 | |
| 	flushed = list_empty(&mcdi->cmd_list) &&
 | |
| 		  (ignore_cleanups || !mcdi->outstanding_cleanups);
 | |
| 	mutex_unlock(&mcdi->iface_lock);
 | |
| 	return flushed;
 | |
| }
 | |
| 
 | |
| /* Wait for outstanding MCDI commands to complete. */
 | |
| static void cdx_mcdi_wait_for_cleanup(struct cdx_mcdi *cdx)
 | |
| {
 | |
| 	struct cdx_mcdi_iface *mcdi = cdx_mcdi_if(cdx);
 | |
| 
 | |
| 	if (!mcdi)
 | |
| 		return;
 | |
| 
 | |
| 	wait_event(mcdi->cmd_complete_wq,
 | |
| 		   cdx_mcdi_flushed(mcdi, false));
 | |
| }
 | |
| 
 | |
| int cdx_mcdi_wait_for_quiescence(struct cdx_mcdi *cdx,
 | |
| 				 unsigned int timeout_jiffies)
 | |
| {
 | |
| 	struct cdx_mcdi_iface *mcdi = cdx_mcdi_if(cdx);
 | |
| 	DEFINE_WAIT_FUNC(wait, woken_wake_function);
 | |
| 	int rc = 0;
 | |
| 
 | |
| 	if (!mcdi)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	flush_workqueue(mcdi->workqueue);
 | |
| 
 | |
| 	add_wait_queue(&mcdi->cmd_complete_wq, &wait);
 | |
| 
 | |
| 	while (!cdx_mcdi_flushed(mcdi, true)) {
 | |
| 		rc = wait_woken(&wait, TASK_IDLE, timeout_jiffies);
 | |
| 		if (rc)
 | |
| 			continue;
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	remove_wait_queue(&mcdi->cmd_complete_wq, &wait);
 | |
| 
 | |
| 	if (rc > 0)
 | |
| 		rc = 0;
 | |
| 	else if (rc == 0)
 | |
| 		rc = -ETIMEDOUT;
 | |
| 
 | |
| 	return rc;
 | |
| }
 | |
| 
 | |
| static u8 cdx_mcdi_payload_csum(const struct cdx_dword *hdr, size_t hdr_len,
 | |
| 				const struct cdx_dword *sdu, size_t sdu_len)
 | |
| {
 | |
| 	u8 *p = (u8 *)hdr;
 | |
| 	u8 csum = 0;
 | |
| 	int i;
 | |
| 
 | |
| 	for (i = 0; i < hdr_len; i++)
 | |
| 		csum += p[i];
 | |
| 
 | |
| 	p = (u8 *)sdu;
 | |
| 	for (i = 0; i < sdu_len; i++)
 | |
| 		csum += p[i];
 | |
| 
 | |
| 	return ~csum & 0xff;
 | |
| }
 | |
| 
 | |
| static void cdx_mcdi_send_request(struct cdx_mcdi *cdx,
 | |
| 				  struct cdx_mcdi_cmd *cmd)
 | |
| {
 | |
| 	struct cdx_mcdi_iface *mcdi = cdx_mcdi_if(cdx);
 | |
| 	const struct cdx_dword *inbuf = cmd->inbuf;
 | |
| 	size_t inlen = cmd->inlen;
 | |
| 	struct cdx_dword hdr[2];
 | |
| 	size_t hdr_len;
 | |
| 	bool not_epoch;
 | |
| 	u32 xflags;
 | |
| 
 | |
| 	if (!mcdi)
 | |
| 		return;
 | |
| 
 | |
| 	mcdi->prev_seq = cmd->seq;
 | |
| 	mcdi->seq_held_by[cmd->seq] = cmd;
 | |
| 	mcdi->db_held_by = cmd;
 | |
| 	cmd->started = jiffies;
 | |
| 
 | |
| 	not_epoch = !mcdi->new_epoch;
 | |
| 	xflags = 0;
 | |
| 
 | |
| 	/* MCDI v2 */
 | |
| 	WARN_ON(inlen > MCDI_CTL_SDU_LEN_MAX_V2);
 | |
| 	CDX_POPULATE_DWORD_7(hdr[0],
 | |
| 			     MCDI_HEADER_RESPONSE, 0,
 | |
| 			     MCDI_HEADER_RESYNC, 1,
 | |
| 			     MCDI_HEADER_CODE, MC_CMD_V2_EXTN,
 | |
| 			     MCDI_HEADER_DATALEN, 0,
 | |
| 			     MCDI_HEADER_SEQ, cmd->seq,
 | |
| 			     MCDI_HEADER_XFLAGS, xflags,
 | |
| 			     MCDI_HEADER_NOT_EPOCH, not_epoch);
 | |
| 	CDX_POPULATE_DWORD_3(hdr[1],
 | |
| 			     MC_CMD_V2_EXTN_IN_EXTENDED_CMD, cmd->cmd,
 | |
| 			     MC_CMD_V2_EXTN_IN_ACTUAL_LEN, inlen,
 | |
| 			     MC_CMD_V2_EXTN_IN_MESSAGE_TYPE,
 | |
| 			     MC_CMD_V2_EXTN_IN_MCDI_MESSAGE_TYPE_PLATFORM);
 | |
| 	hdr_len = 8;
 | |
| 
 | |
| 	hdr[0].cdx_u32 |= (__force __le32)(cdx_mcdi_payload_csum(hdr, hdr_len, inbuf, inlen) <<
 | |
| 			 MCDI_HEADER_XFLAGS_LBN);
 | |
| 
 | |
| 	print_hex_dump_debug("MCDI REQ HEADER: ", DUMP_PREFIX_NONE, 32, 4, hdr, hdr_len, false);
 | |
| 	print_hex_dump_debug("MCDI REQ PAYLOAD: ", DUMP_PREFIX_NONE, 32, 4, inbuf, inlen, false);
 | |
| 
 | |
| 	cdx->mcdi_ops->mcdi_request(cdx, hdr, hdr_len, inbuf, inlen);
 | |
| 
 | |
| 	mcdi->new_epoch = false;
 | |
| }
 | |
| 
 | |
| static int cdx_mcdi_errno(struct cdx_mcdi *cdx, unsigned int mcdi_err)
 | |
| {
 | |
| 	switch (mcdi_err) {
 | |
| 	case 0:
 | |
| 	case MC_CMD_ERR_QUEUE_FULL:
 | |
| 		return mcdi_err;
 | |
| 	case MC_CMD_ERR_EPERM:
 | |
| 		return -EPERM;
 | |
| 	case MC_CMD_ERR_ENOENT:
 | |
| 		return -ENOENT;
 | |
| 	case MC_CMD_ERR_EINTR:
 | |
| 		return -EINTR;
 | |
| 	case MC_CMD_ERR_EAGAIN:
 | |
| 		return -EAGAIN;
 | |
| 	case MC_CMD_ERR_EACCES:
 | |
| 		return -EACCES;
 | |
| 	case MC_CMD_ERR_EBUSY:
 | |
| 		return -EBUSY;
 | |
| 	case MC_CMD_ERR_EINVAL:
 | |
| 		return -EINVAL;
 | |
| 	case MC_CMD_ERR_ERANGE:
 | |
| 		return -ERANGE;
 | |
| 	case MC_CMD_ERR_EDEADLK:
 | |
| 		return -EDEADLK;
 | |
| 	case MC_CMD_ERR_ENOSYS:
 | |
| 		return -EOPNOTSUPP;
 | |
| 	case MC_CMD_ERR_ETIME:
 | |
| 		return -ETIME;
 | |
| 	case MC_CMD_ERR_EALREADY:
 | |
| 		return -EALREADY;
 | |
| 	case MC_CMD_ERR_ENOSPC:
 | |
| 		return -ENOSPC;
 | |
| 	case MC_CMD_ERR_ENOMEM:
 | |
| 		return -ENOMEM;
 | |
| 	case MC_CMD_ERR_ENOTSUP:
 | |
| 		return -EOPNOTSUPP;
 | |
| 	case MC_CMD_ERR_ALLOC_FAIL:
 | |
| 		return -ENOBUFS;
 | |
| 	case MC_CMD_ERR_MAC_EXIST:
 | |
| 		return -EADDRINUSE;
 | |
| 	case MC_CMD_ERR_NO_EVB_PORT:
 | |
| 		return -EAGAIN;
 | |
| 	default:
 | |
| 		return -EPROTO;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void cdx_mcdi_process_cleanup_list(struct cdx_mcdi *cdx,
 | |
| 					  struct list_head *cleanup_list)
 | |
| {
 | |
| 	struct cdx_mcdi_iface *mcdi = cdx_mcdi_if(cdx);
 | |
| 	unsigned int cleanups = 0;
 | |
| 
 | |
| 	if (!mcdi)
 | |
| 		return;
 | |
| 
 | |
| 	while (!list_empty(cleanup_list)) {
 | |
| 		struct cdx_mcdi_cmd *cmd =
 | |
| 			list_first_entry(cleanup_list,
 | |
| 					 struct cdx_mcdi_cmd, cleanup_list);
 | |
| 		cmd->completer(cdx, cmd->cookie, cmd->rc,
 | |
| 			       cmd->outbuf, cmd->outlen);
 | |
| 		list_del(&cmd->cleanup_list);
 | |
| 		kref_put(&cmd->ref, cdx_mcdi_cmd_release);
 | |
| 		++cleanups;
 | |
| 	}
 | |
| 
 | |
| 	if (cleanups) {
 | |
| 		bool all_done;
 | |
| 
 | |
| 		mutex_lock(&mcdi->iface_lock);
 | |
| 		CDX_WARN_ON_PARANOID(cleanups > mcdi->outstanding_cleanups);
 | |
| 		all_done = (mcdi->outstanding_cleanups -= cleanups) == 0;
 | |
| 		mutex_unlock(&mcdi->iface_lock);
 | |
| 		if (all_done)
 | |
| 			wake_up(&mcdi->cmd_complete_wq);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void _cdx_mcdi_cancel_cmd(struct cdx_mcdi_iface *mcdi,
 | |
| 				 unsigned int handle,
 | |
| 				 struct list_head *cleanup_list)
 | |
| {
 | |
| 	struct cdx_mcdi_cmd *cmd;
 | |
| 
 | |
| 	list_for_each_entry(cmd, &mcdi->cmd_list, list)
 | |
| 		if (cdx_mcdi_cmd_handle(cmd) == handle) {
 | |
| 			switch (cmd->state) {
 | |
| 			case MCDI_STATE_QUEUED:
 | |
| 			case MCDI_STATE_RETRY:
 | |
| 				pr_debug("command %#x inlen %zu cancelled in queue\n",
 | |
| 					 cmd->cmd, cmd->inlen);
 | |
| 				/* if not yet running, properly cancel it */
 | |
| 				cmd->rc = -EPIPE;
 | |
| 				cdx_mcdi_remove_cmd(mcdi, cmd, cleanup_list);
 | |
| 				break;
 | |
| 			case MCDI_STATE_RUNNING:
 | |
| 			case MCDI_STATE_RUNNING_CANCELLED:
 | |
| 			case MCDI_STATE_FINISHED:
 | |
| 			default:
 | |
| 				/* invalid state? */
 | |
| 				WARN_ON(1);
 | |
| 			}
 | |
| 			break;
 | |
| 		}
 | |
| }
 | |
| 
 | |
| static void cdx_mcdi_cancel_cmd(struct cdx_mcdi *cdx, struct cdx_mcdi_cmd *cmd)
 | |
| {
 | |
| 	struct cdx_mcdi_iface *mcdi = cdx_mcdi_if(cdx);
 | |
| 	LIST_HEAD(cleanup_list);
 | |
| 
 | |
| 	if (!mcdi)
 | |
| 		return;
 | |
| 
 | |
| 	mutex_lock(&mcdi->iface_lock);
 | |
| 	cdx_mcdi_timeout_cmd(mcdi, cmd, &cleanup_list);
 | |
| 	mutex_unlock(&mcdi->iface_lock);
 | |
| 	cdx_mcdi_process_cleanup_list(cdx, &cleanup_list);
 | |
| }
 | |
| 
 | |
| struct cdx_mcdi_blocking_data {
 | |
| 	struct kref ref;
 | |
| 	bool done;
 | |
| 	wait_queue_head_t wq;
 | |
| 	int rc;
 | |
| 	struct cdx_dword *outbuf;
 | |
| 	size_t outlen;
 | |
| 	size_t outlen_actual;
 | |
| };
 | |
| 
 | |
| static void cdx_mcdi_blocking_data_release(struct kref *ref)
 | |
| {
 | |
| 	kfree(container_of(ref, struct cdx_mcdi_blocking_data, ref));
 | |
| }
 | |
| 
 | |
| static void cdx_mcdi_rpc_completer(struct cdx_mcdi *cdx, unsigned long cookie,
 | |
| 				   int rc, struct cdx_dword *outbuf,
 | |
| 				   size_t outlen_actual)
 | |
| {
 | |
| 	struct cdx_mcdi_blocking_data *wait_data =
 | |
| 		(struct cdx_mcdi_blocking_data *)cookie;
 | |
| 
 | |
| 	wait_data->rc = rc;
 | |
| 	memcpy(wait_data->outbuf, outbuf,
 | |
| 	       min(outlen_actual, wait_data->outlen));
 | |
| 	wait_data->outlen_actual = outlen_actual;
 | |
| 	/* memory barrier */
 | |
| 	smp_wmb();
 | |
| 	wait_data->done = true;
 | |
| 	wake_up(&wait_data->wq);
 | |
| 	kref_put(&wait_data->ref, cdx_mcdi_blocking_data_release);
 | |
| }
 | |
| 
 | |
| static int cdx_mcdi_rpc_sync(struct cdx_mcdi *cdx, unsigned int cmd,
 | |
| 			     const struct cdx_dword *inbuf, size_t inlen,
 | |
| 			     struct cdx_dword *outbuf, size_t outlen,
 | |
| 			     size_t *outlen_actual, bool quiet)
 | |
| {
 | |
| 	struct cdx_mcdi_blocking_data *wait_data;
 | |
| 	struct cdx_mcdi_cmd *cmd_item;
 | |
| 	unsigned int handle;
 | |
| 	int rc;
 | |
| 
 | |
| 	if (outlen_actual)
 | |
| 		*outlen_actual = 0;
 | |
| 
 | |
| 	wait_data = kmalloc(sizeof(*wait_data), GFP_KERNEL);
 | |
| 	if (!wait_data)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	cmd_item = kmalloc(sizeof(*cmd_item), GFP_KERNEL);
 | |
| 	if (!cmd_item) {
 | |
| 		kfree(wait_data);
 | |
| 		return -ENOMEM;
 | |
| 	}
 | |
| 
 | |
| 	kref_init(&wait_data->ref);
 | |
| 	wait_data->done = false;
 | |
| 	init_waitqueue_head(&wait_data->wq);
 | |
| 	wait_data->outbuf = outbuf;
 | |
| 	wait_data->outlen = outlen;
 | |
| 
 | |
| 	kref_init(&cmd_item->ref);
 | |
| 	cmd_item->quiet = quiet;
 | |
| 	cmd_item->cookie = (unsigned long)wait_data;
 | |
| 	cmd_item->completer = &cdx_mcdi_rpc_completer;
 | |
| 	cmd_item->cmd = cmd;
 | |
| 	cmd_item->inlen = inlen;
 | |
| 	cmd_item->inbuf = inbuf;
 | |
| 
 | |
| 	/* Claim an extra reference for the completer to put. */
 | |
| 	kref_get(&wait_data->ref);
 | |
| 	rc = cdx_mcdi_rpc_async_internal(cdx, cmd_item, &handle);
 | |
| 	if (rc) {
 | |
| 		kref_put(&wait_data->ref, cdx_mcdi_blocking_data_release);
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	if (!wait_event_timeout(wait_data->wq, wait_data->done,
 | |
| 				cdx_mcdi_rpc_timeout(cdx, cmd)) &&
 | |
| 	    !wait_data->done) {
 | |
| 		pr_err("MC command 0x%x inlen %zu timed out (sync)\n",
 | |
| 		       cmd, inlen);
 | |
| 
 | |
| 		cdx_mcdi_cancel_cmd(cdx, cmd_item);
 | |
| 
 | |
| 		wait_data->rc = -ETIMEDOUT;
 | |
| 		wait_data->outlen_actual = 0;
 | |
| 	}
 | |
| 
 | |
| 	if (outlen_actual)
 | |
| 		*outlen_actual = wait_data->outlen_actual;
 | |
| 	rc = wait_data->rc;
 | |
| 
 | |
| out:
 | |
| 	kref_put(&wait_data->ref, cdx_mcdi_blocking_data_release);
 | |
| 
 | |
| 	return rc;
 | |
| }
 | |
| 
 | |
| static bool cdx_mcdi_get_seq(struct cdx_mcdi_iface *mcdi, unsigned char *seq)
 | |
| {
 | |
| 	*seq = mcdi->prev_seq;
 | |
| 	do {
 | |
| 		*seq = (*seq + 1) % ARRAY_SIZE(mcdi->seq_held_by);
 | |
| 	} while (mcdi->seq_held_by[*seq] && *seq != mcdi->prev_seq);
 | |
| 	return !mcdi->seq_held_by[*seq];
 | |
| }
 | |
| 
 | |
| static int cdx_mcdi_rpc_async_internal(struct cdx_mcdi *cdx,
 | |
| 				       struct cdx_mcdi_cmd *cmd,
 | |
| 				       unsigned int *handle)
 | |
| {
 | |
| 	struct cdx_mcdi_iface *mcdi = cdx_mcdi_if(cdx);
 | |
| 	LIST_HEAD(cleanup_list);
 | |
| 
 | |
| 	if (!mcdi) {
 | |
| 		kref_put(&cmd->ref, cdx_mcdi_cmd_release);
 | |
| 		return -ENETDOWN;
 | |
| 	}
 | |
| 
 | |
| 	if (mcdi->mode == MCDI_MODE_FAIL) {
 | |
| 		kref_put(&cmd->ref, cdx_mcdi_cmd_release);
 | |
| 		return -ENETDOWN;
 | |
| 	}
 | |
| 
 | |
| 	cmd->mcdi = mcdi;
 | |
| 	INIT_WORK(&cmd->work, cdx_mcdi_cmd_work);
 | |
| 	INIT_LIST_HEAD(&cmd->list);
 | |
| 	INIT_LIST_HEAD(&cmd->cleanup_list);
 | |
| 	cmd->rc = 0;
 | |
| 	cmd->outbuf = NULL;
 | |
| 	cmd->outlen = 0;
 | |
| 
 | |
| 	queue_work(mcdi->workqueue, &cmd->work);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void cdx_mcdi_cmd_start_or_queue(struct cdx_mcdi_iface *mcdi,
 | |
| 					struct cdx_mcdi_cmd *cmd)
 | |
| {
 | |
| 	struct cdx_mcdi *cdx = mcdi->cdx;
 | |
| 	u8 seq;
 | |
| 
 | |
| 	if (!mcdi->db_held_by &&
 | |
| 	    cdx_mcdi_get_seq(mcdi, &seq)) {
 | |
| 		cmd->seq = seq;
 | |
| 		cmd->reboot_seen = false;
 | |
| 		cdx_mcdi_send_request(cdx, cmd);
 | |
| 		cmd->state = MCDI_STATE_RUNNING;
 | |
| 	} else {
 | |
| 		cmd->state = MCDI_STATE_QUEUED;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /* try to advance other commands */
 | |
| static void cdx_mcdi_start_or_queue(struct cdx_mcdi_iface *mcdi,
 | |
| 				    bool allow_retry)
 | |
| {
 | |
| 	struct cdx_mcdi_cmd *cmd, *tmp;
 | |
| 
 | |
| 	list_for_each_entry_safe(cmd, tmp, &mcdi->cmd_list, list)
 | |
| 		if (cmd->state == MCDI_STATE_QUEUED ||
 | |
| 		    (cmd->state == MCDI_STATE_RETRY && allow_retry))
 | |
| 			cdx_mcdi_cmd_start_or_queue(mcdi, cmd);
 | |
| }
 | |
| 
 | |
| void cdx_mcdi_process_cmd(struct cdx_mcdi *cdx, struct cdx_dword *outbuf, int len)
 | |
| {
 | |
| 	struct cdx_mcdi_iface *mcdi;
 | |
| 	struct cdx_mcdi_cmd *cmd;
 | |
| 	LIST_HEAD(cleanup_list);
 | |
| 	unsigned int respseq;
 | |
| 
 | |
| 	if (!len || !outbuf) {
 | |
| 		pr_err("Got empty MC response\n");
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	mcdi = cdx_mcdi_if(cdx);
 | |
| 	if (!mcdi)
 | |
| 		return;
 | |
| 
 | |
| 	respseq = CDX_DWORD_FIELD(outbuf[0], MCDI_HEADER_SEQ);
 | |
| 
 | |
| 	mutex_lock(&mcdi->iface_lock);
 | |
| 	cmd = mcdi->seq_held_by[respseq];
 | |
| 
 | |
| 	if (cmd) {
 | |
| 		if (cmd->state == MCDI_STATE_FINISHED) {
 | |
| 			mutex_unlock(&mcdi->iface_lock);
 | |
| 			kref_put(&cmd->ref, cdx_mcdi_cmd_release);
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		cdx_mcdi_complete_cmd(mcdi, cmd, outbuf, len, &cleanup_list);
 | |
| 	} else {
 | |
| 		pr_err("MC response unexpected for seq : %0X\n", respseq);
 | |
| 	}
 | |
| 
 | |
| 	mutex_unlock(&mcdi->iface_lock);
 | |
| 
 | |
| 	cdx_mcdi_process_cleanup_list(mcdi->cdx, &cleanup_list);
 | |
| }
 | |
| 
 | |
| static void cdx_mcdi_cmd_work(struct work_struct *context)
 | |
| {
 | |
| 	struct cdx_mcdi_cmd *cmd =
 | |
| 		container_of(context, struct cdx_mcdi_cmd, work);
 | |
| 	struct cdx_mcdi_iface *mcdi = cmd->mcdi;
 | |
| 
 | |
| 	mutex_lock(&mcdi->iface_lock);
 | |
| 
 | |
| 	cmd->handle = mcdi->prev_handle++;
 | |
| 	list_add_tail(&cmd->list, &mcdi->cmd_list);
 | |
| 	cdx_mcdi_cmd_start_or_queue(mcdi, cmd);
 | |
| 
 | |
| 	mutex_unlock(&mcdi->iface_lock);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Returns true if the MCDI module is finished with the command.
 | |
|  * (examples of false would be if the command was proxied, or it was
 | |
|  * rejected by the MC due to lack of resources and requeued).
 | |
|  */
 | |
| static bool cdx_mcdi_complete_cmd(struct cdx_mcdi_iface *mcdi,
 | |
| 				  struct cdx_mcdi_cmd *cmd,
 | |
| 				  struct cdx_dword *outbuf,
 | |
| 				  int len,
 | |
| 				  struct list_head *cleanup_list)
 | |
| {
 | |
| 	size_t resp_hdr_len, resp_data_len;
 | |
| 	struct cdx_mcdi *cdx = mcdi->cdx;
 | |
| 	unsigned int respcmd, error;
 | |
| 	bool completed = false;
 | |
| 	int rc;
 | |
| 
 | |
| 	/* ensure the command can't go away before this function returns */
 | |
| 	kref_get(&cmd->ref);
 | |
| 
 | |
| 	respcmd = CDX_DWORD_FIELD(outbuf[0], MCDI_HEADER_CODE);
 | |
| 	error = CDX_DWORD_FIELD(outbuf[0], MCDI_HEADER_ERROR);
 | |
| 
 | |
| 	if (respcmd != MC_CMD_V2_EXTN) {
 | |
| 		resp_hdr_len = 4;
 | |
| 		resp_data_len = CDX_DWORD_FIELD(outbuf[0], MCDI_HEADER_DATALEN);
 | |
| 	} else {
 | |
| 		resp_data_len = 0;
 | |
| 		resp_hdr_len = 8;
 | |
| 		if (len >= 8)
 | |
| 			resp_data_len =
 | |
| 				CDX_DWORD_FIELD(outbuf[1], MC_CMD_V2_EXTN_IN_ACTUAL_LEN);
 | |
| 	}
 | |
| 
 | |
| 	if ((resp_hdr_len + resp_data_len) > len) {
 | |
| 		pr_warn("Incomplete MCDI response received %d. Expected %zu\n",
 | |
| 			len, (resp_hdr_len + resp_data_len));
 | |
| 		resp_data_len = 0;
 | |
| 	}
 | |
| 
 | |
| 	print_hex_dump_debug("MCDI RESP HEADER: ", DUMP_PREFIX_NONE, 32, 4,
 | |
| 			     outbuf, resp_hdr_len, false);
 | |
| 	print_hex_dump_debug("MCDI RESP PAYLOAD: ", DUMP_PREFIX_NONE, 32, 4,
 | |
| 			     outbuf + (resp_hdr_len / 4), resp_data_len, false);
 | |
| 
 | |
| 	if (error && resp_data_len == 0) {
 | |
| 		/* MC rebooted during command */
 | |
| 		rc = -EIO;
 | |
| 	} else {
 | |
| 		if (WARN_ON_ONCE(error && resp_data_len < 4))
 | |
| 			resp_data_len = 4;
 | |
| 		if (error) {
 | |
| 			rc = CDX_DWORD_FIELD(outbuf[resp_hdr_len / 4], CDX_DWORD);
 | |
| 			if (!cmd->quiet) {
 | |
| 				int err_arg = 0;
 | |
| 
 | |
| 				if (resp_data_len >= MC_CMD_ERR_ARG_OFST + 4) {
 | |
| 					int offset = (resp_hdr_len + MC_CMD_ERR_ARG_OFST) / 4;
 | |
| 
 | |
| 					err_arg = CDX_DWORD_VAL(outbuf[offset]);
 | |
| 				}
 | |
| 
 | |
| 				_cdx_mcdi_display_error(cdx, cmd->cmd,
 | |
| 							cmd->inlen, rc, err_arg,
 | |
| 							cdx_mcdi_errno(cdx, rc));
 | |
| 			}
 | |
| 			rc = cdx_mcdi_errno(cdx, rc);
 | |
| 		} else {
 | |
| 			rc = 0;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/* free doorbell */
 | |
| 	if (mcdi->db_held_by == cmd)
 | |
| 		mcdi->db_held_by = NULL;
 | |
| 
 | |
| 	if (cdx_cmd_cancelled(cmd)) {
 | |
| 		list_del(&cmd->list);
 | |
| 		kref_put(&cmd->ref, cdx_mcdi_cmd_release);
 | |
| 		completed = true;
 | |
| 	} else if (rc == MC_CMD_ERR_QUEUE_FULL) {
 | |
| 		cmd->state = MCDI_STATE_RETRY;
 | |
| 	} else {
 | |
| 		cmd->rc = rc;
 | |
| 		cmd->outbuf = outbuf + DIV_ROUND_UP(resp_hdr_len, 4);
 | |
| 		cmd->outlen = resp_data_len;
 | |
| 		cdx_mcdi_remove_cmd(mcdi, cmd, cleanup_list);
 | |
| 		completed = true;
 | |
| 	}
 | |
| 
 | |
| 	/* free sequence number and buffer */
 | |
| 	mcdi->seq_held_by[cmd->seq] = NULL;
 | |
| 
 | |
| 	cdx_mcdi_start_or_queue(mcdi, rc != MC_CMD_ERR_QUEUE_FULL);
 | |
| 
 | |
| 	/* wake up anyone waiting for flush */
 | |
| 	wake_up(&mcdi->cmd_complete_wq);
 | |
| 
 | |
| 	kref_put(&cmd->ref, cdx_mcdi_cmd_release);
 | |
| 
 | |
| 	return completed;
 | |
| }
 | |
| 
 | |
| static void cdx_mcdi_timeout_cmd(struct cdx_mcdi_iface *mcdi,
 | |
| 				 struct cdx_mcdi_cmd *cmd,
 | |
| 				 struct list_head *cleanup_list)
 | |
| {
 | |
| 	struct cdx_mcdi *cdx = mcdi->cdx;
 | |
| 
 | |
| 	pr_err("MC command 0x%x inlen %zu state %d timed out after %u ms\n",
 | |
| 	       cmd->cmd, cmd->inlen, cmd->state,
 | |
| 	       jiffies_to_msecs(jiffies - cmd->started));
 | |
| 
 | |
| 	cmd->rc = -ETIMEDOUT;
 | |
| 	cdx_mcdi_remove_cmd(mcdi, cmd, cleanup_list);
 | |
| 
 | |
| 	cdx_mcdi_mode_fail(cdx, cleanup_list);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * cdx_mcdi_rpc - Issue an MCDI command and wait for completion
 | |
|  * @cdx: NIC through which to issue the command
 | |
|  * @cmd: Command type number
 | |
|  * @inbuf: Command parameters
 | |
|  * @inlen: Length of command parameters, in bytes. Must be a multiple
 | |
|  *	of 4 and no greater than %MCDI_CTL_SDU_LEN_MAX_V1.
 | |
|  * @outbuf: Response buffer. May be %NULL if @outlen is 0.
 | |
|  * @outlen: Length of response buffer, in bytes. If the actual
 | |
|  *	response is longer than @outlen & ~3, it will be truncated
 | |
|  *	to that length.
 | |
|  * @outlen_actual: Pointer through which to return the actual response
 | |
|  *	length. May be %NULL if this is not needed.
 | |
|  *
 | |
|  * This function may sleep and therefore must be called in process
 | |
|  * context.
 | |
|  *
 | |
|  * Return: A negative error code, or zero if successful. The error
 | |
|  *	code may come from the MCDI response or may indicate a failure
 | |
|  *	to communicate with the MC. In the former case, the response
 | |
|  *	will still be copied to @outbuf and *@outlen_actual will be
 | |
|  *	set accordingly. In the latter case, *@outlen_actual will be
 | |
|  *	set to zero.
 | |
|  */
 | |
| int cdx_mcdi_rpc(struct cdx_mcdi *cdx, unsigned int cmd,
 | |
| 		 const struct cdx_dword *inbuf, size_t inlen,
 | |
| 		 struct cdx_dword *outbuf, size_t outlen,
 | |
| 		 size_t *outlen_actual)
 | |
| {
 | |
| 	return cdx_mcdi_rpc_sync(cdx, cmd, inbuf, inlen, outbuf, outlen,
 | |
| 				 outlen_actual, false);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * cdx_mcdi_rpc_async - Schedule an MCDI command to run asynchronously
 | |
|  * @cdx: NIC through which to issue the command
 | |
|  * @cmd: Command type number
 | |
|  * @inbuf: Command parameters
 | |
|  * @inlen: Length of command parameters, in bytes
 | |
|  * @complete: Function to be called on completion or cancellation.
 | |
|  * @cookie: Arbitrary value to be passed to @complete.
 | |
|  *
 | |
|  * This function does not sleep and therefore may be called in atomic
 | |
|  * context.  It will fail if event queues are disabled or if MCDI
 | |
|  * event completions have been disabled due to an error.
 | |
|  *
 | |
|  * If it succeeds, the @complete function will be called exactly once
 | |
|  * in process context, when one of the following occurs:
 | |
|  * (a) the completion event is received (in process context)
 | |
|  * (b) event queues are disabled (in the process that disables them)
 | |
|  */
 | |
| int
 | |
| cdx_mcdi_rpc_async(struct cdx_mcdi *cdx, unsigned int cmd,
 | |
| 		   const struct cdx_dword *inbuf, size_t inlen,
 | |
| 		   cdx_mcdi_async_completer *complete, unsigned long cookie)
 | |
| {
 | |
| 	struct cdx_mcdi_cmd *cmd_item =
 | |
| 		kmalloc(sizeof(struct cdx_mcdi_cmd) + inlen, GFP_ATOMIC);
 | |
| 
 | |
| 	if (!cmd_item)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	kref_init(&cmd_item->ref);
 | |
| 	cmd_item->quiet = true;
 | |
| 	cmd_item->cookie = cookie;
 | |
| 	cmd_item->completer = complete;
 | |
| 	cmd_item->cmd = cmd;
 | |
| 	cmd_item->inlen = inlen;
 | |
| 	/* inbuf is probably not valid after return, so take a copy */
 | |
| 	cmd_item->inbuf = (struct cdx_dword *)(cmd_item + 1);
 | |
| 	memcpy(cmd_item + 1, inbuf, inlen);
 | |
| 
 | |
| 	return cdx_mcdi_rpc_async_internal(cdx, cmd_item, NULL);
 | |
| }
 | |
| 
 | |
| static void _cdx_mcdi_display_error(struct cdx_mcdi *cdx, unsigned int cmd,
 | |
| 				    size_t inlen, int raw, int arg, int err_no)
 | |
| {
 | |
| 	pr_err("MC command 0x%x inlen %d failed err_no=%d (raw=%d) arg=%d\n",
 | |
| 	       cmd, (int)inlen, err_no, raw, arg);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Set MCDI mode to fail to prevent any new commands, then cancel any
 | |
|  * outstanding commands.
 | |
|  * Caller must hold the mcdi iface_lock.
 | |
|  */
 | |
| static void cdx_mcdi_mode_fail(struct cdx_mcdi *cdx, struct list_head *cleanup_list)
 | |
| {
 | |
| 	struct cdx_mcdi_iface *mcdi = cdx_mcdi_if(cdx);
 | |
| 
 | |
| 	if (!mcdi)
 | |
| 		return;
 | |
| 
 | |
| 	mcdi->mode = MCDI_MODE_FAIL;
 | |
| 
 | |
| 	while (!list_empty(&mcdi->cmd_list)) {
 | |
| 		struct cdx_mcdi_cmd *cmd;
 | |
| 
 | |
| 		cmd = list_first_entry(&mcdi->cmd_list, struct cdx_mcdi_cmd,
 | |
| 				       list);
 | |
| 		_cdx_mcdi_cancel_cmd(mcdi, cdx_mcdi_cmd_handle(cmd), cleanup_list);
 | |
| 	}
 | |
| }
 |