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