xref: /openbmc/linux/drivers/net/ethernet/sfc/mcdi_functions.c (revision c900529f3d9161bfde5cca0754f83b4d3c3e0220)
137a5f9dcSAlex Maftei (amaftei) // SPDX-License-Identifier: GPL-2.0-only
237a5f9dcSAlex Maftei (amaftei) /****************************************************************************
337a5f9dcSAlex Maftei (amaftei)  * Driver for Solarflare network controllers and boards
437a5f9dcSAlex Maftei (amaftei)  * Copyright 2019 Solarflare Communications Inc.
537a5f9dcSAlex Maftei (amaftei)  *
637a5f9dcSAlex Maftei (amaftei)  * This program is free software; you can redistribute it and/or modify it
737a5f9dcSAlex Maftei (amaftei)  * under the terms of the GNU General Public License version 2 as published
837a5f9dcSAlex Maftei (amaftei)  * by the Free Software Foundation, incorporated herein by reference.
937a5f9dcSAlex Maftei (amaftei)  */
1037a5f9dcSAlex Maftei (amaftei) 
1137a5f9dcSAlex Maftei (amaftei) #include "net_driver.h"
1237a5f9dcSAlex Maftei (amaftei) #include "efx.h"
1337a5f9dcSAlex Maftei (amaftei) #include "nic.h"
1437a5f9dcSAlex Maftei (amaftei) #include "mcdi_functions.h"
1537a5f9dcSAlex Maftei (amaftei) #include "mcdi.h"
1637a5f9dcSAlex Maftei (amaftei) #include "mcdi_pcol.h"
1737a5f9dcSAlex Maftei (amaftei) 
efx_mcdi_free_vis(struct efx_nic * efx)1837a5f9dcSAlex Maftei (amaftei) int efx_mcdi_free_vis(struct efx_nic *efx)
1937a5f9dcSAlex Maftei (amaftei) {
2037a5f9dcSAlex Maftei (amaftei) 	MCDI_DECLARE_BUF_ERR(outbuf);
2137a5f9dcSAlex Maftei (amaftei) 	size_t outlen;
2237a5f9dcSAlex Maftei (amaftei) 	int rc = efx_mcdi_rpc_quiet(efx, MC_CMD_FREE_VIS, NULL, 0,
2337a5f9dcSAlex Maftei (amaftei) 				    outbuf, sizeof(outbuf), &outlen);
2437a5f9dcSAlex Maftei (amaftei) 
2537a5f9dcSAlex Maftei (amaftei) 	/* -EALREADY means nothing to free, so ignore */
2637a5f9dcSAlex Maftei (amaftei) 	if (rc == -EALREADY)
2737a5f9dcSAlex Maftei (amaftei) 		rc = 0;
2837a5f9dcSAlex Maftei (amaftei) 	if (rc)
2937a5f9dcSAlex Maftei (amaftei) 		efx_mcdi_display_error(efx, MC_CMD_FREE_VIS, 0, outbuf, outlen,
3037a5f9dcSAlex Maftei (amaftei) 				       rc);
3137a5f9dcSAlex Maftei (amaftei) 	return rc;
3237a5f9dcSAlex Maftei (amaftei) }
3337a5f9dcSAlex Maftei (amaftei) 
efx_mcdi_alloc_vis(struct efx_nic * efx,unsigned int min_vis,unsigned int max_vis,unsigned int * vi_base,unsigned int * allocated_vis)3437a5f9dcSAlex Maftei (amaftei) int efx_mcdi_alloc_vis(struct efx_nic *efx, unsigned int min_vis,
3537a5f9dcSAlex Maftei (amaftei) 		       unsigned int max_vis, unsigned int *vi_base,
3637a5f9dcSAlex Maftei (amaftei) 		       unsigned int *allocated_vis)
3737a5f9dcSAlex Maftei (amaftei) {
3837a5f9dcSAlex Maftei (amaftei) 	MCDI_DECLARE_BUF(outbuf, MC_CMD_ALLOC_VIS_OUT_LEN);
3937a5f9dcSAlex Maftei (amaftei) 	MCDI_DECLARE_BUF(inbuf, MC_CMD_ALLOC_VIS_IN_LEN);
4037a5f9dcSAlex Maftei (amaftei) 	size_t outlen;
4137a5f9dcSAlex Maftei (amaftei) 	int rc;
4237a5f9dcSAlex Maftei (amaftei) 
4337a5f9dcSAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, ALLOC_VIS_IN_MIN_VI_COUNT, min_vis);
4437a5f9dcSAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, ALLOC_VIS_IN_MAX_VI_COUNT, max_vis);
4537a5f9dcSAlex Maftei (amaftei) 	rc = efx_mcdi_rpc(efx, MC_CMD_ALLOC_VIS, inbuf, sizeof(inbuf),
4637a5f9dcSAlex Maftei (amaftei) 			  outbuf, sizeof(outbuf), &outlen);
4737a5f9dcSAlex Maftei (amaftei) 	if (rc != 0)
4837a5f9dcSAlex Maftei (amaftei) 		return rc;
4937a5f9dcSAlex Maftei (amaftei) 
5037a5f9dcSAlex Maftei (amaftei) 	if (outlen < MC_CMD_ALLOC_VIS_OUT_LEN)
5137a5f9dcSAlex Maftei (amaftei) 		return -EIO;
5237a5f9dcSAlex Maftei (amaftei) 
5337a5f9dcSAlex Maftei (amaftei) 	netif_dbg(efx, drv, efx->net_dev, "base VI is A0x%03x\n",
5437a5f9dcSAlex Maftei (amaftei) 		  MCDI_DWORD(outbuf, ALLOC_VIS_OUT_VI_BASE));
5537a5f9dcSAlex Maftei (amaftei) 
5637a5f9dcSAlex Maftei (amaftei) 	if (vi_base)
5737a5f9dcSAlex Maftei (amaftei) 		*vi_base = MCDI_DWORD(outbuf, ALLOC_VIS_OUT_VI_BASE);
5837a5f9dcSAlex Maftei (amaftei) 	if (allocated_vis)
5937a5f9dcSAlex Maftei (amaftei) 		*allocated_vis = MCDI_DWORD(outbuf, ALLOC_VIS_OUT_VI_COUNT);
6037a5f9dcSAlex Maftei (amaftei) 	return 0;
6137a5f9dcSAlex Maftei (amaftei) }
624438b587SAlex Maftei (amaftei) 
efx_mcdi_ev_probe(struct efx_channel * channel)634438b587SAlex Maftei (amaftei) int efx_mcdi_ev_probe(struct efx_channel *channel)
644438b587SAlex Maftei (amaftei) {
65*d73e7715SMartin Habets 	return efx_nic_alloc_buffer(channel->efx, &channel->eventq,
664438b587SAlex Maftei (amaftei) 				    (channel->eventq_mask + 1) *
674438b587SAlex Maftei (amaftei) 				    sizeof(efx_qword_t),
684438b587SAlex Maftei (amaftei) 				    GFP_KERNEL);
694438b587SAlex Maftei (amaftei) }
704438b587SAlex Maftei (amaftei) 
efx_mcdi_ev_init(struct efx_channel * channel,bool v1_cut_thru,bool v2)714438b587SAlex Maftei (amaftei) int efx_mcdi_ev_init(struct efx_channel *channel, bool v1_cut_thru, bool v2)
724438b587SAlex Maftei (amaftei) {
734438b587SAlex Maftei (amaftei) 	MCDI_DECLARE_BUF(inbuf,
744438b587SAlex Maftei (amaftei) 			 MC_CMD_INIT_EVQ_V2_IN_LEN(EFX_MAX_EVQ_SIZE * 8 /
754438b587SAlex Maftei (amaftei) 						   EFX_BUF_SIZE));
764438b587SAlex Maftei (amaftei) 	MCDI_DECLARE_BUF(outbuf, MC_CMD_INIT_EVQ_V2_OUT_LEN);
77*d73e7715SMartin Habets 	size_t entries = channel->eventq.len / EFX_BUF_SIZE;
784438b587SAlex Maftei (amaftei) 	struct efx_nic *efx = channel->efx;
794438b587SAlex Maftei (amaftei) 	size_t inlen, outlen;
804438b587SAlex Maftei (amaftei) 	dma_addr_t dma_addr;
814438b587SAlex Maftei (amaftei) 	int rc, i;
824438b587SAlex Maftei (amaftei) 
834438b587SAlex Maftei (amaftei) 	/* Fill event queue with all ones (i.e. empty events) */
84*d73e7715SMartin Habets 	memset(channel->eventq.addr, 0xff, channel->eventq.len);
854438b587SAlex Maftei (amaftei) 
864438b587SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_SIZE, channel->eventq_mask + 1);
874438b587SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_INSTANCE, channel->channel);
884438b587SAlex Maftei (amaftei) 	/* INIT_EVQ expects index in vector table, not absolute */
894438b587SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_IRQ_NUM, channel->channel);
904438b587SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_TMR_MODE,
914438b587SAlex Maftei (amaftei) 		       MC_CMD_INIT_EVQ_IN_TMR_MODE_DIS);
924438b587SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_TMR_LOAD, 0);
934438b587SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_TMR_RELOAD, 0);
944438b587SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_COUNT_MODE,
954438b587SAlex Maftei (amaftei) 		       MC_CMD_INIT_EVQ_IN_COUNT_MODE_DIS);
964438b587SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_COUNT_THRSHLD, 0);
974438b587SAlex Maftei (amaftei) 
984438b587SAlex Maftei (amaftei) 	if (v2) {
994438b587SAlex Maftei (amaftei) 		/* Use the new generic approach to specifying event queue
1004438b587SAlex Maftei (amaftei) 		 * configuration, requesting lower latency or higher throughput.
1014438b587SAlex Maftei (amaftei) 		 * The options that actually get used appear in the output.
1024438b587SAlex Maftei (amaftei) 		 */
1034438b587SAlex Maftei (amaftei) 		MCDI_POPULATE_DWORD_2(inbuf, INIT_EVQ_V2_IN_FLAGS,
1044438b587SAlex Maftei (amaftei) 				      INIT_EVQ_V2_IN_FLAG_INTERRUPTING, 1,
1054438b587SAlex Maftei (amaftei) 				      INIT_EVQ_V2_IN_FLAG_TYPE,
1064438b587SAlex Maftei (amaftei) 				      MC_CMD_INIT_EVQ_V2_IN_FLAG_TYPE_AUTO);
1074438b587SAlex Maftei (amaftei) 	} else {
1084438b587SAlex Maftei (amaftei) 		MCDI_POPULATE_DWORD_4(inbuf, INIT_EVQ_IN_FLAGS,
1094438b587SAlex Maftei (amaftei) 				      INIT_EVQ_IN_FLAG_INTERRUPTING, 1,
1104438b587SAlex Maftei (amaftei) 				      INIT_EVQ_IN_FLAG_RX_MERGE, 1,
1114438b587SAlex Maftei (amaftei) 				      INIT_EVQ_IN_FLAG_TX_MERGE, 1,
1124438b587SAlex Maftei (amaftei) 				      INIT_EVQ_IN_FLAG_CUT_THRU, v1_cut_thru);
1134438b587SAlex Maftei (amaftei) 	}
1144438b587SAlex Maftei (amaftei) 
115*d73e7715SMartin Habets 	dma_addr = channel->eventq.dma_addr;
1164438b587SAlex Maftei (amaftei) 	for (i = 0; i < entries; ++i) {
1174438b587SAlex Maftei (amaftei) 		MCDI_SET_ARRAY_QWORD(inbuf, INIT_EVQ_IN_DMA_ADDR, i, dma_addr);
1184438b587SAlex Maftei (amaftei) 		dma_addr += EFX_BUF_SIZE;
1194438b587SAlex Maftei (amaftei) 	}
1204438b587SAlex Maftei (amaftei) 
1214438b587SAlex Maftei (amaftei) 	inlen = MC_CMD_INIT_EVQ_IN_LEN(entries);
1224438b587SAlex Maftei (amaftei) 
1234438b587SAlex Maftei (amaftei) 	rc = efx_mcdi_rpc(efx, MC_CMD_INIT_EVQ, inbuf, inlen,
1244438b587SAlex Maftei (amaftei) 			  outbuf, sizeof(outbuf), &outlen);
1254438b587SAlex Maftei (amaftei) 
1264438b587SAlex Maftei (amaftei) 	if (outlen >= MC_CMD_INIT_EVQ_V2_OUT_LEN)
1274438b587SAlex Maftei (amaftei) 		netif_dbg(efx, drv, efx->net_dev,
1284438b587SAlex Maftei (amaftei) 			  "Channel %d using event queue flags %08x\n",
1294438b587SAlex Maftei (amaftei) 			  channel->channel,
1304438b587SAlex Maftei (amaftei) 			  MCDI_DWORD(outbuf, INIT_EVQ_V2_OUT_FLAGS));
1314438b587SAlex Maftei (amaftei) 
1324438b587SAlex Maftei (amaftei) 	return rc;
1334438b587SAlex Maftei (amaftei) }
1344438b587SAlex Maftei (amaftei) 
efx_mcdi_ev_remove(struct efx_channel * channel)1354438b587SAlex Maftei (amaftei) void efx_mcdi_ev_remove(struct efx_channel *channel)
1364438b587SAlex Maftei (amaftei) {
137*d73e7715SMartin Habets 	efx_nic_free_buffer(channel->efx, &channel->eventq);
1384438b587SAlex Maftei (amaftei) }
1394438b587SAlex Maftei (amaftei) 
efx_mcdi_ev_fini(struct efx_channel * channel)1404438b587SAlex Maftei (amaftei) void efx_mcdi_ev_fini(struct efx_channel *channel)
1414438b587SAlex Maftei (amaftei) {
1424438b587SAlex Maftei (amaftei) 	MCDI_DECLARE_BUF(inbuf, MC_CMD_FINI_EVQ_IN_LEN);
1434438b587SAlex Maftei (amaftei) 	MCDI_DECLARE_BUF_ERR(outbuf);
1444438b587SAlex Maftei (amaftei) 	struct efx_nic *efx = channel->efx;
1454438b587SAlex Maftei (amaftei) 	size_t outlen;
1464438b587SAlex Maftei (amaftei) 	int rc;
1474438b587SAlex Maftei (amaftei) 
1484438b587SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, FINI_EVQ_IN_INSTANCE, channel->channel);
1494438b587SAlex Maftei (amaftei) 
1504438b587SAlex Maftei (amaftei) 	rc = efx_mcdi_rpc_quiet(efx, MC_CMD_FINI_EVQ, inbuf, sizeof(inbuf),
1514438b587SAlex Maftei (amaftei) 				outbuf, sizeof(outbuf), &outlen);
1524438b587SAlex Maftei (amaftei) 
1534438b587SAlex Maftei (amaftei) 	if (rc && rc != -EALREADY)
1544438b587SAlex Maftei (amaftei) 		goto fail;
1554438b587SAlex Maftei (amaftei) 
1564438b587SAlex Maftei (amaftei) 	return;
1574438b587SAlex Maftei (amaftei) 
1584438b587SAlex Maftei (amaftei) fail:
1594438b587SAlex Maftei (amaftei) 	efx_mcdi_display_error(efx, MC_CMD_FINI_EVQ, MC_CMD_FINI_EVQ_IN_LEN,
1604438b587SAlex Maftei (amaftei) 			       outbuf, outlen, rc);
1614438b587SAlex Maftei (amaftei) }
1628ee4c907SAlex Maftei (amaftei) 
efx_mcdi_tx_init(struct efx_tx_queue * tx_queue)1631679c72cSEdward Cree int efx_mcdi_tx_init(struct efx_tx_queue *tx_queue)
1648ee4c907SAlex Maftei (amaftei) {
1658ee4c907SAlex Maftei (amaftei) 	MCDI_DECLARE_BUF(inbuf, MC_CMD_INIT_TXQ_IN_LEN(EFX_MAX_DMAQ_SIZE * 8 /
1668ee4c907SAlex Maftei (amaftei) 						       EFX_BUF_SIZE));
167044588b9SEdward Cree 	bool csum_offload = tx_queue->type & EFX_TXQ_TYPE_OUTER_CSUM;
16885d43fdbSEdward Cree 	bool inner_csum = tx_queue->type & EFX_TXQ_TYPE_INNER_CSUM;
169*d73e7715SMartin Habets 	size_t entries = tx_queue->txd.len / EFX_BUF_SIZE;
1708ee4c907SAlex Maftei (amaftei) 	struct efx_channel *channel = tx_queue->channel;
1718ee4c907SAlex Maftei (amaftei) 	struct efx_nic *efx = tx_queue->efx;
1728ee4c907SAlex Maftei (amaftei) 	dma_addr_t dma_addr;
1738ee4c907SAlex Maftei (amaftei) 	size_t inlen;
1748ee4c907SAlex Maftei (amaftei) 	int rc, i;
1758ee4c907SAlex Maftei (amaftei) 
1768ee4c907SAlex Maftei (amaftei) 	BUILD_BUG_ON(MC_CMD_INIT_TXQ_OUT_LEN != 0);
1778ee4c907SAlex Maftei (amaftei) 
1788ee4c907SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_TXQ_IN_SIZE, tx_queue->ptr_mask + 1);
1798ee4c907SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_TXQ_IN_TARGET_EVQ, channel->channel);
180a81dcd85SEdward Cree 	MCDI_SET_DWORD(inbuf, INIT_TXQ_IN_LABEL, tx_queue->label);
1818ee4c907SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_TXQ_IN_INSTANCE, tx_queue->queue);
1828ee4c907SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_TXQ_IN_OWNER_ID, 0);
183dfcabb07SEdward Cree 	MCDI_SET_DWORD(inbuf, INIT_TXQ_IN_PORT_ID, efx->vport_id);
1848ee4c907SAlex Maftei (amaftei) 
185*d73e7715SMartin Habets 	dma_addr = tx_queue->txd.dma_addr;
1868ee4c907SAlex Maftei (amaftei) 
1878ee4c907SAlex Maftei (amaftei) 	netif_dbg(efx, hw, efx->net_dev, "pushing TXQ %d. %zu entries (%llx)\n",
1888ee4c907SAlex Maftei (amaftei) 		  tx_queue->queue, entries, (u64)dma_addr);
1898ee4c907SAlex Maftei (amaftei) 
1908ee4c907SAlex Maftei (amaftei) 	for (i = 0; i < entries; ++i) {
1918ee4c907SAlex Maftei (amaftei) 		MCDI_SET_ARRAY_QWORD(inbuf, INIT_TXQ_IN_DMA_ADDR, i, dma_addr);
1928ee4c907SAlex Maftei (amaftei) 		dma_addr += EFX_BUF_SIZE;
1938ee4c907SAlex Maftei (amaftei) 	}
1948ee4c907SAlex Maftei (amaftei) 
1958ee4c907SAlex Maftei (amaftei) 	inlen = MC_CMD_INIT_TXQ_IN_LEN(entries);
1968ee4c907SAlex Maftei (amaftei) 
1978ee4c907SAlex Maftei (amaftei) 	do {
1981679c72cSEdward Cree 		bool tso_v2 = tx_queue->tso_version == 2;
1991679c72cSEdward Cree 
20085d43fdbSEdward Cree 		/* TSOv2 implies IP header checksum offload for TSO frames,
20185d43fdbSEdward Cree 		 * so we can safely disable IP header checksum offload for
20285d43fdbSEdward Cree 		 * everything else.  If we don't have TSOv2, then we have to
20385d43fdbSEdward Cree 		 * enable IP header checksum offload, which is strictly
20485d43fdbSEdward Cree 		 * incorrect but better than breaking TSO.
20585d43fdbSEdward Cree 		 */
20685d43fdbSEdward Cree 		MCDI_POPULATE_DWORD_6(inbuf, INIT_TXQ_IN_FLAGS,
2078ee4c907SAlex Maftei (amaftei) 				/* This flag was removed from mcdi_pcol.h for
2088ee4c907SAlex Maftei (amaftei) 				 * the non-_EXT version of INIT_TXQ.  However,
2098ee4c907SAlex Maftei (amaftei) 				 * firmware still honours it.
2108ee4c907SAlex Maftei (amaftei) 				 */
2118ee4c907SAlex Maftei (amaftei) 				INIT_TXQ_EXT_IN_FLAG_TSOV2_EN, tso_v2,
21285d43fdbSEdward Cree 				INIT_TXQ_IN_FLAG_IP_CSUM_DIS, !(csum_offload && tso_v2),
2138ee4c907SAlex Maftei (amaftei) 				INIT_TXQ_IN_FLAG_TCP_CSUM_DIS, !csum_offload,
21485d43fdbSEdward Cree 				INIT_TXQ_EXT_IN_FLAG_TIMESTAMP, tx_queue->timestamping,
21585d43fdbSEdward Cree 				INIT_TXQ_IN_FLAG_INNER_IP_CSUM_EN, inner_csum && !tso_v2,
21685d43fdbSEdward Cree 				INIT_TXQ_IN_FLAG_INNER_TCP_CSUM_EN, inner_csum);
2178ee4c907SAlex Maftei (amaftei) 
2188ee4c907SAlex Maftei (amaftei) 		rc = efx_mcdi_rpc_quiet(efx, MC_CMD_INIT_TXQ, inbuf, inlen,
2198ee4c907SAlex Maftei (amaftei) 					NULL, 0, NULL);
2208ee4c907SAlex Maftei (amaftei) 		if (rc == -ENOSPC && tso_v2) {
2218ee4c907SAlex Maftei (amaftei) 			/* Retry without TSOv2 if we're short on contexts. */
2221679c72cSEdward Cree 			tx_queue->tso_version = 0;
2238ee4c907SAlex Maftei (amaftei) 			netif_warn(efx, probe, efx->net_dev,
2248ee4c907SAlex Maftei (amaftei) 				   "TSOv2 context not available to segment in "
2258ee4c907SAlex Maftei (amaftei) 				   "hardware. TCP performance may be reduced.\n"
2268ee4c907SAlex Maftei (amaftei) 				   );
2278ee4c907SAlex Maftei (amaftei) 		} else if (rc) {
2288ee4c907SAlex Maftei (amaftei) 			efx_mcdi_display_error(efx, MC_CMD_INIT_TXQ,
2298ee4c907SAlex Maftei (amaftei) 					       MC_CMD_INIT_TXQ_EXT_IN_LEN,
2308ee4c907SAlex Maftei (amaftei) 					       NULL, 0, rc);
2318ee4c907SAlex Maftei (amaftei) 			goto fail;
2328ee4c907SAlex Maftei (amaftei) 		}
2338ee4c907SAlex Maftei (amaftei) 	} while (rc);
2348ee4c907SAlex Maftei (amaftei) 
2358ee4c907SAlex Maftei (amaftei) 	return 0;
2368ee4c907SAlex Maftei (amaftei) 
2378ee4c907SAlex Maftei (amaftei) fail:
2388ee4c907SAlex Maftei (amaftei) 	return rc;
2398ee4c907SAlex Maftei (amaftei) }
2408ee4c907SAlex Maftei (amaftei) 
efx_mcdi_tx_remove(struct efx_tx_queue * tx_queue)2418ee4c907SAlex Maftei (amaftei) void efx_mcdi_tx_remove(struct efx_tx_queue *tx_queue)
2428ee4c907SAlex Maftei (amaftei) {
243*d73e7715SMartin Habets 	efx_nic_free_buffer(tx_queue->efx, &tx_queue->txd);
2448ee4c907SAlex Maftei (amaftei) }
2458ee4c907SAlex Maftei (amaftei) 
efx_mcdi_tx_fini(struct efx_tx_queue * tx_queue)2468ee4c907SAlex Maftei (amaftei) void efx_mcdi_tx_fini(struct efx_tx_queue *tx_queue)
2478ee4c907SAlex Maftei (amaftei) {
2488ee4c907SAlex Maftei (amaftei) 	MCDI_DECLARE_BUF(inbuf, MC_CMD_FINI_TXQ_IN_LEN);
2498ee4c907SAlex Maftei (amaftei) 	MCDI_DECLARE_BUF_ERR(outbuf);
2508ee4c907SAlex Maftei (amaftei) 	struct efx_nic *efx = tx_queue->efx;
2518ee4c907SAlex Maftei (amaftei) 	size_t outlen;
2528ee4c907SAlex Maftei (amaftei) 	int rc;
2538ee4c907SAlex Maftei (amaftei) 
2548ee4c907SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, FINI_TXQ_IN_INSTANCE,
2558ee4c907SAlex Maftei (amaftei) 		       tx_queue->queue);
2568ee4c907SAlex Maftei (amaftei) 
2578ee4c907SAlex Maftei (amaftei) 	rc = efx_mcdi_rpc_quiet(efx, MC_CMD_FINI_TXQ, inbuf, sizeof(inbuf),
2588ee4c907SAlex Maftei (amaftei) 				outbuf, sizeof(outbuf), &outlen);
2598ee4c907SAlex Maftei (amaftei) 
2608ee4c907SAlex Maftei (amaftei) 	if (rc && rc != -EALREADY)
2618ee4c907SAlex Maftei (amaftei) 		goto fail;
2628ee4c907SAlex Maftei (amaftei) 
2638ee4c907SAlex Maftei (amaftei) 	return;
2648ee4c907SAlex Maftei (amaftei) 
2658ee4c907SAlex Maftei (amaftei) fail:
2668ee4c907SAlex Maftei (amaftei) 	efx_mcdi_display_error(efx, MC_CMD_FINI_TXQ, MC_CMD_FINI_TXQ_IN_LEN,
2678ee4c907SAlex Maftei (amaftei) 			       outbuf, outlen, rc);
2688ee4c907SAlex Maftei (amaftei) }
2698da92642SAlex Maftei (amaftei) 
efx_mcdi_rx_probe(struct efx_rx_queue * rx_queue)2708da92642SAlex Maftei (amaftei) int efx_mcdi_rx_probe(struct efx_rx_queue *rx_queue)
2718da92642SAlex Maftei (amaftei) {
272*d73e7715SMartin Habets 	return efx_nic_alloc_buffer(rx_queue->efx, &rx_queue->rxd,
2738da92642SAlex Maftei (amaftei) 				    (rx_queue->ptr_mask + 1) *
2748da92642SAlex Maftei (amaftei) 				    sizeof(efx_qword_t),
2758da92642SAlex Maftei (amaftei) 				    GFP_KERNEL);
2768da92642SAlex Maftei (amaftei) }
2778da92642SAlex Maftei (amaftei) 
efx_mcdi_rx_init(struct efx_rx_queue * rx_queue)2788da92642SAlex Maftei (amaftei) void efx_mcdi_rx_init(struct efx_rx_queue *rx_queue)
2798da92642SAlex Maftei (amaftei) {
2808da92642SAlex Maftei (amaftei) 	struct efx_channel *channel = efx_rx_queue_channel(rx_queue);
281*d73e7715SMartin Habets 	size_t entries = rx_queue->rxd.len / EFX_BUF_SIZE;
28239c965f4SEdward Cree 	MCDI_DECLARE_BUF(inbuf, MC_CMD_INIT_RXQ_V4_IN_LEN);
2838da92642SAlex Maftei (amaftei) 	struct efx_nic *efx = rx_queue->efx;
28439c965f4SEdward Cree 	unsigned int buffer_size;
2858da92642SAlex Maftei (amaftei) 	dma_addr_t dma_addr;
2868da92642SAlex Maftei (amaftei) 	int rc;
2878da92642SAlex Maftei (amaftei) 	int i;
2888da92642SAlex Maftei (amaftei) 	BUILD_BUG_ON(MC_CMD_INIT_RXQ_OUT_LEN != 0);
2898da92642SAlex Maftei (amaftei) 
2908da92642SAlex Maftei (amaftei) 	rx_queue->scatter_n = 0;
2918da92642SAlex Maftei (amaftei) 	rx_queue->scatter_len = 0;
29239c965f4SEdward Cree 	if (efx->type->revision == EFX_REV_EF100)
29339c965f4SEdward Cree 		buffer_size = efx->rx_page_buf_step;
29439c965f4SEdward Cree 	else
29539c965f4SEdward Cree 		buffer_size = 0;
2968da92642SAlex Maftei (amaftei) 
2978da92642SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_RXQ_IN_SIZE, rx_queue->ptr_mask + 1);
2988da92642SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_RXQ_IN_TARGET_EVQ, channel->channel);
2998da92642SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_RXQ_IN_LABEL, efx_rx_queue_index(rx_queue));
3008da92642SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_RXQ_IN_INSTANCE,
3018da92642SAlex Maftei (amaftei) 		       efx_rx_queue_index(rx_queue));
3028da92642SAlex Maftei (amaftei) 	MCDI_POPULATE_DWORD_2(inbuf, INIT_RXQ_IN_FLAGS,
3038da92642SAlex Maftei (amaftei) 			      INIT_RXQ_IN_FLAG_PREFIX, 1,
3048da92642SAlex Maftei (amaftei) 			      INIT_RXQ_IN_FLAG_TIMESTAMP, 1);
3058da92642SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, INIT_RXQ_IN_OWNER_ID, 0);
306dfcabb07SEdward Cree 	MCDI_SET_DWORD(inbuf, INIT_RXQ_IN_PORT_ID, efx->vport_id);
30739c965f4SEdward Cree 	MCDI_SET_DWORD(inbuf, INIT_RXQ_V4_IN_BUFFER_SIZE_BYTES, buffer_size);
3088da92642SAlex Maftei (amaftei) 
309*d73e7715SMartin Habets 	dma_addr = rx_queue->rxd.dma_addr;
3108da92642SAlex Maftei (amaftei) 
3118da92642SAlex Maftei (amaftei) 	netif_dbg(efx, hw, efx->net_dev, "pushing RXQ %d. %zu entries (%llx)\n",
3128da92642SAlex Maftei (amaftei) 		  efx_rx_queue_index(rx_queue), entries, (u64)dma_addr);
3138da92642SAlex Maftei (amaftei) 
3148da92642SAlex Maftei (amaftei) 	for (i = 0; i < entries; ++i) {
3158da92642SAlex Maftei (amaftei) 		MCDI_SET_ARRAY_QWORD(inbuf, INIT_RXQ_IN_DMA_ADDR, i, dma_addr);
3168da92642SAlex Maftei (amaftei) 		dma_addr += EFX_BUF_SIZE;
3178da92642SAlex Maftei (amaftei) 	}
3188da92642SAlex Maftei (amaftei) 
31939c965f4SEdward Cree 	rc = efx_mcdi_rpc(efx, MC_CMD_INIT_RXQ, inbuf, sizeof(inbuf),
3208da92642SAlex Maftei (amaftei) 			  NULL, 0, NULL);
3218da92642SAlex Maftei (amaftei) 	if (rc)
3228da92642SAlex Maftei (amaftei) 		netdev_WARN(efx->net_dev, "failed to initialise RXQ %d\n",
3238da92642SAlex Maftei (amaftei) 			    efx_rx_queue_index(rx_queue));
3248da92642SAlex Maftei (amaftei) }
3258da92642SAlex Maftei (amaftei) 
efx_mcdi_rx_remove(struct efx_rx_queue * rx_queue)3268da92642SAlex Maftei (amaftei) void efx_mcdi_rx_remove(struct efx_rx_queue *rx_queue)
3278da92642SAlex Maftei (amaftei) {
328*d73e7715SMartin Habets 	efx_nic_free_buffer(rx_queue->efx, &rx_queue->rxd);
3298da92642SAlex Maftei (amaftei) }
3308da92642SAlex Maftei (amaftei) 
efx_mcdi_rx_fini(struct efx_rx_queue * rx_queue)3318da92642SAlex Maftei (amaftei) void efx_mcdi_rx_fini(struct efx_rx_queue *rx_queue)
3328da92642SAlex Maftei (amaftei) {
3338da92642SAlex Maftei (amaftei) 	MCDI_DECLARE_BUF(inbuf, MC_CMD_FINI_RXQ_IN_LEN);
3348da92642SAlex Maftei (amaftei) 	MCDI_DECLARE_BUF_ERR(outbuf);
3358da92642SAlex Maftei (amaftei) 	struct efx_nic *efx = rx_queue->efx;
3368da92642SAlex Maftei (amaftei) 	size_t outlen;
3378da92642SAlex Maftei (amaftei) 	int rc;
3388da92642SAlex Maftei (amaftei) 
3398da92642SAlex Maftei (amaftei) 	MCDI_SET_DWORD(inbuf, FINI_RXQ_IN_INSTANCE,
3408da92642SAlex Maftei (amaftei) 		       efx_rx_queue_index(rx_queue));
3418da92642SAlex Maftei (amaftei) 
3428da92642SAlex Maftei (amaftei) 	rc = efx_mcdi_rpc_quiet(efx, MC_CMD_FINI_RXQ, inbuf, sizeof(inbuf),
3438da92642SAlex Maftei (amaftei) 				outbuf, sizeof(outbuf), &outlen);
3448da92642SAlex Maftei (amaftei) 
3458da92642SAlex Maftei (amaftei) 	if (rc && rc != -EALREADY)
3468da92642SAlex Maftei (amaftei) 		goto fail;
3478da92642SAlex Maftei (amaftei) 
3488da92642SAlex Maftei (amaftei) 	return;
3498da92642SAlex Maftei (amaftei) 
3508da92642SAlex Maftei (amaftei) fail:
3518da92642SAlex Maftei (amaftei) 	efx_mcdi_display_error(efx, MC_CMD_FINI_RXQ, MC_CMD_FINI_RXQ_IN_LEN,
3528da92642SAlex Maftei (amaftei) 			       outbuf, outlen, rc);
3538da92642SAlex Maftei (amaftei) }
354190c736aSAlex Maftei (amaftei) 
efx_fini_dmaq(struct efx_nic * efx)355d700fe01SEdward Cree int efx_fini_dmaq(struct efx_nic *efx)
356d700fe01SEdward Cree {
357d700fe01SEdward Cree 	struct efx_tx_queue *tx_queue;
358d700fe01SEdward Cree 	struct efx_rx_queue *rx_queue;
359d700fe01SEdward Cree 	struct efx_channel *channel;
360d700fe01SEdward Cree 	int pending;
361d700fe01SEdward Cree 
362d700fe01SEdward Cree 	/* If the MC has just rebooted, the TX/RX queues will have already been
363d700fe01SEdward Cree 	 * torn down, but efx->active_queues needs to be set to zero.
364d700fe01SEdward Cree 	 */
365d700fe01SEdward Cree 	if (efx->must_realloc_vis) {
366d700fe01SEdward Cree 		atomic_set(&efx->active_queues, 0);
367d700fe01SEdward Cree 		return 0;
368d700fe01SEdward Cree 	}
369d700fe01SEdward Cree 
370d700fe01SEdward Cree 	/* Do not attempt to write to the NIC during EEH recovery */
371d700fe01SEdward Cree 	if (efx->state != STATE_RECOVERY) {
372d700fe01SEdward Cree 		efx_for_each_channel(channel, efx) {
373d700fe01SEdward Cree 			efx_for_each_channel_rx_queue(rx_queue, channel)
374d700fe01SEdward Cree 				efx_mcdi_rx_fini(rx_queue);
375d700fe01SEdward Cree 			efx_for_each_channel_tx_queue(tx_queue, channel)
376d700fe01SEdward Cree 				efx_mcdi_tx_fini(tx_queue);
377d700fe01SEdward Cree 		}
378d700fe01SEdward Cree 
379d700fe01SEdward Cree 		wait_event_timeout(efx->flush_wq,
380d700fe01SEdward Cree 				   atomic_read(&efx->active_queues) == 0,
381d700fe01SEdward Cree 				   msecs_to_jiffies(EFX_MAX_FLUSH_TIME));
382d700fe01SEdward Cree 		pending = atomic_read(&efx->active_queues);
383d700fe01SEdward Cree 		if (pending) {
384d700fe01SEdward Cree 			netif_err(efx, hw, efx->net_dev, "failed to flush %d queues\n",
385d700fe01SEdward Cree 				  pending);
386d700fe01SEdward Cree 			return -ETIMEDOUT;
387d700fe01SEdward Cree 		}
388d700fe01SEdward Cree 	}
389d700fe01SEdward Cree 
390d700fe01SEdward Cree 	return 0;
391d700fe01SEdward Cree }
392d700fe01SEdward Cree 
efx_mcdi_window_mode_to_stride(struct efx_nic * efx,u8 vi_window_mode)393190c736aSAlex Maftei (amaftei) int efx_mcdi_window_mode_to_stride(struct efx_nic *efx, u8 vi_window_mode)
394190c736aSAlex Maftei (amaftei) {
395190c736aSAlex Maftei (amaftei) 	switch (vi_window_mode) {
396190c736aSAlex Maftei (amaftei) 	case MC_CMD_GET_CAPABILITIES_V3_OUT_VI_WINDOW_MODE_8K:
397190c736aSAlex Maftei (amaftei) 		efx->vi_stride = 8192;
398190c736aSAlex Maftei (amaftei) 		break;
399190c736aSAlex Maftei (amaftei) 	case MC_CMD_GET_CAPABILITIES_V3_OUT_VI_WINDOW_MODE_16K:
400190c736aSAlex Maftei (amaftei) 		efx->vi_stride = 16384;
401190c736aSAlex Maftei (amaftei) 		break;
402190c736aSAlex Maftei (amaftei) 	case MC_CMD_GET_CAPABILITIES_V3_OUT_VI_WINDOW_MODE_64K:
403190c736aSAlex Maftei (amaftei) 		efx->vi_stride = 65536;
404190c736aSAlex Maftei (amaftei) 		break;
405190c736aSAlex Maftei (amaftei) 	default:
406190c736aSAlex Maftei (amaftei) 		netif_err(efx, probe, efx->net_dev,
407190c736aSAlex Maftei (amaftei) 			  "Unrecognised VI window mode %d\n",
408190c736aSAlex Maftei (amaftei) 			  vi_window_mode);
409190c736aSAlex Maftei (amaftei) 		return -EIO;
410190c736aSAlex Maftei (amaftei) 	}
411190c736aSAlex Maftei (amaftei) 	netif_dbg(efx, probe, efx->net_dev, "vi_stride = %u\n",
412190c736aSAlex Maftei (amaftei) 		  efx->vi_stride);
413190c736aSAlex Maftei (amaftei) 	return 0;
414190c736aSAlex Maftei (amaftei) }
41531482310SAlex Maftei (amaftei) 
efx_get_pf_index(struct efx_nic * efx,unsigned int * pf_index)41631482310SAlex Maftei (amaftei) int efx_get_pf_index(struct efx_nic *efx, unsigned int *pf_index)
41731482310SAlex Maftei (amaftei) {
41831482310SAlex Maftei (amaftei) 	MCDI_DECLARE_BUF(outbuf, MC_CMD_GET_FUNCTION_INFO_OUT_LEN);
41931482310SAlex Maftei (amaftei) 	size_t outlen;
42031482310SAlex Maftei (amaftei) 	int rc;
42131482310SAlex Maftei (amaftei) 
42231482310SAlex Maftei (amaftei) 	rc = efx_mcdi_rpc(efx, MC_CMD_GET_FUNCTION_INFO, NULL, 0, outbuf,
42331482310SAlex Maftei (amaftei) 			  sizeof(outbuf), &outlen);
42431482310SAlex Maftei (amaftei) 	if (rc)
42531482310SAlex Maftei (amaftei) 		return rc;
42631482310SAlex Maftei (amaftei) 	if (outlen < sizeof(outbuf))
42731482310SAlex Maftei (amaftei) 		return -EIO;
42831482310SAlex Maftei (amaftei) 
42931482310SAlex Maftei (amaftei) 	*pf_index = MCDI_DWORD(outbuf, GET_FUNCTION_INFO_OUT_PF);
43031482310SAlex Maftei (amaftei) 	return 0;
43131482310SAlex Maftei (amaftei) }
432