2009-11-29 23:15:25 +08:00
|
|
|
/****************************************************************************
|
2013-08-30 06:32:48 +08:00
|
|
|
* Driver for Solarflare network controllers and boards
|
|
|
|
* Copyright 2008-2013 Solarflare Communications Inc.
|
2009-11-29 23:15:25 +08:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License version 2 as published
|
|
|
|
* by the Free Software Foundation, incorporated herein by reference.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/delay.h>
|
2015-05-27 20:14:26 +08:00
|
|
|
#include <linux/moduleparam.h>
|
2015-08-26 19:52:46 +08:00
|
|
|
#include <linux/atomic.h>
|
2009-11-29 23:15:25 +08:00
|
|
|
#include "net_driver.h"
|
|
|
|
#include "nic.h"
|
|
|
|
#include "io.h"
|
2012-09-18 08:57:07 +08:00
|
|
|
#include "farch_regs.h"
|
2009-11-29 23:15:25 +08:00
|
|
|
#include "mcdi_pcol.h"
|
|
|
|
|
|
|
|
/**************************************************************************
|
|
|
|
*
|
|
|
|
* Management-Controller-to-Driver Interface
|
|
|
|
*
|
|
|
|
**************************************************************************
|
|
|
|
*/
|
|
|
|
|
2012-12-01 10:21:17 +08:00
|
|
|
#define MCDI_RPC_TIMEOUT (10 * HZ)
|
2009-11-29 23:15:25 +08:00
|
|
|
|
2011-12-21 07:39:31 +08:00
|
|
|
/* A reboot/assertion causes the MCDI status word to be set after the
|
|
|
|
* command word is set or a REBOOT event is sent. If we notice a reboot
|
2013-09-20 23:45:10 +08:00
|
|
|
* via these mechanisms then wait 250ms for the status word to be set.
|
2013-06-20 18:40:07 +08:00
|
|
|
*/
|
2011-12-21 07:39:31 +08:00
|
|
|
#define MCDI_STATUS_DELAY_US 100
|
2013-09-20 23:45:10 +08:00
|
|
|
#define MCDI_STATUS_DELAY_COUNT 2500
|
2011-12-21 07:39:31 +08:00
|
|
|
#define MCDI_STATUS_SLEEP_MS \
|
|
|
|
(MCDI_STATUS_DELAY_US * MCDI_STATUS_DELAY_COUNT / 1000)
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
#define SEQ_MASK \
|
|
|
|
EFX_MASK32(EFX_WIDTH(MCDI_HEADER_SEQ))
|
|
|
|
|
2013-08-28 06:12:31 +08:00
|
|
|
struct efx_mcdi_async_param {
|
|
|
|
struct list_head list;
|
|
|
|
unsigned int cmd;
|
|
|
|
size_t inlen;
|
|
|
|
size_t outlen;
|
2013-06-01 01:36:12 +08:00
|
|
|
bool quiet;
|
2013-08-28 06:12:31 +08:00
|
|
|
efx_mcdi_async_completer *complete;
|
|
|
|
unsigned long cookie;
|
|
|
|
/* followed by request/response buffer */
|
|
|
|
};
|
|
|
|
|
2017-10-24 16:45:59 +08:00
|
|
|
static void efx_mcdi_timeout_async(struct timer_list *t);
|
2013-08-30 02:04:03 +08:00
|
|
|
static int efx_mcdi_drv_attach(struct efx_nic *efx, bool driver_operating,
|
|
|
|
bool *was_attached_out);
|
2013-10-09 18:52:43 +08:00
|
|
|
static bool efx_mcdi_poll_once(struct efx_nic *efx);
|
sfc:On MCDI timeout, issue an FLR (and mark MCDI to fail-fast)
When an MCDI command times out (whether or not we find it
completed when we poll), call efx_mcdi_abandon(), which tells
all subsequent MCDI calls to fail-fast, and queues up an FLR.
Because an FLR doesn't lead to receiving any reboot even from
the MC (unlike most other types of reset), we have to call
efx_ef10_reset_mc_allocations.
In efx_start_all(), if a reset (of any kind) is pending, we
bail out.
Without this, attempts to reconfigure (e.g. change mtu) can
cause driver/mc state inconsistency if the first MCDI call
triggers an FLR.
For similar reasons, on EF10, in
efx_reset_down(method=RESET_TYPE_MCDI_TIMEOUT), set the number
of active queues to zero before calling efx_stop_all().
And, on farch, in efx_reset_up(method=RESET_TYPE_MCDI_TIMEOUT),
set active_queues and flushes pending & outstanding to zero.
efx_mcdi_mode_{poll,event}() should not take us out of fail-fast
mode. Instead, this is done by efx_mcdi_reset() after the FLR
completes.
The new FLR reset_type RESET_TYPE_MCDI_TIMEOUT doesn't really
fit into the hierarchy of reset 'scopes' whereby efx_reset()
decides some resets subsume others. Thus, it uses separate logic.
Also, fixed up some inconsistency around RESET_TYPE_MC_BIST,
which was in the wrong place in that hierarchy.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-04-17 02:27:48 +08:00
|
|
|
static void efx_mcdi_abandon(struct efx_nic *efx);
|
2009-11-29 23:15:25 +08:00
|
|
|
|
2015-05-27 20:14:26 +08:00
|
|
|
#ifdef CONFIG_SFC_MCDI_LOGGING
|
|
|
|
static bool mcdi_logging_default;
|
|
|
|
module_param(mcdi_logging_default, bool, 0644);
|
|
|
|
MODULE_PARM_DESC(mcdi_logging_default,
|
|
|
|
"Enable MCDI logging on newly-probed functions");
|
|
|
|
#endif
|
|
|
|
|
2012-09-18 09:33:55 +08:00
|
|
|
int efx_mcdi_init(struct efx_nic *efx)
|
2009-11-29 23:15:25 +08:00
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi;
|
2013-08-30 02:04:03 +08:00
|
|
|
bool already_attached;
|
2015-05-27 20:13:54 +08:00
|
|
|
int rc = -ENOMEM;
|
2009-11-29 23:15:25 +08:00
|
|
|
|
2012-09-18 09:33:56 +08:00
|
|
|
efx->mcdi = kzalloc(sizeof(*efx->mcdi), GFP_KERNEL);
|
|
|
|
if (!efx->mcdi)
|
2015-05-27 20:13:54 +08:00
|
|
|
goto fail;
|
2012-09-18 09:33:56 +08:00
|
|
|
|
2009-11-29 23:15:25 +08:00
|
|
|
mcdi = efx_mcdi(efx);
|
2013-08-28 06:12:31 +08:00
|
|
|
mcdi->efx = efx;
|
2015-05-27 20:13:54 +08:00
|
|
|
#ifdef CONFIG_SFC_MCDI_LOGGING
|
|
|
|
/* consuming code assumes buffer is page-sized */
|
|
|
|
mcdi->logging_buffer = (char *)__get_free_page(GFP_KERNEL);
|
|
|
|
if (!mcdi->logging_buffer)
|
|
|
|
goto fail1;
|
2015-05-27 20:14:26 +08:00
|
|
|
mcdi->logging_enabled = mcdi_logging_default;
|
2015-05-27 20:13:54 +08:00
|
|
|
#endif
|
2009-11-29 23:15:25 +08:00
|
|
|
init_waitqueue_head(&mcdi->wq);
|
2015-12-23 16:57:18 +08:00
|
|
|
init_waitqueue_head(&mcdi->proxy_rx_wq);
|
2009-11-29 23:15:25 +08:00
|
|
|
spin_lock_init(&mcdi->iface_lock);
|
2013-08-28 06:04:29 +08:00
|
|
|
mcdi->state = MCDI_STATE_QUIESCENT;
|
2009-11-29 23:15:25 +08:00
|
|
|
mcdi->mode = MCDI_MODE_POLL;
|
2013-08-28 06:12:31 +08:00
|
|
|
spin_lock_init(&mcdi->async_lock);
|
|
|
|
INIT_LIST_HEAD(&mcdi->async_list);
|
2017-10-24 16:45:59 +08:00
|
|
|
timer_setup(&mcdi->async_timer, efx_mcdi_timeout_async, 0);
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
(void) efx_mcdi_poll_reboot(efx);
|
2013-06-20 18:40:07 +08:00
|
|
|
mcdi->new_epoch = true;
|
2012-09-18 09:33:55 +08:00
|
|
|
|
|
|
|
/* Recover from a failed assertion before probing */
|
2013-08-30 02:04:03 +08:00
|
|
|
rc = efx_mcdi_handle_assertion(efx);
|
|
|
|
if (rc)
|
2015-05-27 20:13:54 +08:00
|
|
|
goto fail2;
|
2013-08-30 02:04:03 +08:00
|
|
|
|
|
|
|
/* Let the MC (and BMC, if this is a LOM) know that the driver
|
|
|
|
* is loaded. We should do this before we reset the NIC.
|
|
|
|
*/
|
|
|
|
rc = efx_mcdi_drv_attach(efx, true, &already_attached);
|
|
|
|
if (rc) {
|
|
|
|
netif_err(efx, probe, efx->net_dev,
|
|
|
|
"Unable to register driver with MCPU\n");
|
2015-05-27 20:13:54 +08:00
|
|
|
goto fail2;
|
2013-08-30 02:04:03 +08:00
|
|
|
}
|
|
|
|
if (already_attached)
|
|
|
|
/* Not a fatal error */
|
|
|
|
netif_err(efx, probe, efx->net_dev,
|
|
|
|
"Host already registered with MCPU\n");
|
|
|
|
|
2013-10-19 02:21:45 +08:00
|
|
|
if (efx->mcdi->fn_flags &
|
|
|
|
(1 << MC_CMD_DRV_ATTACH_EXT_OUT_FLAG_PRIMARY))
|
|
|
|
efx->primary = efx;
|
|
|
|
|
2013-08-30 02:04:03 +08:00
|
|
|
return 0;
|
2015-05-27 20:13:54 +08:00
|
|
|
fail2:
|
|
|
|
#ifdef CONFIG_SFC_MCDI_LOGGING
|
|
|
|
free_page((unsigned long)mcdi->logging_buffer);
|
|
|
|
fail1:
|
|
|
|
#endif
|
|
|
|
kfree(efx->mcdi);
|
|
|
|
efx->mcdi = NULL;
|
|
|
|
fail:
|
|
|
|
return rc;
|
2009-11-29 23:15:25 +08:00
|
|
|
}
|
|
|
|
|
2017-02-09 00:52:10 +08:00
|
|
|
void efx_mcdi_detach(struct efx_nic *efx)
|
2012-09-18 09:33:56 +08:00
|
|
|
{
|
2013-08-30 02:04:03 +08:00
|
|
|
if (!efx->mcdi)
|
|
|
|
return;
|
|
|
|
|
|
|
|
BUG_ON(efx->mcdi->iface.state != MCDI_STATE_QUIESCENT);
|
|
|
|
|
|
|
|
/* Relinquish the device (back to the BMC, if this is a LOM) */
|
|
|
|
efx_mcdi_drv_attach(efx, false, NULL);
|
2017-02-09 00:52:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void efx_mcdi_fini(struct efx_nic *efx)
|
|
|
|
{
|
|
|
|
if (!efx->mcdi)
|
|
|
|
return;
|
2013-08-30 02:04:03 +08:00
|
|
|
|
2015-05-27 20:13:54 +08:00
|
|
|
#ifdef CONFIG_SFC_MCDI_LOGGING
|
|
|
|
free_page((unsigned long)efx->mcdi->iface.logging_buffer);
|
|
|
|
#endif
|
|
|
|
|
2012-09-18 09:33:56 +08:00
|
|
|
kfree(efx->mcdi);
|
|
|
|
}
|
|
|
|
|
2013-08-23 05:06:09 +08:00
|
|
|
static void efx_mcdi_send_request(struct efx_nic *efx, unsigned cmd,
|
|
|
|
const efx_dword_t *inbuf, size_t inlen)
|
2009-11-29 23:15:25 +08:00
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
2015-05-27 20:13:54 +08:00
|
|
|
#ifdef CONFIG_SFC_MCDI_LOGGING
|
|
|
|
char *buf = mcdi->logging_buffer; /* page-sized */
|
|
|
|
#endif
|
2012-09-19 07:56:18 +08:00
|
|
|
efx_dword_t hdr[2];
|
|
|
|
size_t hdr_len;
|
2009-11-29 23:15:25 +08:00
|
|
|
u32 xflags, seqno;
|
|
|
|
|
2013-08-28 06:04:29 +08:00
|
|
|
BUG_ON(mcdi->state == MCDI_STATE_QUIESCENT);
|
2009-11-29 23:15:25 +08:00
|
|
|
|
2013-08-23 05:06:09 +08:00
|
|
|
/* Serialise with efx_mcdi_ev_cpl() and efx_mcdi_ev_death() */
|
|
|
|
spin_lock_bh(&mcdi->iface_lock);
|
|
|
|
++mcdi->seqno;
|
|
|
|
spin_unlock_bh(&mcdi->iface_lock);
|
|
|
|
|
2009-11-29 23:15:25 +08:00
|
|
|
seqno = mcdi->seqno & SEQ_MASK;
|
|
|
|
xflags = 0;
|
|
|
|
if (mcdi->mode == MCDI_MODE_EVENTS)
|
|
|
|
xflags |= MCDI_HEADER_XFLAGS_EVREQ;
|
|
|
|
|
2012-09-19 07:56:18 +08:00
|
|
|
if (efx->type->mcdi_max_ver == 1) {
|
|
|
|
/* MCDI v1 */
|
2013-06-20 18:40:07 +08:00
|
|
|
EFX_POPULATE_DWORD_7(hdr[0],
|
2012-09-19 07:56:18 +08:00
|
|
|
MCDI_HEADER_RESPONSE, 0,
|
|
|
|
MCDI_HEADER_RESYNC, 1,
|
|
|
|
MCDI_HEADER_CODE, cmd,
|
|
|
|
MCDI_HEADER_DATALEN, inlen,
|
|
|
|
MCDI_HEADER_SEQ, seqno,
|
2013-06-20 18:40:07 +08:00
|
|
|
MCDI_HEADER_XFLAGS, xflags,
|
|
|
|
MCDI_HEADER_NOT_EPOCH, !mcdi->new_epoch);
|
2012-09-19 07:56:18 +08:00
|
|
|
hdr_len = 4;
|
|
|
|
} else {
|
|
|
|
/* MCDI v2 */
|
|
|
|
BUG_ON(inlen > MCDI_CTL_SDU_LEN_MAX_V2);
|
2013-06-20 18:40:07 +08:00
|
|
|
EFX_POPULATE_DWORD_7(hdr[0],
|
2012-09-19 07:56:18 +08:00
|
|
|
MCDI_HEADER_RESPONSE, 0,
|
|
|
|
MCDI_HEADER_RESYNC, 1,
|
|
|
|
MCDI_HEADER_CODE, MC_CMD_V2_EXTN,
|
|
|
|
MCDI_HEADER_DATALEN, 0,
|
|
|
|
MCDI_HEADER_SEQ, seqno,
|
2013-06-20 18:40:07 +08:00
|
|
|
MCDI_HEADER_XFLAGS, xflags,
|
|
|
|
MCDI_HEADER_NOT_EPOCH, !mcdi->new_epoch);
|
2012-09-19 07:56:18 +08:00
|
|
|
EFX_POPULATE_DWORD_2(hdr[1],
|
|
|
|
MC_CMD_V2_EXTN_IN_EXTENDED_CMD, cmd,
|
|
|
|
MC_CMD_V2_EXTN_IN_ACTUAL_LEN, inlen);
|
|
|
|
hdr_len = 8;
|
|
|
|
}
|
2009-11-29 23:15:25 +08:00
|
|
|
|
2015-05-27 20:13:54 +08:00
|
|
|
#ifdef CONFIG_SFC_MCDI_LOGGING
|
2015-05-27 20:14:01 +08:00
|
|
|
if (mcdi->logging_enabled && !WARN_ON_ONCE(!buf)) {
|
2015-05-27 20:13:54 +08:00
|
|
|
int bytes = 0;
|
|
|
|
int i;
|
|
|
|
/* Lengths should always be a whole number of dwords, so scream
|
|
|
|
* if they're not.
|
|
|
|
*/
|
|
|
|
WARN_ON_ONCE(hdr_len % 4);
|
|
|
|
WARN_ON_ONCE(inlen % 4);
|
|
|
|
|
|
|
|
/* We own the logging buffer, as only one MCDI can be in
|
|
|
|
* progress on a NIC at any one time. So no need for locking.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < hdr_len / 4 && bytes < PAGE_SIZE; i++)
|
|
|
|
bytes += snprintf(buf + bytes, PAGE_SIZE - bytes,
|
|
|
|
" %08x", le32_to_cpu(hdr[i].u32[0]));
|
|
|
|
|
|
|
|
for (i = 0; i < inlen / 4 && bytes < PAGE_SIZE; i++)
|
|
|
|
bytes += snprintf(buf + bytes, PAGE_SIZE - bytes,
|
|
|
|
" %08x", le32_to_cpu(inbuf[i].u32[0]));
|
|
|
|
|
|
|
|
netif_info(efx, hw, efx->net_dev, "MCDI RPC REQ:%s\n", buf);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2012-09-19 07:56:18 +08:00
|
|
|
efx->type->mcdi_request(efx, hdr, hdr_len, inbuf, inlen);
|
2013-08-23 05:06:09 +08:00
|
|
|
|
|
|
|
mcdi->new_epoch = false;
|
2009-11-29 23:15:25 +08:00
|
|
|
}
|
|
|
|
|
2012-10-09 04:43:00 +08:00
|
|
|
static int efx_mcdi_errno(unsigned int mcdi_err)
|
|
|
|
{
|
|
|
|
switch (mcdi_err) {
|
|
|
|
case 0:
|
|
|
|
return 0;
|
|
|
|
#define TRANSLATE_ERROR(name) \
|
|
|
|
case MC_CMD_ERR_ ## name: \
|
|
|
|
return -name;
|
2012-09-19 07:56:18 +08:00
|
|
|
TRANSLATE_ERROR(EPERM);
|
2012-10-09 04:43:00 +08:00
|
|
|
TRANSLATE_ERROR(ENOENT);
|
|
|
|
TRANSLATE_ERROR(EINTR);
|
2012-09-19 07:56:18 +08:00
|
|
|
TRANSLATE_ERROR(EAGAIN);
|
2012-10-09 04:43:00 +08:00
|
|
|
TRANSLATE_ERROR(EACCES);
|
|
|
|
TRANSLATE_ERROR(EBUSY);
|
|
|
|
TRANSLATE_ERROR(EINVAL);
|
|
|
|
TRANSLATE_ERROR(EDEADLK);
|
|
|
|
TRANSLATE_ERROR(ENOSYS);
|
|
|
|
TRANSLATE_ERROR(ETIME);
|
2012-09-19 07:56:18 +08:00
|
|
|
TRANSLATE_ERROR(EALREADY);
|
|
|
|
TRANSLATE_ERROR(ENOSPC);
|
2012-10-09 04:43:00 +08:00
|
|
|
#undef TRANSLATE_ERROR
|
2013-10-08 23:36:58 +08:00
|
|
|
case MC_CMD_ERR_ENOTSUP:
|
|
|
|
return -EOPNOTSUPP;
|
2012-09-19 07:56:18 +08:00
|
|
|
case MC_CMD_ERR_ALLOC_FAIL:
|
|
|
|
return -ENOBUFS;
|
|
|
|
case MC_CMD_ERR_MAC_EXIST:
|
|
|
|
return -EADDRINUSE;
|
2012-10-09 04:43:00 +08:00
|
|
|
default:
|
2012-09-19 07:56:18 +08:00
|
|
|
return -EPROTO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void efx_mcdi_read_response_header(struct efx_nic *efx)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
|
|
|
unsigned int respseq, respcmd, error;
|
2015-05-27 20:13:54 +08:00
|
|
|
#ifdef CONFIG_SFC_MCDI_LOGGING
|
|
|
|
char *buf = mcdi->logging_buffer; /* page-sized */
|
|
|
|
#endif
|
2012-09-19 07:56:18 +08:00
|
|
|
efx_dword_t hdr;
|
|
|
|
|
|
|
|
efx->type->mcdi_read_response(efx, &hdr, 0, 4);
|
|
|
|
respseq = EFX_DWORD_FIELD(hdr, MCDI_HEADER_SEQ);
|
|
|
|
respcmd = EFX_DWORD_FIELD(hdr, MCDI_HEADER_CODE);
|
|
|
|
error = EFX_DWORD_FIELD(hdr, MCDI_HEADER_ERROR);
|
|
|
|
|
|
|
|
if (respcmd != MC_CMD_V2_EXTN) {
|
|
|
|
mcdi->resp_hdr_len = 4;
|
|
|
|
mcdi->resp_data_len = EFX_DWORD_FIELD(hdr, MCDI_HEADER_DATALEN);
|
|
|
|
} else {
|
|
|
|
efx->type->mcdi_read_response(efx, &hdr, 4, 4);
|
|
|
|
mcdi->resp_hdr_len = 8;
|
|
|
|
mcdi->resp_data_len =
|
|
|
|
EFX_DWORD_FIELD(hdr, MC_CMD_V2_EXTN_IN_ACTUAL_LEN);
|
|
|
|
}
|
|
|
|
|
2015-05-27 20:13:54 +08:00
|
|
|
#ifdef CONFIG_SFC_MCDI_LOGGING
|
2015-05-27 20:14:01 +08:00
|
|
|
if (mcdi->logging_enabled && !WARN_ON_ONCE(!buf)) {
|
2015-05-27 20:13:54 +08:00
|
|
|
size_t hdr_len, data_len;
|
|
|
|
int bytes = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
WARN_ON_ONCE(mcdi->resp_hdr_len % 4);
|
|
|
|
hdr_len = mcdi->resp_hdr_len / 4;
|
|
|
|
/* MCDI_DECLARE_BUF ensures that underlying buffer is padded
|
|
|
|
* to dword size, and the MCDI buffer is always dword size
|
|
|
|
*/
|
|
|
|
data_len = DIV_ROUND_UP(mcdi->resp_data_len, 4);
|
|
|
|
|
|
|
|
/* We own the logging buffer, as only one MCDI can be in
|
|
|
|
* progress on a NIC at any one time. So no need for locking.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < hdr_len && bytes < PAGE_SIZE; i++) {
|
|
|
|
efx->type->mcdi_read_response(efx, &hdr, (i * 4), 4);
|
|
|
|
bytes += snprintf(buf + bytes, PAGE_SIZE - bytes,
|
|
|
|
" %08x", le32_to_cpu(hdr.u32[0]));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < data_len && bytes < PAGE_SIZE; i++) {
|
|
|
|
efx->type->mcdi_read_response(efx, &hdr,
|
|
|
|
mcdi->resp_hdr_len + (i * 4), 4);
|
|
|
|
bytes += snprintf(buf + bytes, PAGE_SIZE - bytes,
|
|
|
|
" %08x", le32_to_cpu(hdr.u32[0]));
|
|
|
|
}
|
|
|
|
|
|
|
|
netif_info(efx, hw, efx->net_dev, "MCDI RPC RESP:%s\n", buf);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2015-12-23 16:56:40 +08:00
|
|
|
mcdi->resprc_raw = 0;
|
2012-09-19 07:56:18 +08:00
|
|
|
if (error && mcdi->resp_data_len == 0) {
|
|
|
|
netif_err(efx, hw, efx->net_dev, "MC rebooted\n");
|
|
|
|
mcdi->resprc = -EIO;
|
|
|
|
} else if ((respseq ^ mcdi->seqno) & SEQ_MASK) {
|
|
|
|
netif_err(efx, hw, efx->net_dev,
|
|
|
|
"MC response mismatch tx seq 0x%x rx seq 0x%x\n",
|
|
|
|
respseq, mcdi->seqno);
|
|
|
|
mcdi->resprc = -EIO;
|
|
|
|
} else if (error) {
|
|
|
|
efx->type->mcdi_read_response(efx, &hdr, mcdi->resp_hdr_len, 4);
|
2015-12-23 16:56:40 +08:00
|
|
|
mcdi->resprc_raw = EFX_DWORD_FIELD(hdr, EFX_DWORD_0);
|
|
|
|
mcdi->resprc = efx_mcdi_errno(mcdi->resprc_raw);
|
2012-09-19 07:56:18 +08:00
|
|
|
} else {
|
|
|
|
mcdi->resprc = 0;
|
2012-10-09 04:43:00 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-09 18:52:43 +08:00
|
|
|
static bool efx_mcdi_poll_once(struct efx_nic *efx)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
|
|
|
|
|
|
|
rmb();
|
|
|
|
if (!efx->type->mcdi_poll_response(efx))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
spin_lock_bh(&mcdi->iface_lock);
|
|
|
|
efx_mcdi_read_response_header(efx);
|
|
|
|
spin_unlock_bh(&mcdi->iface_lock);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-11-29 23:15:25 +08:00
|
|
|
static int efx_mcdi_poll(struct efx_nic *efx)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
2012-12-01 10:21:17 +08:00
|
|
|
unsigned long time, finish;
|
2012-10-09 04:43:00 +08:00
|
|
|
unsigned int spins;
|
|
|
|
int rc;
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
/* Check for a reboot atomically with respect to efx_mcdi_copyout() */
|
2012-10-09 04:43:00 +08:00
|
|
|
rc = efx_mcdi_poll_reboot(efx);
|
2012-09-19 07:56:18 +08:00
|
|
|
if (rc) {
|
2012-10-27 00:53:12 +08:00
|
|
|
spin_lock_bh(&mcdi->iface_lock);
|
2012-09-19 07:56:18 +08:00
|
|
|
mcdi->resprc = rc;
|
|
|
|
mcdi->resp_hdr_len = 0;
|
|
|
|
mcdi->resp_data_len = 0;
|
2012-10-27 00:53:12 +08:00
|
|
|
spin_unlock_bh(&mcdi->iface_lock);
|
2012-09-19 07:56:18 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
/* Poll for completion. Poll quickly (once a us) for the 1st jiffy,
|
|
|
|
* because generally mcdi responses are fast. After that, back off
|
|
|
|
* and poll once a jiffy (approximately)
|
|
|
|
*/
|
2018-03-20 17:11:28 +08:00
|
|
|
spins = USER_TICK_USEC;
|
2012-12-01 10:21:17 +08:00
|
|
|
finish = jiffies + MCDI_RPC_TIMEOUT;
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
while (1) {
|
|
|
|
if (spins != 0) {
|
|
|
|
--spins;
|
|
|
|
udelay(1);
|
2010-01-13 12:34:25 +08:00
|
|
|
} else {
|
|
|
|
schedule_timeout_uninterruptible(1);
|
|
|
|
}
|
2009-11-29 23:15:25 +08:00
|
|
|
|
2012-12-01 10:21:17 +08:00
|
|
|
time = jiffies;
|
2009-11-29 23:15:25 +08:00
|
|
|
|
2013-10-09 18:52:43 +08:00
|
|
|
if (efx_mcdi_poll_once(efx))
|
2009-11-29 23:15:25 +08:00
|
|
|
break;
|
|
|
|
|
2012-12-01 10:21:17 +08:00
|
|
|
if (time_after(time, finish))
|
2009-11-29 23:15:25 +08:00
|
|
|
return -ETIMEDOUT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Return rc=0 like wait_event_timeout() */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-10-02 03:58:35 +08:00
|
|
|
/* Test and clear MC-rebooted flag for this port/function; reset
|
|
|
|
* software state as necessary.
|
|
|
|
*/
|
2009-11-29 23:15:25 +08:00
|
|
|
int efx_mcdi_poll_reboot(struct efx_nic *efx)
|
|
|
|
{
|
2012-09-18 09:33:56 +08:00
|
|
|
if (!efx->mcdi)
|
|
|
|
return 0;
|
2009-11-29 23:15:25 +08:00
|
|
|
|
2012-12-15 05:52:56 +08:00
|
|
|
return efx->type->mcdi_poll_reboot(efx);
|
2009-11-29 23:15:25 +08:00
|
|
|
}
|
|
|
|
|
2013-08-28 06:12:31 +08:00
|
|
|
static bool efx_mcdi_acquire_async(struct efx_mcdi_iface *mcdi)
|
|
|
|
{
|
|
|
|
return cmpxchg(&mcdi->state,
|
|
|
|
MCDI_STATE_QUIESCENT, MCDI_STATE_RUNNING_ASYNC) ==
|
|
|
|
MCDI_STATE_QUIESCENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void efx_mcdi_acquire_sync(struct efx_mcdi_iface *mcdi)
|
2009-11-29 23:15:25 +08:00
|
|
|
{
|
|
|
|
/* Wait until the interface becomes QUIESCENT and we win the race
|
2013-08-28 06:12:31 +08:00
|
|
|
* to mark it RUNNING_SYNC.
|
|
|
|
*/
|
2009-11-29 23:15:25 +08:00
|
|
|
wait_event(mcdi->wq,
|
2013-08-28 06:04:29 +08:00
|
|
|
cmpxchg(&mcdi->state,
|
2013-08-28 06:12:31 +08:00
|
|
|
MCDI_STATE_QUIESCENT, MCDI_STATE_RUNNING_SYNC) ==
|
2013-08-28 06:04:29 +08:00
|
|
|
MCDI_STATE_QUIESCENT);
|
2009-11-29 23:15:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int efx_mcdi_await_completion(struct efx_nic *efx)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
|
|
|
|
2013-08-28 06:04:29 +08:00
|
|
|
if (wait_event_timeout(mcdi->wq, mcdi->state == MCDI_STATE_COMPLETED,
|
|
|
|
MCDI_RPC_TIMEOUT) == 0)
|
2009-11-29 23:15:25 +08:00
|
|
|
return -ETIMEDOUT;
|
|
|
|
|
|
|
|
/* Check if efx_mcdi_set_mode() switched us back to polled completions.
|
|
|
|
* In which case, poll for completions directly. If efx_mcdi_ev_cpl()
|
|
|
|
* completed the request first, then we'll just end up completing the
|
|
|
|
* request again, which is safe.
|
|
|
|
*
|
|
|
|
* We need an smp_rmb() to synchronise with efx_mcdi_mode_poll(), which
|
|
|
|
* wait_event_timeout() implicitly provides.
|
|
|
|
*/
|
|
|
|
if (mcdi->mode == MCDI_MODE_POLL)
|
|
|
|
return efx_mcdi_poll(efx);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-08-28 06:12:31 +08:00
|
|
|
/* If the interface is RUNNING_SYNC, switch to COMPLETED and wake the
|
|
|
|
* requester. Return whether this was done. Does not take any locks.
|
|
|
|
*/
|
|
|
|
static bool efx_mcdi_complete_sync(struct efx_mcdi_iface *mcdi)
|
2009-11-29 23:15:25 +08:00
|
|
|
{
|
2013-08-28 06:12:31 +08:00
|
|
|
if (cmpxchg(&mcdi->state,
|
|
|
|
MCDI_STATE_RUNNING_SYNC, MCDI_STATE_COMPLETED) ==
|
|
|
|
MCDI_STATE_RUNNING_SYNC) {
|
2009-11-29 23:15:25 +08:00
|
|
|
wake_up(&mcdi->wq);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void efx_mcdi_release(struct efx_mcdi_iface *mcdi)
|
|
|
|
{
|
2013-08-28 06:12:31 +08:00
|
|
|
if (mcdi->mode == MCDI_MODE_EVENTS) {
|
|
|
|
struct efx_mcdi_async_param *async;
|
|
|
|
struct efx_nic *efx = mcdi->efx;
|
|
|
|
|
|
|
|
/* Process the asynchronous request queue */
|
|
|
|
spin_lock_bh(&mcdi->async_lock);
|
|
|
|
async = list_first_entry_or_null(
|
|
|
|
&mcdi->async_list, struct efx_mcdi_async_param, list);
|
|
|
|
if (async) {
|
|
|
|
mcdi->state = MCDI_STATE_RUNNING_ASYNC;
|
|
|
|
efx_mcdi_send_request(efx, async->cmd,
|
|
|
|
(const efx_dword_t *)(async + 1),
|
|
|
|
async->inlen);
|
|
|
|
mod_timer(&mcdi->async_timer,
|
|
|
|
jiffies + MCDI_RPC_TIMEOUT);
|
|
|
|
}
|
|
|
|
spin_unlock_bh(&mcdi->async_lock);
|
|
|
|
|
|
|
|
if (async)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-08-28 06:04:29 +08:00
|
|
|
mcdi->state = MCDI_STATE_QUIESCENT;
|
2009-11-29 23:15:25 +08:00
|
|
|
wake_up(&mcdi->wq);
|
|
|
|
}
|
|
|
|
|
2013-08-28 06:12:31 +08:00
|
|
|
/* If the interface is RUNNING_ASYNC, switch to COMPLETED, call the
|
|
|
|
* asynchronous completion function, and release the interface.
|
|
|
|
* Return whether this was done. Must be called in bh-disabled
|
|
|
|
* context. Will take iface_lock and async_lock.
|
|
|
|
*/
|
|
|
|
static bool efx_mcdi_complete_async(struct efx_mcdi_iface *mcdi, bool timeout)
|
|
|
|
{
|
|
|
|
struct efx_nic *efx = mcdi->efx;
|
|
|
|
struct efx_mcdi_async_param *async;
|
2013-06-01 01:36:12 +08:00
|
|
|
size_t hdr_len, data_len, err_len;
|
2013-08-28 06:12:31 +08:00
|
|
|
efx_dword_t *outbuf;
|
2015-05-20 18:10:41 +08:00
|
|
|
MCDI_DECLARE_BUF_ERR(errbuf);
|
2013-08-28 06:12:31 +08:00
|
|
|
int rc;
|
|
|
|
|
|
|
|
if (cmpxchg(&mcdi->state,
|
|
|
|
MCDI_STATE_RUNNING_ASYNC, MCDI_STATE_COMPLETED) !=
|
|
|
|
MCDI_STATE_RUNNING_ASYNC)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
spin_lock(&mcdi->iface_lock);
|
|
|
|
if (timeout) {
|
|
|
|
/* Ensure that if the completion event arrives later,
|
|
|
|
* the seqno check in efx_mcdi_ev_cpl() will fail
|
|
|
|
*/
|
|
|
|
++mcdi->seqno;
|
|
|
|
++mcdi->credits;
|
|
|
|
rc = -ETIMEDOUT;
|
|
|
|
hdr_len = 0;
|
|
|
|
data_len = 0;
|
|
|
|
} else {
|
|
|
|
rc = mcdi->resprc;
|
|
|
|
hdr_len = mcdi->resp_hdr_len;
|
|
|
|
data_len = mcdi->resp_data_len;
|
|
|
|
}
|
|
|
|
spin_unlock(&mcdi->iface_lock);
|
|
|
|
|
|
|
|
/* Stop the timer. In case the timer function is running, we
|
|
|
|
* must wait for it to return so that there is no possibility
|
|
|
|
* of it aborting the next request.
|
|
|
|
*/
|
|
|
|
if (!timeout)
|
|
|
|
del_timer_sync(&mcdi->async_timer);
|
|
|
|
|
|
|
|
spin_lock(&mcdi->async_lock);
|
|
|
|
async = list_first_entry(&mcdi->async_list,
|
|
|
|
struct efx_mcdi_async_param, list);
|
|
|
|
list_del(&async->list);
|
|
|
|
spin_unlock(&mcdi->async_lock);
|
|
|
|
|
|
|
|
outbuf = (efx_dword_t *)(async + 1);
|
|
|
|
efx->type->mcdi_read_response(efx, outbuf, hdr_len,
|
|
|
|
min(async->outlen, data_len));
|
2013-06-01 01:36:12 +08:00
|
|
|
if (!timeout && rc && !async->quiet) {
|
|
|
|
err_len = min(sizeof(errbuf), data_len);
|
|
|
|
efx->type->mcdi_read_response(efx, errbuf, hdr_len,
|
|
|
|
sizeof(errbuf));
|
|
|
|
efx_mcdi_display_error(efx, async->cmd, async->inlen, errbuf,
|
|
|
|
err_len, rc);
|
|
|
|
}
|
2016-08-11 20:01:21 +08:00
|
|
|
|
|
|
|
if (async->complete)
|
|
|
|
async->complete(efx, async->cookie, rc, outbuf,
|
|
|
|
min(async->outlen, data_len));
|
2013-08-28 06:12:31 +08:00
|
|
|
kfree(async);
|
|
|
|
|
|
|
|
efx_mcdi_release(mcdi);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-11-29 23:15:25 +08:00
|
|
|
static void efx_mcdi_ev_cpl(struct efx_nic *efx, unsigned int seqno,
|
2012-10-09 04:43:00 +08:00
|
|
|
unsigned int datalen, unsigned int mcdi_err)
|
2009-11-29 23:15:25 +08:00
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
|
|
|
bool wake = false;
|
|
|
|
|
|
|
|
spin_lock(&mcdi->iface_lock);
|
|
|
|
|
|
|
|
if ((seqno ^ mcdi->seqno) & SEQ_MASK) {
|
|
|
|
if (mcdi->credits)
|
|
|
|
/* The request has been cancelled */
|
|
|
|
--mcdi->credits;
|
|
|
|
else
|
2010-06-23 19:30:07 +08:00
|
|
|
netif_err(efx, hw, efx->net_dev,
|
|
|
|
"MC response mismatch tx seq 0x%x rx "
|
|
|
|
"seq 0x%x\n", seqno, mcdi->seqno);
|
2009-11-29 23:15:25 +08:00
|
|
|
} else {
|
2012-09-19 07:56:18 +08:00
|
|
|
if (efx->type->mcdi_max_ver >= 2) {
|
|
|
|
/* MCDI v2 responses don't fit in an event */
|
|
|
|
efx_mcdi_read_response_header(efx);
|
|
|
|
} else {
|
|
|
|
mcdi->resprc = efx_mcdi_errno(mcdi_err);
|
|
|
|
mcdi->resp_hdr_len = 4;
|
|
|
|
mcdi->resp_data_len = datalen;
|
|
|
|
}
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
wake = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_unlock(&mcdi->iface_lock);
|
|
|
|
|
2013-08-28 06:12:31 +08:00
|
|
|
if (wake) {
|
|
|
|
if (!efx_mcdi_complete_async(mcdi, false))
|
|
|
|
(void) efx_mcdi_complete_sync(mcdi);
|
|
|
|
|
|
|
|
/* If the interface isn't RUNNING_ASYNC or
|
|
|
|
* RUNNING_SYNC then we've received a duplicate
|
|
|
|
* completion after we've already transitioned back to
|
|
|
|
* QUIESCENT. [A subsequent invocation would increment
|
|
|
|
* seqno, so would have failed the seqno check].
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-24 16:45:59 +08:00
|
|
|
static void efx_mcdi_timeout_async(struct timer_list *t)
|
2013-08-28 06:12:31 +08:00
|
|
|
{
|
2017-10-24 16:45:59 +08:00
|
|
|
struct efx_mcdi_iface *mcdi = from_timer(mcdi, t, async_timer);
|
2013-08-28 06:12:31 +08:00
|
|
|
|
|
|
|
efx_mcdi_complete_async(mcdi, true);
|
2009-11-29 23:15:25 +08:00
|
|
|
}
|
|
|
|
|
2013-08-23 05:06:09 +08:00
|
|
|
static int
|
|
|
|
efx_mcdi_check_supported(struct efx_nic *efx, unsigned int cmd, size_t inlen)
|
|
|
|
{
|
|
|
|
if (efx->type->mcdi_max_ver < 0 ||
|
|
|
|
(efx->type->mcdi_max_ver < 2 &&
|
|
|
|
cmd > MC_CMD_CMD_SPACE_ESCAPE_7))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (inlen > MCDI_CTL_SDU_LEN_MAX_V2 ||
|
|
|
|
(efx->type->mcdi_max_ver < 2 &&
|
|
|
|
inlen > MCDI_CTL_SDU_LEN_MAX_V1))
|
|
|
|
return -EMSGSIZE;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-12-23 16:57:18 +08:00
|
|
|
static bool efx_mcdi_get_proxy_handle(struct efx_nic *efx,
|
|
|
|
size_t hdr_len, size_t data_len,
|
|
|
|
u32 *proxy_handle)
|
|
|
|
{
|
|
|
|
MCDI_DECLARE_BUF_ERR(testbuf);
|
|
|
|
const size_t buflen = sizeof(testbuf);
|
|
|
|
|
|
|
|
if (!proxy_handle || data_len < buflen)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
efx->type->mcdi_read_response(efx, testbuf, hdr_len, buflen);
|
|
|
|
if (MCDI_DWORD(testbuf, ERR_CODE) == MC_CMD_ERR_PROXY_PENDING) {
|
|
|
|
*proxy_handle = MCDI_DWORD(testbuf, ERR_PROXY_PENDING_HANDLE);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _efx_mcdi_rpc_finish(struct efx_nic *efx, unsigned int cmd,
|
|
|
|
size_t inlen,
|
2013-06-01 01:36:12 +08:00
|
|
|
efx_dword_t *outbuf, size_t outlen,
|
2015-12-23 16:56:40 +08:00
|
|
|
size_t *outlen_actual, bool quiet,
|
2015-12-23 16:57:18 +08:00
|
|
|
u32 *proxy_handle, int *raw_rc)
|
2013-06-01 01:36:12 +08:00
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
2015-05-20 18:10:41 +08:00
|
|
|
MCDI_DECLARE_BUF_ERR(errbuf);
|
2013-06-01 01:36:12 +08:00
|
|
|
int rc;
|
|
|
|
|
|
|
|
if (mcdi->mode == MCDI_MODE_POLL)
|
|
|
|
rc = efx_mcdi_poll(efx);
|
|
|
|
else
|
|
|
|
rc = efx_mcdi_await_completion(efx);
|
|
|
|
|
|
|
|
if (rc != 0) {
|
|
|
|
netif_err(efx, hw, efx->net_dev,
|
|
|
|
"MC command 0x%x inlen %d mode %d timed out\n",
|
|
|
|
cmd, (int)inlen, mcdi->mode);
|
|
|
|
|
|
|
|
if (mcdi->mode == MCDI_MODE_EVENTS && efx_mcdi_poll_once(efx)) {
|
|
|
|
netif_err(efx, hw, efx->net_dev,
|
|
|
|
"MCDI request was completed without an event\n");
|
|
|
|
rc = 0;
|
|
|
|
}
|
|
|
|
|
sfc:On MCDI timeout, issue an FLR (and mark MCDI to fail-fast)
When an MCDI command times out (whether or not we find it
completed when we poll), call efx_mcdi_abandon(), which tells
all subsequent MCDI calls to fail-fast, and queues up an FLR.
Because an FLR doesn't lead to receiving any reboot even from
the MC (unlike most other types of reset), we have to call
efx_ef10_reset_mc_allocations.
In efx_start_all(), if a reset (of any kind) is pending, we
bail out.
Without this, attempts to reconfigure (e.g. change mtu) can
cause driver/mc state inconsistency if the first MCDI call
triggers an FLR.
For similar reasons, on EF10, in
efx_reset_down(method=RESET_TYPE_MCDI_TIMEOUT), set the number
of active queues to zero before calling efx_stop_all().
And, on farch, in efx_reset_up(method=RESET_TYPE_MCDI_TIMEOUT),
set active_queues and flushes pending & outstanding to zero.
efx_mcdi_mode_{poll,event}() should not take us out of fail-fast
mode. Instead, this is done by efx_mcdi_reset() after the FLR
completes.
The new FLR reset_type RESET_TYPE_MCDI_TIMEOUT doesn't really
fit into the hierarchy of reset 'scopes' whereby efx_reset()
decides some resets subsume others. Thus, it uses separate logic.
Also, fixed up some inconsistency around RESET_TYPE_MC_BIST,
which was in the wrong place in that hierarchy.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-04-17 02:27:48 +08:00
|
|
|
efx_mcdi_abandon(efx);
|
|
|
|
|
2013-06-01 01:36:12 +08:00
|
|
|
/* Close the race with efx_mcdi_ev_cpl() executing just too late
|
|
|
|
* and completing a request we've just cancelled, by ensuring
|
|
|
|
* that the seqno check therein fails.
|
|
|
|
*/
|
|
|
|
spin_lock_bh(&mcdi->iface_lock);
|
|
|
|
++mcdi->seqno;
|
|
|
|
++mcdi->credits;
|
|
|
|
spin_unlock_bh(&mcdi->iface_lock);
|
|
|
|
}
|
|
|
|
|
2015-12-23 16:57:18 +08:00
|
|
|
if (proxy_handle)
|
|
|
|
*proxy_handle = 0;
|
|
|
|
|
2013-06-01 01:36:12 +08:00
|
|
|
if (rc != 0) {
|
|
|
|
if (outlen_actual)
|
|
|
|
*outlen_actual = 0;
|
|
|
|
} else {
|
|
|
|
size_t hdr_len, data_len, err_len;
|
|
|
|
|
|
|
|
/* At the very least we need a memory barrier here to ensure
|
|
|
|
* we pick up changes from efx_mcdi_ev_cpl(). Protect against
|
|
|
|
* a spurious efx_mcdi_ev_cpl() running concurrently by
|
|
|
|
* acquiring the iface_lock. */
|
|
|
|
spin_lock_bh(&mcdi->iface_lock);
|
|
|
|
rc = mcdi->resprc;
|
2015-12-23 16:56:40 +08:00
|
|
|
if (raw_rc)
|
|
|
|
*raw_rc = mcdi->resprc_raw;
|
2013-06-01 01:36:12 +08:00
|
|
|
hdr_len = mcdi->resp_hdr_len;
|
|
|
|
data_len = mcdi->resp_data_len;
|
|
|
|
err_len = min(sizeof(errbuf), data_len);
|
|
|
|
spin_unlock_bh(&mcdi->iface_lock);
|
|
|
|
|
|
|
|
BUG_ON(rc > 0);
|
|
|
|
|
|
|
|
efx->type->mcdi_read_response(efx, outbuf, hdr_len,
|
|
|
|
min(outlen, data_len));
|
|
|
|
if (outlen_actual)
|
|
|
|
*outlen_actual = data_len;
|
|
|
|
|
|
|
|
efx->type->mcdi_read_response(efx, errbuf, hdr_len, err_len);
|
|
|
|
|
|
|
|
if (cmd == MC_CMD_REBOOT && rc == -EIO) {
|
|
|
|
/* Don't reset if MC_CMD_REBOOT returns EIO */
|
|
|
|
} else if (rc == -EIO || rc == -EINTR) {
|
2017-02-09 00:51:33 +08:00
|
|
|
netif_err(efx, hw, efx->net_dev, "MC reboot detected\n");
|
|
|
|
netif_dbg(efx, hw, efx->net_dev, "MC rebooted during command %d rc %d\n",
|
|
|
|
cmd, -rc);
|
|
|
|
if (efx->type->mcdi_reboot_detected)
|
|
|
|
efx->type->mcdi_reboot_detected(efx);
|
2013-06-01 01:36:12 +08:00
|
|
|
efx_schedule_reset(efx, RESET_TYPE_MC_FAILURE);
|
2015-12-23 16:57:18 +08:00
|
|
|
} else if (proxy_handle && (rc == -EPROTO) &&
|
|
|
|
efx_mcdi_get_proxy_handle(efx, hdr_len, data_len,
|
|
|
|
proxy_handle)) {
|
|
|
|
mcdi->proxy_rx_status = 0;
|
|
|
|
mcdi->proxy_rx_handle = 0;
|
|
|
|
mcdi->state = MCDI_STATE_PROXY_WAIT;
|
2013-06-01 01:36:12 +08:00
|
|
|
} else if (rc && !quiet) {
|
|
|
|
efx_mcdi_display_error(efx, cmd, inlen, errbuf, err_len,
|
|
|
|
rc);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rc == -EIO || rc == -EINTR) {
|
|
|
|
msleep(MCDI_STATUS_SLEEP_MS);
|
|
|
|
efx_mcdi_poll_reboot(efx);
|
|
|
|
mcdi->new_epoch = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-23 16:57:18 +08:00
|
|
|
if (!proxy_handle || !*proxy_handle)
|
|
|
|
efx_mcdi_release(mcdi);
|
2013-06-01 01:36:12 +08:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2015-12-23 16:57:18 +08:00
|
|
|
static void efx_mcdi_proxy_abort(struct efx_mcdi_iface *mcdi)
|
|
|
|
{
|
|
|
|
if (mcdi->state == MCDI_STATE_PROXY_WAIT) {
|
|
|
|
/* Interrupt the proxy wait. */
|
|
|
|
mcdi->proxy_rx_status = -EINTR;
|
|
|
|
wake_up(&mcdi->proxy_rx_wq);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void efx_mcdi_ev_proxy_response(struct efx_nic *efx,
|
|
|
|
u32 handle, int status)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
|
|
|
|
|
|
|
WARN_ON(mcdi->state != MCDI_STATE_PROXY_WAIT);
|
|
|
|
|
|
|
|
mcdi->proxy_rx_status = efx_mcdi_errno(status);
|
|
|
|
/* Ensure the status is written before we update the handle, since the
|
|
|
|
* latter is used to check if we've finished.
|
|
|
|
*/
|
|
|
|
wmb();
|
|
|
|
mcdi->proxy_rx_handle = handle;
|
|
|
|
wake_up(&mcdi->proxy_rx_wq);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int efx_mcdi_proxy_wait(struct efx_nic *efx, u32 handle, bool quiet)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
/* Wait for a proxy event, or timeout. */
|
|
|
|
rc = wait_event_timeout(mcdi->proxy_rx_wq,
|
|
|
|
mcdi->proxy_rx_handle != 0 ||
|
|
|
|
mcdi->proxy_rx_status == -EINTR,
|
|
|
|
MCDI_RPC_TIMEOUT);
|
|
|
|
|
|
|
|
if (rc <= 0) {
|
|
|
|
netif_dbg(efx, hw, efx->net_dev,
|
|
|
|
"MCDI proxy timeout %d\n", handle);
|
|
|
|
return -ETIMEDOUT;
|
|
|
|
} else if (mcdi->proxy_rx_handle != handle) {
|
|
|
|
netif_warn(efx, hw, efx->net_dev,
|
|
|
|
"MCDI proxy unexpected handle %d (expected %d)\n",
|
|
|
|
mcdi->proxy_rx_handle, handle);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return mcdi->proxy_rx_status;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _efx_mcdi_rpc(struct efx_nic *efx, unsigned int cmd,
|
2013-06-01 01:36:12 +08:00
|
|
|
const efx_dword_t *inbuf, size_t inlen,
|
|
|
|
efx_dword_t *outbuf, size_t outlen,
|
2015-12-23 16:56:40 +08:00
|
|
|
size_t *outlen_actual, bool quiet, int *raw_rc)
|
2013-06-01 01:36:12 +08:00
|
|
|
{
|
2015-12-23 16:57:18 +08:00
|
|
|
u32 proxy_handle = 0; /* Zero is an invalid proxy handle. */
|
2013-06-01 01:36:12 +08:00
|
|
|
int rc;
|
|
|
|
|
2015-12-23 16:57:18 +08:00
|
|
|
if (inbuf && inlen && (inbuf == outbuf)) {
|
|
|
|
/* The input buffer can't be aliased with the output. */
|
|
|
|
WARN_ON(1);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2013-06-01 01:36:12 +08:00
|
|
|
rc = efx_mcdi_rpc_start(efx, cmd, inbuf, inlen);
|
2015-12-23 16:56:40 +08:00
|
|
|
if (rc)
|
2013-06-01 01:36:12 +08:00
|
|
|
return rc;
|
2015-12-23 16:56:40 +08:00
|
|
|
|
2015-12-23 16:57:18 +08:00
|
|
|
rc = _efx_mcdi_rpc_finish(efx, cmd, inlen, outbuf, outlen,
|
|
|
|
outlen_actual, quiet, &proxy_handle, raw_rc);
|
|
|
|
|
|
|
|
if (proxy_handle) {
|
|
|
|
/* Handle proxy authorisation. This allows approval of MCDI
|
|
|
|
* operations to be delegated to the admin function, allowing
|
|
|
|
* fine control over (eg) multicast subscriptions.
|
|
|
|
*/
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
|
|
|
|
|
|
|
netif_dbg(efx, hw, efx->net_dev,
|
|
|
|
"MCDI waiting for proxy auth %d\n",
|
|
|
|
proxy_handle);
|
|
|
|
rc = efx_mcdi_proxy_wait(efx, proxy_handle, quiet);
|
|
|
|
|
|
|
|
if (rc == 0) {
|
|
|
|
netif_dbg(efx, hw, efx->net_dev,
|
|
|
|
"MCDI proxy retry %d\n", proxy_handle);
|
|
|
|
|
|
|
|
/* We now retry the original request. */
|
|
|
|
mcdi->state = MCDI_STATE_RUNNING_SYNC;
|
|
|
|
efx_mcdi_send_request(efx, cmd, inbuf, inlen);
|
|
|
|
|
|
|
|
rc = _efx_mcdi_rpc_finish(efx, cmd, inlen,
|
|
|
|
outbuf, outlen, outlen_actual,
|
|
|
|
quiet, NULL, raw_rc);
|
|
|
|
} else {
|
2017-01-27 23:02:39 +08:00
|
|
|
netif_cond_dbg(efx, hw, efx->net_dev, rc == -EPERM, err,
|
|
|
|
"MC command 0x%x failed after proxy auth rc=%d\n",
|
|
|
|
cmd, rc);
|
2015-12-23 16:57:18 +08:00
|
|
|
|
|
|
|
if (rc == -EINTR || rc == -EIO)
|
|
|
|
efx_schedule_reset(efx, RESET_TYPE_MC_FAILURE);
|
|
|
|
efx_mcdi_release(mcdi);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
2015-12-23 16:56:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int _efx_mcdi_rpc_evb_retry(struct efx_nic *efx, unsigned cmd,
|
|
|
|
const efx_dword_t *inbuf, size_t inlen,
|
|
|
|
efx_dword_t *outbuf, size_t outlen,
|
|
|
|
size_t *outlen_actual, bool quiet)
|
|
|
|
{
|
|
|
|
int raw_rc = 0;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
rc = _efx_mcdi_rpc(efx, cmd, inbuf, inlen,
|
|
|
|
outbuf, outlen, outlen_actual, true, &raw_rc);
|
|
|
|
|
|
|
|
if ((rc == -EPROTO) && (raw_rc == MC_CMD_ERR_NO_EVB_PORT) &&
|
|
|
|
efx->type->is_vf) {
|
|
|
|
/* If the EVB port isn't available within a VF this may
|
|
|
|
* mean the PF is still bringing the switch up. We should
|
|
|
|
* retry our request shortly.
|
|
|
|
*/
|
|
|
|
unsigned long abort_time = jiffies + MCDI_RPC_TIMEOUT;
|
|
|
|
unsigned int delay_us = 10000;
|
|
|
|
|
|
|
|
netif_dbg(efx, hw, efx->net_dev,
|
|
|
|
"%s: NO_EVB_PORT; will retry request\n",
|
|
|
|
__func__);
|
|
|
|
|
|
|
|
do {
|
|
|
|
usleep_range(delay_us, delay_us + 10000);
|
|
|
|
rc = _efx_mcdi_rpc(efx, cmd, inbuf, inlen,
|
|
|
|
outbuf, outlen, outlen_actual,
|
|
|
|
true, &raw_rc);
|
|
|
|
if (delay_us < 100000)
|
|
|
|
delay_us <<= 1;
|
|
|
|
} while ((rc == -EPROTO) &&
|
|
|
|
(raw_rc == MC_CMD_ERR_NO_EVB_PORT) &&
|
|
|
|
time_before(jiffies, abort_time));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rc && !quiet && !(cmd == MC_CMD_REBOOT && rc == -EIO))
|
|
|
|
efx_mcdi_display_error(efx, cmd, inlen,
|
|
|
|
outbuf, outlen, rc);
|
|
|
|
|
|
|
|
return rc;
|
2013-06-01 01:36:12 +08:00
|
|
|
}
|
|
|
|
|
2015-12-23 16:56:40 +08:00
|
|
|
/**
|
|
|
|
* efx_mcdi_rpc - Issue an MCDI command and wait for completion
|
|
|
|
* @efx: 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 an appropriate
|
|
|
|
* 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.
|
|
|
|
*/
|
2009-11-29 23:15:25 +08:00
|
|
|
int efx_mcdi_rpc(struct efx_nic *efx, unsigned cmd,
|
2012-09-15 00:31:41 +08:00
|
|
|
const efx_dword_t *inbuf, size_t inlen,
|
|
|
|
efx_dword_t *outbuf, size_t outlen,
|
2009-11-29 23:15:25 +08:00
|
|
|
size_t *outlen_actual)
|
2012-07-17 00:40:47 +08:00
|
|
|
{
|
2015-12-23 16:56:40 +08:00
|
|
|
return _efx_mcdi_rpc_evb_retry(efx, cmd, inbuf, inlen, outbuf, outlen,
|
|
|
|
outlen_actual, false);
|
2013-06-01 01:36:12 +08:00
|
|
|
}
|
2012-09-19 07:56:18 +08:00
|
|
|
|
2013-06-01 01:36:12 +08:00
|
|
|
/* Normally, on receiving an error code in the MCDI response,
|
|
|
|
* efx_mcdi_rpc will log an error message containing (among other
|
|
|
|
* things) the raw error code, by means of efx_mcdi_display_error.
|
|
|
|
* This _quiet version suppresses that; if the caller wishes to log
|
|
|
|
* the error conditionally on the return code, it should call this
|
|
|
|
* function and is then responsible for calling efx_mcdi_display_error
|
|
|
|
* as needed.
|
|
|
|
*/
|
|
|
|
int efx_mcdi_rpc_quiet(struct efx_nic *efx, unsigned cmd,
|
|
|
|
const efx_dword_t *inbuf, size_t inlen,
|
|
|
|
efx_dword_t *outbuf, size_t outlen,
|
|
|
|
size_t *outlen_actual)
|
|
|
|
{
|
2015-12-23 16:56:40 +08:00
|
|
|
return _efx_mcdi_rpc_evb_retry(efx, cmd, inbuf, inlen, outbuf, outlen,
|
|
|
|
outlen_actual, true);
|
2012-07-17 00:40:47 +08:00
|
|
|
}
|
|
|
|
|
2012-09-19 07:56:18 +08:00
|
|
|
int efx_mcdi_rpc_start(struct efx_nic *efx, unsigned cmd,
|
|
|
|
const efx_dword_t *inbuf, size_t inlen)
|
2009-11-29 23:15:25 +08:00
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
2013-08-23 05:06:09 +08:00
|
|
|
int rc;
|
2012-07-17 00:40:47 +08:00
|
|
|
|
2013-08-23 05:06:09 +08:00
|
|
|
rc = efx_mcdi_check_supported(efx, cmd, inlen);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
2012-09-19 07:56:18 +08:00
|
|
|
|
2013-09-16 21:18:51 +08:00
|
|
|
if (efx->mc_bist_for_other_fn)
|
|
|
|
return -ENETDOWN;
|
|
|
|
|
sfc:On MCDI timeout, issue an FLR (and mark MCDI to fail-fast)
When an MCDI command times out (whether or not we find it
completed when we poll), call efx_mcdi_abandon(), which tells
all subsequent MCDI calls to fail-fast, and queues up an FLR.
Because an FLR doesn't lead to receiving any reboot even from
the MC (unlike most other types of reset), we have to call
efx_ef10_reset_mc_allocations.
In efx_start_all(), if a reset (of any kind) is pending, we
bail out.
Without this, attempts to reconfigure (e.g. change mtu) can
cause driver/mc state inconsistency if the first MCDI call
triggers an FLR.
For similar reasons, on EF10, in
efx_reset_down(method=RESET_TYPE_MCDI_TIMEOUT), set the number
of active queues to zero before calling efx_stop_all().
And, on farch, in efx_reset_up(method=RESET_TYPE_MCDI_TIMEOUT),
set active_queues and flushes pending & outstanding to zero.
efx_mcdi_mode_{poll,event}() should not take us out of fail-fast
mode. Instead, this is done by efx_mcdi_reset() after the FLR
completes.
The new FLR reset_type RESET_TYPE_MCDI_TIMEOUT doesn't really
fit into the hierarchy of reset 'scopes' whereby efx_reset()
decides some resets subsume others. Thus, it uses separate logic.
Also, fixed up some inconsistency around RESET_TYPE_MC_BIST,
which was in the wrong place in that hierarchy.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-04-17 02:27:48 +08:00
|
|
|
if (mcdi->mode == MCDI_MODE_FAIL)
|
|
|
|
return -ENETDOWN;
|
|
|
|
|
2013-08-28 06:12:31 +08:00
|
|
|
efx_mcdi_acquire_sync(mcdi);
|
2013-08-23 05:06:09 +08:00
|
|
|
efx_mcdi_send_request(efx, cmd, inbuf, inlen);
|
2012-09-19 07:56:18 +08:00
|
|
|
return 0;
|
2012-07-17 00:40:47 +08:00
|
|
|
}
|
|
|
|
|
2013-06-01 01:36:12 +08:00
|
|
|
static int _efx_mcdi_rpc_async(struct efx_nic *efx, unsigned int cmd,
|
|
|
|
const efx_dword_t *inbuf, size_t inlen,
|
|
|
|
size_t outlen,
|
|
|
|
efx_mcdi_async_completer *complete,
|
|
|
|
unsigned long cookie, bool quiet)
|
2013-08-28 06:12:31 +08:00
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
|
|
|
struct efx_mcdi_async_param *async;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
rc = efx_mcdi_check_supported(efx, cmd, inlen);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
|
2013-09-16 21:18:51 +08:00
|
|
|
if (efx->mc_bist_for_other_fn)
|
|
|
|
return -ENETDOWN;
|
|
|
|
|
2013-08-28 06:12:31 +08:00
|
|
|
async = kmalloc(sizeof(*async) + ALIGN(max(inlen, outlen), 4),
|
|
|
|
GFP_ATOMIC);
|
|
|
|
if (!async)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
async->cmd = cmd;
|
|
|
|
async->inlen = inlen;
|
|
|
|
async->outlen = outlen;
|
2013-06-01 01:36:12 +08:00
|
|
|
async->quiet = quiet;
|
2013-08-28 06:12:31 +08:00
|
|
|
async->complete = complete;
|
|
|
|
async->cookie = cookie;
|
|
|
|
memcpy(async + 1, inbuf, inlen);
|
|
|
|
|
|
|
|
spin_lock_bh(&mcdi->async_lock);
|
|
|
|
|
|
|
|
if (mcdi->mode == MCDI_MODE_EVENTS) {
|
|
|
|
list_add_tail(&async->list, &mcdi->async_list);
|
|
|
|
|
|
|
|
/* If this is at the front of the queue, try to start it
|
|
|
|
* immediately
|
|
|
|
*/
|
|
|
|
if (mcdi->async_list.next == &async->list &&
|
|
|
|
efx_mcdi_acquire_async(mcdi)) {
|
|
|
|
efx_mcdi_send_request(efx, cmd, inbuf, inlen);
|
|
|
|
mod_timer(&mcdi->async_timer,
|
|
|
|
jiffies + MCDI_RPC_TIMEOUT);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
kfree(async);
|
|
|
|
rc = -ENETDOWN;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_unlock_bh(&mcdi->async_lock);
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2013-06-01 01:36:12 +08:00
|
|
|
/**
|
|
|
|
* efx_mcdi_rpc_async - Schedule an MCDI command to run asynchronously
|
|
|
|
* @efx: NIC through which to issue the command
|
|
|
|
* @cmd: Command type number
|
|
|
|
* @inbuf: Command parameters
|
|
|
|
* @inlen: Length of command parameters, in bytes
|
|
|
|
* @outlen: Length to allocate for response buffer, 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 atomic context, when one of the following occurs:
|
|
|
|
* (a) the completion event is received (in NAPI context)
|
|
|
|
* (b) event queues are disabled (in the process that disables them)
|
|
|
|
* (c) the request times-out (in timer context)
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
efx_mcdi_rpc_async(struct efx_nic *efx, unsigned int cmd,
|
|
|
|
const efx_dword_t *inbuf, size_t inlen, size_t outlen,
|
|
|
|
efx_mcdi_async_completer *complete, unsigned long cookie)
|
|
|
|
{
|
|
|
|
return _efx_mcdi_rpc_async(efx, cmd, inbuf, inlen, outlen, complete,
|
|
|
|
cookie, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
int efx_mcdi_rpc_async_quiet(struct efx_nic *efx, unsigned int cmd,
|
|
|
|
const efx_dword_t *inbuf, size_t inlen,
|
|
|
|
size_t outlen, efx_mcdi_async_completer *complete,
|
|
|
|
unsigned long cookie)
|
|
|
|
{
|
|
|
|
return _efx_mcdi_rpc_async(efx, cmd, inbuf, inlen, outlen, complete,
|
|
|
|
cookie, true);
|
|
|
|
}
|
|
|
|
|
2012-07-17 00:40:47 +08:00
|
|
|
int efx_mcdi_rpc_finish(struct efx_nic *efx, unsigned cmd, size_t inlen,
|
2012-09-15 00:31:41 +08:00
|
|
|
efx_dword_t *outbuf, size_t outlen,
|
|
|
|
size_t *outlen_actual)
|
2012-07-17 00:40:47 +08:00
|
|
|
{
|
2013-06-01 01:36:12 +08:00
|
|
|
return _efx_mcdi_rpc_finish(efx, cmd, inlen, outbuf, outlen,
|
2015-12-23 16:57:18 +08:00
|
|
|
outlen_actual, false, NULL, NULL);
|
2013-06-01 01:36:12 +08:00
|
|
|
}
|
2012-10-09 04:43:00 +08:00
|
|
|
|
2013-06-01 01:36:12 +08:00
|
|
|
int efx_mcdi_rpc_finish_quiet(struct efx_nic *efx, unsigned cmd, size_t inlen,
|
|
|
|
efx_dword_t *outbuf, size_t outlen,
|
|
|
|
size_t *outlen_actual)
|
|
|
|
{
|
|
|
|
return _efx_mcdi_rpc_finish(efx, cmd, inlen, outbuf, outlen,
|
2015-12-23 16:57:18 +08:00
|
|
|
outlen_actual, true, NULL, NULL);
|
2013-06-01 01:36:12 +08:00
|
|
|
}
|
2011-12-21 07:39:31 +08:00
|
|
|
|
2013-06-01 01:36:12 +08:00
|
|
|
void efx_mcdi_display_error(struct efx_nic *efx, unsigned cmd,
|
|
|
|
size_t inlen, efx_dword_t *outbuf,
|
|
|
|
size_t outlen, int rc)
|
|
|
|
{
|
|
|
|
int code = 0, err_arg = 0;
|
2009-11-29 23:15:25 +08:00
|
|
|
|
2013-06-01 01:36:12 +08:00
|
|
|
if (outlen >= MC_CMD_ERR_CODE_OFST + 4)
|
|
|
|
code = MCDI_DWORD(outbuf, ERR_CODE);
|
|
|
|
if (outlen >= MC_CMD_ERR_ARG_OFST + 4)
|
|
|
|
err_arg = MCDI_DWORD(outbuf, ERR_ARG);
|
2017-01-27 23:02:39 +08:00
|
|
|
netif_cond_dbg(efx, hw, efx->net_dev, rc == -EPERM, err,
|
|
|
|
"MC command 0x%x inlen %zu failed rc=%d (raw=%d) arg=%d\n",
|
|
|
|
cmd, inlen, rc, code, err_arg);
|
2009-11-29 23:15:25 +08:00
|
|
|
}
|
|
|
|
|
2013-08-28 06:12:31 +08:00
|
|
|
/* Switch to polled MCDI completions. This can be called in various
|
|
|
|
* error conditions with various locks held, so it must be lockless.
|
|
|
|
* Caller is responsible for flushing asynchronous requests later.
|
|
|
|
*/
|
2009-11-29 23:15:25 +08:00
|
|
|
void efx_mcdi_mode_poll(struct efx_nic *efx)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi;
|
|
|
|
|
2012-09-18 09:33:56 +08:00
|
|
|
if (!efx->mcdi)
|
2009-11-29 23:15:25 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
mcdi = efx_mcdi(efx);
|
sfc:On MCDI timeout, issue an FLR (and mark MCDI to fail-fast)
When an MCDI command times out (whether or not we find it
completed when we poll), call efx_mcdi_abandon(), which tells
all subsequent MCDI calls to fail-fast, and queues up an FLR.
Because an FLR doesn't lead to receiving any reboot even from
the MC (unlike most other types of reset), we have to call
efx_ef10_reset_mc_allocations.
In efx_start_all(), if a reset (of any kind) is pending, we
bail out.
Without this, attempts to reconfigure (e.g. change mtu) can
cause driver/mc state inconsistency if the first MCDI call
triggers an FLR.
For similar reasons, on EF10, in
efx_reset_down(method=RESET_TYPE_MCDI_TIMEOUT), set the number
of active queues to zero before calling efx_stop_all().
And, on farch, in efx_reset_up(method=RESET_TYPE_MCDI_TIMEOUT),
set active_queues and flushes pending & outstanding to zero.
efx_mcdi_mode_{poll,event}() should not take us out of fail-fast
mode. Instead, this is done by efx_mcdi_reset() after the FLR
completes.
The new FLR reset_type RESET_TYPE_MCDI_TIMEOUT doesn't really
fit into the hierarchy of reset 'scopes' whereby efx_reset()
decides some resets subsume others. Thus, it uses separate logic.
Also, fixed up some inconsistency around RESET_TYPE_MC_BIST,
which was in the wrong place in that hierarchy.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-04-17 02:27:48 +08:00
|
|
|
/* If already in polling mode, nothing to do.
|
|
|
|
* If in fail-fast state, don't switch to polled completion.
|
|
|
|
* FLR recovery will do that later.
|
|
|
|
*/
|
|
|
|
if (mcdi->mode == MCDI_MODE_POLL || mcdi->mode == MCDI_MODE_FAIL)
|
2009-11-29 23:15:25 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
/* We can switch from event completion to polled completion, because
|
|
|
|
* mcdi requests are always completed in shared memory. We do this by
|
|
|
|
* switching the mode to POLL'd then completing the request.
|
|
|
|
* efx_mcdi_await_completion() will then call efx_mcdi_poll().
|
|
|
|
*
|
|
|
|
* We need an smp_wmb() to synchronise with efx_mcdi_await_completion(),
|
2013-08-28 06:12:31 +08:00
|
|
|
* which efx_mcdi_complete_sync() provides for us.
|
2009-11-29 23:15:25 +08:00
|
|
|
*/
|
|
|
|
mcdi->mode = MCDI_MODE_POLL;
|
|
|
|
|
2013-08-28 06:12:31 +08:00
|
|
|
efx_mcdi_complete_sync(mcdi);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Flush any running or queued asynchronous requests, after event processing
|
|
|
|
* is stopped
|
|
|
|
*/
|
|
|
|
void efx_mcdi_flush_async(struct efx_nic *efx)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_async_param *async, *next;
|
|
|
|
struct efx_mcdi_iface *mcdi;
|
|
|
|
|
|
|
|
if (!efx->mcdi)
|
|
|
|
return;
|
|
|
|
|
|
|
|
mcdi = efx_mcdi(efx);
|
|
|
|
|
sfc:On MCDI timeout, issue an FLR (and mark MCDI to fail-fast)
When an MCDI command times out (whether or not we find it
completed when we poll), call efx_mcdi_abandon(), which tells
all subsequent MCDI calls to fail-fast, and queues up an FLR.
Because an FLR doesn't lead to receiving any reboot even from
the MC (unlike most other types of reset), we have to call
efx_ef10_reset_mc_allocations.
In efx_start_all(), if a reset (of any kind) is pending, we
bail out.
Without this, attempts to reconfigure (e.g. change mtu) can
cause driver/mc state inconsistency if the first MCDI call
triggers an FLR.
For similar reasons, on EF10, in
efx_reset_down(method=RESET_TYPE_MCDI_TIMEOUT), set the number
of active queues to zero before calling efx_stop_all().
And, on farch, in efx_reset_up(method=RESET_TYPE_MCDI_TIMEOUT),
set active_queues and flushes pending & outstanding to zero.
efx_mcdi_mode_{poll,event}() should not take us out of fail-fast
mode. Instead, this is done by efx_mcdi_reset() after the FLR
completes.
The new FLR reset_type RESET_TYPE_MCDI_TIMEOUT doesn't really
fit into the hierarchy of reset 'scopes' whereby efx_reset()
decides some resets subsume others. Thus, it uses separate logic.
Also, fixed up some inconsistency around RESET_TYPE_MC_BIST,
which was in the wrong place in that hierarchy.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-04-17 02:27:48 +08:00
|
|
|
/* We must be in poll or fail mode so no more requests can be queued */
|
|
|
|
BUG_ON(mcdi->mode == MCDI_MODE_EVENTS);
|
2013-08-28 06:12:31 +08:00
|
|
|
|
|
|
|
del_timer_sync(&mcdi->async_timer);
|
|
|
|
|
|
|
|
/* If a request is still running, make sure we give the MC
|
|
|
|
* time to complete it so that the response won't overwrite our
|
|
|
|
* next request.
|
|
|
|
*/
|
|
|
|
if (mcdi->state == MCDI_STATE_RUNNING_ASYNC) {
|
|
|
|
efx_mcdi_poll(efx);
|
|
|
|
mcdi->state = MCDI_STATE_QUIESCENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Nothing else will access the async list now, so it is safe
|
|
|
|
* to walk it without holding async_lock. If we hold it while
|
|
|
|
* calling a completer then lockdep may warn that we have
|
|
|
|
* acquired locks in the wrong order.
|
|
|
|
*/
|
|
|
|
list_for_each_entry_safe(async, next, &mcdi->async_list, list) {
|
2016-09-22 22:47:45 +08:00
|
|
|
if (async->complete)
|
|
|
|
async->complete(efx, async->cookie, -ENETDOWN, NULL, 0);
|
2013-08-28 06:12:31 +08:00
|
|
|
list_del(&async->list);
|
|
|
|
kfree(async);
|
|
|
|
}
|
2009-11-29 23:15:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void efx_mcdi_mode_event(struct efx_nic *efx)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi;
|
|
|
|
|
2012-09-18 09:33:56 +08:00
|
|
|
if (!efx->mcdi)
|
2009-11-29 23:15:25 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
mcdi = efx_mcdi(efx);
|
sfc:On MCDI timeout, issue an FLR (and mark MCDI to fail-fast)
When an MCDI command times out (whether or not we find it
completed when we poll), call efx_mcdi_abandon(), which tells
all subsequent MCDI calls to fail-fast, and queues up an FLR.
Because an FLR doesn't lead to receiving any reboot even from
the MC (unlike most other types of reset), we have to call
efx_ef10_reset_mc_allocations.
In efx_start_all(), if a reset (of any kind) is pending, we
bail out.
Without this, attempts to reconfigure (e.g. change mtu) can
cause driver/mc state inconsistency if the first MCDI call
triggers an FLR.
For similar reasons, on EF10, in
efx_reset_down(method=RESET_TYPE_MCDI_TIMEOUT), set the number
of active queues to zero before calling efx_stop_all().
And, on farch, in efx_reset_up(method=RESET_TYPE_MCDI_TIMEOUT),
set active_queues and flushes pending & outstanding to zero.
efx_mcdi_mode_{poll,event}() should not take us out of fail-fast
mode. Instead, this is done by efx_mcdi_reset() after the FLR
completes.
The new FLR reset_type RESET_TYPE_MCDI_TIMEOUT doesn't really
fit into the hierarchy of reset 'scopes' whereby efx_reset()
decides some resets subsume others. Thus, it uses separate logic.
Also, fixed up some inconsistency around RESET_TYPE_MC_BIST,
which was in the wrong place in that hierarchy.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-04-17 02:27:48 +08:00
|
|
|
/* If already in event completion mode, nothing to do.
|
|
|
|
* If in fail-fast state, don't switch to event completion. FLR
|
|
|
|
* recovery will do that later.
|
|
|
|
*/
|
|
|
|
if (mcdi->mode == MCDI_MODE_EVENTS || mcdi->mode == MCDI_MODE_FAIL)
|
2009-11-29 23:15:25 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
/* We can't switch from polled to event completion in the middle of a
|
|
|
|
* request, because the completion method is specified in the request.
|
|
|
|
* So acquire the interface to serialise the requestors. We don't need
|
|
|
|
* to acquire the iface_lock to change the mode here, but we do need a
|
|
|
|
* write memory barrier ensure that efx_mcdi_rpc() sees it, which
|
|
|
|
* efx_mcdi_acquire() provides.
|
|
|
|
*/
|
2013-08-28 06:12:31 +08:00
|
|
|
efx_mcdi_acquire_sync(mcdi);
|
2009-11-29 23:15:25 +08:00
|
|
|
mcdi->mode = MCDI_MODE_EVENTS;
|
|
|
|
efx_mcdi_release(mcdi);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void efx_mcdi_ev_death(struct efx_nic *efx, int rc)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
|
|
|
|
|
|
|
/* If there is an outstanding MCDI request, it has been terminated
|
|
|
|
* either by a BADASSERT or REBOOT event. If the mcdi interface is
|
|
|
|
* in polled mode, then do nothing because the MC reboot handler will
|
|
|
|
* set the header correctly. However, if the mcdi interface is waiting
|
|
|
|
* for a CMDDONE event it won't receive it [and since all MCDI events
|
|
|
|
* are sent to the same queue, we can't be racing with
|
|
|
|
* efx_mcdi_ev_cpl()]
|
|
|
|
*
|
2013-08-28 06:12:31 +08:00
|
|
|
* If there is an outstanding asynchronous request, we can't
|
|
|
|
* complete it now (efx_mcdi_complete() would deadlock). The
|
|
|
|
* reset process will take care of this.
|
|
|
|
*
|
|
|
|
* There's a race here with efx_mcdi_send_request(), because
|
|
|
|
* we might receive a REBOOT event *before* the request has
|
|
|
|
* been copied out. In polled mode (during startup) this is
|
|
|
|
* irrelevant, because efx_mcdi_complete_sync() is ignored. In
|
|
|
|
* event mode, this condition is just an edge-case of
|
|
|
|
* receiving a REBOOT event after posting the MCDI
|
|
|
|
* request. Did the mc reboot before or after the copyout? The
|
|
|
|
* best we can do always is just return failure.
|
2015-12-23 16:57:18 +08:00
|
|
|
*
|
|
|
|
* If there is an outstanding proxy response expected it is not going
|
|
|
|
* to arrive. We should thus abort it.
|
2009-11-29 23:15:25 +08:00
|
|
|
*/
|
|
|
|
spin_lock(&mcdi->iface_lock);
|
2015-12-23 16:57:18 +08:00
|
|
|
efx_mcdi_proxy_abort(mcdi);
|
|
|
|
|
2013-08-28 06:12:31 +08:00
|
|
|
if (efx_mcdi_complete_sync(mcdi)) {
|
2009-11-29 23:15:25 +08:00
|
|
|
if (mcdi->mode == MCDI_MODE_EVENTS) {
|
|
|
|
mcdi->resprc = rc;
|
2012-09-19 07:56:18 +08:00
|
|
|
mcdi->resp_hdr_len = 0;
|
|
|
|
mcdi->resp_data_len = 0;
|
2010-12-02 21:46:55 +08:00
|
|
|
++mcdi->credits;
|
2009-11-29 23:15:25 +08:00
|
|
|
}
|
2011-12-21 07:39:31 +08:00
|
|
|
} else {
|
|
|
|
int count;
|
|
|
|
|
|
|
|
/* Consume the status word since efx_mcdi_rpc_finish() won't */
|
|
|
|
for (count = 0; count < MCDI_STATUS_DELAY_COUNT; ++count) {
|
2015-10-09 17:40:35 +08:00
|
|
|
rc = efx_mcdi_poll_reboot(efx);
|
|
|
|
if (rc)
|
2011-12-21 07:39:31 +08:00
|
|
|
break;
|
|
|
|
udelay(MCDI_STATUS_DELAY_US);
|
|
|
|
}
|
2015-10-09 17:40:35 +08:00
|
|
|
|
|
|
|
/* On EF10, a CODE_MC_REBOOT event can be received without the
|
|
|
|
* reboot detection in efx_mcdi_poll_reboot() being triggered.
|
|
|
|
* If zero was returned from the final call to
|
|
|
|
* efx_mcdi_poll_reboot(), the MC reboot wasn't noticed but the
|
|
|
|
* MC has definitely rebooted so prepare for the reset.
|
|
|
|
*/
|
|
|
|
if (!rc && efx->type->mcdi_reboot_detected)
|
|
|
|
efx->type->mcdi_reboot_detected(efx);
|
|
|
|
|
2013-06-20 18:40:07 +08:00
|
|
|
mcdi->new_epoch = true;
|
2013-09-18 17:16:24 +08:00
|
|
|
|
|
|
|
/* Nobody was waiting for an MCDI request, so trigger a reset */
|
|
|
|
efx_schedule_reset(efx, RESET_TYPE_MC_FAILURE);
|
2011-12-21 07:39:31 +08:00
|
|
|
}
|
|
|
|
|
2009-11-29 23:15:25 +08:00
|
|
|
spin_unlock(&mcdi->iface_lock);
|
|
|
|
}
|
|
|
|
|
2013-09-16 21:18:51 +08:00
|
|
|
/* The MC is going down in to BIST mode. set the BIST flag to block
|
|
|
|
* new MCDI, cancel any outstanding MCDI and and schedule a BIST-type reset
|
|
|
|
* (which doesn't actually execute a reset, it waits for the controlling
|
|
|
|
* function to reset it).
|
|
|
|
*/
|
|
|
|
static void efx_mcdi_ev_bist(struct efx_nic *efx)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
|
|
|
|
|
|
|
spin_lock(&mcdi->iface_lock);
|
|
|
|
efx->mc_bist_for_other_fn = true;
|
2015-12-23 16:57:18 +08:00
|
|
|
efx_mcdi_proxy_abort(mcdi);
|
|
|
|
|
2013-09-16 21:18:51 +08:00
|
|
|
if (efx_mcdi_complete_sync(mcdi)) {
|
|
|
|
if (mcdi->mode == MCDI_MODE_EVENTS) {
|
|
|
|
mcdi->resprc = -EIO;
|
|
|
|
mcdi->resp_hdr_len = 0;
|
|
|
|
mcdi->resp_data_len = 0;
|
|
|
|
++mcdi->credits;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mcdi->new_epoch = true;
|
|
|
|
efx_schedule_reset(efx, RESET_TYPE_MC_BIST);
|
|
|
|
spin_unlock(&mcdi->iface_lock);
|
|
|
|
}
|
|
|
|
|
sfc:On MCDI timeout, issue an FLR (and mark MCDI to fail-fast)
When an MCDI command times out (whether or not we find it
completed when we poll), call efx_mcdi_abandon(), which tells
all subsequent MCDI calls to fail-fast, and queues up an FLR.
Because an FLR doesn't lead to receiving any reboot even from
the MC (unlike most other types of reset), we have to call
efx_ef10_reset_mc_allocations.
In efx_start_all(), if a reset (of any kind) is pending, we
bail out.
Without this, attempts to reconfigure (e.g. change mtu) can
cause driver/mc state inconsistency if the first MCDI call
triggers an FLR.
For similar reasons, on EF10, in
efx_reset_down(method=RESET_TYPE_MCDI_TIMEOUT), set the number
of active queues to zero before calling efx_stop_all().
And, on farch, in efx_reset_up(method=RESET_TYPE_MCDI_TIMEOUT),
set active_queues and flushes pending & outstanding to zero.
efx_mcdi_mode_{poll,event}() should not take us out of fail-fast
mode. Instead, this is done by efx_mcdi_reset() after the FLR
completes.
The new FLR reset_type RESET_TYPE_MCDI_TIMEOUT doesn't really
fit into the hierarchy of reset 'scopes' whereby efx_reset()
decides some resets subsume others. Thus, it uses separate logic.
Also, fixed up some inconsistency around RESET_TYPE_MC_BIST,
which was in the wrong place in that hierarchy.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-04-17 02:27:48 +08:00
|
|
|
/* MCDI timeouts seen, so make all MCDI calls fail-fast and issue an FLR to try
|
|
|
|
* to recover.
|
|
|
|
*/
|
|
|
|
static void efx_mcdi_abandon(struct efx_nic *efx)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
|
|
|
|
|
|
|
if (xchg(&mcdi->mode, MCDI_MODE_FAIL) == MCDI_MODE_FAIL)
|
|
|
|
return; /* it had already been done */
|
|
|
|
netif_dbg(efx, hw, efx->net_dev, "MCDI is timing out; trying to recover\n");
|
|
|
|
efx_schedule_reset(efx, RESET_TYPE_MCDI_TIMEOUT);
|
|
|
|
}
|
|
|
|
|
2017-06-30 22:55:28 +08:00
|
|
|
/* Called from efx_farch_ev_process and efx_ef10_ev_process for MCDI events */
|
2009-11-29 23:15:25 +08:00
|
|
|
void efx_mcdi_process_event(struct efx_channel *channel,
|
|
|
|
efx_qword_t *event)
|
|
|
|
{
|
|
|
|
struct efx_nic *efx = channel->efx;
|
|
|
|
int code = EFX_QWORD_FIELD(*event, MCDI_EVENT_CODE);
|
|
|
|
u32 data = EFX_QWORD_FIELD(*event, MCDI_EVENT_DATA);
|
|
|
|
|
|
|
|
switch (code) {
|
|
|
|
case MCDI_EVENT_CODE_BADSSERT:
|
2010-06-23 19:30:07 +08:00
|
|
|
netif_err(efx, hw, efx->net_dev,
|
|
|
|
"MC watchdog or assertion failure at 0x%x\n", data);
|
2012-10-09 04:43:00 +08:00
|
|
|
efx_mcdi_ev_death(efx, -EINTR);
|
2009-11-29 23:15:25 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case MCDI_EVENT_CODE_PMNOTICE:
|
2010-06-23 19:30:07 +08:00
|
|
|
netif_info(efx, wol, efx->net_dev, "MCDI PM event.\n");
|
2009-11-29 23:15:25 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case MCDI_EVENT_CODE_CMDDONE:
|
|
|
|
efx_mcdi_ev_cpl(efx,
|
|
|
|
MCDI_EVENT_FIELD(*event, CMDDONE_SEQ),
|
|
|
|
MCDI_EVENT_FIELD(*event, CMDDONE_DATALEN),
|
|
|
|
MCDI_EVENT_FIELD(*event, CMDDONE_ERRNO));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MCDI_EVENT_CODE_LINKCHANGE:
|
|
|
|
efx_mcdi_process_link_change(efx, event);
|
|
|
|
break;
|
|
|
|
case MCDI_EVENT_CODE_SENSOREVT:
|
|
|
|
efx_mcdi_sensor_event(efx, event);
|
|
|
|
break;
|
|
|
|
case MCDI_EVENT_CODE_SCHEDERR:
|
2013-10-08 01:44:17 +08:00
|
|
|
netif_dbg(efx, hw, efx->net_dev,
|
|
|
|
"MC Scheduler alert (0x%x)\n", data);
|
2009-11-29 23:15:25 +08:00
|
|
|
break;
|
|
|
|
case MCDI_EVENT_CODE_REBOOT:
|
2013-08-30 02:19:29 +08:00
|
|
|
case MCDI_EVENT_CODE_MC_REBOOT:
|
2010-06-23 19:30:07 +08:00
|
|
|
netif_info(efx, hw, efx->net_dev, "MC Reboot\n");
|
2012-10-09 04:43:00 +08:00
|
|
|
efx_mcdi_ev_death(efx, -EIO);
|
2009-11-29 23:15:25 +08:00
|
|
|
break;
|
2013-09-16 21:18:51 +08:00
|
|
|
case MCDI_EVENT_CODE_MC_BIST:
|
|
|
|
netif_info(efx, hw, efx->net_dev, "MC entered BIST mode\n");
|
|
|
|
efx_mcdi_ev_bist(efx);
|
|
|
|
break;
|
2009-11-29 23:15:25 +08:00
|
|
|
case MCDI_EVENT_CODE_MAC_STATS_DMA:
|
|
|
|
/* MAC stats are gather lazily. We can ignore this. */
|
|
|
|
break;
|
sfc: Add SR-IOV back-end support for SFC9000 family
On the SFC9000 family, each port has 1024 Virtual Interfaces (VIs),
each with an RX queue, a TX queue, an event queue and a mailbox
register. These may be assigned to up to 127 SR-IOV virtual functions
per port, with up to 64 VIs per VF.
We allocate an extra channel (IRQ and event queue only) to receive
requests from VF drivers.
There is a per-port limit of 4 concurrent RX queue flushes, and queue
flushes may be initiated by the MC in response to a Function Level
Reset (FLR) of a VF. Therefore, when SR-IOV is in use, we submit all
flush requests via the MC.
The RSS indirection table is shared with VFs, so the number of RX
queues used in the PF is limited to the number of VIs per VF.
This is almost entirely the work of Steve Hodgson, formerly
shodgson@solarflare.com.
Signed-off-by: Ben Hutchings <bhutchings@solarflare.com>
2012-02-14 08:48:07 +08:00
|
|
|
case MCDI_EVENT_CODE_FLR:
|
2015-05-06 07:55:13 +08:00
|
|
|
if (efx->type->sriov_flr)
|
|
|
|
efx->type->sriov_flr(efx,
|
|
|
|
MCDI_EVENT_FIELD(*event, FLR_VF));
|
sfc: Add SR-IOV back-end support for SFC9000 family
On the SFC9000 family, each port has 1024 Virtual Interfaces (VIs),
each with an RX queue, a TX queue, an event queue and a mailbox
register. These may be assigned to up to 127 SR-IOV virtual functions
per port, with up to 64 VIs per VF.
We allocate an extra channel (IRQ and event queue only) to receive
requests from VF drivers.
There is a per-port limit of 4 concurrent RX queue flushes, and queue
flushes may be initiated by the MC in response to a Function Level
Reset (FLR) of a VF. Therefore, when SR-IOV is in use, we submit all
flush requests via the MC.
The RSS indirection table is shared with VFs, so the number of RX
queues used in the PF is limited to the number of VIs per VF.
This is almost entirely the work of Steve Hodgson, formerly
shodgson@solarflare.com.
Signed-off-by: Ben Hutchings <bhutchings@solarflare.com>
2012-02-14 08:48:07 +08:00
|
|
|
break;
|
2012-09-03 18:09:36 +08:00
|
|
|
case MCDI_EVENT_CODE_PTP_RX:
|
|
|
|
case MCDI_EVENT_CODE_PTP_FAULT:
|
|
|
|
case MCDI_EVENT_CODE_PTP_PPS:
|
|
|
|
efx_ptp_event(efx, event);
|
|
|
|
break;
|
2013-11-18 20:54:41 +08:00
|
|
|
case MCDI_EVENT_CODE_PTP_TIME:
|
|
|
|
efx_time_sync_event(channel, event);
|
|
|
|
break;
|
2013-08-30 02:19:29 +08:00
|
|
|
case MCDI_EVENT_CODE_TX_FLUSH:
|
|
|
|
case MCDI_EVENT_CODE_RX_FLUSH:
|
|
|
|
/* Two flush events will be sent: one to the same event
|
|
|
|
* queue as completions, and one to event queue 0.
|
|
|
|
* In the latter case the {RX,TX}_FLUSH_TO_DRIVER
|
|
|
|
* flag will be set, and we should ignore the event
|
|
|
|
* because we want to wait for all completions.
|
|
|
|
*/
|
|
|
|
BUILD_BUG_ON(MCDI_EVENT_TX_FLUSH_TO_DRIVER_LBN !=
|
|
|
|
MCDI_EVENT_RX_FLUSH_TO_DRIVER_LBN);
|
|
|
|
if (!MCDI_EVENT_FIELD(*event, TX_FLUSH_TO_DRIVER))
|
|
|
|
efx_ef10_handle_drain_event(efx);
|
|
|
|
break;
|
2013-06-13 18:36:15 +08:00
|
|
|
case MCDI_EVENT_CODE_TX_ERR:
|
|
|
|
case MCDI_EVENT_CODE_RX_ERR:
|
|
|
|
netif_err(efx, hw, efx->net_dev,
|
|
|
|
"%s DMA error (event: "EFX_QWORD_FMT")\n",
|
|
|
|
code == MCDI_EVENT_CODE_TX_ERR ? "TX" : "RX",
|
|
|
|
EFX_QWORD_VAL(*event));
|
|
|
|
efx_schedule_reset(efx, RESET_TYPE_DMA_ERROR);
|
|
|
|
break;
|
2015-12-23 16:57:18 +08:00
|
|
|
case MCDI_EVENT_CODE_PROXY_RESPONSE:
|
|
|
|
efx_mcdi_ev_proxy_response(efx,
|
|
|
|
MCDI_EVENT_FIELD(*event, PROXY_RESPONSE_HANDLE),
|
|
|
|
MCDI_EVENT_FIELD(*event, PROXY_RESPONSE_RC));
|
|
|
|
break;
|
2009-11-29 23:15:25 +08:00
|
|
|
default:
|
2017-06-30 22:54:33 +08:00
|
|
|
netif_err(efx, hw, efx->net_dev,
|
|
|
|
"Unknown MCDI event " EFX_QWORD_FMT "\n",
|
|
|
|
EFX_QWORD_VAL(*event));
|
2009-11-29 23:15:25 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
|
|
|
*
|
|
|
|
* Specific request functions
|
|
|
|
*
|
|
|
|
**************************************************************************
|
|
|
|
*/
|
|
|
|
|
2011-02-25 07:57:47 +08:00
|
|
|
void efx_mcdi_print_fwver(struct efx_nic *efx, char *buf, size_t len)
|
2009-11-29 23:15:25 +08:00
|
|
|
{
|
2015-05-06 07:56:55 +08:00
|
|
|
MCDI_DECLARE_BUF(outbuf, MC_CMD_GET_VERSION_OUT_LEN);
|
2009-11-29 23:15:25 +08:00
|
|
|
size_t outlength;
|
|
|
|
const __le16 *ver_words;
|
2013-08-30 02:19:29 +08:00
|
|
|
size_t offset;
|
2009-11-29 23:15:25 +08:00
|
|
|
int rc;
|
|
|
|
|
|
|
|
BUILD_BUG_ON(MC_CMD_GET_VERSION_IN_LEN != 0);
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_GET_VERSION, NULL, 0,
|
|
|
|
outbuf, sizeof(outbuf), &outlength);
|
|
|
|
if (rc)
|
|
|
|
goto fail;
|
2011-12-20 08:44:06 +08:00
|
|
|
if (outlength < MC_CMD_GET_VERSION_OUT_LEN) {
|
2010-04-28 17:27:14 +08:00
|
|
|
rc = -EIO;
|
2009-11-29 23:15:25 +08:00
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
ver_words = (__le16 *)MCDI_PTR(outbuf, GET_VERSION_OUT_VERSION);
|
2013-08-30 02:19:29 +08:00
|
|
|
offset = snprintf(buf, len, "%u.%u.%u.%u",
|
|
|
|
le16_to_cpu(ver_words[0]), le16_to_cpu(ver_words[1]),
|
|
|
|
le16_to_cpu(ver_words[2]), le16_to_cpu(ver_words[3]));
|
|
|
|
|
|
|
|
/* EF10 may have multiple datapath firmware variants within a
|
|
|
|
* single version. Report which variants are running.
|
|
|
|
*/
|
|
|
|
if (efx_nic_rev(efx) >= EFX_REV_HUNT_A0) {
|
2015-05-06 07:56:55 +08:00
|
|
|
struct efx_ef10_nic_data *nic_data = efx->nic_data;
|
|
|
|
|
|
|
|
offset += snprintf(buf + offset, len - offset, " rx%x tx%x",
|
|
|
|
nic_data->rx_dpcpu_fw_id,
|
|
|
|
nic_data->tx_dpcpu_fw_id);
|
2013-08-30 02:19:29 +08:00
|
|
|
|
|
|
|
/* It's theoretically possible for the string to exceed 31
|
|
|
|
* characters, though in practice the first three version
|
|
|
|
* components are short enough that this doesn't happen.
|
|
|
|
*/
|
|
|
|
if (WARN_ON(offset >= len))
|
|
|
|
buf[0] = 0;
|
|
|
|
}
|
|
|
|
|
2011-02-25 07:57:47 +08:00
|
|
|
return;
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
fail:
|
2010-06-23 19:30:07 +08:00
|
|
|
netif_err(efx, probe, efx->net_dev, "%s: failed rc=%d\n", __func__, rc);
|
2011-02-25 07:57:47 +08:00
|
|
|
buf[0] = 0;
|
2009-11-29 23:15:25 +08:00
|
|
|
}
|
|
|
|
|
2013-08-30 02:04:03 +08:00
|
|
|
static int efx_mcdi_drv_attach(struct efx_nic *efx, bool driver_operating,
|
|
|
|
bool *was_attached)
|
2009-11-29 23:15:25 +08:00
|
|
|
{
|
2012-09-15 00:30:10 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_DRV_ATTACH_IN_LEN);
|
2013-10-08 03:10:11 +08:00
|
|
|
MCDI_DECLARE_BUF(outbuf, MC_CMD_DRV_ATTACH_EXT_OUT_LEN);
|
2009-11-29 23:15:25 +08:00
|
|
|
size_t outlen;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
MCDI_SET_DWORD(inbuf, DRV_ATTACH_IN_NEW_STATE,
|
|
|
|
driver_operating ? 1 : 0);
|
|
|
|
MCDI_SET_DWORD(inbuf, DRV_ATTACH_IN_UPDATE, 1);
|
2013-08-21 03:35:50 +08:00
|
|
|
MCDI_SET_DWORD(inbuf, DRV_ATTACH_IN_FIRMWARE_ID, MC_CMD_FW_LOW_LATENCY);
|
2009-11-29 23:15:25 +08:00
|
|
|
|
sfc: Cope with permissions enforcement added to firmware for SR-IOV
* Accept EPERM in some simple cases, the following cases are handled:
1) efx_mcdi_read_assertion()
Unprivileged PCI functions aren't allowed to GET_ASSERTS.
We return success as it's up to the primary PF to deal with asserts.
2) efx_mcdi_mon_probe() in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to read sensor info, and
worrying about sensor data is the primary PF's job.
3) phy_op->reconfigure() in efx_init_port() and efx_reset_up()
Unprivileged functions aren't allowed to MC_CMD_SET_LINK, they just have
to accept the settings (including flow-control, which is what
efx_init_port() is worried about) they've been given.
4) Fallback to GET_WORKAROUNDS in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to set workarounds. So if
efx_mcdi_set_workaround() fails EPERM, use efx_mcdi_get_workarounds()
to find out if workaround_35388 is enabled.
5) If DRV_ATTACH gets EPERM, try without specifying fw-variant
Unprivileged PCI functions have to use a FIRMWARE_ID of 0xffffffff
(MC_CMD_FW_DONT_CARE).
6) Don't try to exit_assertion unless one had fired
Previously we called efx_mcdi_exit_assertion even if
efx_mcdi_read_assertion had received MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS.
This is unnecessary, and the resulting MC_CMD_REBOOT, even if the
AFTER_ASSERTION flag made it a no-op, would fail EPERM for unprivileged
PCI functions.
So make efx_mcdi_read_assertion return whether an assert happened, and only
call efx_mcdi_exit_assertion if it has.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-05-06 07:59:18 +08:00
|
|
|
rc = efx_mcdi_rpc_quiet(efx, MC_CMD_DRV_ATTACH, inbuf, sizeof(inbuf),
|
|
|
|
outbuf, sizeof(outbuf), &outlen);
|
|
|
|
/* If we're not the primary PF, trying to ATTACH with a FIRMWARE_ID
|
|
|
|
* specified will fail with EPERM, and we have to tell the MC we don't
|
|
|
|
* care what firmware we get.
|
|
|
|
*/
|
|
|
|
if (rc == -EPERM) {
|
|
|
|
netif_dbg(efx, probe, efx->net_dev,
|
|
|
|
"efx_mcdi_drv_attach with fw-variant setting failed EPERM, trying without it\n");
|
|
|
|
MCDI_SET_DWORD(inbuf, DRV_ATTACH_IN_FIRMWARE_ID,
|
|
|
|
MC_CMD_FW_DONT_CARE);
|
|
|
|
rc = efx_mcdi_rpc_quiet(efx, MC_CMD_DRV_ATTACH, inbuf,
|
|
|
|
sizeof(inbuf), outbuf, sizeof(outbuf),
|
|
|
|
&outlen);
|
|
|
|
}
|
|
|
|
if (rc) {
|
|
|
|
efx_mcdi_display_error(efx, MC_CMD_DRV_ATTACH, sizeof(inbuf),
|
|
|
|
outbuf, outlen, rc);
|
2009-11-29 23:15:25 +08:00
|
|
|
goto fail;
|
sfc: Cope with permissions enforcement added to firmware for SR-IOV
* Accept EPERM in some simple cases, the following cases are handled:
1) efx_mcdi_read_assertion()
Unprivileged PCI functions aren't allowed to GET_ASSERTS.
We return success as it's up to the primary PF to deal with asserts.
2) efx_mcdi_mon_probe() in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to read sensor info, and
worrying about sensor data is the primary PF's job.
3) phy_op->reconfigure() in efx_init_port() and efx_reset_up()
Unprivileged functions aren't allowed to MC_CMD_SET_LINK, they just have
to accept the settings (including flow-control, which is what
efx_init_port() is worried about) they've been given.
4) Fallback to GET_WORKAROUNDS in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to set workarounds. So if
efx_mcdi_set_workaround() fails EPERM, use efx_mcdi_get_workarounds()
to find out if workaround_35388 is enabled.
5) If DRV_ATTACH gets EPERM, try without specifying fw-variant
Unprivileged PCI functions have to use a FIRMWARE_ID of 0xffffffff
(MC_CMD_FW_DONT_CARE).
6) Don't try to exit_assertion unless one had fired
Previously we called efx_mcdi_exit_assertion even if
efx_mcdi_read_assertion had received MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS.
This is unnecessary, and the resulting MC_CMD_REBOOT, even if the
AFTER_ASSERTION flag made it a no-op, would fail EPERM for unprivileged
PCI functions.
So make efx_mcdi_read_assertion return whether an assert happened, and only
call efx_mcdi_exit_assertion if it has.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-05-06 07:59:18 +08:00
|
|
|
}
|
2010-04-28 17:27:14 +08:00
|
|
|
if (outlen < MC_CMD_DRV_ATTACH_OUT_LEN) {
|
|
|
|
rc = -EIO;
|
2009-11-29 23:15:25 +08:00
|
|
|
goto fail;
|
2010-04-28 17:27:14 +08:00
|
|
|
}
|
2009-11-29 23:15:25 +08:00
|
|
|
|
2013-10-17 01:32:34 +08:00
|
|
|
if (driver_operating) {
|
|
|
|
if (outlen >= MC_CMD_DRV_ATTACH_EXT_OUT_LEN) {
|
|
|
|
efx->mcdi->fn_flags =
|
|
|
|
MCDI_DWORD(outbuf,
|
|
|
|
DRV_ATTACH_EXT_OUT_FUNC_FLAGS);
|
|
|
|
} else {
|
|
|
|
/* Synthesise flags for Siena */
|
|
|
|
efx->mcdi->fn_flags =
|
|
|
|
1 << MC_CMD_DRV_ATTACH_EXT_OUT_FLAG_LINKCTRL |
|
|
|
|
1 << MC_CMD_DRV_ATTACH_EXT_OUT_FLAG_TRUSTED |
|
|
|
|
(efx_port_num(efx) == 0) <<
|
|
|
|
MC_CMD_DRV_ATTACH_EXT_OUT_FLAG_PRIMARY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-08 03:10:11 +08:00
|
|
|
/* We currently assume we have control of the external link
|
|
|
|
* and are completely trusted by firmware. Abort probing
|
|
|
|
* if that's not true for this function.
|
|
|
|
*/
|
|
|
|
|
2009-11-29 23:15:25 +08:00
|
|
|
if (was_attached != NULL)
|
|
|
|
*was_attached = MCDI_DWORD(outbuf, DRV_ATTACH_OUT_OLD_STATE);
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail:
|
2010-06-23 19:30:07 +08:00
|
|
|
netif_err(efx, probe, efx->net_dev, "%s: failed rc=%d\n", __func__, rc);
|
2009-11-29 23:15:25 +08:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
int efx_mcdi_get_board_cfg(struct efx_nic *efx, u8 *mac_address,
|
2010-07-14 22:36:19 +08:00
|
|
|
u16 *fw_subtype_list, u32 *capabilities)
|
2009-11-29 23:15:25 +08:00
|
|
|
{
|
2012-09-15 00:30:10 +08:00
|
|
|
MCDI_DECLARE_BUF(outbuf, MC_CMD_GET_BOARD_CFG_OUT_LENMAX);
|
2012-09-15 00:31:33 +08:00
|
|
|
size_t outlen, i;
|
2009-11-29 23:15:25 +08:00
|
|
|
int port_num = efx_port_num(efx);
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
BUILD_BUG_ON(MC_CMD_GET_BOARD_CFG_IN_LEN != 0);
|
2014-03-08 02:27:41 +08:00
|
|
|
/* we need __aligned(2) for ether_addr_copy */
|
|
|
|
BUILD_BUG_ON(MC_CMD_GET_BOARD_CFG_OUT_MAC_ADDR_BASE_PORT0_OFST & 1);
|
|
|
|
BUILD_BUG_ON(MC_CMD_GET_BOARD_CFG_OUT_MAC_ADDR_BASE_PORT1_OFST & 1);
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_GET_BOARD_CFG, NULL, 0,
|
|
|
|
outbuf, sizeof(outbuf), &outlen);
|
|
|
|
if (rc)
|
|
|
|
goto fail;
|
|
|
|
|
2011-12-20 08:44:06 +08:00
|
|
|
if (outlen < MC_CMD_GET_BOARD_CFG_OUT_LENMIN) {
|
2010-04-28 17:27:14 +08:00
|
|
|
rc = -EIO;
|
2009-11-29 23:15:25 +08:00
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mac_address)
|
2014-03-08 02:27:41 +08:00
|
|
|
ether_addr_copy(mac_address,
|
|
|
|
port_num ?
|
|
|
|
MCDI_PTR(outbuf, GET_BOARD_CFG_OUT_MAC_ADDR_BASE_PORT1) :
|
|
|
|
MCDI_PTR(outbuf, GET_BOARD_CFG_OUT_MAC_ADDR_BASE_PORT0));
|
2012-09-07 07:58:10 +08:00
|
|
|
if (fw_subtype_list) {
|
|
|
|
for (i = 0;
|
2012-09-15 00:31:33 +08:00
|
|
|
i < MCDI_VAR_ARRAY_LEN(outlen,
|
|
|
|
GET_BOARD_CFG_OUT_FW_SUBTYPE_LIST);
|
|
|
|
i++)
|
|
|
|
fw_subtype_list[i] = MCDI_ARRAY_WORD(
|
|
|
|
outbuf, GET_BOARD_CFG_OUT_FW_SUBTYPE_LIST, i);
|
|
|
|
for (; i < MC_CMD_GET_BOARD_CFG_OUT_FW_SUBTYPE_LIST_MAXNUM; i++)
|
|
|
|
fw_subtype_list[i] = 0;
|
2012-09-07 07:58:10 +08:00
|
|
|
}
|
2010-07-14 22:36:19 +08:00
|
|
|
if (capabilities) {
|
|
|
|
if (port_num)
|
|
|
|
*capabilities = MCDI_DWORD(outbuf,
|
|
|
|
GET_BOARD_CFG_OUT_CAPABILITIES_PORT1);
|
|
|
|
else
|
|
|
|
*capabilities = MCDI_DWORD(outbuf,
|
|
|
|
GET_BOARD_CFG_OUT_CAPABILITIES_PORT0);
|
|
|
|
}
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail:
|
2010-06-23 19:30:07 +08:00
|
|
|
netif_err(efx, hw, efx->net_dev, "%s: failed rc=%d len=%d\n",
|
|
|
|
__func__, rc, (int)outlen);
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
int efx_mcdi_log_ctrl(struct efx_nic *efx, bool evq, bool uart, u32 dest_evq)
|
|
|
|
{
|
2012-09-15 00:30:10 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_LOG_CTRL_IN_LEN);
|
2009-11-29 23:15:25 +08:00
|
|
|
u32 dest = 0;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
if (uart)
|
|
|
|
dest |= MC_CMD_LOG_CTRL_IN_LOG_DEST_UART;
|
|
|
|
if (evq)
|
|
|
|
dest |= MC_CMD_LOG_CTRL_IN_LOG_DEST_EVQ;
|
|
|
|
|
|
|
|
MCDI_SET_DWORD(inbuf, LOG_CTRL_IN_LOG_DEST, dest);
|
|
|
|
MCDI_SET_DWORD(inbuf, LOG_CTRL_IN_LOG_DEST_EVQ, dest_evq);
|
|
|
|
|
|
|
|
BUILD_BUG_ON(MC_CMD_LOG_CTRL_OUT_LEN != 0);
|
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_LOG_CTRL, inbuf, sizeof(inbuf),
|
|
|
|
NULL, 0, NULL);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
int efx_mcdi_nvram_types(struct efx_nic *efx, u32 *nvram_types_out)
|
|
|
|
{
|
2012-09-15 00:30:10 +08:00
|
|
|
MCDI_DECLARE_BUF(outbuf, MC_CMD_NVRAM_TYPES_OUT_LEN);
|
2009-11-29 23:15:25 +08:00
|
|
|
size_t outlen;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
BUILD_BUG_ON(MC_CMD_NVRAM_TYPES_IN_LEN != 0);
|
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_NVRAM_TYPES, NULL, 0,
|
|
|
|
outbuf, sizeof(outbuf), &outlen);
|
|
|
|
if (rc)
|
|
|
|
goto fail;
|
2010-04-28 17:27:14 +08:00
|
|
|
if (outlen < MC_CMD_NVRAM_TYPES_OUT_LEN) {
|
|
|
|
rc = -EIO;
|
2009-11-29 23:15:25 +08:00
|
|
|
goto fail;
|
2010-04-28 17:27:14 +08:00
|
|
|
}
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
*nvram_types_out = MCDI_DWORD(outbuf, NVRAM_TYPES_OUT_TYPES);
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail:
|
2010-06-23 19:30:07 +08:00
|
|
|
netif_err(efx, hw, efx->net_dev, "%s: failed rc=%d\n",
|
|
|
|
__func__, rc);
|
2009-11-29 23:15:25 +08:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
int efx_mcdi_nvram_info(struct efx_nic *efx, unsigned int type,
|
|
|
|
size_t *size_out, size_t *erase_size_out,
|
|
|
|
bool *protected_out)
|
|
|
|
{
|
2012-09-15 00:30:10 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_NVRAM_INFO_IN_LEN);
|
|
|
|
MCDI_DECLARE_BUF(outbuf, MC_CMD_NVRAM_INFO_OUT_LEN);
|
2009-11-29 23:15:25 +08:00
|
|
|
size_t outlen;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
MCDI_SET_DWORD(inbuf, NVRAM_INFO_IN_TYPE, type);
|
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_NVRAM_INFO, inbuf, sizeof(inbuf),
|
|
|
|
outbuf, sizeof(outbuf), &outlen);
|
|
|
|
if (rc)
|
|
|
|
goto fail;
|
2010-04-28 17:27:14 +08:00
|
|
|
if (outlen < MC_CMD_NVRAM_INFO_OUT_LEN) {
|
|
|
|
rc = -EIO;
|
2009-11-29 23:15:25 +08:00
|
|
|
goto fail;
|
2010-04-28 17:27:14 +08:00
|
|
|
}
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
*size_out = MCDI_DWORD(outbuf, NVRAM_INFO_OUT_SIZE);
|
|
|
|
*erase_size_out = MCDI_DWORD(outbuf, NVRAM_INFO_OUT_ERASESIZE);
|
|
|
|
*protected_out = !!(MCDI_DWORD(outbuf, NVRAM_INFO_OUT_FLAGS) &
|
2011-12-20 08:44:06 +08:00
|
|
|
(1 << MC_CMD_NVRAM_INFO_OUT_PROTECTED_LBN));
|
2009-11-29 23:15:25 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail:
|
2010-06-23 19:30:07 +08:00
|
|
|
netif_err(efx, hw, efx->net_dev, "%s: failed rc=%d\n", __func__, rc);
|
2009-11-29 23:15:25 +08:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2010-02-03 17:31:01 +08:00
|
|
|
static int efx_mcdi_nvram_test(struct efx_nic *efx, unsigned int type)
|
|
|
|
{
|
2012-09-15 00:30:10 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_NVRAM_TEST_IN_LEN);
|
|
|
|
MCDI_DECLARE_BUF(outbuf, MC_CMD_NVRAM_TEST_OUT_LEN);
|
2010-02-03 17:31:01 +08:00
|
|
|
int rc;
|
|
|
|
|
|
|
|
MCDI_SET_DWORD(inbuf, NVRAM_TEST_IN_TYPE, type);
|
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_NVRAM_TEST, inbuf, sizeof(inbuf),
|
|
|
|
outbuf, sizeof(outbuf), NULL);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
|
|
|
|
switch (MCDI_DWORD(outbuf, NVRAM_TEST_OUT_RESULT)) {
|
|
|
|
case MC_CMD_NVRAM_TEST_PASS:
|
|
|
|
case MC_CMD_NVRAM_TEST_NOTSUPP:
|
|
|
|
return 0;
|
|
|
|
default:
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int efx_mcdi_nvram_test_all(struct efx_nic *efx)
|
|
|
|
{
|
|
|
|
u32 nvram_types;
|
|
|
|
unsigned int type;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
rc = efx_mcdi_nvram_types(efx, &nvram_types);
|
|
|
|
if (rc)
|
2010-04-28 17:28:36 +08:00
|
|
|
goto fail1;
|
2010-02-03 17:31:01 +08:00
|
|
|
|
|
|
|
type = 0;
|
|
|
|
while (nvram_types != 0) {
|
|
|
|
if (nvram_types & 1) {
|
|
|
|
rc = efx_mcdi_nvram_test(efx, type);
|
|
|
|
if (rc)
|
2010-04-28 17:28:36 +08:00
|
|
|
goto fail2;
|
2010-02-03 17:31:01 +08:00
|
|
|
}
|
|
|
|
type++;
|
|
|
|
nvram_types >>= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2010-04-28 17:28:36 +08:00
|
|
|
|
|
|
|
fail2:
|
2010-06-23 19:30:07 +08:00
|
|
|
netif_err(efx, hw, efx->net_dev, "%s: failed type=%u\n",
|
|
|
|
__func__, type);
|
2010-04-28 17:28:36 +08:00
|
|
|
fail1:
|
2010-06-23 19:30:07 +08:00
|
|
|
netif_err(efx, hw, efx->net_dev, "%s: failed rc=%d\n", __func__, rc);
|
2010-04-28 17:28:36 +08:00
|
|
|
return rc;
|
2010-02-03 17:31:01 +08:00
|
|
|
}
|
|
|
|
|
sfc: Cope with permissions enforcement added to firmware for SR-IOV
* Accept EPERM in some simple cases, the following cases are handled:
1) efx_mcdi_read_assertion()
Unprivileged PCI functions aren't allowed to GET_ASSERTS.
We return success as it's up to the primary PF to deal with asserts.
2) efx_mcdi_mon_probe() in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to read sensor info, and
worrying about sensor data is the primary PF's job.
3) phy_op->reconfigure() in efx_init_port() and efx_reset_up()
Unprivileged functions aren't allowed to MC_CMD_SET_LINK, they just have
to accept the settings (including flow-control, which is what
efx_init_port() is worried about) they've been given.
4) Fallback to GET_WORKAROUNDS in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to set workarounds. So if
efx_mcdi_set_workaround() fails EPERM, use efx_mcdi_get_workarounds()
to find out if workaround_35388 is enabled.
5) If DRV_ATTACH gets EPERM, try without specifying fw-variant
Unprivileged PCI functions have to use a FIRMWARE_ID of 0xffffffff
(MC_CMD_FW_DONT_CARE).
6) Don't try to exit_assertion unless one had fired
Previously we called efx_mcdi_exit_assertion even if
efx_mcdi_read_assertion had received MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS.
This is unnecessary, and the resulting MC_CMD_REBOOT, even if the
AFTER_ASSERTION flag made it a no-op, would fail EPERM for unprivileged
PCI functions.
So make efx_mcdi_read_assertion return whether an assert happened, and only
call efx_mcdi_exit_assertion if it has.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-05-06 07:59:18 +08:00
|
|
|
/* Returns 1 if an assertion was read, 0 if no assertion had fired,
|
|
|
|
* negative on error.
|
|
|
|
*/
|
2010-02-03 17:30:17 +08:00
|
|
|
static int efx_mcdi_read_assertion(struct efx_nic *efx)
|
2009-11-29 23:15:25 +08:00
|
|
|
{
|
2012-09-15 00:30:10 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_GET_ASSERTS_IN_LEN);
|
2015-05-20 18:10:41 +08:00
|
|
|
MCDI_DECLARE_BUF(outbuf, MC_CMD_GET_ASSERTS_OUT_LEN);
|
2012-09-15 00:31:33 +08:00
|
|
|
unsigned int flags, index;
|
2009-11-29 23:15:25 +08:00
|
|
|
const char *reason;
|
|
|
|
size_t outlen;
|
|
|
|
int retry;
|
|
|
|
int rc;
|
|
|
|
|
2010-02-03 17:30:17 +08:00
|
|
|
/* Attempt to read any stored assertion state before we reboot
|
|
|
|
* the mcfw out of the assertion handler. Retry twice, once
|
2009-11-29 23:15:25 +08:00
|
|
|
* because a boot-time assertion might cause this command to fail
|
|
|
|
* with EINTR. And once again because GET_ASSERTS can race with
|
|
|
|
* MC_CMD_REBOOT running on the other port. */
|
|
|
|
retry = 2;
|
|
|
|
do {
|
2010-02-03 17:30:17 +08:00
|
|
|
MCDI_SET_DWORD(inbuf, GET_ASSERTS_IN_CLEAR, 1);
|
2013-06-01 01:36:12 +08:00
|
|
|
rc = efx_mcdi_rpc_quiet(efx, MC_CMD_GET_ASSERTS,
|
|
|
|
inbuf, MC_CMD_GET_ASSERTS_IN_LEN,
|
|
|
|
outbuf, sizeof(outbuf), &outlen);
|
sfc: Cope with permissions enforcement added to firmware for SR-IOV
* Accept EPERM in some simple cases, the following cases are handled:
1) efx_mcdi_read_assertion()
Unprivileged PCI functions aren't allowed to GET_ASSERTS.
We return success as it's up to the primary PF to deal with asserts.
2) efx_mcdi_mon_probe() in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to read sensor info, and
worrying about sensor data is the primary PF's job.
3) phy_op->reconfigure() in efx_init_port() and efx_reset_up()
Unprivileged functions aren't allowed to MC_CMD_SET_LINK, they just have
to accept the settings (including flow-control, which is what
efx_init_port() is worried about) they've been given.
4) Fallback to GET_WORKAROUNDS in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to set workarounds. So if
efx_mcdi_set_workaround() fails EPERM, use efx_mcdi_get_workarounds()
to find out if workaround_35388 is enabled.
5) If DRV_ATTACH gets EPERM, try without specifying fw-variant
Unprivileged PCI functions have to use a FIRMWARE_ID of 0xffffffff
(MC_CMD_FW_DONT_CARE).
6) Don't try to exit_assertion unless one had fired
Previously we called efx_mcdi_exit_assertion even if
efx_mcdi_read_assertion had received MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS.
This is unnecessary, and the resulting MC_CMD_REBOOT, even if the
AFTER_ASSERTION flag made it a no-op, would fail EPERM for unprivileged
PCI functions.
So make efx_mcdi_read_assertion return whether an assert happened, and only
call efx_mcdi_exit_assertion if it has.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-05-06 07:59:18 +08:00
|
|
|
if (rc == -EPERM)
|
|
|
|
return 0;
|
2009-11-29 23:15:25 +08:00
|
|
|
} while ((rc == -EINTR || rc == -EIO) && retry-- > 0);
|
|
|
|
|
2013-06-01 01:36:12 +08:00
|
|
|
if (rc) {
|
|
|
|
efx_mcdi_display_error(efx, MC_CMD_GET_ASSERTS,
|
|
|
|
MC_CMD_GET_ASSERTS_IN_LEN, outbuf,
|
|
|
|
outlen, rc);
|
2009-11-29 23:15:25 +08:00
|
|
|
return rc;
|
2013-06-01 01:36:12 +08:00
|
|
|
}
|
2009-11-29 23:15:25 +08:00
|
|
|
if (outlen < MC_CMD_GET_ASSERTS_OUT_LEN)
|
2010-04-28 17:27:14 +08:00
|
|
|
return -EIO;
|
2009-11-29 23:15:25 +08:00
|
|
|
|
2010-02-03 17:30:17 +08:00
|
|
|
/* Print out any recorded assertion state */
|
|
|
|
flags = MCDI_DWORD(outbuf, GET_ASSERTS_OUT_GLOBAL_FLAGS);
|
2009-11-29 23:15:25 +08:00
|
|
|
if (flags == MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
reason = (flags == MC_CMD_GET_ASSERTS_FLAGS_SYS_FAIL)
|
|
|
|
? "system-level assertion"
|
|
|
|
: (flags == MC_CMD_GET_ASSERTS_FLAGS_THR_FAIL)
|
|
|
|
? "thread-level assertion"
|
|
|
|
: (flags == MC_CMD_GET_ASSERTS_FLAGS_WDOG_FIRED)
|
|
|
|
? "watchdog reset"
|
|
|
|
: "unknown assertion";
|
2010-06-23 19:30:07 +08:00
|
|
|
netif_err(efx, hw, efx->net_dev,
|
|
|
|
"MCPU %s at PC = 0x%.8x in thread 0x%.8x\n", reason,
|
|
|
|
MCDI_DWORD(outbuf, GET_ASSERTS_OUT_SAVED_PC_OFFS),
|
|
|
|
MCDI_DWORD(outbuf, GET_ASSERTS_OUT_THREAD_OFFS));
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
/* Print out the registers */
|
2012-09-15 00:31:33 +08:00
|
|
|
for (index = 0;
|
|
|
|
index < MC_CMD_GET_ASSERTS_OUT_GP_REGS_OFFS_NUM;
|
|
|
|
index++)
|
|
|
|
netif_err(efx, hw, efx->net_dev, "R%.2d (?): 0x%.8x\n",
|
|
|
|
1 + index,
|
|
|
|
MCDI_ARRAY_DWORD(outbuf, GET_ASSERTS_OUT_GP_REGS_OFFS,
|
|
|
|
index));
|
2009-11-29 23:15:25 +08:00
|
|
|
|
sfc: Cope with permissions enforcement added to firmware for SR-IOV
* Accept EPERM in some simple cases, the following cases are handled:
1) efx_mcdi_read_assertion()
Unprivileged PCI functions aren't allowed to GET_ASSERTS.
We return success as it's up to the primary PF to deal with asserts.
2) efx_mcdi_mon_probe() in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to read sensor info, and
worrying about sensor data is the primary PF's job.
3) phy_op->reconfigure() in efx_init_port() and efx_reset_up()
Unprivileged functions aren't allowed to MC_CMD_SET_LINK, they just have
to accept the settings (including flow-control, which is what
efx_init_port() is worried about) they've been given.
4) Fallback to GET_WORKAROUNDS in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to set workarounds. So if
efx_mcdi_set_workaround() fails EPERM, use efx_mcdi_get_workarounds()
to find out if workaround_35388 is enabled.
5) If DRV_ATTACH gets EPERM, try without specifying fw-variant
Unprivileged PCI functions have to use a FIRMWARE_ID of 0xffffffff
(MC_CMD_FW_DONT_CARE).
6) Don't try to exit_assertion unless one had fired
Previously we called efx_mcdi_exit_assertion even if
efx_mcdi_read_assertion had received MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS.
This is unnecessary, and the resulting MC_CMD_REBOOT, even if the
AFTER_ASSERTION flag made it a no-op, would fail EPERM for unprivileged
PCI functions.
So make efx_mcdi_read_assertion return whether an assert happened, and only
call efx_mcdi_exit_assertion if it has.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-05-06 07:59:18 +08:00
|
|
|
return 1;
|
2009-11-29 23:15:25 +08:00
|
|
|
}
|
|
|
|
|
sfc: Cope with permissions enforcement added to firmware for SR-IOV
* Accept EPERM in some simple cases, the following cases are handled:
1) efx_mcdi_read_assertion()
Unprivileged PCI functions aren't allowed to GET_ASSERTS.
We return success as it's up to the primary PF to deal with asserts.
2) efx_mcdi_mon_probe() in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to read sensor info, and
worrying about sensor data is the primary PF's job.
3) phy_op->reconfigure() in efx_init_port() and efx_reset_up()
Unprivileged functions aren't allowed to MC_CMD_SET_LINK, they just have
to accept the settings (including flow-control, which is what
efx_init_port() is worried about) they've been given.
4) Fallback to GET_WORKAROUNDS in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to set workarounds. So if
efx_mcdi_set_workaround() fails EPERM, use efx_mcdi_get_workarounds()
to find out if workaround_35388 is enabled.
5) If DRV_ATTACH gets EPERM, try without specifying fw-variant
Unprivileged PCI functions have to use a FIRMWARE_ID of 0xffffffff
(MC_CMD_FW_DONT_CARE).
6) Don't try to exit_assertion unless one had fired
Previously we called efx_mcdi_exit_assertion even if
efx_mcdi_read_assertion had received MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS.
This is unnecessary, and the resulting MC_CMD_REBOOT, even if the
AFTER_ASSERTION flag made it a no-op, would fail EPERM for unprivileged
PCI functions.
So make efx_mcdi_read_assertion return whether an assert happened, and only
call efx_mcdi_exit_assertion if it has.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-05-06 07:59:18 +08:00
|
|
|
static int efx_mcdi_exit_assertion(struct efx_nic *efx)
|
2010-02-03 17:30:17 +08:00
|
|
|
{
|
2012-09-15 00:30:10 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_REBOOT_IN_LEN);
|
sfc: Cope with permissions enforcement added to firmware for SR-IOV
* Accept EPERM in some simple cases, the following cases are handled:
1) efx_mcdi_read_assertion()
Unprivileged PCI functions aren't allowed to GET_ASSERTS.
We return success as it's up to the primary PF to deal with asserts.
2) efx_mcdi_mon_probe() in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to read sensor info, and
worrying about sensor data is the primary PF's job.
3) phy_op->reconfigure() in efx_init_port() and efx_reset_up()
Unprivileged functions aren't allowed to MC_CMD_SET_LINK, they just have
to accept the settings (including flow-control, which is what
efx_init_port() is worried about) they've been given.
4) Fallback to GET_WORKAROUNDS in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to set workarounds. So if
efx_mcdi_set_workaround() fails EPERM, use efx_mcdi_get_workarounds()
to find out if workaround_35388 is enabled.
5) If DRV_ATTACH gets EPERM, try without specifying fw-variant
Unprivileged PCI functions have to use a FIRMWARE_ID of 0xffffffff
(MC_CMD_FW_DONT_CARE).
6) Don't try to exit_assertion unless one had fired
Previously we called efx_mcdi_exit_assertion even if
efx_mcdi_read_assertion had received MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS.
This is unnecessary, and the resulting MC_CMD_REBOOT, even if the
AFTER_ASSERTION flag made it a no-op, would fail EPERM for unprivileged
PCI functions.
So make efx_mcdi_read_assertion return whether an assert happened, and only
call efx_mcdi_exit_assertion if it has.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-05-06 07:59:18 +08:00
|
|
|
int rc;
|
2010-02-03 17:30:17 +08:00
|
|
|
|
2012-07-03 06:37:40 +08:00
|
|
|
/* If the MC is running debug firmware, it might now be
|
|
|
|
* waiting for a debugger to attach, but we just want it to
|
|
|
|
* reboot. We set a flag that makes the command a no-op if it
|
sfc: Cope with permissions enforcement added to firmware for SR-IOV
* Accept EPERM in some simple cases, the following cases are handled:
1) efx_mcdi_read_assertion()
Unprivileged PCI functions aren't allowed to GET_ASSERTS.
We return success as it's up to the primary PF to deal with asserts.
2) efx_mcdi_mon_probe() in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to read sensor info, and
worrying about sensor data is the primary PF's job.
3) phy_op->reconfigure() in efx_init_port() and efx_reset_up()
Unprivileged functions aren't allowed to MC_CMD_SET_LINK, they just have
to accept the settings (including flow-control, which is what
efx_init_port() is worried about) they've been given.
4) Fallback to GET_WORKAROUNDS in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to set workarounds. So if
efx_mcdi_set_workaround() fails EPERM, use efx_mcdi_get_workarounds()
to find out if workaround_35388 is enabled.
5) If DRV_ATTACH gets EPERM, try without specifying fw-variant
Unprivileged PCI functions have to use a FIRMWARE_ID of 0xffffffff
(MC_CMD_FW_DONT_CARE).
6) Don't try to exit_assertion unless one had fired
Previously we called efx_mcdi_exit_assertion even if
efx_mcdi_read_assertion had received MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS.
This is unnecessary, and the resulting MC_CMD_REBOOT, even if the
AFTER_ASSERTION flag made it a no-op, would fail EPERM for unprivileged
PCI functions.
So make efx_mcdi_read_assertion return whether an assert happened, and only
call efx_mcdi_exit_assertion if it has.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-05-06 07:59:18 +08:00
|
|
|
* has already done so.
|
|
|
|
* The MCDI will thus return either 0 or -EIO.
|
2012-07-03 06:37:40 +08:00
|
|
|
*/
|
2010-02-03 17:30:17 +08:00
|
|
|
BUILD_BUG_ON(MC_CMD_REBOOT_OUT_LEN != 0);
|
|
|
|
MCDI_SET_DWORD(inbuf, REBOOT_IN_FLAGS,
|
|
|
|
MC_CMD_REBOOT_FLAGS_AFTER_ASSERTION);
|
sfc: Cope with permissions enforcement added to firmware for SR-IOV
* Accept EPERM in some simple cases, the following cases are handled:
1) efx_mcdi_read_assertion()
Unprivileged PCI functions aren't allowed to GET_ASSERTS.
We return success as it's up to the primary PF to deal with asserts.
2) efx_mcdi_mon_probe() in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to read sensor info, and
worrying about sensor data is the primary PF's job.
3) phy_op->reconfigure() in efx_init_port() and efx_reset_up()
Unprivileged functions aren't allowed to MC_CMD_SET_LINK, they just have
to accept the settings (including flow-control, which is what
efx_init_port() is worried about) they've been given.
4) Fallback to GET_WORKAROUNDS in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to set workarounds. So if
efx_mcdi_set_workaround() fails EPERM, use efx_mcdi_get_workarounds()
to find out if workaround_35388 is enabled.
5) If DRV_ATTACH gets EPERM, try without specifying fw-variant
Unprivileged PCI functions have to use a FIRMWARE_ID of 0xffffffff
(MC_CMD_FW_DONT_CARE).
6) Don't try to exit_assertion unless one had fired
Previously we called efx_mcdi_exit_assertion even if
efx_mcdi_read_assertion had received MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS.
This is unnecessary, and the resulting MC_CMD_REBOOT, even if the
AFTER_ASSERTION flag made it a no-op, would fail EPERM for unprivileged
PCI functions.
So make efx_mcdi_read_assertion return whether an assert happened, and only
call efx_mcdi_exit_assertion if it has.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-05-06 07:59:18 +08:00
|
|
|
rc = efx_mcdi_rpc_quiet(efx, MC_CMD_REBOOT, inbuf, MC_CMD_REBOOT_IN_LEN,
|
|
|
|
NULL, 0, NULL);
|
|
|
|
if (rc == -EIO)
|
|
|
|
rc = 0;
|
|
|
|
if (rc)
|
|
|
|
efx_mcdi_display_error(efx, MC_CMD_REBOOT, MC_CMD_REBOOT_IN_LEN,
|
|
|
|
NULL, 0, rc);
|
|
|
|
return rc;
|
2010-02-03 17:30:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int efx_mcdi_handle_assertion(struct efx_nic *efx)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
rc = efx_mcdi_read_assertion(efx);
|
sfc: Cope with permissions enforcement added to firmware for SR-IOV
* Accept EPERM in some simple cases, the following cases are handled:
1) efx_mcdi_read_assertion()
Unprivileged PCI functions aren't allowed to GET_ASSERTS.
We return success as it's up to the primary PF to deal with asserts.
2) efx_mcdi_mon_probe() in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to read sensor info, and
worrying about sensor data is the primary PF's job.
3) phy_op->reconfigure() in efx_init_port() and efx_reset_up()
Unprivileged functions aren't allowed to MC_CMD_SET_LINK, they just have
to accept the settings (including flow-control, which is what
efx_init_port() is worried about) they've been given.
4) Fallback to GET_WORKAROUNDS in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to set workarounds. So if
efx_mcdi_set_workaround() fails EPERM, use efx_mcdi_get_workarounds()
to find out if workaround_35388 is enabled.
5) If DRV_ATTACH gets EPERM, try without specifying fw-variant
Unprivileged PCI functions have to use a FIRMWARE_ID of 0xffffffff
(MC_CMD_FW_DONT_CARE).
6) Don't try to exit_assertion unless one had fired
Previously we called efx_mcdi_exit_assertion even if
efx_mcdi_read_assertion had received MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS.
This is unnecessary, and the resulting MC_CMD_REBOOT, even if the
AFTER_ASSERTION flag made it a no-op, would fail EPERM for unprivileged
PCI functions.
So make efx_mcdi_read_assertion return whether an assert happened, and only
call efx_mcdi_exit_assertion if it has.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-05-06 07:59:18 +08:00
|
|
|
if (rc <= 0)
|
2010-02-03 17:30:17 +08:00
|
|
|
return rc;
|
|
|
|
|
sfc: Cope with permissions enforcement added to firmware for SR-IOV
* Accept EPERM in some simple cases, the following cases are handled:
1) efx_mcdi_read_assertion()
Unprivileged PCI functions aren't allowed to GET_ASSERTS.
We return success as it's up to the primary PF to deal with asserts.
2) efx_mcdi_mon_probe() in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to read sensor info, and
worrying about sensor data is the primary PF's job.
3) phy_op->reconfigure() in efx_init_port() and efx_reset_up()
Unprivileged functions aren't allowed to MC_CMD_SET_LINK, they just have
to accept the settings (including flow-control, which is what
efx_init_port() is worried about) they've been given.
4) Fallback to GET_WORKAROUNDS in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to set workarounds. So if
efx_mcdi_set_workaround() fails EPERM, use efx_mcdi_get_workarounds()
to find out if workaround_35388 is enabled.
5) If DRV_ATTACH gets EPERM, try without specifying fw-variant
Unprivileged PCI functions have to use a FIRMWARE_ID of 0xffffffff
(MC_CMD_FW_DONT_CARE).
6) Don't try to exit_assertion unless one had fired
Previously we called efx_mcdi_exit_assertion even if
efx_mcdi_read_assertion had received MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS.
This is unnecessary, and the resulting MC_CMD_REBOOT, even if the
AFTER_ASSERTION flag made it a no-op, would fail EPERM for unprivileged
PCI functions.
So make efx_mcdi_read_assertion return whether an assert happened, and only
call efx_mcdi_exit_assertion if it has.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-05-06 07:59:18 +08:00
|
|
|
return efx_mcdi_exit_assertion(efx);
|
2010-02-03 17:30:17 +08:00
|
|
|
}
|
|
|
|
|
2009-11-29 23:15:25 +08:00
|
|
|
void efx_mcdi_set_id_led(struct efx_nic *efx, enum efx_led_mode mode)
|
|
|
|
{
|
2012-09-15 00:30:10 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_SET_ID_LED_IN_LEN);
|
2009-11-29 23:15:25 +08:00
|
|
|
int rc;
|
|
|
|
|
|
|
|
BUILD_BUG_ON(EFX_LED_OFF != MC_CMD_LED_OFF);
|
|
|
|
BUILD_BUG_ON(EFX_LED_ON != MC_CMD_LED_ON);
|
|
|
|
BUILD_BUG_ON(EFX_LED_DEFAULT != MC_CMD_LED_DEFAULT);
|
|
|
|
|
|
|
|
BUILD_BUG_ON(MC_CMD_SET_ID_LED_OUT_LEN != 0);
|
|
|
|
|
|
|
|
MCDI_SET_DWORD(inbuf, SET_ID_LED_IN_STATE, mode);
|
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_SET_ID_LED, inbuf, sizeof(inbuf),
|
|
|
|
NULL, 0, NULL);
|
|
|
|
}
|
|
|
|
|
2014-01-18 03:48:06 +08:00
|
|
|
static int efx_mcdi_reset_func(struct efx_nic *efx)
|
2009-11-29 23:15:25 +08:00
|
|
|
{
|
2014-01-18 03:48:06 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_ENTITY_RESET_IN_LEN);
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
BUILD_BUG_ON(MC_CMD_ENTITY_RESET_OUT_LEN != 0);
|
|
|
|
MCDI_POPULATE_DWORD_1(inbuf, ENTITY_RESET_IN_FLAG,
|
|
|
|
ENTITY_RESET_IN_FUNCTION_RESOURCE_RESET, 1);
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_ENTITY_RESET, inbuf, sizeof(inbuf),
|
|
|
|
NULL, 0, NULL);
|
|
|
|
return rc;
|
2009-11-29 23:15:25 +08:00
|
|
|
}
|
|
|
|
|
2012-09-18 09:33:52 +08:00
|
|
|
static int efx_mcdi_reset_mc(struct efx_nic *efx)
|
2009-11-29 23:15:25 +08:00
|
|
|
{
|
2012-09-15 00:30:10 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_REBOOT_IN_LEN);
|
2009-11-29 23:15:25 +08:00
|
|
|
int rc;
|
|
|
|
|
|
|
|
BUILD_BUG_ON(MC_CMD_REBOOT_OUT_LEN != 0);
|
|
|
|
MCDI_SET_DWORD(inbuf, REBOOT_IN_FLAGS, 0);
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_REBOOT, inbuf, sizeof(inbuf),
|
|
|
|
NULL, 0, NULL);
|
|
|
|
/* White is black, and up is down */
|
|
|
|
if (rc == -EIO)
|
|
|
|
return 0;
|
|
|
|
if (rc == 0)
|
|
|
|
rc = -EIO;
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2012-09-18 09:33:52 +08:00
|
|
|
enum reset_type efx_mcdi_map_reset_reason(enum reset_type reason)
|
|
|
|
{
|
|
|
|
return RESET_TYPE_RECOVER_OR_ALL;
|
|
|
|
}
|
|
|
|
|
|
|
|
int efx_mcdi_reset(struct efx_nic *efx, enum reset_type method)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
|
sfc:On MCDI timeout, issue an FLR (and mark MCDI to fail-fast)
When an MCDI command times out (whether or not we find it
completed when we poll), call efx_mcdi_abandon(), which tells
all subsequent MCDI calls to fail-fast, and queues up an FLR.
Because an FLR doesn't lead to receiving any reboot even from
the MC (unlike most other types of reset), we have to call
efx_ef10_reset_mc_allocations.
In efx_start_all(), if a reset (of any kind) is pending, we
bail out.
Without this, attempts to reconfigure (e.g. change mtu) can
cause driver/mc state inconsistency if the first MCDI call
triggers an FLR.
For similar reasons, on EF10, in
efx_reset_down(method=RESET_TYPE_MCDI_TIMEOUT), set the number
of active queues to zero before calling efx_stop_all().
And, on farch, in efx_reset_up(method=RESET_TYPE_MCDI_TIMEOUT),
set active_queues and flushes pending & outstanding to zero.
efx_mcdi_mode_{poll,event}() should not take us out of fail-fast
mode. Instead, this is done by efx_mcdi_reset() after the FLR
completes.
The new FLR reset_type RESET_TYPE_MCDI_TIMEOUT doesn't really
fit into the hierarchy of reset 'scopes' whereby efx_reset()
decides some resets subsume others. Thus, it uses separate logic.
Also, fixed up some inconsistency around RESET_TYPE_MC_BIST,
which was in the wrong place in that hierarchy.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-04-17 02:27:48 +08:00
|
|
|
/* If MCDI is down, we can't handle_assertion */
|
|
|
|
if (method == RESET_TYPE_MCDI_TIMEOUT) {
|
|
|
|
rc = pci_reset_function(efx->pci_dev);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
/* Re-enable polled MCDI completion */
|
|
|
|
if (efx->mcdi) {
|
|
|
|
struct efx_mcdi_iface *mcdi = efx_mcdi(efx);
|
|
|
|
mcdi->mode = MCDI_MODE_POLL;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-09-18 09:33:52 +08:00
|
|
|
/* Recover from a failed assertion pre-reset */
|
|
|
|
rc = efx_mcdi_handle_assertion(efx);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
|
2015-05-20 18:11:35 +08:00
|
|
|
if (method == RESET_TYPE_DATAPATH)
|
|
|
|
return 0;
|
|
|
|
else if (method == RESET_TYPE_WORLD)
|
2012-09-18 09:33:52 +08:00
|
|
|
return efx_mcdi_reset_mc(efx);
|
|
|
|
else
|
2014-01-18 03:48:06 +08:00
|
|
|
return efx_mcdi_reset_func(efx);
|
2012-09-18 09:33:52 +08:00
|
|
|
}
|
|
|
|
|
2010-10-18 13:27:31 +08:00
|
|
|
static int efx_mcdi_wol_filter_set(struct efx_nic *efx, u32 type,
|
|
|
|
const u8 *mac, int *id_out)
|
2009-11-29 23:15:25 +08:00
|
|
|
{
|
2012-09-15 00:30:10 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_WOL_FILTER_SET_IN_LEN);
|
|
|
|
MCDI_DECLARE_BUF(outbuf, MC_CMD_WOL_FILTER_SET_OUT_LEN);
|
2009-11-29 23:15:25 +08:00
|
|
|
size_t outlen;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
MCDI_SET_DWORD(inbuf, WOL_FILTER_SET_IN_WOL_TYPE, type);
|
|
|
|
MCDI_SET_DWORD(inbuf, WOL_FILTER_SET_IN_FILTER_MODE,
|
|
|
|
MC_CMD_FILTER_MODE_SIMPLE);
|
2014-03-08 02:27:41 +08:00
|
|
|
ether_addr_copy(MCDI_PTR(inbuf, WOL_FILTER_SET_IN_MAGIC_MAC), mac);
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_WOL_FILTER_SET, inbuf, sizeof(inbuf),
|
|
|
|
outbuf, sizeof(outbuf), &outlen);
|
|
|
|
if (rc)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
if (outlen < MC_CMD_WOL_FILTER_SET_OUT_LEN) {
|
2010-04-28 17:27:14 +08:00
|
|
|
rc = -EIO;
|
2009-11-29 23:15:25 +08:00
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
*id_out = (int)MCDI_DWORD(outbuf, WOL_FILTER_SET_OUT_FILTER_ID);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail:
|
|
|
|
*id_out = -1;
|
2010-06-23 19:30:07 +08:00
|
|
|
netif_err(efx, hw, efx->net_dev, "%s: failed rc=%d\n", __func__, rc);
|
2009-11-29 23:15:25 +08:00
|
|
|
return rc;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
efx_mcdi_wol_filter_set_magic(struct efx_nic *efx, const u8 *mac, int *id_out)
|
|
|
|
{
|
|
|
|
return efx_mcdi_wol_filter_set(efx, MC_CMD_WOL_TYPE_MAGIC, mac, id_out);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int efx_mcdi_wol_filter_get_magic(struct efx_nic *efx, int *id_out)
|
|
|
|
{
|
2012-09-15 00:30:10 +08:00
|
|
|
MCDI_DECLARE_BUF(outbuf, MC_CMD_WOL_FILTER_GET_OUT_LEN);
|
2009-11-29 23:15:25 +08:00
|
|
|
size_t outlen;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_WOL_FILTER_GET, NULL, 0,
|
|
|
|
outbuf, sizeof(outbuf), &outlen);
|
|
|
|
if (rc)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
if (outlen < MC_CMD_WOL_FILTER_GET_OUT_LEN) {
|
2010-04-28 17:27:14 +08:00
|
|
|
rc = -EIO;
|
2009-11-29 23:15:25 +08:00
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
*id_out = (int)MCDI_DWORD(outbuf, WOL_FILTER_GET_OUT_FILTER_ID);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail:
|
|
|
|
*id_out = -1;
|
2010-06-23 19:30:07 +08:00
|
|
|
netif_err(efx, hw, efx->net_dev, "%s: failed rc=%d\n", __func__, rc);
|
2009-11-29 23:15:25 +08:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int efx_mcdi_wol_filter_remove(struct efx_nic *efx, int id)
|
|
|
|
{
|
2012-09-15 00:30:10 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_WOL_FILTER_REMOVE_IN_LEN);
|
2009-11-29 23:15:25 +08:00
|
|
|
int rc;
|
|
|
|
|
|
|
|
MCDI_SET_DWORD(inbuf, WOL_FILTER_REMOVE_IN_FILTER_ID, (u32)id);
|
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_WOL_FILTER_REMOVE, inbuf, sizeof(inbuf),
|
|
|
|
NULL, 0, NULL);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
sfc: Add SR-IOV back-end support for SFC9000 family
On the SFC9000 family, each port has 1024 Virtual Interfaces (VIs),
each with an RX queue, a TX queue, an event queue and a mailbox
register. These may be assigned to up to 127 SR-IOV virtual functions
per port, with up to 64 VIs per VF.
We allocate an extra channel (IRQ and event queue only) to receive
requests from VF drivers.
There is a per-port limit of 4 concurrent RX queue flushes, and queue
flushes may be initiated by the MC in response to a Function Level
Reset (FLR) of a VF. Therefore, when SR-IOV is in use, we submit all
flush requests via the MC.
The RSS indirection table is shared with VFs, so the number of RX
queues used in the PF is limited to the number of VIs per VF.
This is almost entirely the work of Steve Hodgson, formerly
shodgson@solarflare.com.
Signed-off-by: Ben Hutchings <bhutchings@solarflare.com>
2012-02-14 08:48:07 +08:00
|
|
|
int efx_mcdi_flush_rxqs(struct efx_nic *efx)
|
|
|
|
{
|
|
|
|
struct efx_channel *channel;
|
|
|
|
struct efx_rx_queue *rx_queue;
|
2012-09-15 00:31:33 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf,
|
|
|
|
MC_CMD_FLUSH_RX_QUEUES_IN_LEN(EFX_MAX_CHANNELS));
|
sfc: Add SR-IOV back-end support for SFC9000 family
On the SFC9000 family, each port has 1024 Virtual Interfaces (VIs),
each with an RX queue, a TX queue, an event queue and a mailbox
register. These may be assigned to up to 127 SR-IOV virtual functions
per port, with up to 64 VIs per VF.
We allocate an extra channel (IRQ and event queue only) to receive
requests from VF drivers.
There is a per-port limit of 4 concurrent RX queue flushes, and queue
flushes may be initiated by the MC in response to a Function Level
Reset (FLR) of a VF. Therefore, when SR-IOV is in use, we submit all
flush requests via the MC.
The RSS indirection table is shared with VFs, so the number of RX
queues used in the PF is limited to the number of VIs per VF.
This is almost entirely the work of Steve Hodgson, formerly
shodgson@solarflare.com.
Signed-off-by: Ben Hutchings <bhutchings@solarflare.com>
2012-02-14 08:48:07 +08:00
|
|
|
int rc, count;
|
|
|
|
|
2012-09-19 09:53:34 +08:00
|
|
|
BUILD_BUG_ON(EFX_MAX_CHANNELS >
|
|
|
|
MC_CMD_FLUSH_RX_QUEUES_IN_QID_OFST_MAXNUM);
|
|
|
|
|
sfc: Add SR-IOV back-end support for SFC9000 family
On the SFC9000 family, each port has 1024 Virtual Interfaces (VIs),
each with an RX queue, a TX queue, an event queue and a mailbox
register. These may be assigned to up to 127 SR-IOV virtual functions
per port, with up to 64 VIs per VF.
We allocate an extra channel (IRQ and event queue only) to receive
requests from VF drivers.
There is a per-port limit of 4 concurrent RX queue flushes, and queue
flushes may be initiated by the MC in response to a Function Level
Reset (FLR) of a VF. Therefore, when SR-IOV is in use, we submit all
flush requests via the MC.
The RSS indirection table is shared with VFs, so the number of RX
queues used in the PF is limited to the number of VIs per VF.
This is almost entirely the work of Steve Hodgson, formerly
shodgson@solarflare.com.
Signed-off-by: Ben Hutchings <bhutchings@solarflare.com>
2012-02-14 08:48:07 +08:00
|
|
|
count = 0;
|
|
|
|
efx_for_each_channel(channel, efx) {
|
|
|
|
efx_for_each_channel_rx_queue(rx_queue, channel) {
|
|
|
|
if (rx_queue->flush_pending) {
|
|
|
|
rx_queue->flush_pending = false;
|
|
|
|
atomic_dec(&efx->rxq_flush_pending);
|
2012-09-15 00:31:33 +08:00
|
|
|
MCDI_SET_ARRAY_DWORD(
|
|
|
|
inbuf, FLUSH_RX_QUEUES_IN_QID_OFST,
|
|
|
|
count, efx_rx_queue_index(rx_queue));
|
|
|
|
count++;
|
sfc: Add SR-IOV back-end support for SFC9000 family
On the SFC9000 family, each port has 1024 Virtual Interfaces (VIs),
each with an RX queue, a TX queue, an event queue and a mailbox
register. These may be assigned to up to 127 SR-IOV virtual functions
per port, with up to 64 VIs per VF.
We allocate an extra channel (IRQ and event queue only) to receive
requests from VF drivers.
There is a per-port limit of 4 concurrent RX queue flushes, and queue
flushes may be initiated by the MC in response to a Function Level
Reset (FLR) of a VF. Therefore, when SR-IOV is in use, we submit all
flush requests via the MC.
The RSS indirection table is shared with VFs, so the number of RX
queues used in the PF is limited to the number of VIs per VF.
This is almost entirely the work of Steve Hodgson, formerly
shodgson@solarflare.com.
Signed-off-by: Ben Hutchings <bhutchings@solarflare.com>
2012-02-14 08:48:07 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-15 00:31:33 +08:00
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_FLUSH_RX_QUEUES, inbuf,
|
|
|
|
MC_CMD_FLUSH_RX_QUEUES_IN_LEN(count), NULL, 0, NULL);
|
2012-09-12 01:25:13 +08:00
|
|
|
WARN_ON(rc < 0);
|
sfc: Add SR-IOV back-end support for SFC9000 family
On the SFC9000 family, each port has 1024 Virtual Interfaces (VIs),
each with an RX queue, a TX queue, an event queue and a mailbox
register. These may be assigned to up to 127 SR-IOV virtual functions
per port, with up to 64 VIs per VF.
We allocate an extra channel (IRQ and event queue only) to receive
requests from VF drivers.
There is a per-port limit of 4 concurrent RX queue flushes, and queue
flushes may be initiated by the MC in response to a Function Level
Reset (FLR) of a VF. Therefore, when SR-IOV is in use, we submit all
flush requests via the MC.
The RSS indirection table is shared with VFs, so the number of RX
queues used in the PF is limited to the number of VIs per VF.
This is almost entirely the work of Steve Hodgson, formerly
shodgson@solarflare.com.
Signed-off-by: Ben Hutchings <bhutchings@solarflare.com>
2012-02-14 08:48:07 +08:00
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
2009-11-29 23:15:25 +08:00
|
|
|
|
|
|
|
int efx_mcdi_wol_filter_reset(struct efx_nic *efx)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_WOL_FILTER_RESET, NULL, 0, NULL, 0, NULL);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2015-07-21 22:09:43 +08:00
|
|
|
int efx_mcdi_set_workaround(struct efx_nic *efx, u32 type, bool enabled,
|
|
|
|
unsigned int *flags)
|
2013-08-30 02:19:29 +08:00
|
|
|
{
|
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_WORKAROUND_IN_LEN);
|
2015-07-21 22:09:43 +08:00
|
|
|
MCDI_DECLARE_BUF(outbuf, MC_CMD_WORKAROUND_EXT_OUT_LEN);
|
|
|
|
size_t outlen;
|
|
|
|
int rc;
|
2013-08-30 02:19:29 +08:00
|
|
|
|
|
|
|
BUILD_BUG_ON(MC_CMD_WORKAROUND_OUT_LEN != 0);
|
|
|
|
MCDI_SET_DWORD(inbuf, WORKAROUND_IN_TYPE, type);
|
|
|
|
MCDI_SET_DWORD(inbuf, WORKAROUND_IN_ENABLED, enabled);
|
2015-07-21 22:09:43 +08:00
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_WORKAROUND, inbuf, sizeof(inbuf),
|
|
|
|
outbuf, sizeof(outbuf), &outlen);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
|
|
|
|
if (!flags)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (outlen >= MC_CMD_WORKAROUND_EXT_OUT_LEN)
|
|
|
|
*flags = MCDI_DWORD(outbuf, WORKAROUND_EXT_OUT_FLAGS);
|
|
|
|
else
|
|
|
|
*flags = 0;
|
|
|
|
|
|
|
|
return 0;
|
2013-08-30 02:19:29 +08:00
|
|
|
}
|
|
|
|
|
sfc: Cope with permissions enforcement added to firmware for SR-IOV
* Accept EPERM in some simple cases, the following cases are handled:
1) efx_mcdi_read_assertion()
Unprivileged PCI functions aren't allowed to GET_ASSERTS.
We return success as it's up to the primary PF to deal with asserts.
2) efx_mcdi_mon_probe() in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to read sensor info, and
worrying about sensor data is the primary PF's job.
3) phy_op->reconfigure() in efx_init_port() and efx_reset_up()
Unprivileged functions aren't allowed to MC_CMD_SET_LINK, they just have
to accept the settings (including flow-control, which is what
efx_init_port() is worried about) they've been given.
4) Fallback to GET_WORKAROUNDS in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to set workarounds. So if
efx_mcdi_set_workaround() fails EPERM, use efx_mcdi_get_workarounds()
to find out if workaround_35388 is enabled.
5) If DRV_ATTACH gets EPERM, try without specifying fw-variant
Unprivileged PCI functions have to use a FIRMWARE_ID of 0xffffffff
(MC_CMD_FW_DONT_CARE).
6) Don't try to exit_assertion unless one had fired
Previously we called efx_mcdi_exit_assertion even if
efx_mcdi_read_assertion had received MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS.
This is unnecessary, and the resulting MC_CMD_REBOOT, even if the
AFTER_ASSERTION flag made it a no-op, would fail EPERM for unprivileged
PCI functions.
So make efx_mcdi_read_assertion return whether an assert happened, and only
call efx_mcdi_exit_assertion if it has.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-05-06 07:59:18 +08:00
|
|
|
int efx_mcdi_get_workarounds(struct efx_nic *efx, unsigned int *impl_out,
|
|
|
|
unsigned int *enabled_out)
|
|
|
|
{
|
2015-05-20 18:10:41 +08:00
|
|
|
MCDI_DECLARE_BUF(outbuf, MC_CMD_GET_WORKAROUNDS_OUT_LEN);
|
sfc: Cope with permissions enforcement added to firmware for SR-IOV
* Accept EPERM in some simple cases, the following cases are handled:
1) efx_mcdi_read_assertion()
Unprivileged PCI functions aren't allowed to GET_ASSERTS.
We return success as it's up to the primary PF to deal with asserts.
2) efx_mcdi_mon_probe() in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to read sensor info, and
worrying about sensor data is the primary PF's job.
3) phy_op->reconfigure() in efx_init_port() and efx_reset_up()
Unprivileged functions aren't allowed to MC_CMD_SET_LINK, they just have
to accept the settings (including flow-control, which is what
efx_init_port() is worried about) they've been given.
4) Fallback to GET_WORKAROUNDS in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to set workarounds. So if
efx_mcdi_set_workaround() fails EPERM, use efx_mcdi_get_workarounds()
to find out if workaround_35388 is enabled.
5) If DRV_ATTACH gets EPERM, try without specifying fw-variant
Unprivileged PCI functions have to use a FIRMWARE_ID of 0xffffffff
(MC_CMD_FW_DONT_CARE).
6) Don't try to exit_assertion unless one had fired
Previously we called efx_mcdi_exit_assertion even if
efx_mcdi_read_assertion had received MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS.
This is unnecessary, and the resulting MC_CMD_REBOOT, even if the
AFTER_ASSERTION flag made it a no-op, would fail EPERM for unprivileged
PCI functions.
So make efx_mcdi_read_assertion return whether an assert happened, and only
call efx_mcdi_exit_assertion if it has.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-05-06 07:59:18 +08:00
|
|
|
size_t outlen;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_GET_WORKAROUNDS, NULL, 0,
|
|
|
|
outbuf, sizeof(outbuf), &outlen);
|
|
|
|
if (rc)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
if (outlen < MC_CMD_GET_WORKAROUNDS_OUT_LEN) {
|
|
|
|
rc = -EIO;
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl_out)
|
|
|
|
*impl_out = MCDI_DWORD(outbuf, GET_WORKAROUNDS_OUT_IMPLEMENTED);
|
|
|
|
|
|
|
|
if (enabled_out)
|
|
|
|
*enabled_out = MCDI_DWORD(outbuf, GET_WORKAROUNDS_OUT_ENABLED);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail:
|
2015-07-21 22:09:31 +08:00
|
|
|
/* Older firmware lacks GET_WORKAROUNDS and this isn't especially
|
|
|
|
* terrifying. The call site will have to deal with it though.
|
|
|
|
*/
|
2017-01-27 23:02:39 +08:00
|
|
|
netif_cond_dbg(efx, hw, efx->net_dev, rc == -ENOSYS, err,
|
|
|
|
"%s: failed rc=%d\n", __func__, rc);
|
sfc: Cope with permissions enforcement added to firmware for SR-IOV
* Accept EPERM in some simple cases, the following cases are handled:
1) efx_mcdi_read_assertion()
Unprivileged PCI functions aren't allowed to GET_ASSERTS.
We return success as it's up to the primary PF to deal with asserts.
2) efx_mcdi_mon_probe() in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to read sensor info, and
worrying about sensor data is the primary PF's job.
3) phy_op->reconfigure() in efx_init_port() and efx_reset_up()
Unprivileged functions aren't allowed to MC_CMD_SET_LINK, they just have
to accept the settings (including flow-control, which is what
efx_init_port() is worried about) they've been given.
4) Fallback to GET_WORKAROUNDS in efx_ef10_probe()
Unprivileged PCI functions aren't allowed to set workarounds. So if
efx_mcdi_set_workaround() fails EPERM, use efx_mcdi_get_workarounds()
to find out if workaround_35388 is enabled.
5) If DRV_ATTACH gets EPERM, try without specifying fw-variant
Unprivileged PCI functions have to use a FIRMWARE_ID of 0xffffffff
(MC_CMD_FW_DONT_CARE).
6) Don't try to exit_assertion unless one had fired
Previously we called efx_mcdi_exit_assertion even if
efx_mcdi_read_assertion had received MC_CMD_GET_ASSERTS_FLAGS_NO_FAILS.
This is unnecessary, and the resulting MC_CMD_REBOOT, even if the
AFTER_ASSERTION flag made it a no-op, would fail EPERM for unprivileged
PCI functions.
So make efx_mcdi_read_assertion return whether an assert happened, and only
call efx_mcdi_exit_assertion if it has.
Signed-off-by: Shradha Shah <sshah@solarflare.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-05-06 07:59:18 +08:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2012-11-28 12:38:14 +08:00
|
|
|
#ifdef CONFIG_SFC_MTD
|
|
|
|
|
|
|
|
#define EFX_MCDI_NVRAM_LEN_MAX 128
|
|
|
|
|
|
|
|
static int efx_mcdi_nvram_update_start(struct efx_nic *efx, unsigned int type)
|
|
|
|
{
|
2019-01-16 18:00:39 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_NVRAM_UPDATE_START_V2_IN_LEN);
|
2012-11-28 12:38:14 +08:00
|
|
|
int rc;
|
|
|
|
|
|
|
|
MCDI_SET_DWORD(inbuf, NVRAM_UPDATE_START_IN_TYPE, type);
|
2019-01-16 18:00:39 +08:00
|
|
|
MCDI_POPULATE_DWORD_1(inbuf, NVRAM_UPDATE_START_V2_IN_FLAGS,
|
|
|
|
NVRAM_UPDATE_START_V2_IN_FLAG_REPORT_VERIFY_RESULT,
|
|
|
|
1);
|
2012-11-28 12:38:14 +08:00
|
|
|
|
|
|
|
BUILD_BUG_ON(MC_CMD_NVRAM_UPDATE_START_OUT_LEN != 0);
|
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_NVRAM_UPDATE_START, inbuf, sizeof(inbuf),
|
|
|
|
NULL, 0, NULL);
|
2019-01-16 18:00:39 +08:00
|
|
|
|
2012-11-28 12:38:14 +08:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int efx_mcdi_nvram_read(struct efx_nic *efx, unsigned int type,
|
|
|
|
loff_t offset, u8 *buffer, size_t length)
|
|
|
|
{
|
2019-01-16 18:00:39 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_NVRAM_READ_IN_V2_LEN);
|
2012-11-28 12:38:14 +08:00
|
|
|
MCDI_DECLARE_BUF(outbuf,
|
|
|
|
MC_CMD_NVRAM_READ_OUT_LEN(EFX_MCDI_NVRAM_LEN_MAX));
|
|
|
|
size_t outlen;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
MCDI_SET_DWORD(inbuf, NVRAM_READ_IN_TYPE, type);
|
|
|
|
MCDI_SET_DWORD(inbuf, NVRAM_READ_IN_OFFSET, offset);
|
|
|
|
MCDI_SET_DWORD(inbuf, NVRAM_READ_IN_LENGTH, length);
|
2019-01-16 18:00:39 +08:00
|
|
|
MCDI_SET_DWORD(inbuf, NVRAM_READ_IN_V2_MODE,
|
|
|
|
MC_CMD_NVRAM_READ_IN_V2_DEFAULT);
|
2012-11-28 12:38:14 +08:00
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_NVRAM_READ, inbuf, sizeof(inbuf),
|
|
|
|
outbuf, sizeof(outbuf), &outlen);
|
|
|
|
if (rc)
|
2013-06-01 01:36:12 +08:00
|
|
|
return rc;
|
2012-11-28 12:38:14 +08:00
|
|
|
|
|
|
|
memcpy(buffer, MCDI_PTR(outbuf, NVRAM_READ_OUT_READ_BUFFER), length);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int efx_mcdi_nvram_write(struct efx_nic *efx, unsigned int type,
|
|
|
|
loff_t offset, const u8 *buffer, size_t length)
|
|
|
|
{
|
|
|
|
MCDI_DECLARE_BUF(inbuf,
|
|
|
|
MC_CMD_NVRAM_WRITE_IN_LEN(EFX_MCDI_NVRAM_LEN_MAX));
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
MCDI_SET_DWORD(inbuf, NVRAM_WRITE_IN_TYPE, type);
|
|
|
|
MCDI_SET_DWORD(inbuf, NVRAM_WRITE_IN_OFFSET, offset);
|
|
|
|
MCDI_SET_DWORD(inbuf, NVRAM_WRITE_IN_LENGTH, length);
|
|
|
|
memcpy(MCDI_PTR(inbuf, NVRAM_WRITE_IN_WRITE_BUFFER), buffer, length);
|
|
|
|
|
|
|
|
BUILD_BUG_ON(MC_CMD_NVRAM_WRITE_OUT_LEN != 0);
|
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_NVRAM_WRITE, inbuf,
|
|
|
|
ALIGN(MC_CMD_NVRAM_WRITE_IN_LEN(length), 4),
|
|
|
|
NULL, 0, NULL);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int efx_mcdi_nvram_erase(struct efx_nic *efx, unsigned int type,
|
|
|
|
loff_t offset, size_t length)
|
|
|
|
{
|
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_NVRAM_ERASE_IN_LEN);
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
MCDI_SET_DWORD(inbuf, NVRAM_ERASE_IN_TYPE, type);
|
|
|
|
MCDI_SET_DWORD(inbuf, NVRAM_ERASE_IN_OFFSET, offset);
|
|
|
|
MCDI_SET_DWORD(inbuf, NVRAM_ERASE_IN_LENGTH, length);
|
|
|
|
|
|
|
|
BUILD_BUG_ON(MC_CMD_NVRAM_ERASE_OUT_LEN != 0);
|
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_NVRAM_ERASE, inbuf, sizeof(inbuf),
|
|
|
|
NULL, 0, NULL);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int efx_mcdi_nvram_update_finish(struct efx_nic *efx, unsigned int type)
|
|
|
|
{
|
2019-01-16 18:00:39 +08:00
|
|
|
MCDI_DECLARE_BUF(inbuf, MC_CMD_NVRAM_UPDATE_FINISH_V2_IN_LEN);
|
|
|
|
MCDI_DECLARE_BUF(outbuf, MC_CMD_NVRAM_UPDATE_FINISH_V2_OUT_LEN);
|
|
|
|
size_t outlen;
|
|
|
|
int rc, rc2;
|
2012-11-28 12:38:14 +08:00
|
|
|
|
|
|
|
MCDI_SET_DWORD(inbuf, NVRAM_UPDATE_FINISH_IN_TYPE, type);
|
2019-01-16 18:00:39 +08:00
|
|
|
/* Always set this flag. Old firmware ignores it */
|
|
|
|
MCDI_POPULATE_DWORD_1(inbuf, NVRAM_UPDATE_FINISH_V2_IN_FLAGS,
|
|
|
|
NVRAM_UPDATE_FINISH_V2_IN_FLAG_REPORT_VERIFY_RESULT,
|
|
|
|
1);
|
2012-11-28 12:38:14 +08:00
|
|
|
|
|
|
|
rc = efx_mcdi_rpc(efx, MC_CMD_NVRAM_UPDATE_FINISH, inbuf, sizeof(inbuf),
|
2019-01-16 18:00:39 +08:00
|
|
|
outbuf, sizeof(outbuf), &outlen);
|
|
|
|
if (!rc && outlen >= MC_CMD_NVRAM_UPDATE_FINISH_V2_OUT_LEN) {
|
|
|
|
rc2 = MCDI_DWORD(outbuf, NVRAM_UPDATE_FINISH_V2_OUT_RESULT_CODE);
|
|
|
|
if (rc2 != MC_CMD_NVRAM_VERIFY_RC_SUCCESS)
|
|
|
|
netif_err(efx, drv, efx->net_dev,
|
|
|
|
"NVRAM update failed verification with code 0x%x\n",
|
|
|
|
rc2);
|
|
|
|
switch (rc2) {
|
|
|
|
case MC_CMD_NVRAM_VERIFY_RC_SUCCESS:
|
|
|
|
break;
|
|
|
|
case MC_CMD_NVRAM_VERIFY_RC_CMS_CHECK_FAILED:
|
|
|
|
case MC_CMD_NVRAM_VERIFY_RC_MESSAGE_DIGEST_CHECK_FAILED:
|
|
|
|
case MC_CMD_NVRAM_VERIFY_RC_SIGNATURE_CHECK_FAILED:
|
|
|
|
case MC_CMD_NVRAM_VERIFY_RC_TRUSTED_APPROVERS_CHECK_FAILED:
|
|
|
|
case MC_CMD_NVRAM_VERIFY_RC_SIGNATURE_CHAIN_CHECK_FAILED:
|
|
|
|
rc = -EIO;
|
|
|
|
break;
|
|
|
|
case MC_CMD_NVRAM_VERIFY_RC_INVALID_CMS_FORMAT:
|
|
|
|
case MC_CMD_NVRAM_VERIFY_RC_BAD_MESSAGE_DIGEST:
|
|
|
|
rc = -EINVAL;
|
|
|
|
break;
|
|
|
|
case MC_CMD_NVRAM_VERIFY_RC_NO_VALID_SIGNATURES:
|
|
|
|
case MC_CMD_NVRAM_VERIFY_RC_NO_TRUSTED_APPROVERS:
|
|
|
|
case MC_CMD_NVRAM_VERIFY_RC_NO_SIGNATURE_MATCH:
|
|
|
|
rc = -EPERM;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
netif_err(efx, drv, efx->net_dev,
|
|
|
|
"Unknown response to NVRAM_UPDATE_FINISH\n");
|
|
|
|
rc = -EIO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-28 12:38:14 +08:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
int efx_mcdi_mtd_read(struct mtd_info *mtd, loff_t start,
|
|
|
|
size_t len, size_t *retlen, u8 *buffer)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_mtd_partition *part = to_efx_mcdi_mtd_partition(mtd);
|
|
|
|
struct efx_nic *efx = mtd->priv;
|
|
|
|
loff_t offset = start;
|
|
|
|
loff_t end = min_t(loff_t, start + len, mtd->size);
|
|
|
|
size_t chunk;
|
|
|
|
int rc = 0;
|
|
|
|
|
|
|
|
while (offset < end) {
|
|
|
|
chunk = min_t(size_t, end - offset, EFX_MCDI_NVRAM_LEN_MAX);
|
|
|
|
rc = efx_mcdi_nvram_read(efx, part->nvram_type, offset,
|
|
|
|
buffer, chunk);
|
|
|
|
if (rc)
|
|
|
|
goto out;
|
|
|
|
offset += chunk;
|
|
|
|
buffer += chunk;
|
|
|
|
}
|
|
|
|
out:
|
|
|
|
*retlen = offset - start;
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
int efx_mcdi_mtd_erase(struct mtd_info *mtd, loff_t start, size_t len)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_mtd_partition *part = to_efx_mcdi_mtd_partition(mtd);
|
|
|
|
struct efx_nic *efx = mtd->priv;
|
|
|
|
loff_t offset = start & ~((loff_t)(mtd->erasesize - 1));
|
|
|
|
loff_t end = min_t(loff_t, start + len, mtd->size);
|
|
|
|
size_t chunk = part->common.mtd.erasesize;
|
|
|
|
int rc = 0;
|
|
|
|
|
|
|
|
if (!part->updating) {
|
|
|
|
rc = efx_mcdi_nvram_update_start(efx, part->nvram_type);
|
|
|
|
if (rc)
|
|
|
|
goto out;
|
|
|
|
part->updating = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The MCDI interface can in fact do multiple erase blocks at once;
|
|
|
|
* but erasing may be slow, so we make multiple calls here to avoid
|
|
|
|
* tripping the MCDI RPC timeout. */
|
|
|
|
while (offset < end) {
|
|
|
|
rc = efx_mcdi_nvram_erase(efx, part->nvram_type, offset,
|
|
|
|
chunk);
|
|
|
|
if (rc)
|
|
|
|
goto out;
|
|
|
|
offset += chunk;
|
|
|
|
}
|
|
|
|
out:
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
int efx_mcdi_mtd_write(struct mtd_info *mtd, loff_t start,
|
|
|
|
size_t len, size_t *retlen, const u8 *buffer)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_mtd_partition *part = to_efx_mcdi_mtd_partition(mtd);
|
|
|
|
struct efx_nic *efx = mtd->priv;
|
|
|
|
loff_t offset = start;
|
|
|
|
loff_t end = min_t(loff_t, start + len, mtd->size);
|
|
|
|
size_t chunk;
|
|
|
|
int rc = 0;
|
|
|
|
|
|
|
|
if (!part->updating) {
|
|
|
|
rc = efx_mcdi_nvram_update_start(efx, part->nvram_type);
|
|
|
|
if (rc)
|
|
|
|
goto out;
|
|
|
|
part->updating = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (offset < end) {
|
|
|
|
chunk = min_t(size_t, end - offset, EFX_MCDI_NVRAM_LEN_MAX);
|
|
|
|
rc = efx_mcdi_nvram_write(efx, part->nvram_type, offset,
|
|
|
|
buffer, chunk);
|
|
|
|
if (rc)
|
|
|
|
goto out;
|
|
|
|
offset += chunk;
|
|
|
|
buffer += chunk;
|
|
|
|
}
|
|
|
|
out:
|
|
|
|
*retlen = offset - start;
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
int efx_mcdi_mtd_sync(struct mtd_info *mtd)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_mtd_partition *part = to_efx_mcdi_mtd_partition(mtd);
|
|
|
|
struct efx_nic *efx = mtd->priv;
|
|
|
|
int rc = 0;
|
|
|
|
|
|
|
|
if (part->updating) {
|
|
|
|
part->updating = false;
|
|
|
|
rc = efx_mcdi_nvram_update_finish(efx, part->nvram_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
void efx_mcdi_mtd_rename(struct efx_mtd_partition *part)
|
|
|
|
{
|
|
|
|
struct efx_mcdi_mtd_partition *mcdi_part =
|
|
|
|
container_of(part, struct efx_mcdi_mtd_partition, common);
|
|
|
|
struct efx_nic *efx = part->mtd.priv;
|
|
|
|
|
|
|
|
snprintf(part->name, sizeof(part->name), "%s %s:%02x",
|
|
|
|
efx->name, part->type_name, mcdi_part->fw_subtype);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* CONFIG_SFC_MTD */
|