| .. | .. |
|---|
| 1 | +// SPDX-License-Identifier: GPL-2.0-only |
|---|
| 1 | 2 | /**************************************************************************** |
|---|
| 2 | 3 | * Driver for Solarflare network controllers and boards |
|---|
| 3 | 4 | * Copyright 2012-2013 Solarflare Communications Inc. |
|---|
| 4 | | - * |
|---|
| 5 | | - * This program is free software; you can redistribute it and/or modify it |
|---|
| 6 | | - * under the terms of the GNU General Public License version 2 as published |
|---|
| 7 | | - * by the Free Software Foundation, incorporated herein by reference. |
|---|
| 8 | 5 | */ |
|---|
| 9 | 6 | |
|---|
| 10 | 7 | #include "net_driver.h" |
|---|
| 8 | +#include "rx_common.h" |
|---|
| 9 | +#include "tx_common.h" |
|---|
| 11 | 10 | #include "ef10_regs.h" |
|---|
| 12 | 11 | #include "io.h" |
|---|
| 13 | 12 | #include "mcdi.h" |
|---|
| 14 | 13 | #include "mcdi_pcol.h" |
|---|
| 14 | +#include "mcdi_port.h" |
|---|
| 15 | +#include "mcdi_port_common.h" |
|---|
| 16 | +#include "mcdi_functions.h" |
|---|
| 15 | 17 | #include "nic.h" |
|---|
| 18 | +#include "mcdi_filters.h" |
|---|
| 16 | 19 | #include "workarounds.h" |
|---|
| 17 | 20 | #include "selftest.h" |
|---|
| 18 | 21 | #include "ef10_sriov.h" |
|---|
| .. | .. |
|---|
| 20 | 23 | #include <linux/jhash.h> |
|---|
| 21 | 24 | #include <linux/wait.h> |
|---|
| 22 | 25 | #include <linux/workqueue.h> |
|---|
| 26 | +#include <net/udp_tunnel.h> |
|---|
| 23 | 27 | |
|---|
| 24 | 28 | /* Hardware control for EF10 architecture including 'Huntington'. */ |
|---|
| 25 | 29 | |
|---|
| .. | .. |
|---|
| 28 | 32 | EFX_EF10_TEST = 1, |
|---|
| 29 | 33 | EFX_EF10_REFILL, |
|---|
| 30 | 34 | }; |
|---|
| 31 | | -/* The maximum size of a shared RSS context */ |
|---|
| 32 | | -/* TODO: this should really be from the mcdi protocol export */ |
|---|
| 33 | | -#define EFX_EF10_MAX_SHARED_RSS_CONTEXT_SIZE 64UL |
|---|
| 34 | | - |
|---|
| 35 | | -/* The filter table(s) are managed by firmware and we have write-only |
|---|
| 36 | | - * access. When removing filters we must identify them to the |
|---|
| 37 | | - * firmware by a 64-bit handle, but this is too wide for Linux kernel |
|---|
| 38 | | - * interfaces (32-bit for RX NFC, 16-bit for RFS). Also, we need to |
|---|
| 39 | | - * be able to tell in advance whether a requested insertion will |
|---|
| 40 | | - * replace an existing filter. Therefore we maintain a software hash |
|---|
| 41 | | - * table, which should be at least as large as the hardware hash |
|---|
| 42 | | - * table. |
|---|
| 43 | | - * |
|---|
| 44 | | - * Huntington has a single 8K filter table shared between all filter |
|---|
| 45 | | - * types and both ports. |
|---|
| 46 | | - */ |
|---|
| 47 | | -#define HUNT_FILTER_TBL_ROWS 8192 |
|---|
| 48 | | - |
|---|
| 49 | | -#define EFX_EF10_FILTER_ID_INVALID 0xffff |
|---|
| 50 | | - |
|---|
| 51 | | -#define EFX_EF10_FILTER_DEV_UC_MAX 32 |
|---|
| 52 | | -#define EFX_EF10_FILTER_DEV_MC_MAX 256 |
|---|
| 53 | 35 | |
|---|
| 54 | 36 | /* VLAN list entry */ |
|---|
| 55 | 37 | struct efx_ef10_vlan { |
|---|
| .. | .. |
|---|
| 57 | 39 | u16 vid; |
|---|
| 58 | 40 | }; |
|---|
| 59 | 41 | |
|---|
| 60 | | -enum efx_ef10_default_filters { |
|---|
| 61 | | - EFX_EF10_BCAST, |
|---|
| 62 | | - EFX_EF10_UCDEF, |
|---|
| 63 | | - EFX_EF10_MCDEF, |
|---|
| 64 | | - EFX_EF10_VXLAN4_UCDEF, |
|---|
| 65 | | - EFX_EF10_VXLAN4_MCDEF, |
|---|
| 66 | | - EFX_EF10_VXLAN6_UCDEF, |
|---|
| 67 | | - EFX_EF10_VXLAN6_MCDEF, |
|---|
| 68 | | - EFX_EF10_NVGRE4_UCDEF, |
|---|
| 69 | | - EFX_EF10_NVGRE4_MCDEF, |
|---|
| 70 | | - EFX_EF10_NVGRE6_UCDEF, |
|---|
| 71 | | - EFX_EF10_NVGRE6_MCDEF, |
|---|
| 72 | | - EFX_EF10_GENEVE4_UCDEF, |
|---|
| 73 | | - EFX_EF10_GENEVE4_MCDEF, |
|---|
| 74 | | - EFX_EF10_GENEVE6_UCDEF, |
|---|
| 75 | | - EFX_EF10_GENEVE6_MCDEF, |
|---|
| 76 | | - |
|---|
| 77 | | - EFX_EF10_NUM_DEFAULT_FILTERS |
|---|
| 78 | | -}; |
|---|
| 79 | | - |
|---|
| 80 | | -/* Per-VLAN filters information */ |
|---|
| 81 | | -struct efx_ef10_filter_vlan { |
|---|
| 82 | | - struct list_head list; |
|---|
| 83 | | - u16 vid; |
|---|
| 84 | | - u16 uc[EFX_EF10_FILTER_DEV_UC_MAX]; |
|---|
| 85 | | - u16 mc[EFX_EF10_FILTER_DEV_MC_MAX]; |
|---|
| 86 | | - u16 default_filters[EFX_EF10_NUM_DEFAULT_FILTERS]; |
|---|
| 87 | | -}; |
|---|
| 88 | | - |
|---|
| 89 | | -struct efx_ef10_dev_addr { |
|---|
| 90 | | - u8 addr[ETH_ALEN]; |
|---|
| 91 | | -}; |
|---|
| 92 | | - |
|---|
| 93 | | -struct efx_ef10_filter_table { |
|---|
| 94 | | -/* The MCDI match masks supported by this fw & hw, in order of priority */ |
|---|
| 95 | | - u32 rx_match_mcdi_flags[ |
|---|
| 96 | | - MC_CMD_GET_PARSER_DISP_INFO_OUT_SUPPORTED_MATCHES_MAXNUM * 2]; |
|---|
| 97 | | - unsigned int rx_match_count; |
|---|
| 98 | | - |
|---|
| 99 | | - struct rw_semaphore lock; /* Protects entries */ |
|---|
| 100 | | - struct { |
|---|
| 101 | | - unsigned long spec; /* pointer to spec plus flag bits */ |
|---|
| 102 | | -/* AUTO_OLD is used to mark and sweep MAC filters for the device address lists. */ |
|---|
| 103 | | -/* unused flag 1UL */ |
|---|
| 104 | | -#define EFX_EF10_FILTER_FLAG_AUTO_OLD 2UL |
|---|
| 105 | | -#define EFX_EF10_FILTER_FLAGS 3UL |
|---|
| 106 | | - u64 handle; /* firmware handle */ |
|---|
| 107 | | - } *entry; |
|---|
| 108 | | -/* Shadow of net_device address lists, guarded by mac_lock */ |
|---|
| 109 | | - struct efx_ef10_dev_addr dev_uc_list[EFX_EF10_FILTER_DEV_UC_MAX]; |
|---|
| 110 | | - struct efx_ef10_dev_addr dev_mc_list[EFX_EF10_FILTER_DEV_MC_MAX]; |
|---|
| 111 | | - int dev_uc_count; |
|---|
| 112 | | - int dev_mc_count; |
|---|
| 113 | | - bool uc_promisc; |
|---|
| 114 | | - bool mc_promisc; |
|---|
| 115 | | -/* Whether in multicast promiscuous mode when last changed */ |
|---|
| 116 | | - bool mc_promisc_last; |
|---|
| 117 | | - bool mc_overflow; /* Too many MC addrs; should always imply mc_promisc */ |
|---|
| 118 | | - bool vlan_filter; |
|---|
| 119 | | - struct list_head vlan_list; |
|---|
| 120 | | -}; |
|---|
| 121 | | - |
|---|
| 122 | | -/* An arbitrary search limit for the software hash table */ |
|---|
| 123 | | -#define EFX_EF10_FILTER_SEARCH_LIMIT 200 |
|---|
| 124 | | - |
|---|
| 125 | | -static void efx_ef10_rx_free_indir_table(struct efx_nic *efx); |
|---|
| 126 | | -static void efx_ef10_filter_table_remove(struct efx_nic *efx); |
|---|
| 127 | | -static int efx_ef10_filter_add_vlan(struct efx_nic *efx, u16 vid); |
|---|
| 128 | | -static void efx_ef10_filter_del_vlan_internal(struct efx_nic *efx, |
|---|
| 129 | | - struct efx_ef10_filter_vlan *vlan); |
|---|
| 130 | | -static void efx_ef10_filter_del_vlan(struct efx_nic *efx, u16 vid); |
|---|
| 131 | 42 | static int efx_ef10_set_udp_tnl_ports(struct efx_nic *efx, bool unloading); |
|---|
| 132 | | - |
|---|
| 133 | | -static u32 efx_ef10_filter_get_unsafe_id(u32 filter_id) |
|---|
| 134 | | -{ |
|---|
| 135 | | - WARN_ON_ONCE(filter_id == EFX_EF10_FILTER_ID_INVALID); |
|---|
| 136 | | - return filter_id & (HUNT_FILTER_TBL_ROWS - 1); |
|---|
| 137 | | -} |
|---|
| 138 | | - |
|---|
| 139 | | -static unsigned int efx_ef10_filter_get_unsafe_pri(u32 filter_id) |
|---|
| 140 | | -{ |
|---|
| 141 | | - return filter_id / (HUNT_FILTER_TBL_ROWS * 2); |
|---|
| 142 | | -} |
|---|
| 143 | | - |
|---|
| 144 | | -static u32 efx_ef10_make_filter_id(unsigned int pri, u16 idx) |
|---|
| 145 | | -{ |
|---|
| 146 | | - return pri * HUNT_FILTER_TBL_ROWS * 2 + idx; |
|---|
| 147 | | -} |
|---|
| 43 | +static const struct udp_tunnel_nic_info efx_ef10_udp_tunnels; |
|---|
| 148 | 44 | |
|---|
| 149 | 45 | static int efx_ef10_get_warm_boot_count(struct efx_nic *efx) |
|---|
| 150 | 46 | { |
|---|
| .. | .. |
|---|
| 186 | 82 | static bool efx_ef10_is_vf(struct efx_nic *efx) |
|---|
| 187 | 83 | { |
|---|
| 188 | 84 | return efx->type->is_vf; |
|---|
| 189 | | -} |
|---|
| 190 | | - |
|---|
| 191 | | -static int efx_ef10_get_pf_index(struct efx_nic *efx) |
|---|
| 192 | | -{ |
|---|
| 193 | | - MCDI_DECLARE_BUF(outbuf, MC_CMD_GET_FUNCTION_INFO_OUT_LEN); |
|---|
| 194 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 195 | | - size_t outlen; |
|---|
| 196 | | - int rc; |
|---|
| 197 | | - |
|---|
| 198 | | - rc = efx_mcdi_rpc(efx, MC_CMD_GET_FUNCTION_INFO, NULL, 0, outbuf, |
|---|
| 199 | | - sizeof(outbuf), &outlen); |
|---|
| 200 | | - if (rc) |
|---|
| 201 | | - return rc; |
|---|
| 202 | | - if (outlen < sizeof(outbuf)) |
|---|
| 203 | | - return -EIO; |
|---|
| 204 | | - |
|---|
| 205 | | - nic_data->pf_index = MCDI_DWORD(outbuf, GET_FUNCTION_INFO_OUT_PF); |
|---|
| 206 | | - return 0; |
|---|
| 207 | 85 | } |
|---|
| 208 | 86 | |
|---|
| 209 | 87 | #ifdef CONFIG_SFC_SRIOV |
|---|
| .. | .. |
|---|
| 276 | 154 | u8 vi_window_mode = MCDI_BYTE(outbuf, |
|---|
| 277 | 155 | GET_CAPABILITIES_V3_OUT_VI_WINDOW_MODE); |
|---|
| 278 | 156 | |
|---|
| 279 | | - switch (vi_window_mode) { |
|---|
| 280 | | - case MC_CMD_GET_CAPABILITIES_V3_OUT_VI_WINDOW_MODE_8K: |
|---|
| 281 | | - efx->vi_stride = 8192; |
|---|
| 282 | | - break; |
|---|
| 283 | | - case MC_CMD_GET_CAPABILITIES_V3_OUT_VI_WINDOW_MODE_16K: |
|---|
| 284 | | - efx->vi_stride = 16384; |
|---|
| 285 | | - break; |
|---|
| 286 | | - case MC_CMD_GET_CAPABILITIES_V3_OUT_VI_WINDOW_MODE_64K: |
|---|
| 287 | | - efx->vi_stride = 65536; |
|---|
| 288 | | - break; |
|---|
| 289 | | - default: |
|---|
| 290 | | - netif_err(efx, probe, efx->net_dev, |
|---|
| 291 | | - "Unrecognised VI window mode %d\n", |
|---|
| 292 | | - vi_window_mode); |
|---|
| 293 | | - return -EIO; |
|---|
| 294 | | - } |
|---|
| 295 | | - netif_dbg(efx, probe, efx->net_dev, "vi_stride = %u\n", |
|---|
| 296 | | - efx->vi_stride); |
|---|
| 157 | + rc = efx_mcdi_window_mode_to_stride(efx, vi_window_mode); |
|---|
| 158 | + if (rc) |
|---|
| 159 | + return rc; |
|---|
| 297 | 160 | } else { |
|---|
| 298 | 161 | /* keep default VI stride */ |
|---|
| 299 | 162 | netif_dbg(efx, probe, efx->net_dev, |
|---|
| .. | .. |
|---|
| 511 | 374 | struct device_attribute *attr, |
|---|
| 512 | 375 | char *buf) |
|---|
| 513 | 376 | { |
|---|
| 514 | | - struct efx_nic *efx = pci_get_drvdata(to_pci_dev(dev)); |
|---|
| 377 | + struct efx_nic *efx = dev_get_drvdata(dev); |
|---|
| 515 | 378 | |
|---|
| 516 | 379 | return sprintf(buf, "%d\n", |
|---|
| 517 | 380 | ((efx->mcdi->fn_flags) & |
|---|
| .. | .. |
|---|
| 523 | 386 | struct device_attribute *attr, |
|---|
| 524 | 387 | char *buf) |
|---|
| 525 | 388 | { |
|---|
| 526 | | - struct efx_nic *efx = pci_get_drvdata(to_pci_dev(dev)); |
|---|
| 389 | + struct efx_nic *efx = dev_get_drvdata(dev); |
|---|
| 527 | 390 | |
|---|
| 528 | 391 | return sprintf(buf, "%d\n", |
|---|
| 529 | 392 | ((efx->mcdi->fn_flags) & |
|---|
| .. | .. |
|---|
| 579 | 442 | if (efx->filter_state) { |
|---|
| 580 | 443 | mutex_lock(&efx->mac_lock); |
|---|
| 581 | 444 | down_write(&efx->filter_sem); |
|---|
| 582 | | - rc = efx_ef10_filter_add_vlan(efx, vlan->vid); |
|---|
| 445 | + rc = efx_mcdi_filter_add_vlan(efx, vlan->vid); |
|---|
| 583 | 446 | up_write(&efx->filter_sem); |
|---|
| 584 | 447 | mutex_unlock(&efx->mac_lock); |
|---|
| 585 | 448 | if (rc) |
|---|
| .. | .. |
|---|
| 608 | 471 | |
|---|
| 609 | 472 | if (efx->filter_state) { |
|---|
| 610 | 473 | down_write(&efx->filter_sem); |
|---|
| 611 | | - efx_ef10_filter_del_vlan(efx, vlan->vid); |
|---|
| 474 | + efx_mcdi_filter_del_vlan(efx, vlan->vid); |
|---|
| 612 | 475 | up_write(&efx->filter_sem); |
|---|
| 613 | 476 | } |
|---|
| 614 | 477 | |
|---|
| .. | .. |
|---|
| 692 | 555 | } |
|---|
| 693 | 556 | nic_data->warm_boot_count = rc; |
|---|
| 694 | 557 | |
|---|
| 695 | | - efx->rss_context.context_id = EFX_EF10_RSS_CONTEXT_INVALID; |
|---|
| 696 | | - |
|---|
| 697 | | - nic_data->vport_id = EVB_PORT_ID_ASSIGNED; |
|---|
| 698 | | - |
|---|
| 699 | 558 | /* In case we're recovering from a crash (kexec), we want to |
|---|
| 700 | 559 | * cancel any outstanding request by the previous user of this |
|---|
| 701 | 560 | * function. We send a special message using the least |
|---|
| .. | .. |
|---|
| 708 | 567 | goto fail2; |
|---|
| 709 | 568 | |
|---|
| 710 | 569 | mutex_init(&nic_data->udp_tunnels_lock); |
|---|
| 570 | + for (i = 0; i < ARRAY_SIZE(nic_data->udp_tunnels); ++i) |
|---|
| 571 | + nic_data->udp_tunnels[i].type = |
|---|
| 572 | + TUNNEL_ENCAP_UDP_PORT_ENTRY_INVALID; |
|---|
| 711 | 573 | |
|---|
| 712 | 574 | /* Reset (most) configuration for this function */ |
|---|
| 713 | 575 | rc = efx_mcdi_reset(efx, RESET_TYPE_ALL); |
|---|
| .. | .. |
|---|
| 728 | 590 | if (rc) |
|---|
| 729 | 591 | goto fail4; |
|---|
| 730 | 592 | |
|---|
| 731 | | - rc = efx_ef10_get_pf_index(efx); |
|---|
| 593 | + rc = efx_get_pf_index(efx, &nic_data->pf_index); |
|---|
| 732 | 594 | if (rc) |
|---|
| 733 | 595 | goto fail5; |
|---|
| 734 | 596 | |
|---|
| .. | .. |
|---|
| 739 | 601 | efx_ef10_read_licensed_features(efx); |
|---|
| 740 | 602 | |
|---|
| 741 | 603 | /* We can have one VI for each vi_stride-byte region. |
|---|
| 742 | | - * However, until we use TX option descriptors we need two TX queues |
|---|
| 743 | | - * per channel. |
|---|
| 604 | + * However, until we use TX option descriptors we need up to four |
|---|
| 605 | + * TX queues per channel for different checksumming combinations. |
|---|
| 744 | 606 | */ |
|---|
| 745 | | - efx->max_channels = min_t(unsigned int, |
|---|
| 746 | | - EFX_MAX_CHANNELS, |
|---|
| 747 | | - efx_ef10_mem_map_size(efx) / |
|---|
| 748 | | - (efx->vi_stride * EFX_TXQ_TYPES)); |
|---|
| 607 | + if (nic_data->datapath_caps & |
|---|
| 608 | + (1 << MC_CMD_GET_CAPABILITIES_OUT_VXLAN_NVGRE_LBN)) |
|---|
| 609 | + efx->tx_queues_per_channel = 4; |
|---|
| 610 | + else |
|---|
| 611 | + efx->tx_queues_per_channel = 2; |
|---|
| 612 | + efx->max_vis = efx_ef10_mem_map_size(efx) / efx->vi_stride; |
|---|
| 613 | + if (!efx->max_vis) { |
|---|
| 614 | + netif_err(efx, drv, efx->net_dev, "error determining max VIs\n"); |
|---|
| 615 | + rc = -EIO; |
|---|
| 616 | + goto fail5; |
|---|
| 617 | + } |
|---|
| 618 | + efx->max_channels = min_t(unsigned int, EFX_MAX_CHANNELS, |
|---|
| 619 | + efx->max_vis / efx->tx_queues_per_channel); |
|---|
| 749 | 620 | efx->max_tx_channels = efx->max_channels; |
|---|
| 750 | 621 | if (WARN_ON(efx->max_channels == 0)) { |
|---|
| 751 | 622 | rc = -EIO; |
|---|
| .. | .. |
|---|
| 804 | 675 | if (rc) |
|---|
| 805 | 676 | goto fail_add_vid_0; |
|---|
| 806 | 677 | |
|---|
| 678 | + if (nic_data->datapath_caps & |
|---|
| 679 | + (1 << MC_CMD_GET_CAPABILITIES_OUT_VXLAN_NVGRE_LBN) && |
|---|
| 680 | + efx->mcdi->fn_flags & |
|---|
| 681 | + (1 << MC_CMD_DRV_ATTACH_EXT_OUT_FLAG_TRUSTED)) |
|---|
| 682 | + efx->net_dev->udp_tunnel_nic_info = &efx_ef10_udp_tunnels; |
|---|
| 683 | + |
|---|
| 807 | 684 | return 0; |
|---|
| 808 | 685 | |
|---|
| 809 | 686 | fail_add_vid_0: |
|---|
| .. | .. |
|---|
| 831 | 708 | fail1: |
|---|
| 832 | 709 | kfree(nic_data); |
|---|
| 833 | 710 | efx->nic_data = NULL; |
|---|
| 834 | | - return rc; |
|---|
| 835 | | -} |
|---|
| 836 | | - |
|---|
| 837 | | -static int efx_ef10_free_vis(struct efx_nic *efx) |
|---|
| 838 | | -{ |
|---|
| 839 | | - MCDI_DECLARE_BUF_ERR(outbuf); |
|---|
| 840 | | - size_t outlen; |
|---|
| 841 | | - int rc = efx_mcdi_rpc_quiet(efx, MC_CMD_FREE_VIS, NULL, 0, |
|---|
| 842 | | - outbuf, sizeof(outbuf), &outlen); |
|---|
| 843 | | - |
|---|
| 844 | | - /* -EALREADY means nothing to free, so ignore */ |
|---|
| 845 | | - if (rc == -EALREADY) |
|---|
| 846 | | - rc = 0; |
|---|
| 847 | | - if (rc) |
|---|
| 848 | | - efx_mcdi_display_error(efx, MC_CMD_FREE_VIS, 0, outbuf, outlen, |
|---|
| 849 | | - rc); |
|---|
| 850 | 711 | return rc; |
|---|
| 851 | 712 | } |
|---|
| 852 | 713 | |
|---|
| .. | .. |
|---|
| 949 | 810 | /* Extra channels, even those with TXQs (PTP), do not require |
|---|
| 950 | 811 | * PIO resources. |
|---|
| 951 | 812 | */ |
|---|
| 952 | | - if (!channel->type->want_pio) |
|---|
| 813 | + if (!channel->type->want_pio || |
|---|
| 814 | + channel->channel >= efx->xdp_channel_offset) |
|---|
| 953 | 815 | continue; |
|---|
| 816 | + |
|---|
| 954 | 817 | efx_for_each_channel_tx_queue(tx_queue, channel) { |
|---|
| 955 | 818 | /* We assign the PIO buffers to queues in |
|---|
| 956 | 819 | * reverse order to allow for the following |
|---|
| .. | .. |
|---|
| 1085 | 948 | |
|---|
| 1086 | 949 | efx_mcdi_mon_remove(efx); |
|---|
| 1087 | 950 | |
|---|
| 1088 | | - efx_ef10_rx_free_indir_table(efx); |
|---|
| 951 | + efx_mcdi_rx_free_indir_table(efx); |
|---|
| 1089 | 952 | |
|---|
| 1090 | 953 | if (nic_data->wc_membase) |
|---|
| 1091 | 954 | iounmap(nic_data->wc_membase); |
|---|
| 1092 | 955 | |
|---|
| 1093 | | - rc = efx_ef10_free_vis(efx); |
|---|
| 956 | + rc = efx_mcdi_free_vis(efx); |
|---|
| 1094 | 957 | WARN_ON(rc != 0); |
|---|
| 1095 | 958 | |
|---|
| 1096 | 959 | if (!nic_data->must_restore_piobufs) |
|---|
| .. | .. |
|---|
| 1261 | 1124 | static int efx_ef10_alloc_vis(struct efx_nic *efx, |
|---|
| 1262 | 1125 | unsigned int min_vis, unsigned int max_vis) |
|---|
| 1263 | 1126 | { |
|---|
| 1264 | | - MCDI_DECLARE_BUF(inbuf, MC_CMD_ALLOC_VIS_IN_LEN); |
|---|
| 1265 | | - MCDI_DECLARE_BUF(outbuf, MC_CMD_ALLOC_VIS_OUT_LEN); |
|---|
| 1266 | 1127 | struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 1267 | | - size_t outlen; |
|---|
| 1268 | | - int rc; |
|---|
| 1269 | 1128 | |
|---|
| 1270 | | - MCDI_SET_DWORD(inbuf, ALLOC_VIS_IN_MIN_VI_COUNT, min_vis); |
|---|
| 1271 | | - MCDI_SET_DWORD(inbuf, ALLOC_VIS_IN_MAX_VI_COUNT, max_vis); |
|---|
| 1272 | | - rc = efx_mcdi_rpc(efx, MC_CMD_ALLOC_VIS, inbuf, sizeof(inbuf), |
|---|
| 1273 | | - outbuf, sizeof(outbuf), &outlen); |
|---|
| 1274 | | - if (rc != 0) |
|---|
| 1275 | | - return rc; |
|---|
| 1276 | | - |
|---|
| 1277 | | - if (outlen < MC_CMD_ALLOC_VIS_OUT_LEN) |
|---|
| 1278 | | - return -EIO; |
|---|
| 1279 | | - |
|---|
| 1280 | | - netif_dbg(efx, drv, efx->net_dev, "base VI is A0x%03x\n", |
|---|
| 1281 | | - MCDI_DWORD(outbuf, ALLOC_VIS_OUT_VI_BASE)); |
|---|
| 1282 | | - |
|---|
| 1283 | | - nic_data->vi_base = MCDI_DWORD(outbuf, ALLOC_VIS_OUT_VI_BASE); |
|---|
| 1284 | | - nic_data->n_allocated_vis = MCDI_DWORD(outbuf, ALLOC_VIS_OUT_VI_COUNT); |
|---|
| 1285 | | - return 0; |
|---|
| 1129 | + return efx_mcdi_alloc_vis(efx, min_vis, max_vis, &nic_data->vi_base, |
|---|
| 1130 | + &nic_data->n_allocated_vis); |
|---|
| 1286 | 1131 | } |
|---|
| 1287 | 1132 | |
|---|
| 1288 | 1133 | /* Note that the failure path of this function does not free |
|---|
| .. | .. |
|---|
| 1290 | 1135 | */ |
|---|
| 1291 | 1136 | static int efx_ef10_dimension_resources(struct efx_nic *efx) |
|---|
| 1292 | 1137 | { |
|---|
| 1138 | + unsigned int min_vis = max_t(unsigned int, efx->tx_queues_per_channel, |
|---|
| 1139 | + efx_separate_tx_channels ? 2 : 1); |
|---|
| 1140 | + unsigned int channel_vis, pio_write_vi_base, max_vis; |
|---|
| 1293 | 1141 | struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 1294 | 1142 | unsigned int uc_mem_map_size, wc_mem_map_size; |
|---|
| 1295 | | - unsigned int min_vis = max(EFX_TXQ_TYPES, |
|---|
| 1296 | | - efx_separate_tx_channels ? 2 : 1); |
|---|
| 1297 | | - unsigned int channel_vis, pio_write_vi_base, max_vis; |
|---|
| 1298 | 1143 | void __iomem *membase; |
|---|
| 1299 | 1144 | int rc; |
|---|
| 1300 | 1145 | |
|---|
| 1301 | 1146 | channel_vis = max(efx->n_channels, |
|---|
| 1302 | | - (efx->n_tx_channels + efx->n_extra_tx_channels) * |
|---|
| 1303 | | - EFX_TXQ_TYPES); |
|---|
| 1147 | + ((efx->n_tx_channels + efx->n_extra_tx_channels) * |
|---|
| 1148 | + efx->tx_queues_per_channel) + |
|---|
| 1149 | + efx->n_xdp_channels * efx->xdp_tx_per_channel); |
|---|
| 1150 | + if (efx->max_vis && efx->max_vis < channel_vis) { |
|---|
| 1151 | + netif_dbg(efx, drv, efx->net_dev, |
|---|
| 1152 | + "Reducing channel VIs from %u to %u\n", |
|---|
| 1153 | + channel_vis, efx->max_vis); |
|---|
| 1154 | + channel_vis = efx->max_vis; |
|---|
| 1155 | + } |
|---|
| 1304 | 1156 | |
|---|
| 1305 | 1157 | #ifdef EFX_USE_PIO |
|---|
| 1306 | 1158 | /* Try to allocate PIO buffers if wanted and if the full |
|---|
| .. | .. |
|---|
| 1363 | 1215 | } |
|---|
| 1364 | 1216 | |
|---|
| 1365 | 1217 | /* In case the last attached driver failed to free VIs, do it now */ |
|---|
| 1366 | | - rc = efx_ef10_free_vis(efx); |
|---|
| 1218 | + rc = efx_mcdi_free_vis(efx); |
|---|
| 1367 | 1219 | if (rc != 0) |
|---|
| 1368 | 1220 | return rc; |
|---|
| 1369 | 1221 | |
|---|
| .. | .. |
|---|
| 1382 | 1234 | */ |
|---|
| 1383 | 1235 | efx->max_channels = nic_data->n_allocated_vis; |
|---|
| 1384 | 1236 | efx->max_tx_channels = |
|---|
| 1385 | | - nic_data->n_allocated_vis / EFX_TXQ_TYPES; |
|---|
| 1237 | + nic_data->n_allocated_vis / efx->tx_queues_per_channel; |
|---|
| 1386 | 1238 | |
|---|
| 1387 | | - efx_ef10_free_vis(efx); |
|---|
| 1239 | + efx_mcdi_free_vis(efx); |
|---|
| 1388 | 1240 | return -EAGAIN; |
|---|
| 1389 | 1241 | } |
|---|
| 1390 | 1242 | |
|---|
| .. | .. |
|---|
| 1401 | 1253 | } |
|---|
| 1402 | 1254 | |
|---|
| 1403 | 1255 | /* Shrink the original UC mapping of the memory BAR */ |
|---|
| 1404 | | - membase = ioremap_nocache(efx->membase_phys, uc_mem_map_size); |
|---|
| 1256 | + membase = ioremap(efx->membase_phys, uc_mem_map_size); |
|---|
| 1405 | 1257 | if (!membase) { |
|---|
| 1406 | 1258 | netif_err(efx, probe, efx->net_dev, |
|---|
| 1407 | 1259 | "could not shrink memory BAR to %x\n", |
|---|
| .. | .. |
|---|
| 1441 | 1293 | return 0; |
|---|
| 1442 | 1294 | } |
|---|
| 1443 | 1295 | |
|---|
| 1296 | +static void efx_ef10_fini_nic(struct efx_nic *efx) |
|---|
| 1297 | +{ |
|---|
| 1298 | + struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 1299 | + |
|---|
| 1300 | + kfree(nic_data->mc_stats); |
|---|
| 1301 | + nic_data->mc_stats = NULL; |
|---|
| 1302 | +} |
|---|
| 1303 | + |
|---|
| 1444 | 1304 | static int efx_ef10_init_nic(struct efx_nic *efx) |
|---|
| 1445 | 1305 | { |
|---|
| 1446 | 1306 | struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 1307 | + netdev_features_t hw_enc_features = 0; |
|---|
| 1447 | 1308 | int rc; |
|---|
| 1448 | 1309 | |
|---|
| 1449 | 1310 | if (nic_data->must_check_datapath_caps) { |
|---|
| .. | .. |
|---|
| 1453 | 1314 | nic_data->must_check_datapath_caps = false; |
|---|
| 1454 | 1315 | } |
|---|
| 1455 | 1316 | |
|---|
| 1456 | | - if (nic_data->must_realloc_vis) { |
|---|
| 1317 | + if (efx->must_realloc_vis) { |
|---|
| 1457 | 1318 | /* We cannot let the number of VIs change now */ |
|---|
| 1458 | 1319 | rc = efx_ef10_alloc_vis(efx, nic_data->n_allocated_vis, |
|---|
| 1459 | 1320 | nic_data->n_allocated_vis); |
|---|
| 1460 | 1321 | if (rc) |
|---|
| 1461 | 1322 | return rc; |
|---|
| 1462 | | - nic_data->must_realloc_vis = false; |
|---|
| 1323 | + efx->must_realloc_vis = false; |
|---|
| 1463 | 1324 | } |
|---|
| 1325 | + |
|---|
| 1326 | + nic_data->mc_stats = kmalloc(efx->num_mac_stats * sizeof(__le64), |
|---|
| 1327 | + GFP_KERNEL); |
|---|
| 1328 | + if (!nic_data->mc_stats) |
|---|
| 1329 | + return -ENOMEM; |
|---|
| 1464 | 1330 | |
|---|
| 1465 | 1331 | if (nic_data->must_restore_piobufs && nic_data->n_piobufs) { |
|---|
| 1466 | 1332 | rc = efx_ef10_alloc_piobufs(efx, nic_data->n_piobufs); |
|---|
| .. | .. |
|---|
| 1483 | 1349 | nic_data->must_restore_piobufs = false; |
|---|
| 1484 | 1350 | } |
|---|
| 1485 | 1351 | |
|---|
| 1352 | + /* add encapsulated checksum offload features */ |
|---|
| 1353 | + if (efx_has_cap(efx, VXLAN_NVGRE) && !efx_ef10_is_vf(efx)) |
|---|
| 1354 | + hw_enc_features |= NETIF_F_IP_CSUM | NETIF_F_IPV6_CSUM; |
|---|
| 1355 | + /* add encapsulated TSO features */ |
|---|
| 1356 | + if (efx_has_cap(efx, TX_TSO_V2_ENCAP)) { |
|---|
| 1357 | + netdev_features_t encap_tso_features; |
|---|
| 1358 | + |
|---|
| 1359 | + encap_tso_features = NETIF_F_GSO_UDP_TUNNEL | NETIF_F_GSO_GRE | |
|---|
| 1360 | + NETIF_F_GSO_UDP_TUNNEL_CSUM | NETIF_F_GSO_GRE_CSUM; |
|---|
| 1361 | + |
|---|
| 1362 | + hw_enc_features |= encap_tso_features | NETIF_F_TSO; |
|---|
| 1363 | + efx->net_dev->features |= encap_tso_features; |
|---|
| 1364 | + } |
|---|
| 1365 | + efx->net_dev->hw_enc_features = hw_enc_features; |
|---|
| 1366 | + |
|---|
| 1486 | 1367 | /* don't fail init if RSS setup doesn't work */ |
|---|
| 1487 | 1368 | rc = efx->type->rx_push_rss_config(efx, false, |
|---|
| 1488 | 1369 | efx->rss_context.rx_indir_table, NULL); |
|---|
| .. | .. |
|---|
| 1490 | 1371 | return 0; |
|---|
| 1491 | 1372 | } |
|---|
| 1492 | 1373 | |
|---|
| 1493 | | -static void efx_ef10_reset_mc_allocations(struct efx_nic *efx) |
|---|
| 1374 | +static void efx_ef10_table_reset_mc_allocations(struct efx_nic *efx) |
|---|
| 1494 | 1375 | { |
|---|
| 1495 | 1376 | struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 1496 | 1377 | #ifdef CONFIG_SFC_SRIOV |
|---|
| .. | .. |
|---|
| 1498 | 1379 | #endif |
|---|
| 1499 | 1380 | |
|---|
| 1500 | 1381 | /* All our allocations have been reset */ |
|---|
| 1501 | | - nic_data->must_realloc_vis = true; |
|---|
| 1502 | | - nic_data->must_restore_rss_contexts = true; |
|---|
| 1503 | | - nic_data->must_restore_filters = true; |
|---|
| 1382 | + efx->must_realloc_vis = true; |
|---|
| 1383 | + efx_mcdi_filter_table_reset_mc_allocations(efx); |
|---|
| 1504 | 1384 | nic_data->must_restore_piobufs = true; |
|---|
| 1505 | 1385 | efx_ef10_forget_old_piobufs(efx); |
|---|
| 1506 | | - efx->rss_context.context_id = EFX_EF10_RSS_CONTEXT_INVALID; |
|---|
| 1386 | + efx->rss_context.context_id = EFX_MCDI_RSS_CONTEXT_INVALID; |
|---|
| 1507 | 1387 | |
|---|
| 1508 | 1388 | /* Driver-created vswitches and vports must be re-created */ |
|---|
| 1509 | 1389 | nic_data->must_probe_vswitching = true; |
|---|
| 1510 | | - nic_data->vport_id = EVB_PORT_ID_ASSIGNED; |
|---|
| 1390 | + efx->vport_id = EVB_PORT_ID_ASSIGNED; |
|---|
| 1511 | 1391 | #ifdef CONFIG_SFC_SRIOV |
|---|
| 1512 | 1392 | if (nic_data->vf) |
|---|
| 1513 | 1393 | for (i = 0; i < efx->vf_count; i++) |
|---|
| .. | .. |
|---|
| 1571 | 1451 | */ |
|---|
| 1572 | 1452 | if ((reset_type == RESET_TYPE_ALL || |
|---|
| 1573 | 1453 | reset_type == RESET_TYPE_MCDI_TIMEOUT) && !rc) |
|---|
| 1574 | | - efx_ef10_reset_mc_allocations(efx); |
|---|
| 1454 | + efx_ef10_table_reset_mc_allocations(efx); |
|---|
| 1575 | 1455 | return rc; |
|---|
| 1576 | 1456 | } |
|---|
| 1577 | 1457 | |
|---|
| .. | .. |
|---|
| 1583 | 1463 | { NULL, 64, 8 * MC_CMD_MAC_ ## mcdi_name } |
|---|
| 1584 | 1464 | #define EF10_OTHER_STAT(ext_name) \ |
|---|
| 1585 | 1465 | [EF10_STAT_ ## ext_name] = { #ext_name, 0, 0 } |
|---|
| 1586 | | -#define GENERIC_SW_STAT(ext_name) \ |
|---|
| 1587 | | - [GENERIC_STAT_ ## ext_name] = { #ext_name, 0, 0 } |
|---|
| 1588 | 1466 | |
|---|
| 1589 | 1467 | static const struct efx_hw_stat_desc efx_ef10_stat_desc[EF10_STAT_COUNT] = { |
|---|
| 1590 | 1468 | EF10_DMA_STAT(port_tx_bytes, TX_BYTES), |
|---|
| .. | .. |
|---|
| 1628 | 1506 | EF10_DMA_STAT(port_rx_align_error, RX_ALIGN_ERROR_PKTS), |
|---|
| 1629 | 1507 | EF10_DMA_STAT(port_rx_length_error, RX_LENGTH_ERROR_PKTS), |
|---|
| 1630 | 1508 | EF10_DMA_STAT(port_rx_nodesc_drops, RX_NODESC_DROPS), |
|---|
| 1631 | | - GENERIC_SW_STAT(rx_nodesc_trunc), |
|---|
| 1632 | | - GENERIC_SW_STAT(rx_noskb_drops), |
|---|
| 1509 | + EFX_GENERIC_SW_STAT(rx_nodesc_trunc), |
|---|
| 1510 | + EFX_GENERIC_SW_STAT(rx_noskb_drops), |
|---|
| 1633 | 1511 | EF10_DMA_STAT(port_rx_pm_trunc_bb_overflow, PM_TRUNC_BB_OVERFLOW), |
|---|
| 1634 | 1512 | EF10_DMA_STAT(port_rx_pm_discard_bb_overflow, PM_DISCARD_BB_OVERFLOW), |
|---|
| 1635 | 1513 | EF10_DMA_STAT(port_rx_pm_trunc_vfifo_full, PM_TRUNC_VFIFO_FULL), |
|---|
| .. | .. |
|---|
| 1938 | 1816 | return stats_count; |
|---|
| 1939 | 1817 | } |
|---|
| 1940 | 1818 | |
|---|
| 1941 | | -static int efx_ef10_try_update_nic_stats_pf(struct efx_nic *efx) |
|---|
| 1819 | +static size_t efx_ef10_update_stats_pf(struct efx_nic *efx, u64 *full_stats, |
|---|
| 1820 | + struct rtnl_link_stats64 *core_stats) |
|---|
| 1942 | 1821 | { |
|---|
| 1943 | 1822 | struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 1944 | 1823 | DECLARE_BITMAP(mask, EF10_STAT_COUNT); |
|---|
| 1945 | | - __le64 generation_start, generation_end; |
|---|
| 1946 | 1824 | u64 *stats = nic_data->stats; |
|---|
| 1947 | | - __le64 *dma_stats; |
|---|
| 1948 | 1825 | |
|---|
| 1949 | 1826 | efx_ef10_get_stat_mask(efx, mask); |
|---|
| 1950 | 1827 | |
|---|
| 1951 | | - dma_stats = efx->stats_buffer.addr; |
|---|
| 1952 | | - |
|---|
| 1953 | | - generation_end = dma_stats[efx->num_mac_stats - 1]; |
|---|
| 1954 | | - if (generation_end == EFX_MC_STATS_GENERATION_INVALID) |
|---|
| 1955 | | - return 0; |
|---|
| 1956 | | - rmb(); |
|---|
| 1957 | | - efx_nic_update_stats(efx_ef10_stat_desc, EF10_STAT_COUNT, mask, |
|---|
| 1958 | | - stats, efx->stats_buffer.addr, false); |
|---|
| 1959 | | - rmb(); |
|---|
| 1960 | | - generation_start = dma_stats[MC_CMD_MAC_GENERATION_START]; |
|---|
| 1961 | | - if (generation_end != generation_start) |
|---|
| 1962 | | - return -EAGAIN; |
|---|
| 1828 | + efx_nic_copy_stats(efx, nic_data->mc_stats); |
|---|
| 1829 | + efx_nic_update_stats(efx_ef10_stat_desc, EF10_STAT_COUNT, |
|---|
| 1830 | + mask, stats, nic_data->mc_stats, false); |
|---|
| 1963 | 1831 | |
|---|
| 1964 | 1832 | /* Update derived statistics */ |
|---|
| 1965 | 1833 | efx_nic_fix_nodesc_drop_stat(efx, |
|---|
| 1966 | 1834 | &stats[EF10_STAT_port_rx_nodesc_drops]); |
|---|
| 1835 | + /* MC Firmware reads RX_BYTES and RX_GOOD_BYTES from the MAC. |
|---|
| 1836 | + * It then calculates RX_BAD_BYTES and DMAs it to us with RX_BYTES. |
|---|
| 1837 | + * We report these as port_rx_ stats. We are not given RX_GOOD_BYTES. |
|---|
| 1838 | + * Here we calculate port_rx_good_bytes. |
|---|
| 1839 | + */ |
|---|
| 1967 | 1840 | stats[EF10_STAT_port_rx_good_bytes] = |
|---|
| 1968 | 1841 | stats[EF10_STAT_port_rx_bytes] - |
|---|
| 1969 | 1842 | stats[EF10_STAT_port_rx_bytes_minus_good_bytes]; |
|---|
| 1843 | + |
|---|
| 1844 | + /* The asynchronous reads used to calculate RX_BAD_BYTES in |
|---|
| 1845 | + * MC Firmware are done such that we should not see an increase in |
|---|
| 1846 | + * RX_BAD_BYTES when a good packet has arrived. Unfortunately this |
|---|
| 1847 | + * does mean that the stat can decrease at times. Here we do not |
|---|
| 1848 | + * update the stat unless it has increased or has gone to zero |
|---|
| 1849 | + * (In the case of the NIC rebooting). |
|---|
| 1850 | + * Please see Bug 33781 for a discussion of why things work this way. |
|---|
| 1851 | + */ |
|---|
| 1970 | 1852 | efx_update_diff_stat(&stats[EF10_STAT_port_rx_bad_bytes], |
|---|
| 1971 | 1853 | stats[EF10_STAT_port_rx_bytes_minus_good_bytes]); |
|---|
| 1972 | 1854 | efx_update_sw_stats(efx, stats); |
|---|
| 1973 | | - return 0; |
|---|
| 1974 | | -} |
|---|
| 1975 | | - |
|---|
| 1976 | | - |
|---|
| 1977 | | -static size_t efx_ef10_update_stats_pf(struct efx_nic *efx, u64 *full_stats, |
|---|
| 1978 | | - struct rtnl_link_stats64 *core_stats) |
|---|
| 1979 | | -{ |
|---|
| 1980 | | - int retry; |
|---|
| 1981 | | - |
|---|
| 1982 | | - /* If we're unlucky enough to read statistics during the DMA, wait |
|---|
| 1983 | | - * up to 10ms for it to finish (typically takes <500us) |
|---|
| 1984 | | - */ |
|---|
| 1985 | | - for (retry = 0; retry < 100; ++retry) { |
|---|
| 1986 | | - if (efx_ef10_try_update_nic_stats_pf(efx) == 0) |
|---|
| 1987 | | - break; |
|---|
| 1988 | | - udelay(100); |
|---|
| 1989 | | - } |
|---|
| 1990 | 1855 | |
|---|
| 1991 | 1856 | return efx_ef10_update_stats_common(efx, full_stats, core_stats); |
|---|
| 1992 | 1857 | } |
|---|
| 1993 | 1858 | |
|---|
| 1994 | 1859 | static int efx_ef10_try_update_nic_stats_vf(struct efx_nic *efx) |
|---|
| 1860 | + __must_hold(&efx->stats_lock) |
|---|
| 1995 | 1861 | { |
|---|
| 1996 | 1862 | MCDI_DECLARE_BUF(inbuf, MC_CMD_MAC_STATS_IN_LEN); |
|---|
| 1997 | 1863 | struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| .. | .. |
|---|
| 2005 | 1871 | |
|---|
| 2006 | 1872 | spin_unlock_bh(&efx->stats_lock); |
|---|
| 2007 | 1873 | |
|---|
| 2008 | | - if (in_interrupt()) { |
|---|
| 2009 | | - /* If in atomic context, cannot update stats. Just update the |
|---|
| 2010 | | - * software stats and return so the caller can continue. |
|---|
| 2011 | | - */ |
|---|
| 2012 | | - spin_lock_bh(&efx->stats_lock); |
|---|
| 2013 | | - efx_update_sw_stats(efx, stats); |
|---|
| 2014 | | - return 0; |
|---|
| 2015 | | - } |
|---|
| 2016 | | - |
|---|
| 2017 | 1874 | efx_ef10_get_stat_mask(efx, mask); |
|---|
| 2018 | 1875 | |
|---|
| 2019 | | - rc = efx_nic_alloc_buffer(efx, &stats_buf, dma_len, GFP_ATOMIC); |
|---|
| 1876 | + rc = efx_nic_alloc_buffer(efx, &stats_buf, dma_len, GFP_KERNEL); |
|---|
| 2020 | 1877 | if (rc) { |
|---|
| 2021 | 1878 | spin_lock_bh(&efx->stats_lock); |
|---|
| 2022 | 1879 | return rc; |
|---|
| .. | .. |
|---|
| 2059 | 1916 | |
|---|
| 2060 | 1917 | efx_update_sw_stats(efx, stats); |
|---|
| 2061 | 1918 | out: |
|---|
| 1919 | + /* releasing a DMA coherent buffer with BH disabled can panic */ |
|---|
| 1920 | + spin_unlock_bh(&efx->stats_lock); |
|---|
| 2062 | 1921 | efx_nic_free_buffer(efx, &stats_buf); |
|---|
| 1922 | + spin_lock_bh(&efx->stats_lock); |
|---|
| 2063 | 1923 | return rc; |
|---|
| 2064 | 1924 | } |
|---|
| 2065 | 1925 | |
|---|
| .. | .. |
|---|
| 2069 | 1929 | if (efx_ef10_try_update_nic_stats_vf(efx)) |
|---|
| 2070 | 1930 | return 0; |
|---|
| 2071 | 1931 | |
|---|
| 1932 | + return efx_ef10_update_stats_common(efx, full_stats, core_stats); |
|---|
| 1933 | +} |
|---|
| 1934 | + |
|---|
| 1935 | +static size_t efx_ef10_update_stats_atomic_vf(struct efx_nic *efx, u64 *full_stats, |
|---|
| 1936 | + struct rtnl_link_stats64 *core_stats) |
|---|
| 1937 | +{ |
|---|
| 1938 | + struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 1939 | + |
|---|
| 1940 | + /* In atomic context, cannot update HW stats. Just update the |
|---|
| 1941 | + * software stats and return so the caller can continue. |
|---|
| 1942 | + */ |
|---|
| 1943 | + efx_update_sw_stats(efx, nic_data->stats); |
|---|
| 2072 | 1944 | return efx_ef10_update_stats_common(efx, full_stats, core_stats); |
|---|
| 2073 | 1945 | } |
|---|
| 2074 | 1946 | |
|---|
| .. | .. |
|---|
| 2187 | 2059 | struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 2188 | 2060 | |
|---|
| 2189 | 2061 | /* All our allocations have been reset */ |
|---|
| 2190 | | - efx_ef10_reset_mc_allocations(efx); |
|---|
| 2062 | + efx_ef10_table_reset_mc_allocations(efx); |
|---|
| 2191 | 2063 | |
|---|
| 2192 | 2064 | /* The datapath firmware might have been changed */ |
|---|
| 2193 | 2065 | nic_data->must_check_datapath_caps = true; |
|---|
| .. | .. |
|---|
| 2300 | 2172 | |
|---|
| 2301 | 2173 | static int efx_ef10_tx_probe(struct efx_tx_queue *tx_queue) |
|---|
| 2302 | 2174 | { |
|---|
| 2175 | + /* low two bits of label are what we want for type */ |
|---|
| 2176 | + BUILD_BUG_ON((EFX_TXQ_TYPE_OUTER_CSUM | EFX_TXQ_TYPE_INNER_CSUM) != 3); |
|---|
| 2177 | + tx_queue->type = tx_queue->label & 3; |
|---|
| 2303 | 2178 | return efx_nic_alloc_buffer(tx_queue->efx, &tx_queue->txd.buf, |
|---|
| 2304 | 2179 | (tx_queue->ptr_mask + 1) * |
|---|
| 2305 | 2180 | sizeof(efx_qword_t), |
|---|
| .. | .. |
|---|
| 2322 | 2197 | |
|---|
| 2323 | 2198 | /* Add Firmware-Assisted TSO v2 option descriptors to a queue. |
|---|
| 2324 | 2199 | */ |
|---|
| 2325 | | -static int efx_ef10_tx_tso_desc(struct efx_tx_queue *tx_queue, |
|---|
| 2326 | | - struct sk_buff *skb, |
|---|
| 2327 | | - bool *data_mapped) |
|---|
| 2200 | +int efx_ef10_tx_tso_desc(struct efx_tx_queue *tx_queue, struct sk_buff *skb, |
|---|
| 2201 | + bool *data_mapped) |
|---|
| 2328 | 2202 | { |
|---|
| 2329 | 2203 | struct efx_tx_buffer *buffer; |
|---|
| 2204 | + u16 inner_ipv4_id = 0; |
|---|
| 2205 | + u16 outer_ipv4_id = 0; |
|---|
| 2330 | 2206 | struct tcphdr *tcp; |
|---|
| 2331 | 2207 | struct iphdr *ip; |
|---|
| 2332 | | - |
|---|
| 2333 | | - u16 ipv4_id; |
|---|
| 2208 | + u16 ip_tot_len; |
|---|
| 2334 | 2209 | u32 seqnum; |
|---|
| 2335 | 2210 | u32 mss; |
|---|
| 2336 | 2211 | |
|---|
| .. | .. |
|---|
| 2343 | 2218 | return -EINVAL; |
|---|
| 2344 | 2219 | } |
|---|
| 2345 | 2220 | |
|---|
| 2346 | | - ip = ip_hdr(skb); |
|---|
| 2347 | | - if (ip->version == 4) { |
|---|
| 2348 | | - /* Modify IPv4 header if needed. */ |
|---|
| 2349 | | - ip->tot_len = 0; |
|---|
| 2350 | | - ip->check = 0; |
|---|
| 2351 | | - ipv4_id = ntohs(ip->id); |
|---|
| 2352 | | - } else { |
|---|
| 2353 | | - /* Modify IPv6 header if needed. */ |
|---|
| 2354 | | - struct ipv6hdr *ipv6 = ipv6_hdr(skb); |
|---|
| 2221 | + if (skb->encapsulation) { |
|---|
| 2222 | + if (!tx_queue->tso_encap) |
|---|
| 2223 | + return -EINVAL; |
|---|
| 2224 | + ip = ip_hdr(skb); |
|---|
| 2225 | + if (ip->version == 4) |
|---|
| 2226 | + outer_ipv4_id = ntohs(ip->id); |
|---|
| 2355 | 2227 | |
|---|
| 2356 | | - ipv6->payload_len = 0; |
|---|
| 2357 | | - ipv4_id = 0; |
|---|
| 2228 | + ip = inner_ip_hdr(skb); |
|---|
| 2229 | + tcp = inner_tcp_hdr(skb); |
|---|
| 2230 | + } else { |
|---|
| 2231 | + ip = ip_hdr(skb); |
|---|
| 2232 | + tcp = tcp_hdr(skb); |
|---|
| 2358 | 2233 | } |
|---|
| 2359 | 2234 | |
|---|
| 2360 | | - tcp = tcp_hdr(skb); |
|---|
| 2235 | + /* 8000-series EF10 hardware requires that IP Total Length be |
|---|
| 2236 | + * greater than or equal to the value it will have in each segment |
|---|
| 2237 | + * (which is at most mss + 208 + TCP header length), but also less |
|---|
| 2238 | + * than (0x10000 - inner_network_header). Otherwise the TCP |
|---|
| 2239 | + * checksum calculation will be broken for encapsulated packets. |
|---|
| 2240 | + * We fill in ip->tot_len with 0xff30, which should satisfy the |
|---|
| 2241 | + * first requirement unless the MSS is ridiculously large (which |
|---|
| 2242 | + * should be impossible as the driver max MTU is 9216); it is |
|---|
| 2243 | + * guaranteed to satisfy the second as we only attempt TSO if |
|---|
| 2244 | + * inner_network_header <= 208. |
|---|
| 2245 | + */ |
|---|
| 2246 | + ip_tot_len = 0x10000 - EFX_TSO2_MAX_HDRLEN; |
|---|
| 2247 | + EFX_WARN_ON_ONCE_PARANOID(mss + EFX_TSO2_MAX_HDRLEN + |
|---|
| 2248 | + (tcp->doff << 2u) > ip_tot_len); |
|---|
| 2249 | + |
|---|
| 2250 | + if (ip->version == 4) { |
|---|
| 2251 | + ip->tot_len = htons(ip_tot_len); |
|---|
| 2252 | + ip->check = 0; |
|---|
| 2253 | + inner_ipv4_id = ntohs(ip->id); |
|---|
| 2254 | + } else { |
|---|
| 2255 | + ((struct ipv6hdr *)ip)->payload_len = htons(ip_tot_len); |
|---|
| 2256 | + } |
|---|
| 2257 | + |
|---|
| 2361 | 2258 | seqnum = ntohl(tcp->seq); |
|---|
| 2362 | 2259 | |
|---|
| 2363 | 2260 | buffer = efx_tx_queue_get_insert_buffer(tx_queue); |
|---|
| .. | .. |
|---|
| 2370 | 2267 | ESF_DZ_TX_OPTION_TYPE, ESE_DZ_TX_OPTION_DESC_TSO, |
|---|
| 2371 | 2268 | ESF_DZ_TX_TSO_OPTION_TYPE, |
|---|
| 2372 | 2269 | ESE_DZ_TX_TSO_OPTION_DESC_FATSO2A, |
|---|
| 2373 | | - ESF_DZ_TX_TSO_IP_ID, ipv4_id, |
|---|
| 2270 | + ESF_DZ_TX_TSO_IP_ID, inner_ipv4_id, |
|---|
| 2374 | 2271 | ESF_DZ_TX_TSO_TCP_SEQNO, seqnum |
|---|
| 2375 | 2272 | ); |
|---|
| 2376 | 2273 | ++tx_queue->insert_count; |
|---|
| .. | .. |
|---|
| 2380 | 2277 | buffer->flags = EFX_TX_BUF_OPTION; |
|---|
| 2381 | 2278 | buffer->len = 0; |
|---|
| 2382 | 2279 | buffer->unmap_len = 0; |
|---|
| 2383 | | - EFX_POPULATE_QWORD_4(buffer->option, |
|---|
| 2280 | + EFX_POPULATE_QWORD_5(buffer->option, |
|---|
| 2384 | 2281 | ESF_DZ_TX_DESC_IS_OPT, 1, |
|---|
| 2385 | 2282 | ESF_DZ_TX_OPTION_TYPE, ESE_DZ_TX_OPTION_DESC_TSO, |
|---|
| 2386 | 2283 | ESF_DZ_TX_TSO_OPTION_TYPE, |
|---|
| 2387 | 2284 | ESE_DZ_TX_TSO_OPTION_DESC_FATSO2B, |
|---|
| 2285 | + ESF_DZ_TX_TSO_OUTER_IPID, outer_ipv4_id, |
|---|
| 2388 | 2286 | ESF_DZ_TX_TSO_TCP_MSS, mss |
|---|
| 2389 | 2287 | ); |
|---|
| 2390 | 2288 | ++tx_queue->insert_count; |
|---|
| .. | .. |
|---|
| 2408 | 2306 | |
|---|
| 2409 | 2307 | static void efx_ef10_tx_init(struct efx_tx_queue *tx_queue) |
|---|
| 2410 | 2308 | { |
|---|
| 2411 | | - MCDI_DECLARE_BUF(inbuf, MC_CMD_INIT_TXQ_IN_LEN(EFX_MAX_DMAQ_SIZE * 8 / |
|---|
| 2412 | | - EFX_BUF_SIZE)); |
|---|
| 2413 | | - bool csum_offload = tx_queue->queue & EFX_TXQ_TYPE_OFFLOAD; |
|---|
| 2414 | | - size_t entries = tx_queue->txd.buf.len / EFX_BUF_SIZE; |
|---|
| 2309 | + bool csum_offload = tx_queue->type & EFX_TXQ_TYPE_OUTER_CSUM; |
|---|
| 2310 | + bool inner_csum = tx_queue->type & EFX_TXQ_TYPE_INNER_CSUM; |
|---|
| 2415 | 2311 | struct efx_channel *channel = tx_queue->channel; |
|---|
| 2416 | 2312 | struct efx_nic *efx = tx_queue->efx; |
|---|
| 2417 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 2418 | | - bool tso_v2 = false; |
|---|
| 2419 | | - size_t inlen; |
|---|
| 2420 | | - dma_addr_t dma_addr; |
|---|
| 2313 | + struct efx_ef10_nic_data *nic_data; |
|---|
| 2421 | 2314 | efx_qword_t *txd; |
|---|
| 2422 | 2315 | int rc; |
|---|
| 2423 | | - int i; |
|---|
| 2424 | | - BUILD_BUG_ON(MC_CMD_INIT_TXQ_OUT_LEN != 0); |
|---|
| 2316 | + |
|---|
| 2317 | + nic_data = efx->nic_data; |
|---|
| 2425 | 2318 | |
|---|
| 2426 | 2319 | /* Only attempt to enable TX timestamping if we have the license for it, |
|---|
| 2427 | 2320 | * otherwise TXQ init will fail |
|---|
| .. | .. |
|---|
| 2437 | 2330 | /* TSOv2 is a limited resource that can only be configured on a limited |
|---|
| 2438 | 2331 | * number of queues. TSO without checksum offload is not really a thing, |
|---|
| 2439 | 2332 | * so we only enable it for those queues. |
|---|
| 2440 | | - * TSOv2 cannot be used with Hardware timestamping. |
|---|
| 2333 | + * TSOv2 cannot be used with Hardware timestamping, and is never needed |
|---|
| 2334 | + * for XDP tx. |
|---|
| 2441 | 2335 | */ |
|---|
| 2442 | | - if (csum_offload && (nic_data->datapath_caps2 & |
|---|
| 2443 | | - (1 << MC_CMD_GET_CAPABILITIES_V2_OUT_TX_TSO_V2_LBN)) && |
|---|
| 2444 | | - !tx_queue->timestamping) { |
|---|
| 2445 | | - tso_v2 = true; |
|---|
| 2446 | | - netif_dbg(efx, hw, efx->net_dev, "Using TSOv2 for channel %u\n", |
|---|
| 2447 | | - channel->channel); |
|---|
| 2448 | | - } |
|---|
| 2449 | | - |
|---|
| 2450 | | - MCDI_SET_DWORD(inbuf, INIT_TXQ_IN_SIZE, tx_queue->ptr_mask + 1); |
|---|
| 2451 | | - MCDI_SET_DWORD(inbuf, INIT_TXQ_IN_TARGET_EVQ, channel->channel); |
|---|
| 2452 | | - MCDI_SET_DWORD(inbuf, INIT_TXQ_IN_LABEL, tx_queue->queue); |
|---|
| 2453 | | - MCDI_SET_DWORD(inbuf, INIT_TXQ_IN_INSTANCE, tx_queue->queue); |
|---|
| 2454 | | - MCDI_SET_DWORD(inbuf, INIT_TXQ_IN_OWNER_ID, 0); |
|---|
| 2455 | | - MCDI_SET_DWORD(inbuf, INIT_TXQ_IN_PORT_ID, nic_data->vport_id); |
|---|
| 2456 | | - |
|---|
| 2457 | | - dma_addr = tx_queue->txd.buf.dma_addr; |
|---|
| 2458 | | - |
|---|
| 2459 | | - netif_dbg(efx, hw, efx->net_dev, "pushing TXQ %d. %zu entries (%llx)\n", |
|---|
| 2460 | | - tx_queue->queue, entries, (u64)dma_addr); |
|---|
| 2461 | | - |
|---|
| 2462 | | - for (i = 0; i < entries; ++i) { |
|---|
| 2463 | | - MCDI_SET_ARRAY_QWORD(inbuf, INIT_TXQ_IN_DMA_ADDR, i, dma_addr); |
|---|
| 2464 | | - dma_addr += EFX_BUF_SIZE; |
|---|
| 2465 | | - } |
|---|
| 2466 | | - |
|---|
| 2467 | | - inlen = MC_CMD_INIT_TXQ_IN_LEN(entries); |
|---|
| 2468 | | - |
|---|
| 2469 | | - do { |
|---|
| 2470 | | - MCDI_POPULATE_DWORD_4(inbuf, INIT_TXQ_IN_FLAGS, |
|---|
| 2471 | | - /* This flag was removed from mcdi_pcol.h for |
|---|
| 2472 | | - * the non-_EXT version of INIT_TXQ. However, |
|---|
| 2473 | | - * firmware still honours it. |
|---|
| 2474 | | - */ |
|---|
| 2475 | | - INIT_TXQ_EXT_IN_FLAG_TSOV2_EN, tso_v2, |
|---|
| 2476 | | - INIT_TXQ_IN_FLAG_IP_CSUM_DIS, !csum_offload, |
|---|
| 2477 | | - INIT_TXQ_IN_FLAG_TCP_CSUM_DIS, !csum_offload, |
|---|
| 2478 | | - INIT_TXQ_EXT_IN_FLAG_TIMESTAMP, |
|---|
| 2479 | | - tx_queue->timestamping); |
|---|
| 2480 | | - |
|---|
| 2481 | | - rc = efx_mcdi_rpc_quiet(efx, MC_CMD_INIT_TXQ, inbuf, inlen, |
|---|
| 2482 | | - NULL, 0, NULL); |
|---|
| 2483 | | - if (rc == -ENOSPC && tso_v2) { |
|---|
| 2484 | | - /* Retry without TSOv2 if we're short on contexts. */ |
|---|
| 2485 | | - tso_v2 = false; |
|---|
| 2486 | | - netif_warn(efx, probe, efx->net_dev, |
|---|
| 2487 | | - "TSOv2 context not available to segment in hardware. TCP performance may be reduced.\n"); |
|---|
| 2488 | | - } else if (rc) { |
|---|
| 2489 | | - efx_mcdi_display_error(efx, MC_CMD_INIT_TXQ, |
|---|
| 2490 | | - MC_CMD_INIT_TXQ_EXT_IN_LEN, |
|---|
| 2491 | | - NULL, 0, rc); |
|---|
| 2492 | | - goto fail; |
|---|
| 2336 | + if (efx_has_cap(efx, TX_TSO_V2)) { |
|---|
| 2337 | + if ((csum_offload || inner_csum) && |
|---|
| 2338 | + !tx_queue->timestamping && !tx_queue->xdp_tx) { |
|---|
| 2339 | + tx_queue->tso_version = 2; |
|---|
| 2340 | + netif_dbg(efx, hw, efx->net_dev, "Using TSOv2 for channel %u\n", |
|---|
| 2341 | + channel->channel); |
|---|
| 2493 | 2342 | } |
|---|
| 2494 | | - } while (rc); |
|---|
| 2343 | + } else if (efx_has_cap(efx, TX_TSO)) { |
|---|
| 2344 | + tx_queue->tso_version = 1; |
|---|
| 2345 | + } |
|---|
| 2346 | + |
|---|
| 2347 | + rc = efx_mcdi_tx_init(tx_queue); |
|---|
| 2348 | + if (rc) |
|---|
| 2349 | + goto fail; |
|---|
| 2495 | 2350 | |
|---|
| 2496 | 2351 | /* A previous user of this TX queue might have set us up the |
|---|
| 2497 | 2352 | * bomb by writing a descriptor to the TX push collector but |
|---|
| .. | .. |
|---|
| 2502 | 2357 | tx_queue->buffer[0].flags = EFX_TX_BUF_OPTION; |
|---|
| 2503 | 2358 | tx_queue->insert_count = 1; |
|---|
| 2504 | 2359 | txd = efx_tx_desc(tx_queue, 0); |
|---|
| 2505 | | - EFX_POPULATE_QWORD_5(*txd, |
|---|
| 2360 | + EFX_POPULATE_QWORD_7(*txd, |
|---|
| 2506 | 2361 | ESF_DZ_TX_DESC_IS_OPT, true, |
|---|
| 2507 | 2362 | ESF_DZ_TX_OPTION_TYPE, |
|---|
| 2508 | 2363 | ESE_DZ_TX_OPTION_DESC_CRC_CSUM, |
|---|
| 2509 | 2364 | ESF_DZ_TX_OPTION_UDP_TCP_CSUM, csum_offload, |
|---|
| 2510 | | - ESF_DZ_TX_OPTION_IP_CSUM, csum_offload, |
|---|
| 2365 | + ESF_DZ_TX_OPTION_IP_CSUM, csum_offload && tx_queue->tso_version != 2, |
|---|
| 2366 | + ESF_DZ_TX_OPTION_INNER_UDP_TCP_CSUM, inner_csum, |
|---|
| 2367 | + ESF_DZ_TX_OPTION_INNER_IP_CSUM, inner_csum && tx_queue->tso_version != 2, |
|---|
| 2511 | 2368 | ESF_DZ_TX_TIMESTAMP, tx_queue->timestamping); |
|---|
| 2512 | 2369 | tx_queue->write_count = 1; |
|---|
| 2513 | 2370 | |
|---|
| 2514 | | - if (tso_v2) { |
|---|
| 2515 | | - tx_queue->handle_tso = efx_ef10_tx_tso_desc; |
|---|
| 2516 | | - tx_queue->tso_version = 2; |
|---|
| 2517 | | - } else if (nic_data->datapath_caps & |
|---|
| 2518 | | - (1 << MC_CMD_GET_CAPABILITIES_OUT_TX_TSO_LBN)) { |
|---|
| 2519 | | - tx_queue->tso_version = 1; |
|---|
| 2520 | | - } |
|---|
| 2371 | + if (tx_queue->tso_version == 2 && efx_has_cap(efx, TX_TSO_V2_ENCAP)) |
|---|
| 2372 | + tx_queue->tso_encap = true; |
|---|
| 2521 | 2373 | |
|---|
| 2522 | 2374 | wmb(); |
|---|
| 2523 | 2375 | efx_ef10_push_tx_desc(tx_queue, txd); |
|---|
| .. | .. |
|---|
| 2527 | 2379 | fail: |
|---|
| 2528 | 2380 | netdev_WARN(efx->net_dev, "failed to initialise TXQ %d\n", |
|---|
| 2529 | 2381 | tx_queue->queue); |
|---|
| 2530 | | -} |
|---|
| 2531 | | - |
|---|
| 2532 | | -static void efx_ef10_tx_fini(struct efx_tx_queue *tx_queue) |
|---|
| 2533 | | -{ |
|---|
| 2534 | | - MCDI_DECLARE_BUF(inbuf, MC_CMD_FINI_TXQ_IN_LEN); |
|---|
| 2535 | | - MCDI_DECLARE_BUF_ERR(outbuf); |
|---|
| 2536 | | - struct efx_nic *efx = tx_queue->efx; |
|---|
| 2537 | | - size_t outlen; |
|---|
| 2538 | | - int rc; |
|---|
| 2539 | | - |
|---|
| 2540 | | - MCDI_SET_DWORD(inbuf, FINI_TXQ_IN_INSTANCE, |
|---|
| 2541 | | - tx_queue->queue); |
|---|
| 2542 | | - |
|---|
| 2543 | | - rc = efx_mcdi_rpc_quiet(efx, MC_CMD_FINI_TXQ, inbuf, sizeof(inbuf), |
|---|
| 2544 | | - outbuf, sizeof(outbuf), &outlen); |
|---|
| 2545 | | - |
|---|
| 2546 | | - if (rc && rc != -EALREADY) |
|---|
| 2547 | | - goto fail; |
|---|
| 2548 | | - |
|---|
| 2549 | | - return; |
|---|
| 2550 | | - |
|---|
| 2551 | | -fail: |
|---|
| 2552 | | - efx_mcdi_display_error(efx, MC_CMD_FINI_TXQ, MC_CMD_FINI_TXQ_IN_LEN, |
|---|
| 2553 | | - outbuf, outlen, rc); |
|---|
| 2554 | | -} |
|---|
| 2555 | | - |
|---|
| 2556 | | -static void efx_ef10_tx_remove(struct efx_tx_queue *tx_queue) |
|---|
| 2557 | | -{ |
|---|
| 2558 | | - efx_nic_free_buffer(tx_queue->efx, &tx_queue->txd.buf); |
|---|
| 2559 | 2382 | } |
|---|
| 2560 | 2383 | |
|---|
| 2561 | 2384 | /* This writes to the TX_DESC_WPTR; write pointer for TX descriptor ring */ |
|---|
| .. | .. |
|---|
| 2596 | 2419 | unsigned int write_ptr; |
|---|
| 2597 | 2420 | efx_qword_t *txd; |
|---|
| 2598 | 2421 | |
|---|
| 2599 | | - tx_queue->xmit_more_available = false; |
|---|
| 2422 | + tx_queue->xmit_pending = false; |
|---|
| 2600 | 2423 | if (unlikely(tx_queue->write_count == tx_queue->insert_count)) |
|---|
| 2601 | 2424 | return; |
|---|
| 2602 | 2425 | |
|---|
| .. | .. |
|---|
| 2636 | 2459 | } |
|---|
| 2637 | 2460 | } |
|---|
| 2638 | 2461 | |
|---|
| 2639 | | -#define RSS_MODE_HASH_ADDRS (1 << RSS_MODE_HASH_SRC_ADDR_LBN |\ |
|---|
| 2640 | | - 1 << RSS_MODE_HASH_DST_ADDR_LBN) |
|---|
| 2641 | | -#define RSS_MODE_HASH_PORTS (1 << RSS_MODE_HASH_SRC_PORT_LBN |\ |
|---|
| 2642 | | - 1 << RSS_MODE_HASH_DST_PORT_LBN) |
|---|
| 2643 | | -#define RSS_CONTEXT_FLAGS_DEFAULT (1 << MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_TOEPLITZ_IPV4_EN_LBN |\ |
|---|
| 2644 | | - 1 << MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_TOEPLITZ_TCPV4_EN_LBN |\ |
|---|
| 2645 | | - 1 << MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_TOEPLITZ_IPV6_EN_LBN |\ |
|---|
| 2646 | | - 1 << MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_TOEPLITZ_TCPV6_EN_LBN |\ |
|---|
| 2647 | | - (RSS_MODE_HASH_ADDRS | RSS_MODE_HASH_PORTS) << MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_TCP_IPV4_RSS_MODE_LBN |\ |
|---|
| 2648 | | - RSS_MODE_HASH_ADDRS << MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_UDP_IPV4_RSS_MODE_LBN |\ |
|---|
| 2649 | | - RSS_MODE_HASH_ADDRS << MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_OTHER_IPV4_RSS_MODE_LBN |\ |
|---|
| 2650 | | - (RSS_MODE_HASH_ADDRS | RSS_MODE_HASH_PORTS) << MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_TCP_IPV6_RSS_MODE_LBN |\ |
|---|
| 2651 | | - RSS_MODE_HASH_ADDRS << MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_UDP_IPV6_RSS_MODE_LBN |\ |
|---|
| 2652 | | - RSS_MODE_HASH_ADDRS << MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_OTHER_IPV6_RSS_MODE_LBN) |
|---|
| 2653 | | - |
|---|
| 2654 | | -static int efx_ef10_get_rss_flags(struct efx_nic *efx, u32 context, u32 *flags) |
|---|
| 2462 | +static int efx_ef10_probe_multicast_chaining(struct efx_nic *efx) |
|---|
| 2655 | 2463 | { |
|---|
| 2656 | | - /* Firmware had a bug (sfc bug 61952) where it would not actually |
|---|
| 2657 | | - * fill in the flags field in the response to MC_CMD_RSS_CONTEXT_GET_FLAGS. |
|---|
| 2658 | | - * This meant that it would always contain whatever was previously |
|---|
| 2659 | | - * in the MCDI buffer. Fortunately, all firmware versions with |
|---|
| 2660 | | - * this bug have the same default flags value for a newly-allocated |
|---|
| 2661 | | - * RSS context, and the only time we want to get the flags is just |
|---|
| 2662 | | - * after allocating. Moreover, the response has a 32-bit hole |
|---|
| 2663 | | - * where the context ID would be in the request, so we can use an |
|---|
| 2664 | | - * overlength buffer in the request and pre-fill the flags field |
|---|
| 2665 | | - * with what we believe the default to be. Thus if the firmware |
|---|
| 2666 | | - * has the bug, it will leave our pre-filled value in the flags |
|---|
| 2667 | | - * field of the response, and we will get the right answer. |
|---|
| 2668 | | - * |
|---|
| 2669 | | - * However, this does mean that this function should NOT be used if |
|---|
| 2670 | | - * the RSS context flags might not be their defaults - it is ONLY |
|---|
| 2671 | | - * reliably correct for a newly-allocated RSS context. |
|---|
| 2672 | | - */ |
|---|
| 2673 | | - MCDI_DECLARE_BUF(inbuf, MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_LEN); |
|---|
| 2674 | | - MCDI_DECLARE_BUF(outbuf, MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_LEN); |
|---|
| 2675 | | - size_t outlen; |
|---|
| 2676 | | - int rc; |
|---|
| 2677 | | - |
|---|
| 2678 | | - /* Check we have a hole for the context ID */ |
|---|
| 2679 | | - BUILD_BUG_ON(MC_CMD_RSS_CONTEXT_GET_FLAGS_IN_LEN != MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_FLAGS_OFST); |
|---|
| 2680 | | - MCDI_SET_DWORD(inbuf, RSS_CONTEXT_GET_FLAGS_IN_RSS_CONTEXT_ID, context); |
|---|
| 2681 | | - MCDI_SET_DWORD(inbuf, RSS_CONTEXT_GET_FLAGS_OUT_FLAGS, |
|---|
| 2682 | | - RSS_CONTEXT_FLAGS_DEFAULT); |
|---|
| 2683 | | - rc = efx_mcdi_rpc(efx, MC_CMD_RSS_CONTEXT_GET_FLAGS, inbuf, |
|---|
| 2684 | | - sizeof(inbuf), outbuf, sizeof(outbuf), &outlen); |
|---|
| 2685 | | - if (rc == 0) { |
|---|
| 2686 | | - if (outlen < MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_LEN) |
|---|
| 2687 | | - rc = -EIO; |
|---|
| 2688 | | - else |
|---|
| 2689 | | - *flags = MCDI_DWORD(outbuf, RSS_CONTEXT_GET_FLAGS_OUT_FLAGS); |
|---|
| 2690 | | - } |
|---|
| 2691 | | - return rc; |
|---|
| 2692 | | -} |
|---|
| 2693 | | - |
|---|
| 2694 | | -/* Attempt to enable 4-tuple UDP hashing on the specified RSS context. |
|---|
| 2695 | | - * If we fail, we just leave the RSS context at its default hash settings, |
|---|
| 2696 | | - * which is safe but may slightly reduce performance. |
|---|
| 2697 | | - * Defaults are 4-tuple for TCP and 2-tuple for UDP and other-IP, so we |
|---|
| 2698 | | - * just need to set the UDP ports flags (for both IP versions). |
|---|
| 2699 | | - */ |
|---|
| 2700 | | -static void efx_ef10_set_rss_flags(struct efx_nic *efx, |
|---|
| 2701 | | - struct efx_rss_context *ctx) |
|---|
| 2702 | | -{ |
|---|
| 2703 | | - MCDI_DECLARE_BUF(inbuf, MC_CMD_RSS_CONTEXT_SET_FLAGS_IN_LEN); |
|---|
| 2704 | | - u32 flags; |
|---|
| 2705 | | - |
|---|
| 2706 | | - BUILD_BUG_ON(MC_CMD_RSS_CONTEXT_SET_FLAGS_OUT_LEN != 0); |
|---|
| 2707 | | - |
|---|
| 2708 | | - if (efx_ef10_get_rss_flags(efx, ctx->context_id, &flags) != 0) |
|---|
| 2709 | | - return; |
|---|
| 2710 | | - MCDI_SET_DWORD(inbuf, RSS_CONTEXT_SET_FLAGS_IN_RSS_CONTEXT_ID, |
|---|
| 2711 | | - ctx->context_id); |
|---|
| 2712 | | - flags |= RSS_MODE_HASH_PORTS << MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_UDP_IPV4_RSS_MODE_LBN; |
|---|
| 2713 | | - flags |= RSS_MODE_HASH_PORTS << MC_CMD_RSS_CONTEXT_GET_FLAGS_OUT_UDP_IPV6_RSS_MODE_LBN; |
|---|
| 2714 | | - MCDI_SET_DWORD(inbuf, RSS_CONTEXT_SET_FLAGS_IN_FLAGS, flags); |
|---|
| 2715 | | - if (!efx_mcdi_rpc(efx, MC_CMD_RSS_CONTEXT_SET_FLAGS, inbuf, sizeof(inbuf), |
|---|
| 2716 | | - NULL, 0, NULL)) |
|---|
| 2717 | | - /* Succeeded, so UDP 4-tuple is now enabled */ |
|---|
| 2718 | | - ctx->rx_hash_udp_4tuple = true; |
|---|
| 2719 | | -} |
|---|
| 2720 | | - |
|---|
| 2721 | | -static int efx_ef10_alloc_rss_context(struct efx_nic *efx, bool exclusive, |
|---|
| 2722 | | - struct efx_rss_context *ctx, |
|---|
| 2723 | | - unsigned *context_size) |
|---|
| 2724 | | -{ |
|---|
| 2725 | | - MCDI_DECLARE_BUF(inbuf, MC_CMD_RSS_CONTEXT_ALLOC_IN_LEN); |
|---|
| 2726 | | - MCDI_DECLARE_BUF(outbuf, MC_CMD_RSS_CONTEXT_ALLOC_OUT_LEN); |
|---|
| 2727 | 2464 | struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 2728 | | - size_t outlen; |
|---|
| 2465 | + unsigned int enabled, implemented; |
|---|
| 2466 | + bool want_workaround_26807; |
|---|
| 2729 | 2467 | int rc; |
|---|
| 2730 | | - u32 alloc_type = exclusive ? |
|---|
| 2731 | | - MC_CMD_RSS_CONTEXT_ALLOC_IN_TYPE_EXCLUSIVE : |
|---|
| 2732 | | - MC_CMD_RSS_CONTEXT_ALLOC_IN_TYPE_SHARED; |
|---|
| 2733 | | - unsigned rss_spread = exclusive ? |
|---|
| 2734 | | - efx->rss_spread : |
|---|
| 2735 | | - min(rounddown_pow_of_two(efx->rss_spread), |
|---|
| 2736 | | - EFX_EF10_MAX_SHARED_RSS_CONTEXT_SIZE); |
|---|
| 2737 | 2468 | |
|---|
| 2738 | | - if (!exclusive && rss_spread == 1) { |
|---|
| 2739 | | - ctx->context_id = EFX_EF10_RSS_CONTEXT_INVALID; |
|---|
| 2740 | | - if (context_size) |
|---|
| 2741 | | - *context_size = 1; |
|---|
| 2469 | + rc = efx_mcdi_get_workarounds(efx, &implemented, &enabled); |
|---|
| 2470 | + if (rc == -ENOSYS) { |
|---|
| 2471 | + /* GET_WORKAROUNDS was implemented before this workaround, |
|---|
| 2472 | + * thus it must be unavailable in this firmware. |
|---|
| 2473 | + */ |
|---|
| 2474 | + nic_data->workaround_26807 = false; |
|---|
| 2742 | 2475 | return 0; |
|---|
| 2743 | 2476 | } |
|---|
| 2744 | | - |
|---|
| 2745 | | - if (nic_data->datapath_caps & |
|---|
| 2746 | | - 1 << MC_CMD_GET_CAPABILITIES_OUT_RX_RSS_LIMITED_LBN) |
|---|
| 2747 | | - return -EOPNOTSUPP; |
|---|
| 2748 | | - |
|---|
| 2749 | | - MCDI_SET_DWORD(inbuf, RSS_CONTEXT_ALLOC_IN_UPSTREAM_PORT_ID, |
|---|
| 2750 | | - nic_data->vport_id); |
|---|
| 2751 | | - MCDI_SET_DWORD(inbuf, RSS_CONTEXT_ALLOC_IN_TYPE, alloc_type); |
|---|
| 2752 | | - MCDI_SET_DWORD(inbuf, RSS_CONTEXT_ALLOC_IN_NUM_QUEUES, rss_spread); |
|---|
| 2753 | | - |
|---|
| 2754 | | - rc = efx_mcdi_rpc(efx, MC_CMD_RSS_CONTEXT_ALLOC, inbuf, sizeof(inbuf), |
|---|
| 2755 | | - outbuf, sizeof(outbuf), &outlen); |
|---|
| 2756 | | - if (rc != 0) |
|---|
| 2757 | | - return rc; |
|---|
| 2758 | | - |
|---|
| 2759 | | - if (outlen < MC_CMD_RSS_CONTEXT_ALLOC_OUT_LEN) |
|---|
| 2760 | | - return -EIO; |
|---|
| 2761 | | - |
|---|
| 2762 | | - ctx->context_id = MCDI_DWORD(outbuf, RSS_CONTEXT_ALLOC_OUT_RSS_CONTEXT_ID); |
|---|
| 2763 | | - |
|---|
| 2764 | | - if (context_size) |
|---|
| 2765 | | - *context_size = rss_spread; |
|---|
| 2766 | | - |
|---|
| 2767 | | - if (nic_data->datapath_caps & |
|---|
| 2768 | | - 1 << MC_CMD_GET_CAPABILITIES_OUT_ADDITIONAL_RSS_MODES_LBN) |
|---|
| 2769 | | - efx_ef10_set_rss_flags(efx, ctx); |
|---|
| 2770 | | - |
|---|
| 2771 | | - return 0; |
|---|
| 2772 | | -} |
|---|
| 2773 | | - |
|---|
| 2774 | | -static int efx_ef10_free_rss_context(struct efx_nic *efx, u32 context) |
|---|
| 2775 | | -{ |
|---|
| 2776 | | - MCDI_DECLARE_BUF(inbuf, MC_CMD_RSS_CONTEXT_FREE_IN_LEN); |
|---|
| 2777 | | - |
|---|
| 2778 | | - MCDI_SET_DWORD(inbuf, RSS_CONTEXT_FREE_IN_RSS_CONTEXT_ID, |
|---|
| 2779 | | - context); |
|---|
| 2780 | | - return efx_mcdi_rpc(efx, MC_CMD_RSS_CONTEXT_FREE, inbuf, sizeof(inbuf), |
|---|
| 2781 | | - NULL, 0, NULL); |
|---|
| 2782 | | -} |
|---|
| 2783 | | - |
|---|
| 2784 | | -static int efx_ef10_populate_rss_table(struct efx_nic *efx, u32 context, |
|---|
| 2785 | | - const u32 *rx_indir_table, const u8 *key) |
|---|
| 2786 | | -{ |
|---|
| 2787 | | - MCDI_DECLARE_BUF(tablebuf, MC_CMD_RSS_CONTEXT_SET_TABLE_IN_LEN); |
|---|
| 2788 | | - MCDI_DECLARE_BUF(keybuf, MC_CMD_RSS_CONTEXT_SET_KEY_IN_LEN); |
|---|
| 2789 | | - int i, rc; |
|---|
| 2790 | | - |
|---|
| 2791 | | - MCDI_SET_DWORD(tablebuf, RSS_CONTEXT_SET_TABLE_IN_RSS_CONTEXT_ID, |
|---|
| 2792 | | - context); |
|---|
| 2793 | | - BUILD_BUG_ON(ARRAY_SIZE(efx->rss_context.rx_indir_table) != |
|---|
| 2794 | | - MC_CMD_RSS_CONTEXT_SET_TABLE_IN_INDIRECTION_TABLE_LEN); |
|---|
| 2795 | | - |
|---|
| 2796 | | - /* This iterates over the length of efx->rss_context.rx_indir_table, but |
|---|
| 2797 | | - * copies bytes from rx_indir_table. That's because the latter is a |
|---|
| 2798 | | - * pointer rather than an array, but should have the same length. |
|---|
| 2799 | | - * The efx->rss_context.rx_hash_key loop below is similar. |
|---|
| 2800 | | - */ |
|---|
| 2801 | | - for (i = 0; i < ARRAY_SIZE(efx->rss_context.rx_indir_table); ++i) |
|---|
| 2802 | | - MCDI_PTR(tablebuf, |
|---|
| 2803 | | - RSS_CONTEXT_SET_TABLE_IN_INDIRECTION_TABLE)[i] = |
|---|
| 2804 | | - (u8) rx_indir_table[i]; |
|---|
| 2805 | | - |
|---|
| 2806 | | - rc = efx_mcdi_rpc(efx, MC_CMD_RSS_CONTEXT_SET_TABLE, tablebuf, |
|---|
| 2807 | | - sizeof(tablebuf), NULL, 0, NULL); |
|---|
| 2808 | | - if (rc != 0) |
|---|
| 2809 | | - return rc; |
|---|
| 2810 | | - |
|---|
| 2811 | | - MCDI_SET_DWORD(keybuf, RSS_CONTEXT_SET_KEY_IN_RSS_CONTEXT_ID, |
|---|
| 2812 | | - context); |
|---|
| 2813 | | - BUILD_BUG_ON(ARRAY_SIZE(efx->rss_context.rx_hash_key) != |
|---|
| 2814 | | - MC_CMD_RSS_CONTEXT_SET_KEY_IN_TOEPLITZ_KEY_LEN); |
|---|
| 2815 | | - for (i = 0; i < ARRAY_SIZE(efx->rss_context.rx_hash_key); ++i) |
|---|
| 2816 | | - MCDI_PTR(keybuf, RSS_CONTEXT_SET_KEY_IN_TOEPLITZ_KEY)[i] = key[i]; |
|---|
| 2817 | | - |
|---|
| 2818 | | - return efx_mcdi_rpc(efx, MC_CMD_RSS_CONTEXT_SET_KEY, keybuf, |
|---|
| 2819 | | - sizeof(keybuf), NULL, 0, NULL); |
|---|
| 2820 | | -} |
|---|
| 2821 | | - |
|---|
| 2822 | | -static void efx_ef10_rx_free_indir_table(struct efx_nic *efx) |
|---|
| 2823 | | -{ |
|---|
| 2824 | | - int rc; |
|---|
| 2825 | | - |
|---|
| 2826 | | - if (efx->rss_context.context_id != EFX_EF10_RSS_CONTEXT_INVALID) { |
|---|
| 2827 | | - rc = efx_ef10_free_rss_context(efx, efx->rss_context.context_id); |
|---|
| 2828 | | - WARN_ON(rc != 0); |
|---|
| 2829 | | - } |
|---|
| 2830 | | - efx->rss_context.context_id = EFX_EF10_RSS_CONTEXT_INVALID; |
|---|
| 2831 | | -} |
|---|
| 2832 | | - |
|---|
| 2833 | | -static int efx_ef10_rx_push_shared_rss_config(struct efx_nic *efx, |
|---|
| 2834 | | - unsigned *context_size) |
|---|
| 2835 | | -{ |
|---|
| 2836 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 2837 | | - int rc = efx_ef10_alloc_rss_context(efx, false, &efx->rss_context, |
|---|
| 2838 | | - context_size); |
|---|
| 2839 | | - |
|---|
| 2840 | | - if (rc != 0) |
|---|
| 2841 | | - return rc; |
|---|
| 2842 | | - |
|---|
| 2843 | | - nic_data->rx_rss_context_exclusive = false; |
|---|
| 2844 | | - efx_set_default_rx_indir_table(efx, &efx->rss_context); |
|---|
| 2845 | | - return 0; |
|---|
| 2846 | | -} |
|---|
| 2847 | | - |
|---|
| 2848 | | -static int efx_ef10_rx_push_exclusive_rss_config(struct efx_nic *efx, |
|---|
| 2849 | | - const u32 *rx_indir_table, |
|---|
| 2850 | | - const u8 *key) |
|---|
| 2851 | | -{ |
|---|
| 2852 | | - u32 old_rx_rss_context = efx->rss_context.context_id; |
|---|
| 2853 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 2854 | | - int rc; |
|---|
| 2855 | | - |
|---|
| 2856 | | - if (efx->rss_context.context_id == EFX_EF10_RSS_CONTEXT_INVALID || |
|---|
| 2857 | | - !nic_data->rx_rss_context_exclusive) { |
|---|
| 2858 | | - rc = efx_ef10_alloc_rss_context(efx, true, &efx->rss_context, |
|---|
| 2859 | | - NULL); |
|---|
| 2860 | | - if (rc == -EOPNOTSUPP) |
|---|
| 2861 | | - return rc; |
|---|
| 2862 | | - else if (rc != 0) |
|---|
| 2863 | | - goto fail1; |
|---|
| 2864 | | - } |
|---|
| 2865 | | - |
|---|
| 2866 | | - rc = efx_ef10_populate_rss_table(efx, efx->rss_context.context_id, |
|---|
| 2867 | | - rx_indir_table, key); |
|---|
| 2868 | | - if (rc != 0) |
|---|
| 2869 | | - goto fail2; |
|---|
| 2870 | | - |
|---|
| 2871 | | - if (efx->rss_context.context_id != old_rx_rss_context && |
|---|
| 2872 | | - old_rx_rss_context != EFX_EF10_RSS_CONTEXT_INVALID) |
|---|
| 2873 | | - WARN_ON(efx_ef10_free_rss_context(efx, old_rx_rss_context) != 0); |
|---|
| 2874 | | - nic_data->rx_rss_context_exclusive = true; |
|---|
| 2875 | | - if (rx_indir_table != efx->rss_context.rx_indir_table) |
|---|
| 2876 | | - memcpy(efx->rss_context.rx_indir_table, rx_indir_table, |
|---|
| 2877 | | - sizeof(efx->rss_context.rx_indir_table)); |
|---|
| 2878 | | - if (key != efx->rss_context.rx_hash_key) |
|---|
| 2879 | | - memcpy(efx->rss_context.rx_hash_key, key, |
|---|
| 2880 | | - efx->type->rx_hash_key_size); |
|---|
| 2881 | | - |
|---|
| 2882 | | - return 0; |
|---|
| 2883 | | - |
|---|
| 2884 | | -fail2: |
|---|
| 2885 | | - if (old_rx_rss_context != efx->rss_context.context_id) { |
|---|
| 2886 | | - WARN_ON(efx_ef10_free_rss_context(efx, efx->rss_context.context_id) != 0); |
|---|
| 2887 | | - efx->rss_context.context_id = old_rx_rss_context; |
|---|
| 2888 | | - } |
|---|
| 2889 | | -fail1: |
|---|
| 2890 | | - netif_err(efx, hw, efx->net_dev, "%s: failed rc=%d\n", __func__, rc); |
|---|
| 2891 | | - return rc; |
|---|
| 2892 | | -} |
|---|
| 2893 | | - |
|---|
| 2894 | | -static int efx_ef10_rx_push_rss_context_config(struct efx_nic *efx, |
|---|
| 2895 | | - struct efx_rss_context *ctx, |
|---|
| 2896 | | - const u32 *rx_indir_table, |
|---|
| 2897 | | - const u8 *key) |
|---|
| 2898 | | -{ |
|---|
| 2899 | | - int rc; |
|---|
| 2900 | | - |
|---|
| 2901 | | - WARN_ON(!mutex_is_locked(&efx->rss_lock)); |
|---|
| 2902 | | - |
|---|
| 2903 | | - if (ctx->context_id == EFX_EF10_RSS_CONTEXT_INVALID) { |
|---|
| 2904 | | - rc = efx_ef10_alloc_rss_context(efx, true, ctx, NULL); |
|---|
| 2905 | | - if (rc) |
|---|
| 2906 | | - return rc; |
|---|
| 2907 | | - } |
|---|
| 2908 | | - |
|---|
| 2909 | | - if (!rx_indir_table) /* Delete this context */ |
|---|
| 2910 | | - return efx_ef10_free_rss_context(efx, ctx->context_id); |
|---|
| 2911 | | - |
|---|
| 2912 | | - rc = efx_ef10_populate_rss_table(efx, ctx->context_id, |
|---|
| 2913 | | - rx_indir_table, key); |
|---|
| 2914 | 2477 | if (rc) |
|---|
| 2915 | 2478 | return rc; |
|---|
| 2479 | + want_workaround_26807 = |
|---|
| 2480 | + implemented & MC_CMD_GET_WORKAROUNDS_OUT_BUG26807; |
|---|
| 2481 | + nic_data->workaround_26807 = |
|---|
| 2482 | + !!(enabled & MC_CMD_GET_WORKAROUNDS_OUT_BUG26807); |
|---|
| 2916 | 2483 | |
|---|
| 2917 | | - memcpy(ctx->rx_indir_table, rx_indir_table, |
|---|
| 2918 | | - sizeof(efx->rss_context.rx_indir_table)); |
|---|
| 2919 | | - memcpy(ctx->rx_hash_key, key, efx->type->rx_hash_key_size); |
|---|
| 2484 | + if (want_workaround_26807 && !nic_data->workaround_26807) { |
|---|
| 2485 | + unsigned int flags; |
|---|
| 2920 | 2486 | |
|---|
| 2921 | | - return 0; |
|---|
| 2922 | | -} |
|---|
| 2487 | + rc = efx_mcdi_set_workaround(efx, |
|---|
| 2488 | + MC_CMD_WORKAROUND_BUG26807, |
|---|
| 2489 | + true, &flags); |
|---|
| 2490 | + if (!rc) { |
|---|
| 2491 | + if (flags & |
|---|
| 2492 | + 1 << MC_CMD_WORKAROUND_EXT_OUT_FLR_DONE_LBN) { |
|---|
| 2493 | + netif_info(efx, drv, efx->net_dev, |
|---|
| 2494 | + "other functions on NIC have been reset\n"); |
|---|
| 2923 | 2495 | |
|---|
| 2924 | | -static int efx_ef10_rx_pull_rss_context_config(struct efx_nic *efx, |
|---|
| 2925 | | - struct efx_rss_context *ctx) |
|---|
| 2926 | | -{ |
|---|
| 2927 | | - MCDI_DECLARE_BUF(inbuf, MC_CMD_RSS_CONTEXT_GET_TABLE_IN_LEN); |
|---|
| 2928 | | - MCDI_DECLARE_BUF(tablebuf, MC_CMD_RSS_CONTEXT_GET_TABLE_OUT_LEN); |
|---|
| 2929 | | - MCDI_DECLARE_BUF(keybuf, MC_CMD_RSS_CONTEXT_GET_KEY_OUT_LEN); |
|---|
| 2930 | | - size_t outlen; |
|---|
| 2931 | | - int rc, i; |
|---|
| 2932 | | - |
|---|
| 2933 | | - WARN_ON(!mutex_is_locked(&efx->rss_lock)); |
|---|
| 2934 | | - |
|---|
| 2935 | | - BUILD_BUG_ON(MC_CMD_RSS_CONTEXT_GET_TABLE_IN_LEN != |
|---|
| 2936 | | - MC_CMD_RSS_CONTEXT_GET_KEY_IN_LEN); |
|---|
| 2937 | | - |
|---|
| 2938 | | - if (ctx->context_id == EFX_EF10_RSS_CONTEXT_INVALID) |
|---|
| 2939 | | - return -ENOENT; |
|---|
| 2940 | | - |
|---|
| 2941 | | - MCDI_SET_DWORD(inbuf, RSS_CONTEXT_GET_TABLE_IN_RSS_CONTEXT_ID, |
|---|
| 2942 | | - ctx->context_id); |
|---|
| 2943 | | - BUILD_BUG_ON(ARRAY_SIZE(ctx->rx_indir_table) != |
|---|
| 2944 | | - MC_CMD_RSS_CONTEXT_GET_TABLE_OUT_INDIRECTION_TABLE_LEN); |
|---|
| 2945 | | - rc = efx_mcdi_rpc(efx, MC_CMD_RSS_CONTEXT_GET_TABLE, inbuf, sizeof(inbuf), |
|---|
| 2946 | | - tablebuf, sizeof(tablebuf), &outlen); |
|---|
| 2947 | | - if (rc != 0) |
|---|
| 2948 | | - return rc; |
|---|
| 2949 | | - |
|---|
| 2950 | | - if (WARN_ON(outlen != MC_CMD_RSS_CONTEXT_GET_TABLE_OUT_LEN)) |
|---|
| 2951 | | - return -EIO; |
|---|
| 2952 | | - |
|---|
| 2953 | | - for (i = 0; i < ARRAY_SIZE(ctx->rx_indir_table); i++) |
|---|
| 2954 | | - ctx->rx_indir_table[i] = MCDI_PTR(tablebuf, |
|---|
| 2955 | | - RSS_CONTEXT_GET_TABLE_OUT_INDIRECTION_TABLE)[i]; |
|---|
| 2956 | | - |
|---|
| 2957 | | - MCDI_SET_DWORD(inbuf, RSS_CONTEXT_GET_KEY_IN_RSS_CONTEXT_ID, |
|---|
| 2958 | | - ctx->context_id); |
|---|
| 2959 | | - BUILD_BUG_ON(ARRAY_SIZE(ctx->rx_hash_key) != |
|---|
| 2960 | | - MC_CMD_RSS_CONTEXT_SET_KEY_IN_TOEPLITZ_KEY_LEN); |
|---|
| 2961 | | - rc = efx_mcdi_rpc(efx, MC_CMD_RSS_CONTEXT_GET_KEY, inbuf, sizeof(inbuf), |
|---|
| 2962 | | - keybuf, sizeof(keybuf), &outlen); |
|---|
| 2963 | | - if (rc != 0) |
|---|
| 2964 | | - return rc; |
|---|
| 2965 | | - |
|---|
| 2966 | | - if (WARN_ON(outlen != MC_CMD_RSS_CONTEXT_GET_KEY_OUT_LEN)) |
|---|
| 2967 | | - return -EIO; |
|---|
| 2968 | | - |
|---|
| 2969 | | - for (i = 0; i < ARRAY_SIZE(ctx->rx_hash_key); ++i) |
|---|
| 2970 | | - ctx->rx_hash_key[i] = MCDI_PTR( |
|---|
| 2971 | | - keybuf, RSS_CONTEXT_GET_KEY_OUT_TOEPLITZ_KEY)[i]; |
|---|
| 2972 | | - |
|---|
| 2973 | | - return 0; |
|---|
| 2974 | | -} |
|---|
| 2975 | | - |
|---|
| 2976 | | -static int efx_ef10_rx_pull_rss_config(struct efx_nic *efx) |
|---|
| 2977 | | -{ |
|---|
| 2978 | | - int rc; |
|---|
| 2979 | | - |
|---|
| 2980 | | - mutex_lock(&efx->rss_lock); |
|---|
| 2981 | | - rc = efx_ef10_rx_pull_rss_context_config(efx, &efx->rss_context); |
|---|
| 2982 | | - mutex_unlock(&efx->rss_lock); |
|---|
| 2983 | | - return rc; |
|---|
| 2984 | | -} |
|---|
| 2985 | | - |
|---|
| 2986 | | -static void efx_ef10_rx_restore_rss_contexts(struct efx_nic *efx) |
|---|
| 2987 | | -{ |
|---|
| 2988 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 2989 | | - struct efx_rss_context *ctx; |
|---|
| 2990 | | - int rc; |
|---|
| 2991 | | - |
|---|
| 2992 | | - WARN_ON(!mutex_is_locked(&efx->rss_lock)); |
|---|
| 2993 | | - |
|---|
| 2994 | | - if (!nic_data->must_restore_rss_contexts) |
|---|
| 2995 | | - return; |
|---|
| 2996 | | - |
|---|
| 2997 | | - list_for_each_entry(ctx, &efx->rss_context.list, list) { |
|---|
| 2998 | | - /* previous NIC RSS context is gone */ |
|---|
| 2999 | | - ctx->context_id = EFX_EF10_RSS_CONTEXT_INVALID; |
|---|
| 3000 | | - /* so try to allocate a new one */ |
|---|
| 3001 | | - rc = efx_ef10_rx_push_rss_context_config(efx, ctx, |
|---|
| 3002 | | - ctx->rx_indir_table, |
|---|
| 3003 | | - ctx->rx_hash_key); |
|---|
| 3004 | | - if (rc) |
|---|
| 3005 | | - netif_warn(efx, probe, efx->net_dev, |
|---|
| 3006 | | - "failed to restore RSS context %u, rc=%d" |
|---|
| 3007 | | - "; RSS filters may fail to be applied\n", |
|---|
| 3008 | | - ctx->user_id, rc); |
|---|
| 3009 | | - } |
|---|
| 3010 | | - nic_data->must_restore_rss_contexts = false; |
|---|
| 3011 | | -} |
|---|
| 3012 | | - |
|---|
| 3013 | | -static int efx_ef10_pf_rx_push_rss_config(struct efx_nic *efx, bool user, |
|---|
| 3014 | | - const u32 *rx_indir_table, |
|---|
| 3015 | | - const u8 *key) |
|---|
| 3016 | | -{ |
|---|
| 3017 | | - int rc; |
|---|
| 3018 | | - |
|---|
| 3019 | | - if (efx->rss_spread == 1) |
|---|
| 3020 | | - return 0; |
|---|
| 3021 | | - |
|---|
| 3022 | | - if (!key) |
|---|
| 3023 | | - key = efx->rss_context.rx_hash_key; |
|---|
| 3024 | | - |
|---|
| 3025 | | - rc = efx_ef10_rx_push_exclusive_rss_config(efx, rx_indir_table, key); |
|---|
| 3026 | | - |
|---|
| 3027 | | - if (rc == -ENOBUFS && !user) { |
|---|
| 3028 | | - unsigned context_size; |
|---|
| 3029 | | - bool mismatch = false; |
|---|
| 3030 | | - size_t i; |
|---|
| 3031 | | - |
|---|
| 3032 | | - for (i = 0; |
|---|
| 3033 | | - i < ARRAY_SIZE(efx->rss_context.rx_indir_table) && !mismatch; |
|---|
| 3034 | | - i++) |
|---|
| 3035 | | - mismatch = rx_indir_table[i] != |
|---|
| 3036 | | - ethtool_rxfh_indir_default(i, efx->rss_spread); |
|---|
| 3037 | | - |
|---|
| 3038 | | - rc = efx_ef10_rx_push_shared_rss_config(efx, &context_size); |
|---|
| 3039 | | - if (rc == 0) { |
|---|
| 3040 | | - if (context_size != efx->rss_spread) |
|---|
| 3041 | | - netif_warn(efx, probe, efx->net_dev, |
|---|
| 3042 | | - "Could not allocate an exclusive RSS" |
|---|
| 3043 | | - " context; allocated a shared one of" |
|---|
| 3044 | | - " different size." |
|---|
| 3045 | | - " Wanted %u, got %u.\n", |
|---|
| 3046 | | - efx->rss_spread, context_size); |
|---|
| 3047 | | - else if (mismatch) |
|---|
| 3048 | | - netif_warn(efx, probe, efx->net_dev, |
|---|
| 3049 | | - "Could not allocate an exclusive RSS" |
|---|
| 3050 | | - " context; allocated a shared one but" |
|---|
| 3051 | | - " could not apply custom" |
|---|
| 3052 | | - " indirection.\n"); |
|---|
| 3053 | | - else |
|---|
| 3054 | | - netif_info(efx, probe, efx->net_dev, |
|---|
| 3055 | | - "Could not allocate an exclusive RSS" |
|---|
| 3056 | | - " context; allocated a shared one.\n"); |
|---|
| 2496 | + /* With MCFW v4.6.x and earlier, the |
|---|
| 2497 | + * boot count will have incremented, |
|---|
| 2498 | + * so re-read the warm_boot_count |
|---|
| 2499 | + * value now to ensure this function |
|---|
| 2500 | + * doesn't think it has changed next |
|---|
| 2501 | + * time it checks. |
|---|
| 2502 | + */ |
|---|
| 2503 | + rc = efx_ef10_get_warm_boot_count(efx); |
|---|
| 2504 | + if (rc >= 0) { |
|---|
| 2505 | + nic_data->warm_boot_count = rc; |
|---|
| 2506 | + rc = 0; |
|---|
| 2507 | + } |
|---|
| 2508 | + } |
|---|
| 2509 | + nic_data->workaround_26807 = true; |
|---|
| 2510 | + } else if (rc == -EPERM) { |
|---|
| 2511 | + rc = 0; |
|---|
| 3057 | 2512 | } |
|---|
| 3058 | 2513 | } |
|---|
| 3059 | 2514 | return rc; |
|---|
| 3060 | 2515 | } |
|---|
| 3061 | 2516 | |
|---|
| 3062 | | -static int efx_ef10_vf_rx_push_rss_config(struct efx_nic *efx, bool user, |
|---|
| 3063 | | - const u32 *rx_indir_table |
|---|
| 3064 | | - __attribute__ ((unused)), |
|---|
| 3065 | | - const u8 *key |
|---|
| 3066 | | - __attribute__ ((unused))) |
|---|
| 2517 | +static int efx_ef10_filter_table_probe(struct efx_nic *efx) |
|---|
| 3067 | 2518 | { |
|---|
| 3068 | | - if (user) |
|---|
| 3069 | | - return -EOPNOTSUPP; |
|---|
| 3070 | | - if (efx->rss_context.context_id != EFX_EF10_RSS_CONTEXT_INVALID) |
|---|
| 3071 | | - return 0; |
|---|
| 3072 | | - return efx_ef10_rx_push_shared_rss_config(efx, NULL); |
|---|
| 3073 | | -} |
|---|
| 3074 | | - |
|---|
| 3075 | | -static int efx_ef10_rx_probe(struct efx_rx_queue *rx_queue) |
|---|
| 3076 | | -{ |
|---|
| 3077 | | - return efx_nic_alloc_buffer(rx_queue->efx, &rx_queue->rxd.buf, |
|---|
| 3078 | | - (rx_queue->ptr_mask + 1) * |
|---|
| 3079 | | - sizeof(efx_qword_t), |
|---|
| 3080 | | - GFP_KERNEL); |
|---|
| 3081 | | -} |
|---|
| 3082 | | - |
|---|
| 3083 | | -static void efx_ef10_rx_init(struct efx_rx_queue *rx_queue) |
|---|
| 3084 | | -{ |
|---|
| 3085 | | - MCDI_DECLARE_BUF(inbuf, |
|---|
| 3086 | | - MC_CMD_INIT_RXQ_IN_LEN(EFX_MAX_DMAQ_SIZE * 8 / |
|---|
| 3087 | | - EFX_BUF_SIZE)); |
|---|
| 3088 | | - struct efx_channel *channel = efx_rx_queue_channel(rx_queue); |
|---|
| 3089 | | - size_t entries = rx_queue->rxd.buf.len / EFX_BUF_SIZE; |
|---|
| 3090 | | - struct efx_nic *efx = rx_queue->efx; |
|---|
| 3091 | 2519 | struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 3092 | | - size_t inlen; |
|---|
| 3093 | | - dma_addr_t dma_addr; |
|---|
| 3094 | | - int rc; |
|---|
| 3095 | | - int i; |
|---|
| 3096 | | - BUILD_BUG_ON(MC_CMD_INIT_RXQ_OUT_LEN != 0); |
|---|
| 2520 | + int rc = efx_ef10_probe_multicast_chaining(efx); |
|---|
| 2521 | + struct efx_mcdi_filter_vlan *vlan; |
|---|
| 3097 | 2522 | |
|---|
| 3098 | | - rx_queue->scatter_n = 0; |
|---|
| 3099 | | - rx_queue->scatter_len = 0; |
|---|
| 3100 | | - |
|---|
| 3101 | | - MCDI_SET_DWORD(inbuf, INIT_RXQ_IN_SIZE, rx_queue->ptr_mask + 1); |
|---|
| 3102 | | - MCDI_SET_DWORD(inbuf, INIT_RXQ_IN_TARGET_EVQ, channel->channel); |
|---|
| 3103 | | - MCDI_SET_DWORD(inbuf, INIT_RXQ_IN_LABEL, efx_rx_queue_index(rx_queue)); |
|---|
| 3104 | | - MCDI_SET_DWORD(inbuf, INIT_RXQ_IN_INSTANCE, |
|---|
| 3105 | | - efx_rx_queue_index(rx_queue)); |
|---|
| 3106 | | - MCDI_POPULATE_DWORD_2(inbuf, INIT_RXQ_IN_FLAGS, |
|---|
| 3107 | | - INIT_RXQ_IN_FLAG_PREFIX, 1, |
|---|
| 3108 | | - INIT_RXQ_IN_FLAG_TIMESTAMP, 1); |
|---|
| 3109 | | - MCDI_SET_DWORD(inbuf, INIT_RXQ_IN_OWNER_ID, 0); |
|---|
| 3110 | | - MCDI_SET_DWORD(inbuf, INIT_RXQ_IN_PORT_ID, nic_data->vport_id); |
|---|
| 3111 | | - |
|---|
| 3112 | | - dma_addr = rx_queue->rxd.buf.dma_addr; |
|---|
| 3113 | | - |
|---|
| 3114 | | - netif_dbg(efx, hw, efx->net_dev, "pushing RXQ %d. %zu entries (%llx)\n", |
|---|
| 3115 | | - efx_rx_queue_index(rx_queue), entries, (u64)dma_addr); |
|---|
| 3116 | | - |
|---|
| 3117 | | - for (i = 0; i < entries; ++i) { |
|---|
| 3118 | | - MCDI_SET_ARRAY_QWORD(inbuf, INIT_RXQ_IN_DMA_ADDR, i, dma_addr); |
|---|
| 3119 | | - dma_addr += EFX_BUF_SIZE; |
|---|
| 3120 | | - } |
|---|
| 3121 | | - |
|---|
| 3122 | | - inlen = MC_CMD_INIT_RXQ_IN_LEN(entries); |
|---|
| 3123 | | - |
|---|
| 3124 | | - rc = efx_mcdi_rpc(efx, MC_CMD_INIT_RXQ, inbuf, inlen, |
|---|
| 3125 | | - NULL, 0, NULL); |
|---|
| 3126 | 2523 | if (rc) |
|---|
| 3127 | | - netdev_WARN(efx->net_dev, "failed to initialise RXQ %d\n", |
|---|
| 3128 | | - efx_rx_queue_index(rx_queue)); |
|---|
| 3129 | | -} |
|---|
| 2524 | + return rc; |
|---|
| 2525 | + rc = efx_mcdi_filter_table_probe(efx, nic_data->workaround_26807); |
|---|
| 3130 | 2526 | |
|---|
| 3131 | | -static void efx_ef10_rx_fini(struct efx_rx_queue *rx_queue) |
|---|
| 3132 | | -{ |
|---|
| 3133 | | - MCDI_DECLARE_BUF(inbuf, MC_CMD_FINI_RXQ_IN_LEN); |
|---|
| 3134 | | - MCDI_DECLARE_BUF_ERR(outbuf); |
|---|
| 3135 | | - struct efx_nic *efx = rx_queue->efx; |
|---|
| 3136 | | - size_t outlen; |
|---|
| 3137 | | - int rc; |
|---|
| 2527 | + if (rc) |
|---|
| 2528 | + return rc; |
|---|
| 3138 | 2529 | |
|---|
| 3139 | | - MCDI_SET_DWORD(inbuf, FINI_RXQ_IN_INSTANCE, |
|---|
| 3140 | | - efx_rx_queue_index(rx_queue)); |
|---|
| 2530 | + list_for_each_entry(vlan, &nic_data->vlan_list, list) { |
|---|
| 2531 | + rc = efx_mcdi_filter_add_vlan(efx, vlan->vid); |
|---|
| 2532 | + if (rc) |
|---|
| 2533 | + goto fail_add_vlan; |
|---|
| 2534 | + } |
|---|
| 2535 | + return 0; |
|---|
| 3141 | 2536 | |
|---|
| 3142 | | - rc = efx_mcdi_rpc_quiet(efx, MC_CMD_FINI_RXQ, inbuf, sizeof(inbuf), |
|---|
| 3143 | | - outbuf, sizeof(outbuf), &outlen); |
|---|
| 3144 | | - |
|---|
| 3145 | | - if (rc && rc != -EALREADY) |
|---|
| 3146 | | - goto fail; |
|---|
| 3147 | | - |
|---|
| 3148 | | - return; |
|---|
| 3149 | | - |
|---|
| 3150 | | -fail: |
|---|
| 3151 | | - efx_mcdi_display_error(efx, MC_CMD_FINI_RXQ, MC_CMD_FINI_RXQ_IN_LEN, |
|---|
| 3152 | | - outbuf, outlen, rc); |
|---|
| 3153 | | -} |
|---|
| 3154 | | - |
|---|
| 3155 | | -static void efx_ef10_rx_remove(struct efx_rx_queue *rx_queue) |
|---|
| 3156 | | -{ |
|---|
| 3157 | | - efx_nic_free_buffer(rx_queue->efx, &rx_queue->rxd.buf); |
|---|
| 2537 | +fail_add_vlan: |
|---|
| 2538 | + efx_mcdi_filter_table_remove(efx); |
|---|
| 2539 | + return rc; |
|---|
| 3158 | 2540 | } |
|---|
| 3159 | 2541 | |
|---|
| 3160 | 2542 | /* This creates an entry in the RX descriptor queue */ |
|---|
| .. | .. |
|---|
| 3228 | 2610 | /* nothing to do */ |
|---|
| 3229 | 2611 | } |
|---|
| 3230 | 2612 | |
|---|
| 3231 | | -static int efx_ef10_ev_probe(struct efx_channel *channel) |
|---|
| 3232 | | -{ |
|---|
| 3233 | | - return efx_nic_alloc_buffer(channel->efx, &channel->eventq.buf, |
|---|
| 3234 | | - (channel->eventq_mask + 1) * |
|---|
| 3235 | | - sizeof(efx_qword_t), |
|---|
| 3236 | | - GFP_KERNEL); |
|---|
| 3237 | | -} |
|---|
| 3238 | | - |
|---|
| 3239 | | -static void efx_ef10_ev_fini(struct efx_channel *channel) |
|---|
| 3240 | | -{ |
|---|
| 3241 | | - MCDI_DECLARE_BUF(inbuf, MC_CMD_FINI_EVQ_IN_LEN); |
|---|
| 3242 | | - MCDI_DECLARE_BUF_ERR(outbuf); |
|---|
| 3243 | | - struct efx_nic *efx = channel->efx; |
|---|
| 3244 | | - size_t outlen; |
|---|
| 3245 | | - int rc; |
|---|
| 3246 | | - |
|---|
| 3247 | | - MCDI_SET_DWORD(inbuf, FINI_EVQ_IN_INSTANCE, channel->channel); |
|---|
| 3248 | | - |
|---|
| 3249 | | - rc = efx_mcdi_rpc_quiet(efx, MC_CMD_FINI_EVQ, inbuf, sizeof(inbuf), |
|---|
| 3250 | | - outbuf, sizeof(outbuf), &outlen); |
|---|
| 3251 | | - |
|---|
| 3252 | | - if (rc && rc != -EALREADY) |
|---|
| 3253 | | - goto fail; |
|---|
| 3254 | | - |
|---|
| 3255 | | - return; |
|---|
| 3256 | | - |
|---|
| 3257 | | -fail: |
|---|
| 3258 | | - efx_mcdi_display_error(efx, MC_CMD_FINI_EVQ, MC_CMD_FINI_EVQ_IN_LEN, |
|---|
| 3259 | | - outbuf, outlen, rc); |
|---|
| 3260 | | -} |
|---|
| 3261 | | - |
|---|
| 3262 | 2613 | static int efx_ef10_ev_init(struct efx_channel *channel) |
|---|
| 3263 | 2614 | { |
|---|
| 3264 | | - MCDI_DECLARE_BUF(inbuf, |
|---|
| 3265 | | - MC_CMD_INIT_EVQ_V2_IN_LEN(EFX_MAX_EVQ_SIZE * 8 / |
|---|
| 3266 | | - EFX_BUF_SIZE)); |
|---|
| 3267 | | - MCDI_DECLARE_BUF(outbuf, MC_CMD_INIT_EVQ_V2_OUT_LEN); |
|---|
| 3268 | | - size_t entries = channel->eventq.buf.len / EFX_BUF_SIZE; |
|---|
| 3269 | 2615 | struct efx_nic *efx = channel->efx; |
|---|
| 3270 | 2616 | struct efx_ef10_nic_data *nic_data; |
|---|
| 3271 | | - size_t inlen, outlen; |
|---|
| 3272 | | - unsigned int enabled, implemented; |
|---|
| 3273 | | - dma_addr_t dma_addr; |
|---|
| 3274 | | - int rc; |
|---|
| 3275 | | - int i; |
|---|
| 2617 | + bool use_v2, cut_thru; |
|---|
| 3276 | 2618 | |
|---|
| 3277 | 2619 | nic_data = efx->nic_data; |
|---|
| 3278 | | - |
|---|
| 3279 | | - /* Fill event queue with all ones (i.e. empty events) */ |
|---|
| 3280 | | - memset(channel->eventq.buf.addr, 0xff, channel->eventq.buf.len); |
|---|
| 3281 | | - |
|---|
| 3282 | | - MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_SIZE, channel->eventq_mask + 1); |
|---|
| 3283 | | - MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_INSTANCE, channel->channel); |
|---|
| 3284 | | - /* INIT_EVQ expects index in vector table, not absolute */ |
|---|
| 3285 | | - MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_IRQ_NUM, channel->channel); |
|---|
| 3286 | | - MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_TMR_MODE, |
|---|
| 3287 | | - MC_CMD_INIT_EVQ_IN_TMR_MODE_DIS); |
|---|
| 3288 | | - MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_TMR_LOAD, 0); |
|---|
| 3289 | | - MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_TMR_RELOAD, 0); |
|---|
| 3290 | | - MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_COUNT_MODE, |
|---|
| 3291 | | - MC_CMD_INIT_EVQ_IN_COUNT_MODE_DIS); |
|---|
| 3292 | | - MCDI_SET_DWORD(inbuf, INIT_EVQ_IN_COUNT_THRSHLD, 0); |
|---|
| 3293 | | - |
|---|
| 3294 | | - if (nic_data->datapath_caps2 & |
|---|
| 3295 | | - 1 << MC_CMD_GET_CAPABILITIES_V2_OUT_INIT_EVQ_V2_LBN) { |
|---|
| 3296 | | - /* Use the new generic approach to specifying event queue |
|---|
| 3297 | | - * configuration, requesting lower latency or higher throughput. |
|---|
| 3298 | | - * The options that actually get used appear in the output. |
|---|
| 3299 | | - */ |
|---|
| 3300 | | - MCDI_POPULATE_DWORD_2(inbuf, INIT_EVQ_V2_IN_FLAGS, |
|---|
| 3301 | | - INIT_EVQ_V2_IN_FLAG_INTERRUPTING, 1, |
|---|
| 3302 | | - INIT_EVQ_V2_IN_FLAG_TYPE, |
|---|
| 3303 | | - MC_CMD_INIT_EVQ_V2_IN_FLAG_TYPE_AUTO); |
|---|
| 3304 | | - } else { |
|---|
| 3305 | | - bool cut_thru = !(nic_data->datapath_caps & |
|---|
| 3306 | | - 1 << MC_CMD_GET_CAPABILITIES_OUT_RX_BATCHING_LBN); |
|---|
| 3307 | | - |
|---|
| 3308 | | - MCDI_POPULATE_DWORD_4(inbuf, INIT_EVQ_IN_FLAGS, |
|---|
| 3309 | | - INIT_EVQ_IN_FLAG_INTERRUPTING, 1, |
|---|
| 3310 | | - INIT_EVQ_IN_FLAG_RX_MERGE, 1, |
|---|
| 3311 | | - INIT_EVQ_IN_FLAG_TX_MERGE, 1, |
|---|
| 3312 | | - INIT_EVQ_IN_FLAG_CUT_THRU, cut_thru); |
|---|
| 3313 | | - } |
|---|
| 3314 | | - |
|---|
| 3315 | | - dma_addr = channel->eventq.buf.dma_addr; |
|---|
| 3316 | | - for (i = 0; i < entries; ++i) { |
|---|
| 3317 | | - MCDI_SET_ARRAY_QWORD(inbuf, INIT_EVQ_IN_DMA_ADDR, i, dma_addr); |
|---|
| 3318 | | - dma_addr += EFX_BUF_SIZE; |
|---|
| 3319 | | - } |
|---|
| 3320 | | - |
|---|
| 3321 | | - inlen = MC_CMD_INIT_EVQ_IN_LEN(entries); |
|---|
| 3322 | | - |
|---|
| 3323 | | - rc = efx_mcdi_rpc(efx, MC_CMD_INIT_EVQ, inbuf, inlen, |
|---|
| 3324 | | - outbuf, sizeof(outbuf), &outlen); |
|---|
| 3325 | | - |
|---|
| 3326 | | - if (outlen >= MC_CMD_INIT_EVQ_V2_OUT_LEN) |
|---|
| 3327 | | - netif_dbg(efx, drv, efx->net_dev, |
|---|
| 3328 | | - "Channel %d using event queue flags %08x\n", |
|---|
| 3329 | | - channel->channel, |
|---|
| 3330 | | - MCDI_DWORD(outbuf, INIT_EVQ_V2_OUT_FLAGS)); |
|---|
| 3331 | | - |
|---|
| 3332 | | - /* IRQ return is ignored */ |
|---|
| 3333 | | - if (channel->channel || rc) |
|---|
| 3334 | | - return rc; |
|---|
| 3335 | | - |
|---|
| 3336 | | - /* Successfully created event queue on channel 0 */ |
|---|
| 3337 | | - rc = efx_mcdi_get_workarounds(efx, &implemented, &enabled); |
|---|
| 3338 | | - if (rc == -ENOSYS) { |
|---|
| 3339 | | - /* GET_WORKAROUNDS was implemented before this workaround, |
|---|
| 3340 | | - * thus it must be unavailable in this firmware. |
|---|
| 3341 | | - */ |
|---|
| 3342 | | - nic_data->workaround_26807 = false; |
|---|
| 3343 | | - rc = 0; |
|---|
| 3344 | | - } else if (rc) { |
|---|
| 3345 | | - goto fail; |
|---|
| 3346 | | - } else { |
|---|
| 3347 | | - nic_data->workaround_26807 = |
|---|
| 3348 | | - !!(enabled & MC_CMD_GET_WORKAROUNDS_OUT_BUG26807); |
|---|
| 3349 | | - |
|---|
| 3350 | | - if (implemented & MC_CMD_GET_WORKAROUNDS_OUT_BUG26807 && |
|---|
| 3351 | | - !nic_data->workaround_26807) { |
|---|
| 3352 | | - unsigned int flags; |
|---|
| 3353 | | - |
|---|
| 3354 | | - rc = efx_mcdi_set_workaround(efx, |
|---|
| 3355 | | - MC_CMD_WORKAROUND_BUG26807, |
|---|
| 3356 | | - true, &flags); |
|---|
| 3357 | | - |
|---|
| 3358 | | - if (!rc) { |
|---|
| 3359 | | - if (flags & |
|---|
| 3360 | | - 1 << MC_CMD_WORKAROUND_EXT_OUT_FLR_DONE_LBN) { |
|---|
| 3361 | | - netif_info(efx, drv, efx->net_dev, |
|---|
| 3362 | | - "other functions on NIC have been reset\n"); |
|---|
| 3363 | | - |
|---|
| 3364 | | - /* With MCFW v4.6.x and earlier, the |
|---|
| 3365 | | - * boot count will have incremented, |
|---|
| 3366 | | - * so re-read the warm_boot_count |
|---|
| 3367 | | - * value now to ensure this function |
|---|
| 3368 | | - * doesn't think it has changed next |
|---|
| 3369 | | - * time it checks. |
|---|
| 3370 | | - */ |
|---|
| 3371 | | - rc = efx_ef10_get_warm_boot_count(efx); |
|---|
| 3372 | | - if (rc >= 0) { |
|---|
| 3373 | | - nic_data->warm_boot_count = rc; |
|---|
| 3374 | | - rc = 0; |
|---|
| 3375 | | - } |
|---|
| 3376 | | - } |
|---|
| 3377 | | - nic_data->workaround_26807 = true; |
|---|
| 3378 | | - } else if (rc == -EPERM) { |
|---|
| 3379 | | - rc = 0; |
|---|
| 3380 | | - } |
|---|
| 3381 | | - } |
|---|
| 3382 | | - } |
|---|
| 3383 | | - |
|---|
| 3384 | | - if (!rc) |
|---|
| 3385 | | - return 0; |
|---|
| 3386 | | - |
|---|
| 3387 | | -fail: |
|---|
| 3388 | | - efx_ef10_ev_fini(channel); |
|---|
| 3389 | | - return rc; |
|---|
| 3390 | | -} |
|---|
| 3391 | | - |
|---|
| 3392 | | -static void efx_ef10_ev_remove(struct efx_channel *channel) |
|---|
| 3393 | | -{ |
|---|
| 3394 | | - efx_nic_free_buffer(channel->efx, &channel->eventq.buf); |
|---|
| 2620 | + use_v2 = nic_data->datapath_caps2 & |
|---|
| 2621 | + 1 << MC_CMD_GET_CAPABILITIES_V2_OUT_INIT_EVQ_V2_LBN; |
|---|
| 2622 | + cut_thru = !(nic_data->datapath_caps & |
|---|
| 2623 | + 1 << MC_CMD_GET_CAPABILITIES_OUT_RX_BATCHING_LBN); |
|---|
| 2624 | + return efx_mcdi_ev_init(channel, cut_thru, use_v2); |
|---|
| 3395 | 2625 | } |
|---|
| 3396 | 2626 | |
|---|
| 3397 | 2627 | static void efx_ef10_handle_rx_wrong_queue(struct efx_rx_queue *rx_queue, |
|---|
| .. | .. |
|---|
| 3701 | 2931 | |
|---|
| 3702 | 2932 | /* Get the transmit queue */ |
|---|
| 3703 | 2933 | tx_ev_q_label = EFX_QWORD_FIELD(*event, ESF_DZ_TX_QLABEL); |
|---|
| 3704 | | - tx_queue = efx_channel_get_tx_queue(channel, |
|---|
| 3705 | | - tx_ev_q_label % EFX_TXQ_TYPES); |
|---|
| 2934 | + tx_queue = channel->tx_queue + (tx_ev_q_label % EFX_MAX_TXQ_PER_CHANNEL); |
|---|
| 3706 | 2935 | |
|---|
| 3707 | 2936 | if (!tx_queue->timestamping) { |
|---|
| 3708 | 2937 | /* Transmit completion */ |
|---|
| .. | .. |
|---|
| 3712 | 2941 | } |
|---|
| 3713 | 2942 | |
|---|
| 3714 | 2943 | /* Transmit timestamps are only available for 8XXX series. They result |
|---|
| 3715 | | - * in three events per packet. These occur in order, and are: |
|---|
| 3716 | | - * - the normal completion event |
|---|
| 2944 | + * in up to three events per packet. These occur in order, and are: |
|---|
| 2945 | + * - the normal completion event (may be omitted) |
|---|
| 3717 | 2946 | * - the low part of the timestamp |
|---|
| 3718 | 2947 | * - the high part of the timestamp |
|---|
| 2948 | + * |
|---|
| 2949 | + * It's possible for multiple completion events to appear before the |
|---|
| 2950 | + * corresponding timestamps. So we can for example get: |
|---|
| 2951 | + * COMP N |
|---|
| 2952 | + * COMP N+1 |
|---|
| 2953 | + * TS_LO N |
|---|
| 2954 | + * TS_HI N |
|---|
| 2955 | + * TS_LO N+1 |
|---|
| 2956 | + * TS_HI N+1 |
|---|
| 2957 | + * |
|---|
| 2958 | + * In addition it's also possible for the adjacent completions to be |
|---|
| 2959 | + * merged, so we may not see COMP N above. As such, the completion |
|---|
| 2960 | + * events are not very useful here. |
|---|
| 3719 | 2961 | * |
|---|
| 3720 | 2962 | * Each part of the timestamp is itself split across two 16 bit |
|---|
| 3721 | 2963 | * fields in the event. |
|---|
| .. | .. |
|---|
| 3724 | 2966 | |
|---|
| 3725 | 2967 | switch (tx_ev_type) { |
|---|
| 3726 | 2968 | case TX_TIMESTAMP_EVENT_TX_EV_COMPLETION: |
|---|
| 3727 | | - /* In case of Queue flush or FLR, we might have received |
|---|
| 3728 | | - * the previous TX completion event but not the Timestamp |
|---|
| 3729 | | - * events. |
|---|
| 3730 | | - */ |
|---|
| 3731 | | - if (tx_queue->completed_desc_ptr != tx_queue->ptr_mask) |
|---|
| 3732 | | - efx_xmit_done(tx_queue, tx_queue->completed_desc_ptr); |
|---|
| 3733 | | - |
|---|
| 3734 | | - tx_ev_desc_ptr = EFX_QWORD_FIELD(*event, |
|---|
| 3735 | | - ESF_DZ_TX_DESCR_INDX); |
|---|
| 3736 | | - tx_queue->completed_desc_ptr = |
|---|
| 3737 | | - tx_ev_desc_ptr & tx_queue->ptr_mask; |
|---|
| 2969 | + /* Ignore this event - see above. */ |
|---|
| 3738 | 2970 | break; |
|---|
| 3739 | 2971 | |
|---|
| 3740 | 2972 | case TX_TIMESTAMP_EVENT_TX_EV_TSTAMP_LO: |
|---|
| .. | .. |
|---|
| 3746 | 2978 | ts_part = efx_ef10_extract_event_ts(event); |
|---|
| 3747 | 2979 | tx_queue->completed_timestamp_major = ts_part; |
|---|
| 3748 | 2980 | |
|---|
| 3749 | | - efx_xmit_done(tx_queue, tx_queue->completed_desc_ptr); |
|---|
| 3750 | | - tx_queue->completed_desc_ptr = tx_queue->ptr_mask; |
|---|
| 2981 | + efx_xmit_done_single(tx_queue); |
|---|
| 3751 | 2982 | break; |
|---|
| 3752 | 2983 | |
|---|
| 3753 | 2984 | default: |
|---|
| .. | .. |
|---|
| 3947 | 3178 | netif_err(efx, hw, efx->net_dev, "%s: failed rc=%d\n", __func__, rc); |
|---|
| 3948 | 3179 | } |
|---|
| 3949 | 3180 | |
|---|
| 3950 | | -void efx_ef10_handle_drain_event(struct efx_nic *efx) |
|---|
| 3951 | | -{ |
|---|
| 3952 | | - if (atomic_dec_and_test(&efx->active_queues)) |
|---|
| 3953 | | - wake_up(&efx->flush_wq); |
|---|
| 3954 | | - |
|---|
| 3955 | | - WARN_ON(atomic_read(&efx->active_queues) < 0); |
|---|
| 3956 | | -} |
|---|
| 3957 | | - |
|---|
| 3958 | | -static int efx_ef10_fini_dmaq(struct efx_nic *efx) |
|---|
| 3959 | | -{ |
|---|
| 3960 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 3961 | | - struct efx_channel *channel; |
|---|
| 3962 | | - struct efx_tx_queue *tx_queue; |
|---|
| 3963 | | - struct efx_rx_queue *rx_queue; |
|---|
| 3964 | | - int pending; |
|---|
| 3965 | | - |
|---|
| 3966 | | - /* If the MC has just rebooted, the TX/RX queues will have already been |
|---|
| 3967 | | - * torn down, but efx->active_queues needs to be set to zero. |
|---|
| 3968 | | - */ |
|---|
| 3969 | | - if (nic_data->must_realloc_vis) { |
|---|
| 3970 | | - atomic_set(&efx->active_queues, 0); |
|---|
| 3971 | | - return 0; |
|---|
| 3972 | | - } |
|---|
| 3973 | | - |
|---|
| 3974 | | - /* Do not attempt to write to the NIC during EEH recovery */ |
|---|
| 3975 | | - if (efx->state != STATE_RECOVERY) { |
|---|
| 3976 | | - efx_for_each_channel(channel, efx) { |
|---|
| 3977 | | - efx_for_each_channel_rx_queue(rx_queue, channel) |
|---|
| 3978 | | - efx_ef10_rx_fini(rx_queue); |
|---|
| 3979 | | - efx_for_each_channel_tx_queue(tx_queue, channel) |
|---|
| 3980 | | - efx_ef10_tx_fini(tx_queue); |
|---|
| 3981 | | - } |
|---|
| 3982 | | - |
|---|
| 3983 | | - wait_event_timeout(efx->flush_wq, |
|---|
| 3984 | | - atomic_read(&efx->active_queues) == 0, |
|---|
| 3985 | | - msecs_to_jiffies(EFX_MAX_FLUSH_TIME)); |
|---|
| 3986 | | - pending = atomic_read(&efx->active_queues); |
|---|
| 3987 | | - if (pending) { |
|---|
| 3988 | | - netif_err(efx, hw, efx->net_dev, "failed to flush %d queues\n", |
|---|
| 3989 | | - pending); |
|---|
| 3990 | | - return -ETIMEDOUT; |
|---|
| 3991 | | - } |
|---|
| 3992 | | - } |
|---|
| 3993 | | - |
|---|
| 3994 | | - return 0; |
|---|
| 3995 | | -} |
|---|
| 3996 | | - |
|---|
| 3997 | 3181 | static void efx_ef10_prepare_flr(struct efx_nic *efx) |
|---|
| 3998 | 3182 | { |
|---|
| 3999 | 3183 | atomic_set(&efx->active_queues, 0); |
|---|
| 4000 | | -} |
|---|
| 4001 | | - |
|---|
| 4002 | | -/* Decide whether a filter should be exclusive or else should allow |
|---|
| 4003 | | - * delivery to additional recipients. Currently we decide that |
|---|
| 4004 | | - * filters for specific local unicast MAC and IP addresses are |
|---|
| 4005 | | - * exclusive. |
|---|
| 4006 | | - */ |
|---|
| 4007 | | -static bool efx_ef10_filter_is_exclusive(const struct efx_filter_spec *spec) |
|---|
| 4008 | | -{ |
|---|
| 4009 | | - if (spec->match_flags & EFX_FILTER_MATCH_LOC_MAC && |
|---|
| 4010 | | - !is_multicast_ether_addr(spec->loc_mac)) |
|---|
| 4011 | | - return true; |
|---|
| 4012 | | - |
|---|
| 4013 | | - if ((spec->match_flags & |
|---|
| 4014 | | - (EFX_FILTER_MATCH_ETHER_TYPE | EFX_FILTER_MATCH_LOC_HOST)) == |
|---|
| 4015 | | - (EFX_FILTER_MATCH_ETHER_TYPE | EFX_FILTER_MATCH_LOC_HOST)) { |
|---|
| 4016 | | - if (spec->ether_type == htons(ETH_P_IP) && |
|---|
| 4017 | | - !ipv4_is_multicast(spec->loc_host[0])) |
|---|
| 4018 | | - return true; |
|---|
| 4019 | | - if (spec->ether_type == htons(ETH_P_IPV6) && |
|---|
| 4020 | | - ((const u8 *)spec->loc_host)[0] != 0xff) |
|---|
| 4021 | | - return true; |
|---|
| 4022 | | - } |
|---|
| 4023 | | - |
|---|
| 4024 | | - return false; |
|---|
| 4025 | | -} |
|---|
| 4026 | | - |
|---|
| 4027 | | -static struct efx_filter_spec * |
|---|
| 4028 | | -efx_ef10_filter_entry_spec(const struct efx_ef10_filter_table *table, |
|---|
| 4029 | | - unsigned int filter_idx) |
|---|
| 4030 | | -{ |
|---|
| 4031 | | - return (struct efx_filter_spec *)(table->entry[filter_idx].spec & |
|---|
| 4032 | | - ~EFX_EF10_FILTER_FLAGS); |
|---|
| 4033 | | -} |
|---|
| 4034 | | - |
|---|
| 4035 | | -static unsigned int |
|---|
| 4036 | | -efx_ef10_filter_entry_flags(const struct efx_ef10_filter_table *table, |
|---|
| 4037 | | - unsigned int filter_idx) |
|---|
| 4038 | | -{ |
|---|
| 4039 | | - return table->entry[filter_idx].spec & EFX_EF10_FILTER_FLAGS; |
|---|
| 4040 | | -} |
|---|
| 4041 | | - |
|---|
| 4042 | | -static void |
|---|
| 4043 | | -efx_ef10_filter_set_entry(struct efx_ef10_filter_table *table, |
|---|
| 4044 | | - unsigned int filter_idx, |
|---|
| 4045 | | - const struct efx_filter_spec *spec, |
|---|
| 4046 | | - unsigned int flags) |
|---|
| 4047 | | -{ |
|---|
| 4048 | | - table->entry[filter_idx].spec = (unsigned long)spec | flags; |
|---|
| 4049 | | -} |
|---|
| 4050 | | - |
|---|
| 4051 | | -static void |
|---|
| 4052 | | -efx_ef10_filter_push_prep_set_match_fields(struct efx_nic *efx, |
|---|
| 4053 | | - const struct efx_filter_spec *spec, |
|---|
| 4054 | | - efx_dword_t *inbuf) |
|---|
| 4055 | | -{ |
|---|
| 4056 | | - enum efx_encap_type encap_type = efx_filter_get_encap_type(spec); |
|---|
| 4057 | | - u32 match_fields = 0, uc_match, mc_match; |
|---|
| 4058 | | - |
|---|
| 4059 | | - MCDI_SET_DWORD(inbuf, FILTER_OP_IN_OP, |
|---|
| 4060 | | - efx_ef10_filter_is_exclusive(spec) ? |
|---|
| 4061 | | - MC_CMD_FILTER_OP_IN_OP_INSERT : |
|---|
| 4062 | | - MC_CMD_FILTER_OP_IN_OP_SUBSCRIBE); |
|---|
| 4063 | | - |
|---|
| 4064 | | - /* Convert match flags and values. Unlike almost |
|---|
| 4065 | | - * everything else in MCDI, these fields are in |
|---|
| 4066 | | - * network byte order. |
|---|
| 4067 | | - */ |
|---|
| 4068 | | -#define COPY_VALUE(value, mcdi_field) \ |
|---|
| 4069 | | - do { \ |
|---|
| 4070 | | - match_fields |= \ |
|---|
| 4071 | | - 1 << MC_CMD_FILTER_OP_IN_MATCH_ ## \ |
|---|
| 4072 | | - mcdi_field ## _LBN; \ |
|---|
| 4073 | | - BUILD_BUG_ON( \ |
|---|
| 4074 | | - MC_CMD_FILTER_OP_IN_ ## mcdi_field ## _LEN < \ |
|---|
| 4075 | | - sizeof(value)); \ |
|---|
| 4076 | | - memcpy(MCDI_PTR(inbuf, FILTER_OP_IN_ ## mcdi_field), \ |
|---|
| 4077 | | - &value, sizeof(value)); \ |
|---|
| 4078 | | - } while (0) |
|---|
| 4079 | | -#define COPY_FIELD(gen_flag, gen_field, mcdi_field) \ |
|---|
| 4080 | | - if (spec->match_flags & EFX_FILTER_MATCH_ ## gen_flag) { \ |
|---|
| 4081 | | - COPY_VALUE(spec->gen_field, mcdi_field); \ |
|---|
| 4082 | | - } |
|---|
| 4083 | | - /* Handle encap filters first. They will always be mismatch |
|---|
| 4084 | | - * (unknown UC or MC) filters |
|---|
| 4085 | | - */ |
|---|
| 4086 | | - if (encap_type) { |
|---|
| 4087 | | - /* ether_type and outer_ip_proto need to be variables |
|---|
| 4088 | | - * because COPY_VALUE wants to memcpy them |
|---|
| 4089 | | - */ |
|---|
| 4090 | | - __be16 ether_type = |
|---|
| 4091 | | - htons(encap_type & EFX_ENCAP_FLAG_IPV6 ? |
|---|
| 4092 | | - ETH_P_IPV6 : ETH_P_IP); |
|---|
| 4093 | | - u8 vni_type = MC_CMD_FILTER_OP_EXT_IN_VNI_TYPE_GENEVE; |
|---|
| 4094 | | - u8 outer_ip_proto; |
|---|
| 4095 | | - |
|---|
| 4096 | | - switch (encap_type & EFX_ENCAP_TYPES_MASK) { |
|---|
| 4097 | | - case EFX_ENCAP_TYPE_VXLAN: |
|---|
| 4098 | | - vni_type = MC_CMD_FILTER_OP_EXT_IN_VNI_TYPE_VXLAN; |
|---|
| 4099 | | - /* fallthrough */ |
|---|
| 4100 | | - case EFX_ENCAP_TYPE_GENEVE: |
|---|
| 4101 | | - COPY_VALUE(ether_type, ETHER_TYPE); |
|---|
| 4102 | | - outer_ip_proto = IPPROTO_UDP; |
|---|
| 4103 | | - COPY_VALUE(outer_ip_proto, IP_PROTO); |
|---|
| 4104 | | - /* We always need to set the type field, even |
|---|
| 4105 | | - * though we're not matching on the TNI. |
|---|
| 4106 | | - */ |
|---|
| 4107 | | - MCDI_POPULATE_DWORD_1(inbuf, |
|---|
| 4108 | | - FILTER_OP_EXT_IN_VNI_OR_VSID, |
|---|
| 4109 | | - FILTER_OP_EXT_IN_VNI_TYPE, |
|---|
| 4110 | | - vni_type); |
|---|
| 4111 | | - break; |
|---|
| 4112 | | - case EFX_ENCAP_TYPE_NVGRE: |
|---|
| 4113 | | - COPY_VALUE(ether_type, ETHER_TYPE); |
|---|
| 4114 | | - outer_ip_proto = IPPROTO_GRE; |
|---|
| 4115 | | - COPY_VALUE(outer_ip_proto, IP_PROTO); |
|---|
| 4116 | | - break; |
|---|
| 4117 | | - default: |
|---|
| 4118 | | - WARN_ON(1); |
|---|
| 4119 | | - } |
|---|
| 4120 | | - |
|---|
| 4121 | | - uc_match = MC_CMD_FILTER_OP_EXT_IN_MATCH_IFRM_UNKNOWN_UCAST_DST_LBN; |
|---|
| 4122 | | - mc_match = MC_CMD_FILTER_OP_EXT_IN_MATCH_IFRM_UNKNOWN_MCAST_DST_LBN; |
|---|
| 4123 | | - } else { |
|---|
| 4124 | | - uc_match = MC_CMD_FILTER_OP_EXT_IN_MATCH_UNKNOWN_UCAST_DST_LBN; |
|---|
| 4125 | | - mc_match = MC_CMD_FILTER_OP_EXT_IN_MATCH_UNKNOWN_MCAST_DST_LBN; |
|---|
| 4126 | | - } |
|---|
| 4127 | | - |
|---|
| 4128 | | - if (spec->match_flags & EFX_FILTER_MATCH_LOC_MAC_IG) |
|---|
| 4129 | | - match_fields |= |
|---|
| 4130 | | - is_multicast_ether_addr(spec->loc_mac) ? |
|---|
| 4131 | | - 1 << mc_match : |
|---|
| 4132 | | - 1 << uc_match; |
|---|
| 4133 | | - COPY_FIELD(REM_HOST, rem_host, SRC_IP); |
|---|
| 4134 | | - COPY_FIELD(LOC_HOST, loc_host, DST_IP); |
|---|
| 4135 | | - COPY_FIELD(REM_MAC, rem_mac, SRC_MAC); |
|---|
| 4136 | | - COPY_FIELD(REM_PORT, rem_port, SRC_PORT); |
|---|
| 4137 | | - COPY_FIELD(LOC_MAC, loc_mac, DST_MAC); |
|---|
| 4138 | | - COPY_FIELD(LOC_PORT, loc_port, DST_PORT); |
|---|
| 4139 | | - COPY_FIELD(ETHER_TYPE, ether_type, ETHER_TYPE); |
|---|
| 4140 | | - COPY_FIELD(INNER_VID, inner_vid, INNER_VLAN); |
|---|
| 4141 | | - COPY_FIELD(OUTER_VID, outer_vid, OUTER_VLAN); |
|---|
| 4142 | | - COPY_FIELD(IP_PROTO, ip_proto, IP_PROTO); |
|---|
| 4143 | | -#undef COPY_FIELD |
|---|
| 4144 | | -#undef COPY_VALUE |
|---|
| 4145 | | - MCDI_SET_DWORD(inbuf, FILTER_OP_IN_MATCH_FIELDS, |
|---|
| 4146 | | - match_fields); |
|---|
| 4147 | | -} |
|---|
| 4148 | | - |
|---|
| 4149 | | -static void efx_ef10_filter_push_prep(struct efx_nic *efx, |
|---|
| 4150 | | - const struct efx_filter_spec *spec, |
|---|
| 4151 | | - efx_dword_t *inbuf, u64 handle, |
|---|
| 4152 | | - struct efx_rss_context *ctx, |
|---|
| 4153 | | - bool replacing) |
|---|
| 4154 | | -{ |
|---|
| 4155 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 4156 | | - u32 flags = spec->flags; |
|---|
| 4157 | | - |
|---|
| 4158 | | - memset(inbuf, 0, MC_CMD_FILTER_OP_EXT_IN_LEN); |
|---|
| 4159 | | - |
|---|
| 4160 | | - /* If RSS filter, caller better have given us an RSS context */ |
|---|
| 4161 | | - if (flags & EFX_FILTER_FLAG_RX_RSS) { |
|---|
| 4162 | | - /* We don't have the ability to return an error, so we'll just |
|---|
| 4163 | | - * log a warning and disable RSS for the filter. |
|---|
| 4164 | | - */ |
|---|
| 4165 | | - if (WARN_ON_ONCE(!ctx)) |
|---|
| 4166 | | - flags &= ~EFX_FILTER_FLAG_RX_RSS; |
|---|
| 4167 | | - else if (WARN_ON_ONCE(ctx->context_id == EFX_EF10_RSS_CONTEXT_INVALID)) |
|---|
| 4168 | | - flags &= ~EFX_FILTER_FLAG_RX_RSS; |
|---|
| 4169 | | - } |
|---|
| 4170 | | - |
|---|
| 4171 | | - if (replacing) { |
|---|
| 4172 | | - MCDI_SET_DWORD(inbuf, FILTER_OP_IN_OP, |
|---|
| 4173 | | - MC_CMD_FILTER_OP_IN_OP_REPLACE); |
|---|
| 4174 | | - MCDI_SET_QWORD(inbuf, FILTER_OP_IN_HANDLE, handle); |
|---|
| 4175 | | - } else { |
|---|
| 4176 | | - efx_ef10_filter_push_prep_set_match_fields(efx, spec, inbuf); |
|---|
| 4177 | | - } |
|---|
| 4178 | | - |
|---|
| 4179 | | - MCDI_SET_DWORD(inbuf, FILTER_OP_IN_PORT_ID, nic_data->vport_id); |
|---|
| 4180 | | - MCDI_SET_DWORD(inbuf, FILTER_OP_IN_RX_DEST, |
|---|
| 4181 | | - spec->dmaq_id == EFX_FILTER_RX_DMAQ_ID_DROP ? |
|---|
| 4182 | | - MC_CMD_FILTER_OP_IN_RX_DEST_DROP : |
|---|
| 4183 | | - MC_CMD_FILTER_OP_IN_RX_DEST_HOST); |
|---|
| 4184 | | - MCDI_SET_DWORD(inbuf, FILTER_OP_IN_TX_DOMAIN, 0); |
|---|
| 4185 | | - MCDI_SET_DWORD(inbuf, FILTER_OP_IN_TX_DEST, |
|---|
| 4186 | | - MC_CMD_FILTER_OP_IN_TX_DEST_DEFAULT); |
|---|
| 4187 | | - MCDI_SET_DWORD(inbuf, FILTER_OP_IN_RX_QUEUE, |
|---|
| 4188 | | - spec->dmaq_id == EFX_FILTER_RX_DMAQ_ID_DROP ? |
|---|
| 4189 | | - 0 : spec->dmaq_id); |
|---|
| 4190 | | - MCDI_SET_DWORD(inbuf, FILTER_OP_IN_RX_MODE, |
|---|
| 4191 | | - (flags & EFX_FILTER_FLAG_RX_RSS) ? |
|---|
| 4192 | | - MC_CMD_FILTER_OP_IN_RX_MODE_RSS : |
|---|
| 4193 | | - MC_CMD_FILTER_OP_IN_RX_MODE_SIMPLE); |
|---|
| 4194 | | - if (flags & EFX_FILTER_FLAG_RX_RSS) |
|---|
| 4195 | | - MCDI_SET_DWORD(inbuf, FILTER_OP_IN_RX_CONTEXT, ctx->context_id); |
|---|
| 4196 | | -} |
|---|
| 4197 | | - |
|---|
| 4198 | | -static int efx_ef10_filter_push(struct efx_nic *efx, |
|---|
| 4199 | | - const struct efx_filter_spec *spec, u64 *handle, |
|---|
| 4200 | | - struct efx_rss_context *ctx, bool replacing) |
|---|
| 4201 | | -{ |
|---|
| 4202 | | - MCDI_DECLARE_BUF(inbuf, MC_CMD_FILTER_OP_EXT_IN_LEN); |
|---|
| 4203 | | - MCDI_DECLARE_BUF(outbuf, MC_CMD_FILTER_OP_EXT_OUT_LEN); |
|---|
| 4204 | | - int rc; |
|---|
| 4205 | | - |
|---|
| 4206 | | - efx_ef10_filter_push_prep(efx, spec, inbuf, *handle, ctx, replacing); |
|---|
| 4207 | | - rc = efx_mcdi_rpc(efx, MC_CMD_FILTER_OP, inbuf, sizeof(inbuf), |
|---|
| 4208 | | - outbuf, sizeof(outbuf), NULL); |
|---|
| 4209 | | - if (rc == 0) |
|---|
| 4210 | | - *handle = MCDI_QWORD(outbuf, FILTER_OP_OUT_HANDLE); |
|---|
| 4211 | | - if (rc == -ENOSPC) |
|---|
| 4212 | | - rc = -EBUSY; /* to match efx_farch_filter_insert() */ |
|---|
| 4213 | | - return rc; |
|---|
| 4214 | | -} |
|---|
| 4215 | | - |
|---|
| 4216 | | -static u32 efx_ef10_filter_mcdi_flags_from_spec(const struct efx_filter_spec *spec) |
|---|
| 4217 | | -{ |
|---|
| 4218 | | - enum efx_encap_type encap_type = efx_filter_get_encap_type(spec); |
|---|
| 4219 | | - unsigned int match_flags = spec->match_flags; |
|---|
| 4220 | | - unsigned int uc_match, mc_match; |
|---|
| 4221 | | - u32 mcdi_flags = 0; |
|---|
| 4222 | | - |
|---|
| 4223 | | -#define MAP_FILTER_TO_MCDI_FLAG(gen_flag, mcdi_field, encap) { \ |
|---|
| 4224 | | - unsigned int old_match_flags = match_flags; \ |
|---|
| 4225 | | - match_flags &= ~EFX_FILTER_MATCH_ ## gen_flag; \ |
|---|
| 4226 | | - if (match_flags != old_match_flags) \ |
|---|
| 4227 | | - mcdi_flags |= \ |
|---|
| 4228 | | - (1 << ((encap) ? \ |
|---|
| 4229 | | - MC_CMD_FILTER_OP_EXT_IN_MATCH_IFRM_ ## \ |
|---|
| 4230 | | - mcdi_field ## _LBN : \ |
|---|
| 4231 | | - MC_CMD_FILTER_OP_EXT_IN_MATCH_ ##\ |
|---|
| 4232 | | - mcdi_field ## _LBN)); \ |
|---|
| 4233 | | - } |
|---|
| 4234 | | - /* inner or outer based on encap type */ |
|---|
| 4235 | | - MAP_FILTER_TO_MCDI_FLAG(REM_HOST, SRC_IP, encap_type); |
|---|
| 4236 | | - MAP_FILTER_TO_MCDI_FLAG(LOC_HOST, DST_IP, encap_type); |
|---|
| 4237 | | - MAP_FILTER_TO_MCDI_FLAG(REM_MAC, SRC_MAC, encap_type); |
|---|
| 4238 | | - MAP_FILTER_TO_MCDI_FLAG(REM_PORT, SRC_PORT, encap_type); |
|---|
| 4239 | | - MAP_FILTER_TO_MCDI_FLAG(LOC_MAC, DST_MAC, encap_type); |
|---|
| 4240 | | - MAP_FILTER_TO_MCDI_FLAG(LOC_PORT, DST_PORT, encap_type); |
|---|
| 4241 | | - MAP_FILTER_TO_MCDI_FLAG(ETHER_TYPE, ETHER_TYPE, encap_type); |
|---|
| 4242 | | - MAP_FILTER_TO_MCDI_FLAG(IP_PROTO, IP_PROTO, encap_type); |
|---|
| 4243 | | - /* always outer */ |
|---|
| 4244 | | - MAP_FILTER_TO_MCDI_FLAG(INNER_VID, INNER_VLAN, false); |
|---|
| 4245 | | - MAP_FILTER_TO_MCDI_FLAG(OUTER_VID, OUTER_VLAN, false); |
|---|
| 4246 | | -#undef MAP_FILTER_TO_MCDI_FLAG |
|---|
| 4247 | | - |
|---|
| 4248 | | - /* special handling for encap type, and mismatch */ |
|---|
| 4249 | | - if (encap_type) { |
|---|
| 4250 | | - match_flags &= ~EFX_FILTER_MATCH_ENCAP_TYPE; |
|---|
| 4251 | | - mcdi_flags |= |
|---|
| 4252 | | - (1 << MC_CMD_FILTER_OP_EXT_IN_MATCH_ETHER_TYPE_LBN); |
|---|
| 4253 | | - mcdi_flags |= (1 << MC_CMD_FILTER_OP_EXT_IN_MATCH_IP_PROTO_LBN); |
|---|
| 4254 | | - |
|---|
| 4255 | | - uc_match = MC_CMD_FILTER_OP_EXT_IN_MATCH_IFRM_UNKNOWN_UCAST_DST_LBN; |
|---|
| 4256 | | - mc_match = MC_CMD_FILTER_OP_EXT_IN_MATCH_IFRM_UNKNOWN_MCAST_DST_LBN; |
|---|
| 4257 | | - } else { |
|---|
| 4258 | | - uc_match = MC_CMD_FILTER_OP_EXT_IN_MATCH_UNKNOWN_UCAST_DST_LBN; |
|---|
| 4259 | | - mc_match = MC_CMD_FILTER_OP_EXT_IN_MATCH_UNKNOWN_MCAST_DST_LBN; |
|---|
| 4260 | | - } |
|---|
| 4261 | | - |
|---|
| 4262 | | - if (match_flags & EFX_FILTER_MATCH_LOC_MAC_IG) { |
|---|
| 4263 | | - match_flags &= ~EFX_FILTER_MATCH_LOC_MAC_IG; |
|---|
| 4264 | | - mcdi_flags |= |
|---|
| 4265 | | - is_multicast_ether_addr(spec->loc_mac) ? |
|---|
| 4266 | | - 1 << mc_match : |
|---|
| 4267 | | - 1 << uc_match; |
|---|
| 4268 | | - } |
|---|
| 4269 | | - |
|---|
| 4270 | | - /* Did we map them all? */ |
|---|
| 4271 | | - WARN_ON_ONCE(match_flags); |
|---|
| 4272 | | - |
|---|
| 4273 | | - return mcdi_flags; |
|---|
| 4274 | | -} |
|---|
| 4275 | | - |
|---|
| 4276 | | -static int efx_ef10_filter_pri(struct efx_ef10_filter_table *table, |
|---|
| 4277 | | - const struct efx_filter_spec *spec) |
|---|
| 4278 | | -{ |
|---|
| 4279 | | - u32 mcdi_flags = efx_ef10_filter_mcdi_flags_from_spec(spec); |
|---|
| 4280 | | - unsigned int match_pri; |
|---|
| 4281 | | - |
|---|
| 4282 | | - for (match_pri = 0; |
|---|
| 4283 | | - match_pri < table->rx_match_count; |
|---|
| 4284 | | - match_pri++) |
|---|
| 4285 | | - if (table->rx_match_mcdi_flags[match_pri] == mcdi_flags) |
|---|
| 4286 | | - return match_pri; |
|---|
| 4287 | | - |
|---|
| 4288 | | - return -EPROTONOSUPPORT; |
|---|
| 4289 | | -} |
|---|
| 4290 | | - |
|---|
| 4291 | | -static s32 efx_ef10_filter_insert_locked(struct efx_nic *efx, |
|---|
| 4292 | | - struct efx_filter_spec *spec, |
|---|
| 4293 | | - bool replace_equal) |
|---|
| 4294 | | -{ |
|---|
| 4295 | | - DECLARE_BITMAP(mc_rem_map, EFX_EF10_FILTER_SEARCH_LIMIT); |
|---|
| 4296 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 4297 | | - struct efx_ef10_filter_table *table; |
|---|
| 4298 | | - struct efx_filter_spec *saved_spec; |
|---|
| 4299 | | - struct efx_rss_context *ctx = NULL; |
|---|
| 4300 | | - unsigned int match_pri, hash; |
|---|
| 4301 | | - unsigned int priv_flags; |
|---|
| 4302 | | - bool rss_locked = false; |
|---|
| 4303 | | - bool replacing = false; |
|---|
| 4304 | | - unsigned int depth, i; |
|---|
| 4305 | | - int ins_index = -1; |
|---|
| 4306 | | - DEFINE_WAIT(wait); |
|---|
| 4307 | | - bool is_mc_recip; |
|---|
| 4308 | | - s32 rc; |
|---|
| 4309 | | - |
|---|
| 4310 | | - WARN_ON(!rwsem_is_locked(&efx->filter_sem)); |
|---|
| 4311 | | - table = efx->filter_state; |
|---|
| 4312 | | - down_write(&table->lock); |
|---|
| 4313 | | - |
|---|
| 4314 | | - /* For now, only support RX filters */ |
|---|
| 4315 | | - if ((spec->flags & (EFX_FILTER_FLAG_RX | EFX_FILTER_FLAG_TX)) != |
|---|
| 4316 | | - EFX_FILTER_FLAG_RX) { |
|---|
| 4317 | | - rc = -EINVAL; |
|---|
| 4318 | | - goto out_unlock; |
|---|
| 4319 | | - } |
|---|
| 4320 | | - |
|---|
| 4321 | | - rc = efx_ef10_filter_pri(table, spec); |
|---|
| 4322 | | - if (rc < 0) |
|---|
| 4323 | | - goto out_unlock; |
|---|
| 4324 | | - match_pri = rc; |
|---|
| 4325 | | - |
|---|
| 4326 | | - hash = efx_filter_spec_hash(spec); |
|---|
| 4327 | | - is_mc_recip = efx_filter_is_mc_recipient(spec); |
|---|
| 4328 | | - if (is_mc_recip) |
|---|
| 4329 | | - bitmap_zero(mc_rem_map, EFX_EF10_FILTER_SEARCH_LIMIT); |
|---|
| 4330 | | - |
|---|
| 4331 | | - if (spec->flags & EFX_FILTER_FLAG_RX_RSS) { |
|---|
| 4332 | | - mutex_lock(&efx->rss_lock); |
|---|
| 4333 | | - rss_locked = true; |
|---|
| 4334 | | - if (spec->rss_context) |
|---|
| 4335 | | - ctx = efx_find_rss_context_entry(efx, spec->rss_context); |
|---|
| 4336 | | - else |
|---|
| 4337 | | - ctx = &efx->rss_context; |
|---|
| 4338 | | - if (!ctx) { |
|---|
| 4339 | | - rc = -ENOENT; |
|---|
| 4340 | | - goto out_unlock; |
|---|
| 4341 | | - } |
|---|
| 4342 | | - if (ctx->context_id == EFX_EF10_RSS_CONTEXT_INVALID) { |
|---|
| 4343 | | - rc = -EOPNOTSUPP; |
|---|
| 4344 | | - goto out_unlock; |
|---|
| 4345 | | - } |
|---|
| 4346 | | - } |
|---|
| 4347 | | - |
|---|
| 4348 | | - /* Find any existing filters with the same match tuple or |
|---|
| 4349 | | - * else a free slot to insert at. |
|---|
| 4350 | | - */ |
|---|
| 4351 | | - for (depth = 1; depth < EFX_EF10_FILTER_SEARCH_LIMIT; depth++) { |
|---|
| 4352 | | - i = (hash + depth) & (HUNT_FILTER_TBL_ROWS - 1); |
|---|
| 4353 | | - saved_spec = efx_ef10_filter_entry_spec(table, i); |
|---|
| 4354 | | - |
|---|
| 4355 | | - if (!saved_spec) { |
|---|
| 4356 | | - if (ins_index < 0) |
|---|
| 4357 | | - ins_index = i; |
|---|
| 4358 | | - } else if (efx_filter_spec_equal(spec, saved_spec)) { |
|---|
| 4359 | | - if (spec->priority < saved_spec->priority && |
|---|
| 4360 | | - spec->priority != EFX_FILTER_PRI_AUTO) { |
|---|
| 4361 | | - rc = -EPERM; |
|---|
| 4362 | | - goto out_unlock; |
|---|
| 4363 | | - } |
|---|
| 4364 | | - if (!is_mc_recip) { |
|---|
| 4365 | | - /* This is the only one */ |
|---|
| 4366 | | - if (spec->priority == |
|---|
| 4367 | | - saved_spec->priority && |
|---|
| 4368 | | - !replace_equal) { |
|---|
| 4369 | | - rc = -EEXIST; |
|---|
| 4370 | | - goto out_unlock; |
|---|
| 4371 | | - } |
|---|
| 4372 | | - ins_index = i; |
|---|
| 4373 | | - break; |
|---|
| 4374 | | - } else if (spec->priority > |
|---|
| 4375 | | - saved_spec->priority || |
|---|
| 4376 | | - (spec->priority == |
|---|
| 4377 | | - saved_spec->priority && |
|---|
| 4378 | | - replace_equal)) { |
|---|
| 4379 | | - if (ins_index < 0) |
|---|
| 4380 | | - ins_index = i; |
|---|
| 4381 | | - else |
|---|
| 4382 | | - __set_bit(depth, mc_rem_map); |
|---|
| 4383 | | - } |
|---|
| 4384 | | - } |
|---|
| 4385 | | - } |
|---|
| 4386 | | - |
|---|
| 4387 | | - /* Once we reach the maximum search depth, use the first suitable |
|---|
| 4388 | | - * slot, or return -EBUSY if there was none |
|---|
| 4389 | | - */ |
|---|
| 4390 | | - if (ins_index < 0) { |
|---|
| 4391 | | - rc = -EBUSY; |
|---|
| 4392 | | - goto out_unlock; |
|---|
| 4393 | | - } |
|---|
| 4394 | | - |
|---|
| 4395 | | - /* Create a software table entry if necessary. */ |
|---|
| 4396 | | - saved_spec = efx_ef10_filter_entry_spec(table, ins_index); |
|---|
| 4397 | | - if (saved_spec) { |
|---|
| 4398 | | - if (spec->priority == EFX_FILTER_PRI_AUTO && |
|---|
| 4399 | | - saved_spec->priority >= EFX_FILTER_PRI_AUTO) { |
|---|
| 4400 | | - /* Just make sure it won't be removed */ |
|---|
| 4401 | | - if (saved_spec->priority > EFX_FILTER_PRI_AUTO) |
|---|
| 4402 | | - saved_spec->flags |= EFX_FILTER_FLAG_RX_OVER_AUTO; |
|---|
| 4403 | | - table->entry[ins_index].spec &= |
|---|
| 4404 | | - ~EFX_EF10_FILTER_FLAG_AUTO_OLD; |
|---|
| 4405 | | - rc = ins_index; |
|---|
| 4406 | | - goto out_unlock; |
|---|
| 4407 | | - } |
|---|
| 4408 | | - replacing = true; |
|---|
| 4409 | | - priv_flags = efx_ef10_filter_entry_flags(table, ins_index); |
|---|
| 4410 | | - } else { |
|---|
| 4411 | | - saved_spec = kmalloc(sizeof(*spec), GFP_ATOMIC); |
|---|
| 4412 | | - if (!saved_spec) { |
|---|
| 4413 | | - rc = -ENOMEM; |
|---|
| 4414 | | - goto out_unlock; |
|---|
| 4415 | | - } |
|---|
| 4416 | | - *saved_spec = *spec; |
|---|
| 4417 | | - priv_flags = 0; |
|---|
| 4418 | | - } |
|---|
| 4419 | | - efx_ef10_filter_set_entry(table, ins_index, saved_spec, priv_flags); |
|---|
| 4420 | | - |
|---|
| 4421 | | - /* Actually insert the filter on the HW */ |
|---|
| 4422 | | - rc = efx_ef10_filter_push(efx, spec, &table->entry[ins_index].handle, |
|---|
| 4423 | | - ctx, replacing); |
|---|
| 4424 | | - |
|---|
| 4425 | | - if (rc == -EINVAL && nic_data->must_realloc_vis) |
|---|
| 4426 | | - /* The MC rebooted under us, causing it to reject our filter |
|---|
| 4427 | | - * insertion as pointing to an invalid VI (spec->dmaq_id). |
|---|
| 4428 | | - */ |
|---|
| 4429 | | - rc = -EAGAIN; |
|---|
| 4430 | | - |
|---|
| 4431 | | - /* Finalise the software table entry */ |
|---|
| 4432 | | - if (rc == 0) { |
|---|
| 4433 | | - if (replacing) { |
|---|
| 4434 | | - /* Update the fields that may differ */ |
|---|
| 4435 | | - if (saved_spec->priority == EFX_FILTER_PRI_AUTO) |
|---|
| 4436 | | - saved_spec->flags |= |
|---|
| 4437 | | - EFX_FILTER_FLAG_RX_OVER_AUTO; |
|---|
| 4438 | | - saved_spec->priority = spec->priority; |
|---|
| 4439 | | - saved_spec->flags &= EFX_FILTER_FLAG_RX_OVER_AUTO; |
|---|
| 4440 | | - saved_spec->flags |= spec->flags; |
|---|
| 4441 | | - saved_spec->rss_context = spec->rss_context; |
|---|
| 4442 | | - saved_spec->dmaq_id = spec->dmaq_id; |
|---|
| 4443 | | - } |
|---|
| 4444 | | - } else if (!replacing) { |
|---|
| 4445 | | - kfree(saved_spec); |
|---|
| 4446 | | - saved_spec = NULL; |
|---|
| 4447 | | - } else { |
|---|
| 4448 | | - /* We failed to replace, so the old filter is still present. |
|---|
| 4449 | | - * Roll back the software table to reflect this. In fact the |
|---|
| 4450 | | - * efx_ef10_filter_set_entry() call below will do the right |
|---|
| 4451 | | - * thing, so nothing extra is needed here. |
|---|
| 4452 | | - */ |
|---|
| 4453 | | - } |
|---|
| 4454 | | - efx_ef10_filter_set_entry(table, ins_index, saved_spec, priv_flags); |
|---|
| 4455 | | - |
|---|
| 4456 | | - /* Remove and finalise entries for lower-priority multicast |
|---|
| 4457 | | - * recipients |
|---|
| 4458 | | - */ |
|---|
| 4459 | | - if (is_mc_recip) { |
|---|
| 4460 | | - MCDI_DECLARE_BUF(inbuf, MC_CMD_FILTER_OP_EXT_IN_LEN); |
|---|
| 4461 | | - unsigned int depth, i; |
|---|
| 4462 | | - |
|---|
| 4463 | | - memset(inbuf, 0, sizeof(inbuf)); |
|---|
| 4464 | | - |
|---|
| 4465 | | - for (depth = 0; depth < EFX_EF10_FILTER_SEARCH_LIMIT; depth++) { |
|---|
| 4466 | | - if (!test_bit(depth, mc_rem_map)) |
|---|
| 4467 | | - continue; |
|---|
| 4468 | | - |
|---|
| 4469 | | - i = (hash + depth) & (HUNT_FILTER_TBL_ROWS - 1); |
|---|
| 4470 | | - saved_spec = efx_ef10_filter_entry_spec(table, i); |
|---|
| 4471 | | - priv_flags = efx_ef10_filter_entry_flags(table, i); |
|---|
| 4472 | | - |
|---|
| 4473 | | - if (rc == 0) { |
|---|
| 4474 | | - MCDI_SET_DWORD(inbuf, FILTER_OP_IN_OP, |
|---|
| 4475 | | - MC_CMD_FILTER_OP_IN_OP_UNSUBSCRIBE); |
|---|
| 4476 | | - MCDI_SET_QWORD(inbuf, FILTER_OP_IN_HANDLE, |
|---|
| 4477 | | - table->entry[i].handle); |
|---|
| 4478 | | - rc = efx_mcdi_rpc(efx, MC_CMD_FILTER_OP, |
|---|
| 4479 | | - inbuf, sizeof(inbuf), |
|---|
| 4480 | | - NULL, 0, NULL); |
|---|
| 4481 | | - } |
|---|
| 4482 | | - |
|---|
| 4483 | | - if (rc == 0) { |
|---|
| 4484 | | - kfree(saved_spec); |
|---|
| 4485 | | - saved_spec = NULL; |
|---|
| 4486 | | - priv_flags = 0; |
|---|
| 4487 | | - } |
|---|
| 4488 | | - efx_ef10_filter_set_entry(table, i, saved_spec, |
|---|
| 4489 | | - priv_flags); |
|---|
| 4490 | | - } |
|---|
| 4491 | | - } |
|---|
| 4492 | | - |
|---|
| 4493 | | - /* If successful, return the inserted filter ID */ |
|---|
| 4494 | | - if (rc == 0) |
|---|
| 4495 | | - rc = efx_ef10_make_filter_id(match_pri, ins_index); |
|---|
| 4496 | | - |
|---|
| 4497 | | -out_unlock: |
|---|
| 4498 | | - if (rss_locked) |
|---|
| 4499 | | - mutex_unlock(&efx->rss_lock); |
|---|
| 4500 | | - up_write(&table->lock); |
|---|
| 4501 | | - return rc; |
|---|
| 4502 | | -} |
|---|
| 4503 | | - |
|---|
| 4504 | | -static s32 efx_ef10_filter_insert(struct efx_nic *efx, |
|---|
| 4505 | | - struct efx_filter_spec *spec, |
|---|
| 4506 | | - bool replace_equal) |
|---|
| 4507 | | -{ |
|---|
| 4508 | | - s32 ret; |
|---|
| 4509 | | - |
|---|
| 4510 | | - down_read(&efx->filter_sem); |
|---|
| 4511 | | - ret = efx_ef10_filter_insert_locked(efx, spec, replace_equal); |
|---|
| 4512 | | - up_read(&efx->filter_sem); |
|---|
| 4513 | | - |
|---|
| 4514 | | - return ret; |
|---|
| 4515 | | -} |
|---|
| 4516 | | - |
|---|
| 4517 | | -static void efx_ef10_filter_update_rx_scatter(struct efx_nic *efx) |
|---|
| 4518 | | -{ |
|---|
| 4519 | | - /* no need to do anything here on EF10 */ |
|---|
| 4520 | | -} |
|---|
| 4521 | | - |
|---|
| 4522 | | -/* Remove a filter. |
|---|
| 4523 | | - * If !by_index, remove by ID |
|---|
| 4524 | | - * If by_index, remove by index |
|---|
| 4525 | | - * Filter ID may come from userland and must be range-checked. |
|---|
| 4526 | | - * Caller must hold efx->filter_sem for read, and efx->filter_state->lock |
|---|
| 4527 | | - * for write. |
|---|
| 4528 | | - */ |
|---|
| 4529 | | -static int efx_ef10_filter_remove_internal(struct efx_nic *efx, |
|---|
| 4530 | | - unsigned int priority_mask, |
|---|
| 4531 | | - u32 filter_id, bool by_index) |
|---|
| 4532 | | -{ |
|---|
| 4533 | | - unsigned int filter_idx = efx_ef10_filter_get_unsafe_id(filter_id); |
|---|
| 4534 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 4535 | | - MCDI_DECLARE_BUF(inbuf, |
|---|
| 4536 | | - MC_CMD_FILTER_OP_IN_HANDLE_OFST + |
|---|
| 4537 | | - MC_CMD_FILTER_OP_IN_HANDLE_LEN); |
|---|
| 4538 | | - struct efx_filter_spec *spec; |
|---|
| 4539 | | - DEFINE_WAIT(wait); |
|---|
| 4540 | | - int rc; |
|---|
| 4541 | | - |
|---|
| 4542 | | - spec = efx_ef10_filter_entry_spec(table, filter_idx); |
|---|
| 4543 | | - if (!spec || |
|---|
| 4544 | | - (!by_index && |
|---|
| 4545 | | - efx_ef10_filter_pri(table, spec) != |
|---|
| 4546 | | - efx_ef10_filter_get_unsafe_pri(filter_id))) |
|---|
| 4547 | | - return -ENOENT; |
|---|
| 4548 | | - |
|---|
| 4549 | | - if (spec->flags & EFX_FILTER_FLAG_RX_OVER_AUTO && |
|---|
| 4550 | | - priority_mask == (1U << EFX_FILTER_PRI_AUTO)) { |
|---|
| 4551 | | - /* Just remove flags */ |
|---|
| 4552 | | - spec->flags &= ~EFX_FILTER_FLAG_RX_OVER_AUTO; |
|---|
| 4553 | | - table->entry[filter_idx].spec &= ~EFX_EF10_FILTER_FLAG_AUTO_OLD; |
|---|
| 4554 | | - return 0; |
|---|
| 4555 | | - } |
|---|
| 4556 | | - |
|---|
| 4557 | | - if (!(priority_mask & (1U << spec->priority))) |
|---|
| 4558 | | - return -ENOENT; |
|---|
| 4559 | | - |
|---|
| 4560 | | - if (spec->flags & EFX_FILTER_FLAG_RX_OVER_AUTO) { |
|---|
| 4561 | | - /* Reset to an automatic filter */ |
|---|
| 4562 | | - |
|---|
| 4563 | | - struct efx_filter_spec new_spec = *spec; |
|---|
| 4564 | | - |
|---|
| 4565 | | - new_spec.priority = EFX_FILTER_PRI_AUTO; |
|---|
| 4566 | | - new_spec.flags = (EFX_FILTER_FLAG_RX | |
|---|
| 4567 | | - (efx_rss_active(&efx->rss_context) ? |
|---|
| 4568 | | - EFX_FILTER_FLAG_RX_RSS : 0)); |
|---|
| 4569 | | - new_spec.dmaq_id = 0; |
|---|
| 4570 | | - new_spec.rss_context = 0; |
|---|
| 4571 | | - rc = efx_ef10_filter_push(efx, &new_spec, |
|---|
| 4572 | | - &table->entry[filter_idx].handle, |
|---|
| 4573 | | - &efx->rss_context, |
|---|
| 4574 | | - true); |
|---|
| 4575 | | - |
|---|
| 4576 | | - if (rc == 0) |
|---|
| 4577 | | - *spec = new_spec; |
|---|
| 4578 | | - } else { |
|---|
| 4579 | | - /* Really remove the filter */ |
|---|
| 4580 | | - |
|---|
| 4581 | | - MCDI_SET_DWORD(inbuf, FILTER_OP_IN_OP, |
|---|
| 4582 | | - efx_ef10_filter_is_exclusive(spec) ? |
|---|
| 4583 | | - MC_CMD_FILTER_OP_IN_OP_REMOVE : |
|---|
| 4584 | | - MC_CMD_FILTER_OP_IN_OP_UNSUBSCRIBE); |
|---|
| 4585 | | - MCDI_SET_QWORD(inbuf, FILTER_OP_IN_HANDLE, |
|---|
| 4586 | | - table->entry[filter_idx].handle); |
|---|
| 4587 | | - rc = efx_mcdi_rpc_quiet(efx, MC_CMD_FILTER_OP, |
|---|
| 4588 | | - inbuf, sizeof(inbuf), NULL, 0, NULL); |
|---|
| 4589 | | - |
|---|
| 4590 | | - if ((rc == 0) || (rc == -ENOENT)) { |
|---|
| 4591 | | - /* Filter removed OK or didn't actually exist */ |
|---|
| 4592 | | - kfree(spec); |
|---|
| 4593 | | - efx_ef10_filter_set_entry(table, filter_idx, NULL, 0); |
|---|
| 4594 | | - } else { |
|---|
| 4595 | | - efx_mcdi_display_error(efx, MC_CMD_FILTER_OP, |
|---|
| 4596 | | - MC_CMD_FILTER_OP_EXT_IN_LEN, |
|---|
| 4597 | | - NULL, 0, rc); |
|---|
| 4598 | | - } |
|---|
| 4599 | | - } |
|---|
| 4600 | | - |
|---|
| 4601 | | - return rc; |
|---|
| 4602 | | -} |
|---|
| 4603 | | - |
|---|
| 4604 | | -static int efx_ef10_filter_remove_safe(struct efx_nic *efx, |
|---|
| 4605 | | - enum efx_filter_priority priority, |
|---|
| 4606 | | - u32 filter_id) |
|---|
| 4607 | | -{ |
|---|
| 4608 | | - struct efx_ef10_filter_table *table; |
|---|
| 4609 | | - int rc; |
|---|
| 4610 | | - |
|---|
| 4611 | | - down_read(&efx->filter_sem); |
|---|
| 4612 | | - table = efx->filter_state; |
|---|
| 4613 | | - down_write(&table->lock); |
|---|
| 4614 | | - rc = efx_ef10_filter_remove_internal(efx, 1U << priority, filter_id, |
|---|
| 4615 | | - false); |
|---|
| 4616 | | - up_write(&table->lock); |
|---|
| 4617 | | - up_read(&efx->filter_sem); |
|---|
| 4618 | | - return rc; |
|---|
| 4619 | | -} |
|---|
| 4620 | | - |
|---|
| 4621 | | -/* Caller must hold efx->filter_sem for read */ |
|---|
| 4622 | | -static void efx_ef10_filter_remove_unsafe(struct efx_nic *efx, |
|---|
| 4623 | | - enum efx_filter_priority priority, |
|---|
| 4624 | | - u32 filter_id) |
|---|
| 4625 | | -{ |
|---|
| 4626 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 4627 | | - |
|---|
| 4628 | | - if (filter_id == EFX_EF10_FILTER_ID_INVALID) |
|---|
| 4629 | | - return; |
|---|
| 4630 | | - |
|---|
| 4631 | | - down_write(&table->lock); |
|---|
| 4632 | | - efx_ef10_filter_remove_internal(efx, 1U << priority, filter_id, |
|---|
| 4633 | | - true); |
|---|
| 4634 | | - up_write(&table->lock); |
|---|
| 4635 | | -} |
|---|
| 4636 | | - |
|---|
| 4637 | | -static int efx_ef10_filter_get_safe(struct efx_nic *efx, |
|---|
| 4638 | | - enum efx_filter_priority priority, |
|---|
| 4639 | | - u32 filter_id, struct efx_filter_spec *spec) |
|---|
| 4640 | | -{ |
|---|
| 4641 | | - unsigned int filter_idx = efx_ef10_filter_get_unsafe_id(filter_id); |
|---|
| 4642 | | - const struct efx_filter_spec *saved_spec; |
|---|
| 4643 | | - struct efx_ef10_filter_table *table; |
|---|
| 4644 | | - int rc; |
|---|
| 4645 | | - |
|---|
| 4646 | | - down_read(&efx->filter_sem); |
|---|
| 4647 | | - table = efx->filter_state; |
|---|
| 4648 | | - down_read(&table->lock); |
|---|
| 4649 | | - saved_spec = efx_ef10_filter_entry_spec(table, filter_idx); |
|---|
| 4650 | | - if (saved_spec && saved_spec->priority == priority && |
|---|
| 4651 | | - efx_ef10_filter_pri(table, saved_spec) == |
|---|
| 4652 | | - efx_ef10_filter_get_unsafe_pri(filter_id)) { |
|---|
| 4653 | | - *spec = *saved_spec; |
|---|
| 4654 | | - rc = 0; |
|---|
| 4655 | | - } else { |
|---|
| 4656 | | - rc = -ENOENT; |
|---|
| 4657 | | - } |
|---|
| 4658 | | - up_read(&table->lock); |
|---|
| 4659 | | - up_read(&efx->filter_sem); |
|---|
| 4660 | | - return rc; |
|---|
| 4661 | | -} |
|---|
| 4662 | | - |
|---|
| 4663 | | -static int efx_ef10_filter_clear_rx(struct efx_nic *efx, |
|---|
| 4664 | | - enum efx_filter_priority priority) |
|---|
| 4665 | | -{ |
|---|
| 4666 | | - struct efx_ef10_filter_table *table; |
|---|
| 4667 | | - unsigned int priority_mask; |
|---|
| 4668 | | - unsigned int i; |
|---|
| 4669 | | - int rc; |
|---|
| 4670 | | - |
|---|
| 4671 | | - priority_mask = (((1U << (priority + 1)) - 1) & |
|---|
| 4672 | | - ~(1U << EFX_FILTER_PRI_AUTO)); |
|---|
| 4673 | | - |
|---|
| 4674 | | - down_read(&efx->filter_sem); |
|---|
| 4675 | | - table = efx->filter_state; |
|---|
| 4676 | | - down_write(&table->lock); |
|---|
| 4677 | | - for (i = 0; i < HUNT_FILTER_TBL_ROWS; i++) { |
|---|
| 4678 | | - rc = efx_ef10_filter_remove_internal(efx, priority_mask, |
|---|
| 4679 | | - i, true); |
|---|
| 4680 | | - if (rc && rc != -ENOENT) |
|---|
| 4681 | | - break; |
|---|
| 4682 | | - rc = 0; |
|---|
| 4683 | | - } |
|---|
| 4684 | | - |
|---|
| 4685 | | - up_write(&table->lock); |
|---|
| 4686 | | - up_read(&efx->filter_sem); |
|---|
| 4687 | | - return rc; |
|---|
| 4688 | | -} |
|---|
| 4689 | | - |
|---|
| 4690 | | -static u32 efx_ef10_filter_count_rx_used(struct efx_nic *efx, |
|---|
| 4691 | | - enum efx_filter_priority priority) |
|---|
| 4692 | | -{ |
|---|
| 4693 | | - struct efx_ef10_filter_table *table; |
|---|
| 4694 | | - unsigned int filter_idx; |
|---|
| 4695 | | - s32 count = 0; |
|---|
| 4696 | | - |
|---|
| 4697 | | - down_read(&efx->filter_sem); |
|---|
| 4698 | | - table = efx->filter_state; |
|---|
| 4699 | | - down_read(&table->lock); |
|---|
| 4700 | | - for (filter_idx = 0; filter_idx < HUNT_FILTER_TBL_ROWS; filter_idx++) { |
|---|
| 4701 | | - if (table->entry[filter_idx].spec && |
|---|
| 4702 | | - efx_ef10_filter_entry_spec(table, filter_idx)->priority == |
|---|
| 4703 | | - priority) |
|---|
| 4704 | | - ++count; |
|---|
| 4705 | | - } |
|---|
| 4706 | | - up_read(&table->lock); |
|---|
| 4707 | | - up_read(&efx->filter_sem); |
|---|
| 4708 | | - return count; |
|---|
| 4709 | | -} |
|---|
| 4710 | | - |
|---|
| 4711 | | -static u32 efx_ef10_filter_get_rx_id_limit(struct efx_nic *efx) |
|---|
| 4712 | | -{ |
|---|
| 4713 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 4714 | | - |
|---|
| 4715 | | - return table->rx_match_count * HUNT_FILTER_TBL_ROWS * 2; |
|---|
| 4716 | | -} |
|---|
| 4717 | | - |
|---|
| 4718 | | -static s32 efx_ef10_filter_get_rx_ids(struct efx_nic *efx, |
|---|
| 4719 | | - enum efx_filter_priority priority, |
|---|
| 4720 | | - u32 *buf, u32 size) |
|---|
| 4721 | | -{ |
|---|
| 4722 | | - struct efx_ef10_filter_table *table; |
|---|
| 4723 | | - struct efx_filter_spec *spec; |
|---|
| 4724 | | - unsigned int filter_idx; |
|---|
| 4725 | | - s32 count = 0; |
|---|
| 4726 | | - |
|---|
| 4727 | | - down_read(&efx->filter_sem); |
|---|
| 4728 | | - table = efx->filter_state; |
|---|
| 4729 | | - down_read(&table->lock); |
|---|
| 4730 | | - |
|---|
| 4731 | | - for (filter_idx = 0; filter_idx < HUNT_FILTER_TBL_ROWS; filter_idx++) { |
|---|
| 4732 | | - spec = efx_ef10_filter_entry_spec(table, filter_idx); |
|---|
| 4733 | | - if (spec && spec->priority == priority) { |
|---|
| 4734 | | - if (count == size) { |
|---|
| 4735 | | - count = -EMSGSIZE; |
|---|
| 4736 | | - break; |
|---|
| 4737 | | - } |
|---|
| 4738 | | - buf[count++] = |
|---|
| 4739 | | - efx_ef10_make_filter_id( |
|---|
| 4740 | | - efx_ef10_filter_pri(table, spec), |
|---|
| 4741 | | - filter_idx); |
|---|
| 4742 | | - } |
|---|
| 4743 | | - } |
|---|
| 4744 | | - up_read(&table->lock); |
|---|
| 4745 | | - up_read(&efx->filter_sem); |
|---|
| 4746 | | - return count; |
|---|
| 4747 | | -} |
|---|
| 4748 | | - |
|---|
| 4749 | | -#ifdef CONFIG_RFS_ACCEL |
|---|
| 4750 | | - |
|---|
| 4751 | | -static bool efx_ef10_filter_rfs_expire_one(struct efx_nic *efx, u32 flow_id, |
|---|
| 4752 | | - unsigned int filter_idx) |
|---|
| 4753 | | -{ |
|---|
| 4754 | | - struct efx_filter_spec *spec, saved_spec; |
|---|
| 4755 | | - struct efx_ef10_filter_table *table; |
|---|
| 4756 | | - struct efx_arfs_rule *rule = NULL; |
|---|
| 4757 | | - bool ret = true, force = false; |
|---|
| 4758 | | - u16 arfs_id; |
|---|
| 4759 | | - |
|---|
| 4760 | | - down_read(&efx->filter_sem); |
|---|
| 4761 | | - table = efx->filter_state; |
|---|
| 4762 | | - down_write(&table->lock); |
|---|
| 4763 | | - spec = efx_ef10_filter_entry_spec(table, filter_idx); |
|---|
| 4764 | | - |
|---|
| 4765 | | - if (!spec || spec->priority != EFX_FILTER_PRI_HINT) |
|---|
| 4766 | | - goto out_unlock; |
|---|
| 4767 | | - |
|---|
| 4768 | | - spin_lock_bh(&efx->rps_hash_lock); |
|---|
| 4769 | | - if (!efx->rps_hash_table) { |
|---|
| 4770 | | - /* In the absence of the table, we always return 0 to ARFS. */ |
|---|
| 4771 | | - arfs_id = 0; |
|---|
| 4772 | | - } else { |
|---|
| 4773 | | - rule = efx_rps_hash_find(efx, spec); |
|---|
| 4774 | | - if (!rule) |
|---|
| 4775 | | - /* ARFS table doesn't know of this filter, so remove it */ |
|---|
| 4776 | | - goto expire; |
|---|
| 4777 | | - arfs_id = rule->arfs_id; |
|---|
| 4778 | | - ret = efx_rps_check_rule(rule, filter_idx, &force); |
|---|
| 4779 | | - if (force) |
|---|
| 4780 | | - goto expire; |
|---|
| 4781 | | - if (!ret) { |
|---|
| 4782 | | - spin_unlock_bh(&efx->rps_hash_lock); |
|---|
| 4783 | | - goto out_unlock; |
|---|
| 4784 | | - } |
|---|
| 4785 | | - } |
|---|
| 4786 | | - if (!rps_may_expire_flow(efx->net_dev, spec->dmaq_id, flow_id, arfs_id)) |
|---|
| 4787 | | - ret = false; |
|---|
| 4788 | | - else if (rule) |
|---|
| 4789 | | - rule->filter_id = EFX_ARFS_FILTER_ID_REMOVING; |
|---|
| 4790 | | -expire: |
|---|
| 4791 | | - saved_spec = *spec; /* remove operation will kfree spec */ |
|---|
| 4792 | | - spin_unlock_bh(&efx->rps_hash_lock); |
|---|
| 4793 | | - /* At this point (since we dropped the lock), another thread might queue |
|---|
| 4794 | | - * up a fresh insertion request (but the actual insertion will be held |
|---|
| 4795 | | - * up by our possession of the filter table lock). In that case, it |
|---|
| 4796 | | - * will set rule->filter_id to EFX_ARFS_FILTER_ID_PENDING, meaning that |
|---|
| 4797 | | - * the rule is not removed by efx_rps_hash_del() below. |
|---|
| 4798 | | - */ |
|---|
| 4799 | | - if (ret) |
|---|
| 4800 | | - ret = efx_ef10_filter_remove_internal(efx, 1U << spec->priority, |
|---|
| 4801 | | - filter_idx, true) == 0; |
|---|
| 4802 | | - /* While we can't safely dereference rule (we dropped the lock), we can |
|---|
| 4803 | | - * still test it for NULL. |
|---|
| 4804 | | - */ |
|---|
| 4805 | | - if (ret && rule) { |
|---|
| 4806 | | - /* Expiring, so remove entry from ARFS table */ |
|---|
| 4807 | | - spin_lock_bh(&efx->rps_hash_lock); |
|---|
| 4808 | | - efx_rps_hash_del(efx, &saved_spec); |
|---|
| 4809 | | - spin_unlock_bh(&efx->rps_hash_lock); |
|---|
| 4810 | | - } |
|---|
| 4811 | | -out_unlock: |
|---|
| 4812 | | - up_write(&table->lock); |
|---|
| 4813 | | - up_read(&efx->filter_sem); |
|---|
| 4814 | | - return ret; |
|---|
| 4815 | | -} |
|---|
| 4816 | | - |
|---|
| 4817 | | -#endif /* CONFIG_RFS_ACCEL */ |
|---|
| 4818 | | - |
|---|
| 4819 | | -static int efx_ef10_filter_match_flags_from_mcdi(bool encap, u32 mcdi_flags) |
|---|
| 4820 | | -{ |
|---|
| 4821 | | - int match_flags = 0; |
|---|
| 4822 | | - |
|---|
| 4823 | | -#define MAP_FLAG(gen_flag, mcdi_field) do { \ |
|---|
| 4824 | | - u32 old_mcdi_flags = mcdi_flags; \ |
|---|
| 4825 | | - mcdi_flags &= ~(1 << MC_CMD_FILTER_OP_EXT_IN_MATCH_ ## \ |
|---|
| 4826 | | - mcdi_field ## _LBN); \ |
|---|
| 4827 | | - if (mcdi_flags != old_mcdi_flags) \ |
|---|
| 4828 | | - match_flags |= EFX_FILTER_MATCH_ ## gen_flag; \ |
|---|
| 4829 | | - } while (0) |
|---|
| 4830 | | - |
|---|
| 4831 | | - if (encap) { |
|---|
| 4832 | | - /* encap filters must specify encap type */ |
|---|
| 4833 | | - match_flags |= EFX_FILTER_MATCH_ENCAP_TYPE; |
|---|
| 4834 | | - /* and imply ethertype and ip proto */ |
|---|
| 4835 | | - mcdi_flags &= |
|---|
| 4836 | | - ~(1 << MC_CMD_FILTER_OP_EXT_IN_MATCH_IP_PROTO_LBN); |
|---|
| 4837 | | - mcdi_flags &= |
|---|
| 4838 | | - ~(1 << MC_CMD_FILTER_OP_EXT_IN_MATCH_ETHER_TYPE_LBN); |
|---|
| 4839 | | - /* VLAN tags refer to the outer packet */ |
|---|
| 4840 | | - MAP_FLAG(INNER_VID, INNER_VLAN); |
|---|
| 4841 | | - MAP_FLAG(OUTER_VID, OUTER_VLAN); |
|---|
| 4842 | | - /* everything else refers to the inner packet */ |
|---|
| 4843 | | - MAP_FLAG(LOC_MAC_IG, IFRM_UNKNOWN_UCAST_DST); |
|---|
| 4844 | | - MAP_FLAG(LOC_MAC_IG, IFRM_UNKNOWN_MCAST_DST); |
|---|
| 4845 | | - MAP_FLAG(REM_HOST, IFRM_SRC_IP); |
|---|
| 4846 | | - MAP_FLAG(LOC_HOST, IFRM_DST_IP); |
|---|
| 4847 | | - MAP_FLAG(REM_MAC, IFRM_SRC_MAC); |
|---|
| 4848 | | - MAP_FLAG(REM_PORT, IFRM_SRC_PORT); |
|---|
| 4849 | | - MAP_FLAG(LOC_MAC, IFRM_DST_MAC); |
|---|
| 4850 | | - MAP_FLAG(LOC_PORT, IFRM_DST_PORT); |
|---|
| 4851 | | - MAP_FLAG(ETHER_TYPE, IFRM_ETHER_TYPE); |
|---|
| 4852 | | - MAP_FLAG(IP_PROTO, IFRM_IP_PROTO); |
|---|
| 4853 | | - } else { |
|---|
| 4854 | | - MAP_FLAG(LOC_MAC_IG, UNKNOWN_UCAST_DST); |
|---|
| 4855 | | - MAP_FLAG(LOC_MAC_IG, UNKNOWN_MCAST_DST); |
|---|
| 4856 | | - MAP_FLAG(REM_HOST, SRC_IP); |
|---|
| 4857 | | - MAP_FLAG(LOC_HOST, DST_IP); |
|---|
| 4858 | | - MAP_FLAG(REM_MAC, SRC_MAC); |
|---|
| 4859 | | - MAP_FLAG(REM_PORT, SRC_PORT); |
|---|
| 4860 | | - MAP_FLAG(LOC_MAC, DST_MAC); |
|---|
| 4861 | | - MAP_FLAG(LOC_PORT, DST_PORT); |
|---|
| 4862 | | - MAP_FLAG(ETHER_TYPE, ETHER_TYPE); |
|---|
| 4863 | | - MAP_FLAG(INNER_VID, INNER_VLAN); |
|---|
| 4864 | | - MAP_FLAG(OUTER_VID, OUTER_VLAN); |
|---|
| 4865 | | - MAP_FLAG(IP_PROTO, IP_PROTO); |
|---|
| 4866 | | - } |
|---|
| 4867 | | -#undef MAP_FLAG |
|---|
| 4868 | | - |
|---|
| 4869 | | - /* Did we map them all? */ |
|---|
| 4870 | | - if (mcdi_flags) |
|---|
| 4871 | | - return -EINVAL; |
|---|
| 4872 | | - |
|---|
| 4873 | | - return match_flags; |
|---|
| 4874 | | -} |
|---|
| 4875 | | - |
|---|
| 4876 | | -static void efx_ef10_filter_cleanup_vlans(struct efx_nic *efx) |
|---|
| 4877 | | -{ |
|---|
| 4878 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 4879 | | - struct efx_ef10_filter_vlan *vlan, *next_vlan; |
|---|
| 4880 | | - |
|---|
| 4881 | | - /* See comment in efx_ef10_filter_table_remove() */ |
|---|
| 4882 | | - if (!efx_rwsem_assert_write_locked(&efx->filter_sem)) |
|---|
| 4883 | | - return; |
|---|
| 4884 | | - |
|---|
| 4885 | | - if (!table) |
|---|
| 4886 | | - return; |
|---|
| 4887 | | - |
|---|
| 4888 | | - list_for_each_entry_safe(vlan, next_vlan, &table->vlan_list, list) |
|---|
| 4889 | | - efx_ef10_filter_del_vlan_internal(efx, vlan); |
|---|
| 4890 | | -} |
|---|
| 4891 | | - |
|---|
| 4892 | | -static bool efx_ef10_filter_match_supported(struct efx_ef10_filter_table *table, |
|---|
| 4893 | | - bool encap, |
|---|
| 4894 | | - enum efx_filter_match_flags match_flags) |
|---|
| 4895 | | -{ |
|---|
| 4896 | | - unsigned int match_pri; |
|---|
| 4897 | | - int mf; |
|---|
| 4898 | | - |
|---|
| 4899 | | - for (match_pri = 0; |
|---|
| 4900 | | - match_pri < table->rx_match_count; |
|---|
| 4901 | | - match_pri++) { |
|---|
| 4902 | | - mf = efx_ef10_filter_match_flags_from_mcdi(encap, |
|---|
| 4903 | | - table->rx_match_mcdi_flags[match_pri]); |
|---|
| 4904 | | - if (mf == match_flags) |
|---|
| 4905 | | - return true; |
|---|
| 4906 | | - } |
|---|
| 4907 | | - |
|---|
| 4908 | | - return false; |
|---|
| 4909 | | -} |
|---|
| 4910 | | - |
|---|
| 4911 | | -static int |
|---|
| 4912 | | -efx_ef10_filter_table_probe_matches(struct efx_nic *efx, |
|---|
| 4913 | | - struct efx_ef10_filter_table *table, |
|---|
| 4914 | | - bool encap) |
|---|
| 4915 | | -{ |
|---|
| 4916 | | - MCDI_DECLARE_BUF(inbuf, MC_CMD_GET_PARSER_DISP_INFO_IN_LEN); |
|---|
| 4917 | | - MCDI_DECLARE_BUF(outbuf, MC_CMD_GET_PARSER_DISP_INFO_OUT_LENMAX); |
|---|
| 4918 | | - unsigned int pd_match_pri, pd_match_count; |
|---|
| 4919 | | - size_t outlen; |
|---|
| 4920 | | - int rc; |
|---|
| 4921 | | - |
|---|
| 4922 | | - /* Find out which RX filter types are supported, and their priorities */ |
|---|
| 4923 | | - MCDI_SET_DWORD(inbuf, GET_PARSER_DISP_INFO_IN_OP, |
|---|
| 4924 | | - encap ? |
|---|
| 4925 | | - MC_CMD_GET_PARSER_DISP_INFO_IN_OP_GET_SUPPORTED_ENCAP_RX_MATCHES : |
|---|
| 4926 | | - MC_CMD_GET_PARSER_DISP_INFO_IN_OP_GET_SUPPORTED_RX_MATCHES); |
|---|
| 4927 | | - rc = efx_mcdi_rpc(efx, MC_CMD_GET_PARSER_DISP_INFO, |
|---|
| 4928 | | - inbuf, sizeof(inbuf), outbuf, sizeof(outbuf), |
|---|
| 4929 | | - &outlen); |
|---|
| 4930 | | - if (rc) |
|---|
| 4931 | | - return rc; |
|---|
| 4932 | | - |
|---|
| 4933 | | - pd_match_count = MCDI_VAR_ARRAY_LEN( |
|---|
| 4934 | | - outlen, GET_PARSER_DISP_INFO_OUT_SUPPORTED_MATCHES); |
|---|
| 4935 | | - |
|---|
| 4936 | | - for (pd_match_pri = 0; pd_match_pri < pd_match_count; pd_match_pri++) { |
|---|
| 4937 | | - u32 mcdi_flags = |
|---|
| 4938 | | - MCDI_ARRAY_DWORD( |
|---|
| 4939 | | - outbuf, |
|---|
| 4940 | | - GET_PARSER_DISP_INFO_OUT_SUPPORTED_MATCHES, |
|---|
| 4941 | | - pd_match_pri); |
|---|
| 4942 | | - rc = efx_ef10_filter_match_flags_from_mcdi(encap, mcdi_flags); |
|---|
| 4943 | | - if (rc < 0) { |
|---|
| 4944 | | - netif_dbg(efx, probe, efx->net_dev, |
|---|
| 4945 | | - "%s: fw flags %#x pri %u not supported in driver\n", |
|---|
| 4946 | | - __func__, mcdi_flags, pd_match_pri); |
|---|
| 4947 | | - } else { |
|---|
| 4948 | | - netif_dbg(efx, probe, efx->net_dev, |
|---|
| 4949 | | - "%s: fw flags %#x pri %u supported as driver flags %#x pri %u\n", |
|---|
| 4950 | | - __func__, mcdi_flags, pd_match_pri, |
|---|
| 4951 | | - rc, table->rx_match_count); |
|---|
| 4952 | | - table->rx_match_mcdi_flags[table->rx_match_count] = mcdi_flags; |
|---|
| 4953 | | - table->rx_match_count++; |
|---|
| 4954 | | - } |
|---|
| 4955 | | - } |
|---|
| 4956 | | - |
|---|
| 4957 | | - return 0; |
|---|
| 4958 | | -} |
|---|
| 4959 | | - |
|---|
| 4960 | | -static int efx_ef10_filter_table_probe(struct efx_nic *efx) |
|---|
| 4961 | | -{ |
|---|
| 4962 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 4963 | | - struct net_device *net_dev = efx->net_dev; |
|---|
| 4964 | | - struct efx_ef10_filter_table *table; |
|---|
| 4965 | | - struct efx_ef10_vlan *vlan; |
|---|
| 4966 | | - int rc; |
|---|
| 4967 | | - |
|---|
| 4968 | | - if (!efx_rwsem_assert_write_locked(&efx->filter_sem)) |
|---|
| 4969 | | - return -EINVAL; |
|---|
| 4970 | | - |
|---|
| 4971 | | - if (efx->filter_state) /* already probed */ |
|---|
| 4972 | | - return 0; |
|---|
| 4973 | | - |
|---|
| 4974 | | - table = kzalloc(sizeof(*table), GFP_KERNEL); |
|---|
| 4975 | | - if (!table) |
|---|
| 4976 | | - return -ENOMEM; |
|---|
| 4977 | | - |
|---|
| 4978 | | - table->rx_match_count = 0; |
|---|
| 4979 | | - rc = efx_ef10_filter_table_probe_matches(efx, table, false); |
|---|
| 4980 | | - if (rc) |
|---|
| 4981 | | - goto fail; |
|---|
| 4982 | | - if (nic_data->datapath_caps & |
|---|
| 4983 | | - (1 << MC_CMD_GET_CAPABILITIES_OUT_VXLAN_NVGRE_LBN)) |
|---|
| 4984 | | - rc = efx_ef10_filter_table_probe_matches(efx, table, true); |
|---|
| 4985 | | - if (rc) |
|---|
| 4986 | | - goto fail; |
|---|
| 4987 | | - if ((efx_supported_features(efx) & NETIF_F_HW_VLAN_CTAG_FILTER) && |
|---|
| 4988 | | - !(efx_ef10_filter_match_supported(table, false, |
|---|
| 4989 | | - (EFX_FILTER_MATCH_OUTER_VID | EFX_FILTER_MATCH_LOC_MAC)) && |
|---|
| 4990 | | - efx_ef10_filter_match_supported(table, false, |
|---|
| 4991 | | - (EFX_FILTER_MATCH_OUTER_VID | EFX_FILTER_MATCH_LOC_MAC_IG)))) { |
|---|
| 4992 | | - netif_info(efx, probe, net_dev, |
|---|
| 4993 | | - "VLAN filters are not supported in this firmware variant\n"); |
|---|
| 4994 | | - net_dev->features &= ~NETIF_F_HW_VLAN_CTAG_FILTER; |
|---|
| 4995 | | - efx->fixed_features &= ~NETIF_F_HW_VLAN_CTAG_FILTER; |
|---|
| 4996 | | - net_dev->hw_features &= ~NETIF_F_HW_VLAN_CTAG_FILTER; |
|---|
| 4997 | | - } |
|---|
| 4998 | | - |
|---|
| 4999 | | - table->entry = vzalloc(array_size(HUNT_FILTER_TBL_ROWS, |
|---|
| 5000 | | - sizeof(*table->entry))); |
|---|
| 5001 | | - if (!table->entry) { |
|---|
| 5002 | | - rc = -ENOMEM; |
|---|
| 5003 | | - goto fail; |
|---|
| 5004 | | - } |
|---|
| 5005 | | - |
|---|
| 5006 | | - table->mc_promisc_last = false; |
|---|
| 5007 | | - table->vlan_filter = |
|---|
| 5008 | | - !!(efx->net_dev->features & NETIF_F_HW_VLAN_CTAG_FILTER); |
|---|
| 5009 | | - INIT_LIST_HEAD(&table->vlan_list); |
|---|
| 5010 | | - init_rwsem(&table->lock); |
|---|
| 5011 | | - |
|---|
| 5012 | | - efx->filter_state = table; |
|---|
| 5013 | | - |
|---|
| 5014 | | - list_for_each_entry(vlan, &nic_data->vlan_list, list) { |
|---|
| 5015 | | - rc = efx_ef10_filter_add_vlan(efx, vlan->vid); |
|---|
| 5016 | | - if (rc) |
|---|
| 5017 | | - goto fail_add_vlan; |
|---|
| 5018 | | - } |
|---|
| 5019 | | - |
|---|
| 5020 | | - return 0; |
|---|
| 5021 | | - |
|---|
| 5022 | | -fail_add_vlan: |
|---|
| 5023 | | - efx_ef10_filter_cleanup_vlans(efx); |
|---|
| 5024 | | - efx->filter_state = NULL; |
|---|
| 5025 | | -fail: |
|---|
| 5026 | | - kfree(table); |
|---|
| 5027 | | - return rc; |
|---|
| 5028 | | -} |
|---|
| 5029 | | - |
|---|
| 5030 | | -/* Caller must hold efx->filter_sem for read if race against |
|---|
| 5031 | | - * efx_ef10_filter_table_remove() is possible |
|---|
| 5032 | | - */ |
|---|
| 5033 | | -static void efx_ef10_filter_table_restore(struct efx_nic *efx) |
|---|
| 5034 | | -{ |
|---|
| 5035 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 5036 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 5037 | | - unsigned int invalid_filters = 0, failed = 0; |
|---|
| 5038 | | - struct efx_ef10_filter_vlan *vlan; |
|---|
| 5039 | | - struct efx_filter_spec *spec; |
|---|
| 5040 | | - struct efx_rss_context *ctx; |
|---|
| 5041 | | - unsigned int filter_idx; |
|---|
| 5042 | | - u32 mcdi_flags; |
|---|
| 5043 | | - int match_pri; |
|---|
| 5044 | | - int rc, i; |
|---|
| 5045 | | - |
|---|
| 5046 | | - WARN_ON(!rwsem_is_locked(&efx->filter_sem)); |
|---|
| 5047 | | - |
|---|
| 5048 | | - if (!nic_data->must_restore_filters) |
|---|
| 5049 | | - return; |
|---|
| 5050 | | - |
|---|
| 5051 | | - if (!table) |
|---|
| 5052 | | - return; |
|---|
| 5053 | | - |
|---|
| 5054 | | - down_write(&table->lock); |
|---|
| 5055 | | - mutex_lock(&efx->rss_lock); |
|---|
| 5056 | | - |
|---|
| 5057 | | - for (filter_idx = 0; filter_idx < HUNT_FILTER_TBL_ROWS; filter_idx++) { |
|---|
| 5058 | | - spec = efx_ef10_filter_entry_spec(table, filter_idx); |
|---|
| 5059 | | - if (!spec) |
|---|
| 5060 | | - continue; |
|---|
| 5061 | | - |
|---|
| 5062 | | - mcdi_flags = efx_ef10_filter_mcdi_flags_from_spec(spec); |
|---|
| 5063 | | - match_pri = 0; |
|---|
| 5064 | | - while (match_pri < table->rx_match_count && |
|---|
| 5065 | | - table->rx_match_mcdi_flags[match_pri] != mcdi_flags) |
|---|
| 5066 | | - ++match_pri; |
|---|
| 5067 | | - if (match_pri >= table->rx_match_count) { |
|---|
| 5068 | | - invalid_filters++; |
|---|
| 5069 | | - goto not_restored; |
|---|
| 5070 | | - } |
|---|
| 5071 | | - if (spec->rss_context) |
|---|
| 5072 | | - ctx = efx_find_rss_context_entry(efx, spec->rss_context); |
|---|
| 5073 | | - else |
|---|
| 5074 | | - ctx = &efx->rss_context; |
|---|
| 5075 | | - if (spec->flags & EFX_FILTER_FLAG_RX_RSS) { |
|---|
| 5076 | | - if (!ctx) { |
|---|
| 5077 | | - netif_warn(efx, drv, efx->net_dev, |
|---|
| 5078 | | - "Warning: unable to restore a filter with nonexistent RSS context %u.\n", |
|---|
| 5079 | | - spec->rss_context); |
|---|
| 5080 | | - invalid_filters++; |
|---|
| 5081 | | - goto not_restored; |
|---|
| 5082 | | - } |
|---|
| 5083 | | - if (ctx->context_id == EFX_EF10_RSS_CONTEXT_INVALID) { |
|---|
| 5084 | | - netif_warn(efx, drv, efx->net_dev, |
|---|
| 5085 | | - "Warning: unable to restore a filter with RSS context %u as it was not created.\n", |
|---|
| 5086 | | - spec->rss_context); |
|---|
| 5087 | | - invalid_filters++; |
|---|
| 5088 | | - goto not_restored; |
|---|
| 5089 | | - } |
|---|
| 5090 | | - } |
|---|
| 5091 | | - |
|---|
| 5092 | | - rc = efx_ef10_filter_push(efx, spec, |
|---|
| 5093 | | - &table->entry[filter_idx].handle, |
|---|
| 5094 | | - ctx, false); |
|---|
| 5095 | | - if (rc) |
|---|
| 5096 | | - failed++; |
|---|
| 5097 | | - |
|---|
| 5098 | | - if (rc) { |
|---|
| 5099 | | -not_restored: |
|---|
| 5100 | | - list_for_each_entry(vlan, &table->vlan_list, list) |
|---|
| 5101 | | - for (i = 0; i < EFX_EF10_NUM_DEFAULT_FILTERS; ++i) |
|---|
| 5102 | | - if (vlan->default_filters[i] == filter_idx) |
|---|
| 5103 | | - vlan->default_filters[i] = |
|---|
| 5104 | | - EFX_EF10_FILTER_ID_INVALID; |
|---|
| 5105 | | - |
|---|
| 5106 | | - kfree(spec); |
|---|
| 5107 | | - efx_ef10_filter_set_entry(table, filter_idx, NULL, 0); |
|---|
| 5108 | | - } |
|---|
| 5109 | | - } |
|---|
| 5110 | | - |
|---|
| 5111 | | - mutex_unlock(&efx->rss_lock); |
|---|
| 5112 | | - up_write(&table->lock); |
|---|
| 5113 | | - |
|---|
| 5114 | | - /* This can happen validly if the MC's capabilities have changed, so |
|---|
| 5115 | | - * is not an error. |
|---|
| 5116 | | - */ |
|---|
| 5117 | | - if (invalid_filters) |
|---|
| 5118 | | - netif_dbg(efx, drv, efx->net_dev, |
|---|
| 5119 | | - "Did not restore %u filters that are now unsupported.\n", |
|---|
| 5120 | | - invalid_filters); |
|---|
| 5121 | | - |
|---|
| 5122 | | - if (failed) |
|---|
| 5123 | | - netif_err(efx, hw, efx->net_dev, |
|---|
| 5124 | | - "unable to restore %u filters\n", failed); |
|---|
| 5125 | | - else |
|---|
| 5126 | | - nic_data->must_restore_filters = false; |
|---|
| 5127 | | -} |
|---|
| 5128 | | - |
|---|
| 5129 | | -static void efx_ef10_filter_table_remove(struct efx_nic *efx) |
|---|
| 5130 | | -{ |
|---|
| 5131 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 5132 | | - MCDI_DECLARE_BUF(inbuf, MC_CMD_FILTER_OP_EXT_IN_LEN); |
|---|
| 5133 | | - struct efx_filter_spec *spec; |
|---|
| 5134 | | - unsigned int filter_idx; |
|---|
| 5135 | | - int rc; |
|---|
| 5136 | | - |
|---|
| 5137 | | - efx_ef10_filter_cleanup_vlans(efx); |
|---|
| 5138 | | - efx->filter_state = NULL; |
|---|
| 5139 | | - /* If we were called without locking, then it's not safe to free |
|---|
| 5140 | | - * the table as others might be using it. So we just WARN, leak |
|---|
| 5141 | | - * the memory, and potentially get an inconsistent filter table |
|---|
| 5142 | | - * state. |
|---|
| 5143 | | - * This should never actually happen. |
|---|
| 5144 | | - */ |
|---|
| 5145 | | - if (!efx_rwsem_assert_write_locked(&efx->filter_sem)) |
|---|
| 5146 | | - return; |
|---|
| 5147 | | - |
|---|
| 5148 | | - if (!table) |
|---|
| 5149 | | - return; |
|---|
| 5150 | | - |
|---|
| 5151 | | - for (filter_idx = 0; filter_idx < HUNT_FILTER_TBL_ROWS; filter_idx++) { |
|---|
| 5152 | | - spec = efx_ef10_filter_entry_spec(table, filter_idx); |
|---|
| 5153 | | - if (!spec) |
|---|
| 5154 | | - continue; |
|---|
| 5155 | | - |
|---|
| 5156 | | - MCDI_SET_DWORD(inbuf, FILTER_OP_IN_OP, |
|---|
| 5157 | | - efx_ef10_filter_is_exclusive(spec) ? |
|---|
| 5158 | | - MC_CMD_FILTER_OP_IN_OP_REMOVE : |
|---|
| 5159 | | - MC_CMD_FILTER_OP_IN_OP_UNSUBSCRIBE); |
|---|
| 5160 | | - MCDI_SET_QWORD(inbuf, FILTER_OP_IN_HANDLE, |
|---|
| 5161 | | - table->entry[filter_idx].handle); |
|---|
| 5162 | | - rc = efx_mcdi_rpc_quiet(efx, MC_CMD_FILTER_OP, inbuf, |
|---|
| 5163 | | - sizeof(inbuf), NULL, 0, NULL); |
|---|
| 5164 | | - if (rc) |
|---|
| 5165 | | - netif_info(efx, drv, efx->net_dev, |
|---|
| 5166 | | - "%s: filter %04x remove failed\n", |
|---|
| 5167 | | - __func__, filter_idx); |
|---|
| 5168 | | - kfree(spec); |
|---|
| 5169 | | - } |
|---|
| 5170 | | - |
|---|
| 5171 | | - vfree(table->entry); |
|---|
| 5172 | | - kfree(table); |
|---|
| 5173 | | -} |
|---|
| 5174 | | - |
|---|
| 5175 | | -static void efx_ef10_filter_mark_one_old(struct efx_nic *efx, uint16_t *id) |
|---|
| 5176 | | -{ |
|---|
| 5177 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 5178 | | - unsigned int filter_idx; |
|---|
| 5179 | | - |
|---|
| 5180 | | - efx_rwsem_assert_write_locked(&table->lock); |
|---|
| 5181 | | - |
|---|
| 5182 | | - if (*id != EFX_EF10_FILTER_ID_INVALID) { |
|---|
| 5183 | | - filter_idx = efx_ef10_filter_get_unsafe_id(*id); |
|---|
| 5184 | | - if (!table->entry[filter_idx].spec) |
|---|
| 5185 | | - netif_dbg(efx, drv, efx->net_dev, |
|---|
| 5186 | | - "marked null spec old %04x:%04x\n", *id, |
|---|
| 5187 | | - filter_idx); |
|---|
| 5188 | | - table->entry[filter_idx].spec |= EFX_EF10_FILTER_FLAG_AUTO_OLD; |
|---|
| 5189 | | - *id = EFX_EF10_FILTER_ID_INVALID; |
|---|
| 5190 | | - } |
|---|
| 5191 | | -} |
|---|
| 5192 | | - |
|---|
| 5193 | | -/* Mark old per-VLAN filters that may need to be removed */ |
|---|
| 5194 | | -static void _efx_ef10_filter_vlan_mark_old(struct efx_nic *efx, |
|---|
| 5195 | | - struct efx_ef10_filter_vlan *vlan) |
|---|
| 5196 | | -{ |
|---|
| 5197 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 5198 | | - unsigned int i; |
|---|
| 5199 | | - |
|---|
| 5200 | | - for (i = 0; i < table->dev_uc_count; i++) |
|---|
| 5201 | | - efx_ef10_filter_mark_one_old(efx, &vlan->uc[i]); |
|---|
| 5202 | | - for (i = 0; i < table->dev_mc_count; i++) |
|---|
| 5203 | | - efx_ef10_filter_mark_one_old(efx, &vlan->mc[i]); |
|---|
| 5204 | | - for (i = 0; i < EFX_EF10_NUM_DEFAULT_FILTERS; i++) |
|---|
| 5205 | | - efx_ef10_filter_mark_one_old(efx, &vlan->default_filters[i]); |
|---|
| 5206 | | -} |
|---|
| 5207 | | - |
|---|
| 5208 | | -/* Mark old filters that may need to be removed. |
|---|
| 5209 | | - * Caller must hold efx->filter_sem for read if race against |
|---|
| 5210 | | - * efx_ef10_filter_table_remove() is possible |
|---|
| 5211 | | - */ |
|---|
| 5212 | | -static void efx_ef10_filter_mark_old(struct efx_nic *efx) |
|---|
| 5213 | | -{ |
|---|
| 5214 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 5215 | | - struct efx_ef10_filter_vlan *vlan; |
|---|
| 5216 | | - |
|---|
| 5217 | | - down_write(&table->lock); |
|---|
| 5218 | | - list_for_each_entry(vlan, &table->vlan_list, list) |
|---|
| 5219 | | - _efx_ef10_filter_vlan_mark_old(efx, vlan); |
|---|
| 5220 | | - up_write(&table->lock); |
|---|
| 5221 | | -} |
|---|
| 5222 | | - |
|---|
| 5223 | | -static void efx_ef10_filter_uc_addr_list(struct efx_nic *efx) |
|---|
| 5224 | | -{ |
|---|
| 5225 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 5226 | | - struct net_device *net_dev = efx->net_dev; |
|---|
| 5227 | | - struct netdev_hw_addr *uc; |
|---|
| 5228 | | - unsigned int i; |
|---|
| 5229 | | - |
|---|
| 5230 | | - table->uc_promisc = !!(net_dev->flags & IFF_PROMISC); |
|---|
| 5231 | | - ether_addr_copy(table->dev_uc_list[0].addr, net_dev->dev_addr); |
|---|
| 5232 | | - i = 1; |
|---|
| 5233 | | - netdev_for_each_uc_addr(uc, net_dev) { |
|---|
| 5234 | | - if (i >= EFX_EF10_FILTER_DEV_UC_MAX) { |
|---|
| 5235 | | - table->uc_promisc = true; |
|---|
| 5236 | | - break; |
|---|
| 5237 | | - } |
|---|
| 5238 | | - ether_addr_copy(table->dev_uc_list[i].addr, uc->addr); |
|---|
| 5239 | | - i++; |
|---|
| 5240 | | - } |
|---|
| 5241 | | - |
|---|
| 5242 | | - table->dev_uc_count = i; |
|---|
| 5243 | | -} |
|---|
| 5244 | | - |
|---|
| 5245 | | -static void efx_ef10_filter_mc_addr_list(struct efx_nic *efx) |
|---|
| 5246 | | -{ |
|---|
| 5247 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 5248 | | - struct net_device *net_dev = efx->net_dev; |
|---|
| 5249 | | - struct netdev_hw_addr *mc; |
|---|
| 5250 | | - unsigned int i; |
|---|
| 5251 | | - |
|---|
| 5252 | | - table->mc_overflow = false; |
|---|
| 5253 | | - table->mc_promisc = !!(net_dev->flags & (IFF_PROMISC | IFF_ALLMULTI)); |
|---|
| 5254 | | - |
|---|
| 5255 | | - i = 0; |
|---|
| 5256 | | - netdev_for_each_mc_addr(mc, net_dev) { |
|---|
| 5257 | | - if (i >= EFX_EF10_FILTER_DEV_MC_MAX) { |
|---|
| 5258 | | - table->mc_promisc = true; |
|---|
| 5259 | | - table->mc_overflow = true; |
|---|
| 5260 | | - break; |
|---|
| 5261 | | - } |
|---|
| 5262 | | - ether_addr_copy(table->dev_mc_list[i].addr, mc->addr); |
|---|
| 5263 | | - i++; |
|---|
| 5264 | | - } |
|---|
| 5265 | | - |
|---|
| 5266 | | - table->dev_mc_count = i; |
|---|
| 5267 | | -} |
|---|
| 5268 | | - |
|---|
| 5269 | | -static int efx_ef10_filter_insert_addr_list(struct efx_nic *efx, |
|---|
| 5270 | | - struct efx_ef10_filter_vlan *vlan, |
|---|
| 5271 | | - bool multicast, bool rollback) |
|---|
| 5272 | | -{ |
|---|
| 5273 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 5274 | | - struct efx_ef10_dev_addr *addr_list; |
|---|
| 5275 | | - enum efx_filter_flags filter_flags; |
|---|
| 5276 | | - struct efx_filter_spec spec; |
|---|
| 5277 | | - u8 baddr[ETH_ALEN]; |
|---|
| 5278 | | - unsigned int i, j; |
|---|
| 5279 | | - int addr_count; |
|---|
| 5280 | | - u16 *ids; |
|---|
| 5281 | | - int rc; |
|---|
| 5282 | | - |
|---|
| 5283 | | - if (multicast) { |
|---|
| 5284 | | - addr_list = table->dev_mc_list; |
|---|
| 5285 | | - addr_count = table->dev_mc_count; |
|---|
| 5286 | | - ids = vlan->mc; |
|---|
| 5287 | | - } else { |
|---|
| 5288 | | - addr_list = table->dev_uc_list; |
|---|
| 5289 | | - addr_count = table->dev_uc_count; |
|---|
| 5290 | | - ids = vlan->uc; |
|---|
| 5291 | | - } |
|---|
| 5292 | | - |
|---|
| 5293 | | - filter_flags = efx_rss_active(&efx->rss_context) ? EFX_FILTER_FLAG_RX_RSS : 0; |
|---|
| 5294 | | - |
|---|
| 5295 | | - /* Insert/renew filters */ |
|---|
| 5296 | | - for (i = 0; i < addr_count; i++) { |
|---|
| 5297 | | - EFX_WARN_ON_PARANOID(ids[i] != EFX_EF10_FILTER_ID_INVALID); |
|---|
| 5298 | | - efx_filter_init_rx(&spec, EFX_FILTER_PRI_AUTO, filter_flags, 0); |
|---|
| 5299 | | - efx_filter_set_eth_local(&spec, vlan->vid, addr_list[i].addr); |
|---|
| 5300 | | - rc = efx_ef10_filter_insert_locked(efx, &spec, true); |
|---|
| 5301 | | - if (rc < 0) { |
|---|
| 5302 | | - if (rollback) { |
|---|
| 5303 | | - netif_info(efx, drv, efx->net_dev, |
|---|
| 5304 | | - "efx_ef10_filter_insert failed rc=%d\n", |
|---|
| 5305 | | - rc); |
|---|
| 5306 | | - /* Fall back to promiscuous */ |
|---|
| 5307 | | - for (j = 0; j < i; j++) { |
|---|
| 5308 | | - efx_ef10_filter_remove_unsafe( |
|---|
| 5309 | | - efx, EFX_FILTER_PRI_AUTO, |
|---|
| 5310 | | - ids[j]); |
|---|
| 5311 | | - ids[j] = EFX_EF10_FILTER_ID_INVALID; |
|---|
| 5312 | | - } |
|---|
| 5313 | | - return rc; |
|---|
| 5314 | | - } else { |
|---|
| 5315 | | - /* keep invalid ID, and carry on */ |
|---|
| 5316 | | - } |
|---|
| 5317 | | - } else { |
|---|
| 5318 | | - ids[i] = efx_ef10_filter_get_unsafe_id(rc); |
|---|
| 5319 | | - } |
|---|
| 5320 | | - } |
|---|
| 5321 | | - |
|---|
| 5322 | | - if (multicast && rollback) { |
|---|
| 5323 | | - /* Also need an Ethernet broadcast filter */ |
|---|
| 5324 | | - EFX_WARN_ON_PARANOID(vlan->default_filters[EFX_EF10_BCAST] != |
|---|
| 5325 | | - EFX_EF10_FILTER_ID_INVALID); |
|---|
| 5326 | | - efx_filter_init_rx(&spec, EFX_FILTER_PRI_AUTO, filter_flags, 0); |
|---|
| 5327 | | - eth_broadcast_addr(baddr); |
|---|
| 5328 | | - efx_filter_set_eth_local(&spec, vlan->vid, baddr); |
|---|
| 5329 | | - rc = efx_ef10_filter_insert_locked(efx, &spec, true); |
|---|
| 5330 | | - if (rc < 0) { |
|---|
| 5331 | | - netif_warn(efx, drv, efx->net_dev, |
|---|
| 5332 | | - "Broadcast filter insert failed rc=%d\n", rc); |
|---|
| 5333 | | - /* Fall back to promiscuous */ |
|---|
| 5334 | | - for (j = 0; j < i; j++) { |
|---|
| 5335 | | - efx_ef10_filter_remove_unsafe( |
|---|
| 5336 | | - efx, EFX_FILTER_PRI_AUTO, |
|---|
| 5337 | | - ids[j]); |
|---|
| 5338 | | - ids[j] = EFX_EF10_FILTER_ID_INVALID; |
|---|
| 5339 | | - } |
|---|
| 5340 | | - return rc; |
|---|
| 5341 | | - } else { |
|---|
| 5342 | | - vlan->default_filters[EFX_EF10_BCAST] = |
|---|
| 5343 | | - efx_ef10_filter_get_unsafe_id(rc); |
|---|
| 5344 | | - } |
|---|
| 5345 | | - } |
|---|
| 5346 | | - |
|---|
| 5347 | | - return 0; |
|---|
| 5348 | | -} |
|---|
| 5349 | | - |
|---|
| 5350 | | -static int efx_ef10_filter_insert_def(struct efx_nic *efx, |
|---|
| 5351 | | - struct efx_ef10_filter_vlan *vlan, |
|---|
| 5352 | | - enum efx_encap_type encap_type, |
|---|
| 5353 | | - bool multicast, bool rollback) |
|---|
| 5354 | | -{ |
|---|
| 5355 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 5356 | | - enum efx_filter_flags filter_flags; |
|---|
| 5357 | | - struct efx_filter_spec spec; |
|---|
| 5358 | | - u8 baddr[ETH_ALEN]; |
|---|
| 5359 | | - int rc; |
|---|
| 5360 | | - u16 *id; |
|---|
| 5361 | | - |
|---|
| 5362 | | - filter_flags = efx_rss_active(&efx->rss_context) ? EFX_FILTER_FLAG_RX_RSS : 0; |
|---|
| 5363 | | - |
|---|
| 5364 | | - efx_filter_init_rx(&spec, EFX_FILTER_PRI_AUTO, filter_flags, 0); |
|---|
| 5365 | | - |
|---|
| 5366 | | - if (multicast) |
|---|
| 5367 | | - efx_filter_set_mc_def(&spec); |
|---|
| 5368 | | - else |
|---|
| 5369 | | - efx_filter_set_uc_def(&spec); |
|---|
| 5370 | | - |
|---|
| 5371 | | - if (encap_type) { |
|---|
| 5372 | | - if (nic_data->datapath_caps & |
|---|
| 5373 | | - (1 << MC_CMD_GET_CAPABILITIES_OUT_VXLAN_NVGRE_LBN)) |
|---|
| 5374 | | - efx_filter_set_encap_type(&spec, encap_type); |
|---|
| 5375 | | - else |
|---|
| 5376 | | - /* don't insert encap filters on non-supporting |
|---|
| 5377 | | - * platforms. ID will be left as INVALID. |
|---|
| 5378 | | - */ |
|---|
| 5379 | | - return 0; |
|---|
| 5380 | | - } |
|---|
| 5381 | | - |
|---|
| 5382 | | - if (vlan->vid != EFX_FILTER_VID_UNSPEC) |
|---|
| 5383 | | - efx_filter_set_eth_local(&spec, vlan->vid, NULL); |
|---|
| 5384 | | - |
|---|
| 5385 | | - rc = efx_ef10_filter_insert_locked(efx, &spec, true); |
|---|
| 5386 | | - if (rc < 0) { |
|---|
| 5387 | | - const char *um = multicast ? "Multicast" : "Unicast"; |
|---|
| 5388 | | - const char *encap_name = ""; |
|---|
| 5389 | | - const char *encap_ipv = ""; |
|---|
| 5390 | | - |
|---|
| 5391 | | - if ((encap_type & EFX_ENCAP_TYPES_MASK) == |
|---|
| 5392 | | - EFX_ENCAP_TYPE_VXLAN) |
|---|
| 5393 | | - encap_name = "VXLAN "; |
|---|
| 5394 | | - else if ((encap_type & EFX_ENCAP_TYPES_MASK) == |
|---|
| 5395 | | - EFX_ENCAP_TYPE_NVGRE) |
|---|
| 5396 | | - encap_name = "NVGRE "; |
|---|
| 5397 | | - else if ((encap_type & EFX_ENCAP_TYPES_MASK) == |
|---|
| 5398 | | - EFX_ENCAP_TYPE_GENEVE) |
|---|
| 5399 | | - encap_name = "GENEVE "; |
|---|
| 5400 | | - if (encap_type & EFX_ENCAP_FLAG_IPV6) |
|---|
| 5401 | | - encap_ipv = "IPv6 "; |
|---|
| 5402 | | - else if (encap_type) |
|---|
| 5403 | | - encap_ipv = "IPv4 "; |
|---|
| 5404 | | - |
|---|
| 5405 | | - /* unprivileged functions can't insert mismatch filters |
|---|
| 5406 | | - * for encapsulated or unicast traffic, so downgrade |
|---|
| 5407 | | - * those warnings to debug. |
|---|
| 5408 | | - */ |
|---|
| 5409 | | - netif_cond_dbg(efx, drv, efx->net_dev, |
|---|
| 5410 | | - rc == -EPERM && (encap_type || !multicast), warn, |
|---|
| 5411 | | - "%s%s%s mismatch filter insert failed rc=%d\n", |
|---|
| 5412 | | - encap_name, encap_ipv, um, rc); |
|---|
| 5413 | | - } else if (multicast) { |
|---|
| 5414 | | - /* mapping from encap types to default filter IDs (multicast) */ |
|---|
| 5415 | | - static enum efx_ef10_default_filters map[] = { |
|---|
| 5416 | | - [EFX_ENCAP_TYPE_NONE] = EFX_EF10_MCDEF, |
|---|
| 5417 | | - [EFX_ENCAP_TYPE_VXLAN] = EFX_EF10_VXLAN4_MCDEF, |
|---|
| 5418 | | - [EFX_ENCAP_TYPE_NVGRE] = EFX_EF10_NVGRE4_MCDEF, |
|---|
| 5419 | | - [EFX_ENCAP_TYPE_GENEVE] = EFX_EF10_GENEVE4_MCDEF, |
|---|
| 5420 | | - [EFX_ENCAP_TYPE_VXLAN | EFX_ENCAP_FLAG_IPV6] = |
|---|
| 5421 | | - EFX_EF10_VXLAN6_MCDEF, |
|---|
| 5422 | | - [EFX_ENCAP_TYPE_NVGRE | EFX_ENCAP_FLAG_IPV6] = |
|---|
| 5423 | | - EFX_EF10_NVGRE6_MCDEF, |
|---|
| 5424 | | - [EFX_ENCAP_TYPE_GENEVE | EFX_ENCAP_FLAG_IPV6] = |
|---|
| 5425 | | - EFX_EF10_GENEVE6_MCDEF, |
|---|
| 5426 | | - }; |
|---|
| 5427 | | - |
|---|
| 5428 | | - /* quick bounds check (BCAST result impossible) */ |
|---|
| 5429 | | - BUILD_BUG_ON(EFX_EF10_BCAST != 0); |
|---|
| 5430 | | - if (encap_type >= ARRAY_SIZE(map) || map[encap_type] == 0) { |
|---|
| 5431 | | - WARN_ON(1); |
|---|
| 5432 | | - return -EINVAL; |
|---|
| 5433 | | - } |
|---|
| 5434 | | - /* then follow map */ |
|---|
| 5435 | | - id = &vlan->default_filters[map[encap_type]]; |
|---|
| 5436 | | - |
|---|
| 5437 | | - EFX_WARN_ON_PARANOID(*id != EFX_EF10_FILTER_ID_INVALID); |
|---|
| 5438 | | - *id = efx_ef10_filter_get_unsafe_id(rc); |
|---|
| 5439 | | - if (!nic_data->workaround_26807 && !encap_type) { |
|---|
| 5440 | | - /* Also need an Ethernet broadcast filter */ |
|---|
| 5441 | | - efx_filter_init_rx(&spec, EFX_FILTER_PRI_AUTO, |
|---|
| 5442 | | - filter_flags, 0); |
|---|
| 5443 | | - eth_broadcast_addr(baddr); |
|---|
| 5444 | | - efx_filter_set_eth_local(&spec, vlan->vid, baddr); |
|---|
| 5445 | | - rc = efx_ef10_filter_insert_locked(efx, &spec, true); |
|---|
| 5446 | | - if (rc < 0) { |
|---|
| 5447 | | - netif_warn(efx, drv, efx->net_dev, |
|---|
| 5448 | | - "Broadcast filter insert failed rc=%d\n", |
|---|
| 5449 | | - rc); |
|---|
| 5450 | | - if (rollback) { |
|---|
| 5451 | | - /* Roll back the mc_def filter */ |
|---|
| 5452 | | - efx_ef10_filter_remove_unsafe( |
|---|
| 5453 | | - efx, EFX_FILTER_PRI_AUTO, |
|---|
| 5454 | | - *id); |
|---|
| 5455 | | - *id = EFX_EF10_FILTER_ID_INVALID; |
|---|
| 5456 | | - return rc; |
|---|
| 5457 | | - } |
|---|
| 5458 | | - } else { |
|---|
| 5459 | | - EFX_WARN_ON_PARANOID( |
|---|
| 5460 | | - vlan->default_filters[EFX_EF10_BCAST] != |
|---|
| 5461 | | - EFX_EF10_FILTER_ID_INVALID); |
|---|
| 5462 | | - vlan->default_filters[EFX_EF10_BCAST] = |
|---|
| 5463 | | - efx_ef10_filter_get_unsafe_id(rc); |
|---|
| 5464 | | - } |
|---|
| 5465 | | - } |
|---|
| 5466 | | - rc = 0; |
|---|
| 5467 | | - } else { |
|---|
| 5468 | | - /* mapping from encap types to default filter IDs (unicast) */ |
|---|
| 5469 | | - static enum efx_ef10_default_filters map[] = { |
|---|
| 5470 | | - [EFX_ENCAP_TYPE_NONE] = EFX_EF10_UCDEF, |
|---|
| 5471 | | - [EFX_ENCAP_TYPE_VXLAN] = EFX_EF10_VXLAN4_UCDEF, |
|---|
| 5472 | | - [EFX_ENCAP_TYPE_NVGRE] = EFX_EF10_NVGRE4_UCDEF, |
|---|
| 5473 | | - [EFX_ENCAP_TYPE_GENEVE] = EFX_EF10_GENEVE4_UCDEF, |
|---|
| 5474 | | - [EFX_ENCAP_TYPE_VXLAN | EFX_ENCAP_FLAG_IPV6] = |
|---|
| 5475 | | - EFX_EF10_VXLAN6_UCDEF, |
|---|
| 5476 | | - [EFX_ENCAP_TYPE_NVGRE | EFX_ENCAP_FLAG_IPV6] = |
|---|
| 5477 | | - EFX_EF10_NVGRE6_UCDEF, |
|---|
| 5478 | | - [EFX_ENCAP_TYPE_GENEVE | EFX_ENCAP_FLAG_IPV6] = |
|---|
| 5479 | | - EFX_EF10_GENEVE6_UCDEF, |
|---|
| 5480 | | - }; |
|---|
| 5481 | | - |
|---|
| 5482 | | - /* quick bounds check (BCAST result impossible) */ |
|---|
| 5483 | | - BUILD_BUG_ON(EFX_EF10_BCAST != 0); |
|---|
| 5484 | | - if (encap_type >= ARRAY_SIZE(map) || map[encap_type] == 0) { |
|---|
| 5485 | | - WARN_ON(1); |
|---|
| 5486 | | - return -EINVAL; |
|---|
| 5487 | | - } |
|---|
| 5488 | | - /* then follow map */ |
|---|
| 5489 | | - id = &vlan->default_filters[map[encap_type]]; |
|---|
| 5490 | | - EFX_WARN_ON_PARANOID(*id != EFX_EF10_FILTER_ID_INVALID); |
|---|
| 5491 | | - *id = rc; |
|---|
| 5492 | | - rc = 0; |
|---|
| 5493 | | - } |
|---|
| 5494 | | - return rc; |
|---|
| 5495 | | -} |
|---|
| 5496 | | - |
|---|
| 5497 | | -/* Remove filters that weren't renewed. */ |
|---|
| 5498 | | -static void efx_ef10_filter_remove_old(struct efx_nic *efx) |
|---|
| 5499 | | -{ |
|---|
| 5500 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 5501 | | - int remove_failed = 0; |
|---|
| 5502 | | - int remove_noent = 0; |
|---|
| 5503 | | - int rc; |
|---|
| 5504 | | - int i; |
|---|
| 5505 | | - |
|---|
| 5506 | | - down_write(&table->lock); |
|---|
| 5507 | | - for (i = 0; i < HUNT_FILTER_TBL_ROWS; i++) { |
|---|
| 5508 | | - if (READ_ONCE(table->entry[i].spec) & |
|---|
| 5509 | | - EFX_EF10_FILTER_FLAG_AUTO_OLD) { |
|---|
| 5510 | | - rc = efx_ef10_filter_remove_internal(efx, |
|---|
| 5511 | | - 1U << EFX_FILTER_PRI_AUTO, i, true); |
|---|
| 5512 | | - if (rc == -ENOENT) |
|---|
| 5513 | | - remove_noent++; |
|---|
| 5514 | | - else if (rc) |
|---|
| 5515 | | - remove_failed++; |
|---|
| 5516 | | - } |
|---|
| 5517 | | - } |
|---|
| 5518 | | - up_write(&table->lock); |
|---|
| 5519 | | - |
|---|
| 5520 | | - if (remove_failed) |
|---|
| 5521 | | - netif_info(efx, drv, efx->net_dev, |
|---|
| 5522 | | - "%s: failed to remove %d filters\n", |
|---|
| 5523 | | - __func__, remove_failed); |
|---|
| 5524 | | - if (remove_noent) |
|---|
| 5525 | | - netif_info(efx, drv, efx->net_dev, |
|---|
| 5526 | | - "%s: failed to remove %d non-existent filters\n", |
|---|
| 5527 | | - __func__, remove_noent); |
|---|
| 5528 | 3184 | } |
|---|
| 5529 | 3185 | |
|---|
| 5530 | 3186 | static int efx_ef10_vport_set_mac_address(struct efx_nic *efx) |
|---|
| .. | .. |
|---|
| 5540 | 3196 | efx_device_detach_sync(efx); |
|---|
| 5541 | 3197 | efx_net_stop(efx->net_dev); |
|---|
| 5542 | 3198 | down_write(&efx->filter_sem); |
|---|
| 5543 | | - efx_ef10_filter_table_remove(efx); |
|---|
| 3199 | + efx_mcdi_filter_table_remove(efx); |
|---|
| 5544 | 3200 | up_write(&efx->filter_sem); |
|---|
| 5545 | 3201 | |
|---|
| 5546 | | - rc = efx_ef10_vadaptor_free(efx, nic_data->vport_id); |
|---|
| 3202 | + rc = efx_ef10_vadaptor_free(efx, efx->vport_id); |
|---|
| 5547 | 3203 | if (rc) |
|---|
| 5548 | 3204 | goto restore_filters; |
|---|
| 5549 | 3205 | |
|---|
| 5550 | 3206 | ether_addr_copy(mac_old, nic_data->vport_mac); |
|---|
| 5551 | | - rc = efx_ef10_vport_del_mac(efx, nic_data->vport_id, |
|---|
| 3207 | + rc = efx_ef10_vport_del_mac(efx, efx->vport_id, |
|---|
| 5552 | 3208 | nic_data->vport_mac); |
|---|
| 5553 | 3209 | if (rc) |
|---|
| 5554 | 3210 | goto restore_vadaptor; |
|---|
| 5555 | 3211 | |
|---|
| 5556 | | - rc = efx_ef10_vport_add_mac(efx, nic_data->vport_id, |
|---|
| 3212 | + rc = efx_ef10_vport_add_mac(efx, efx->vport_id, |
|---|
| 5557 | 3213 | efx->net_dev->dev_addr); |
|---|
| 5558 | 3214 | if (!rc) { |
|---|
| 5559 | 3215 | ether_addr_copy(nic_data->vport_mac, efx->net_dev->dev_addr); |
|---|
| 5560 | 3216 | } else { |
|---|
| 5561 | | - rc2 = efx_ef10_vport_add_mac(efx, nic_data->vport_id, mac_old); |
|---|
| 3217 | + rc2 = efx_ef10_vport_add_mac(efx, efx->vport_id, mac_old); |
|---|
| 5562 | 3218 | if (rc2) { |
|---|
| 5563 | 3219 | /* Failed to add original MAC, so clear vport_mac */ |
|---|
| 5564 | 3220 | eth_zero_addr(nic_data->vport_mac); |
|---|
| .. | .. |
|---|
| 5567 | 3223 | } |
|---|
| 5568 | 3224 | |
|---|
| 5569 | 3225 | restore_vadaptor: |
|---|
| 5570 | | - rc2 = efx_ef10_vadaptor_alloc(efx, nic_data->vport_id); |
|---|
| 3226 | + rc2 = efx_ef10_vadaptor_alloc(efx, efx->vport_id); |
|---|
| 5571 | 3227 | if (rc2) |
|---|
| 5572 | 3228 | goto reset_nic; |
|---|
| 5573 | 3229 | restore_filters: |
|---|
| .. | .. |
|---|
| 5593 | 3249 | return rc ? rc : rc2; |
|---|
| 5594 | 3250 | } |
|---|
| 5595 | 3251 | |
|---|
| 5596 | | -/* Caller must hold efx->filter_sem for read if race against |
|---|
| 5597 | | - * efx_ef10_filter_table_remove() is possible |
|---|
| 5598 | | - */ |
|---|
| 5599 | | -static void efx_ef10_filter_vlan_sync_rx_mode(struct efx_nic *efx, |
|---|
| 5600 | | - struct efx_ef10_filter_vlan *vlan) |
|---|
| 5601 | | -{ |
|---|
| 5602 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 5603 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 5604 | | - |
|---|
| 5605 | | - /* Do not install unspecified VID if VLAN filtering is enabled. |
|---|
| 5606 | | - * Do not install all specified VIDs if VLAN filtering is disabled. |
|---|
| 5607 | | - */ |
|---|
| 5608 | | - if ((vlan->vid == EFX_FILTER_VID_UNSPEC) == table->vlan_filter) |
|---|
| 5609 | | - return; |
|---|
| 5610 | | - |
|---|
| 5611 | | - /* Insert/renew unicast filters */ |
|---|
| 5612 | | - if (table->uc_promisc) { |
|---|
| 5613 | | - efx_ef10_filter_insert_def(efx, vlan, EFX_ENCAP_TYPE_NONE, |
|---|
| 5614 | | - false, false); |
|---|
| 5615 | | - efx_ef10_filter_insert_addr_list(efx, vlan, false, false); |
|---|
| 5616 | | - } else { |
|---|
| 5617 | | - /* If any of the filters failed to insert, fall back to |
|---|
| 5618 | | - * promiscuous mode - add in the uc_def filter. But keep |
|---|
| 5619 | | - * our individual unicast filters. |
|---|
| 5620 | | - */ |
|---|
| 5621 | | - if (efx_ef10_filter_insert_addr_list(efx, vlan, false, false)) |
|---|
| 5622 | | - efx_ef10_filter_insert_def(efx, vlan, |
|---|
| 5623 | | - EFX_ENCAP_TYPE_NONE, |
|---|
| 5624 | | - false, false); |
|---|
| 5625 | | - } |
|---|
| 5626 | | - efx_ef10_filter_insert_def(efx, vlan, EFX_ENCAP_TYPE_VXLAN, |
|---|
| 5627 | | - false, false); |
|---|
| 5628 | | - efx_ef10_filter_insert_def(efx, vlan, EFX_ENCAP_TYPE_VXLAN | |
|---|
| 5629 | | - EFX_ENCAP_FLAG_IPV6, |
|---|
| 5630 | | - false, false); |
|---|
| 5631 | | - efx_ef10_filter_insert_def(efx, vlan, EFX_ENCAP_TYPE_NVGRE, |
|---|
| 5632 | | - false, false); |
|---|
| 5633 | | - efx_ef10_filter_insert_def(efx, vlan, EFX_ENCAP_TYPE_NVGRE | |
|---|
| 5634 | | - EFX_ENCAP_FLAG_IPV6, |
|---|
| 5635 | | - false, false); |
|---|
| 5636 | | - efx_ef10_filter_insert_def(efx, vlan, EFX_ENCAP_TYPE_GENEVE, |
|---|
| 5637 | | - false, false); |
|---|
| 5638 | | - efx_ef10_filter_insert_def(efx, vlan, EFX_ENCAP_TYPE_GENEVE | |
|---|
| 5639 | | - EFX_ENCAP_FLAG_IPV6, |
|---|
| 5640 | | - false, false); |
|---|
| 5641 | | - |
|---|
| 5642 | | - /* Insert/renew multicast filters */ |
|---|
| 5643 | | - /* If changing promiscuous state with cascaded multicast filters, remove |
|---|
| 5644 | | - * old filters first, so that packets are dropped rather than duplicated |
|---|
| 5645 | | - */ |
|---|
| 5646 | | - if (nic_data->workaround_26807 && |
|---|
| 5647 | | - table->mc_promisc_last != table->mc_promisc) |
|---|
| 5648 | | - efx_ef10_filter_remove_old(efx); |
|---|
| 5649 | | - if (table->mc_promisc) { |
|---|
| 5650 | | - if (nic_data->workaround_26807) { |
|---|
| 5651 | | - /* If we failed to insert promiscuous filters, rollback |
|---|
| 5652 | | - * and fall back to individual multicast filters |
|---|
| 5653 | | - */ |
|---|
| 5654 | | - if (efx_ef10_filter_insert_def(efx, vlan, |
|---|
| 5655 | | - EFX_ENCAP_TYPE_NONE, |
|---|
| 5656 | | - true, true)) { |
|---|
| 5657 | | - /* Changing promisc state, so remove old filters */ |
|---|
| 5658 | | - efx_ef10_filter_remove_old(efx); |
|---|
| 5659 | | - efx_ef10_filter_insert_addr_list(efx, vlan, |
|---|
| 5660 | | - true, false); |
|---|
| 5661 | | - } |
|---|
| 5662 | | - } else { |
|---|
| 5663 | | - /* If we failed to insert promiscuous filters, don't |
|---|
| 5664 | | - * rollback. Regardless, also insert the mc_list, |
|---|
| 5665 | | - * unless it's incomplete due to overflow |
|---|
| 5666 | | - */ |
|---|
| 5667 | | - efx_ef10_filter_insert_def(efx, vlan, |
|---|
| 5668 | | - EFX_ENCAP_TYPE_NONE, |
|---|
| 5669 | | - true, false); |
|---|
| 5670 | | - if (!table->mc_overflow) |
|---|
| 5671 | | - efx_ef10_filter_insert_addr_list(efx, vlan, |
|---|
| 5672 | | - true, false); |
|---|
| 5673 | | - } |
|---|
| 5674 | | - } else { |
|---|
| 5675 | | - /* If any filters failed to insert, rollback and fall back to |
|---|
| 5676 | | - * promiscuous mode - mc_def filter and maybe broadcast. If |
|---|
| 5677 | | - * that fails, roll back again and insert as many of our |
|---|
| 5678 | | - * individual multicast filters as we can. |
|---|
| 5679 | | - */ |
|---|
| 5680 | | - if (efx_ef10_filter_insert_addr_list(efx, vlan, true, true)) { |
|---|
| 5681 | | - /* Changing promisc state, so remove old filters */ |
|---|
| 5682 | | - if (nic_data->workaround_26807) |
|---|
| 5683 | | - efx_ef10_filter_remove_old(efx); |
|---|
| 5684 | | - if (efx_ef10_filter_insert_def(efx, vlan, |
|---|
| 5685 | | - EFX_ENCAP_TYPE_NONE, |
|---|
| 5686 | | - true, true)) |
|---|
| 5687 | | - efx_ef10_filter_insert_addr_list(efx, vlan, |
|---|
| 5688 | | - true, false); |
|---|
| 5689 | | - } |
|---|
| 5690 | | - } |
|---|
| 5691 | | - efx_ef10_filter_insert_def(efx, vlan, EFX_ENCAP_TYPE_VXLAN, |
|---|
| 5692 | | - true, false); |
|---|
| 5693 | | - efx_ef10_filter_insert_def(efx, vlan, EFX_ENCAP_TYPE_VXLAN | |
|---|
| 5694 | | - EFX_ENCAP_FLAG_IPV6, |
|---|
| 5695 | | - true, false); |
|---|
| 5696 | | - efx_ef10_filter_insert_def(efx, vlan, EFX_ENCAP_TYPE_NVGRE, |
|---|
| 5697 | | - true, false); |
|---|
| 5698 | | - efx_ef10_filter_insert_def(efx, vlan, EFX_ENCAP_TYPE_NVGRE | |
|---|
| 5699 | | - EFX_ENCAP_FLAG_IPV6, |
|---|
| 5700 | | - true, false); |
|---|
| 5701 | | - efx_ef10_filter_insert_def(efx, vlan, EFX_ENCAP_TYPE_GENEVE, |
|---|
| 5702 | | - true, false); |
|---|
| 5703 | | - efx_ef10_filter_insert_def(efx, vlan, EFX_ENCAP_TYPE_GENEVE | |
|---|
| 5704 | | - EFX_ENCAP_FLAG_IPV6, |
|---|
| 5705 | | - true, false); |
|---|
| 5706 | | -} |
|---|
| 5707 | | - |
|---|
| 5708 | | -/* Caller must hold efx->filter_sem for read if race against |
|---|
| 5709 | | - * efx_ef10_filter_table_remove() is possible |
|---|
| 5710 | | - */ |
|---|
| 5711 | | -static void efx_ef10_filter_sync_rx_mode(struct efx_nic *efx) |
|---|
| 5712 | | -{ |
|---|
| 5713 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 5714 | | - struct net_device *net_dev = efx->net_dev; |
|---|
| 5715 | | - struct efx_ef10_filter_vlan *vlan; |
|---|
| 5716 | | - bool vlan_filter; |
|---|
| 5717 | | - |
|---|
| 5718 | | - if (!efx_dev_registered(efx)) |
|---|
| 5719 | | - return; |
|---|
| 5720 | | - |
|---|
| 5721 | | - if (!table) |
|---|
| 5722 | | - return; |
|---|
| 5723 | | - |
|---|
| 5724 | | - efx_ef10_filter_mark_old(efx); |
|---|
| 5725 | | - |
|---|
| 5726 | | - /* Copy/convert the address lists; add the primary station |
|---|
| 5727 | | - * address and broadcast address |
|---|
| 5728 | | - */ |
|---|
| 5729 | | - netif_addr_lock_bh(net_dev); |
|---|
| 5730 | | - efx_ef10_filter_uc_addr_list(efx); |
|---|
| 5731 | | - efx_ef10_filter_mc_addr_list(efx); |
|---|
| 5732 | | - netif_addr_unlock_bh(net_dev); |
|---|
| 5733 | | - |
|---|
| 5734 | | - /* If VLAN filtering changes, all old filters are finally removed. |
|---|
| 5735 | | - * Do it in advance to avoid conflicts for unicast untagged and |
|---|
| 5736 | | - * VLAN 0 tagged filters. |
|---|
| 5737 | | - */ |
|---|
| 5738 | | - vlan_filter = !!(net_dev->features & NETIF_F_HW_VLAN_CTAG_FILTER); |
|---|
| 5739 | | - if (table->vlan_filter != vlan_filter) { |
|---|
| 5740 | | - table->vlan_filter = vlan_filter; |
|---|
| 5741 | | - efx_ef10_filter_remove_old(efx); |
|---|
| 5742 | | - } |
|---|
| 5743 | | - |
|---|
| 5744 | | - list_for_each_entry(vlan, &table->vlan_list, list) |
|---|
| 5745 | | - efx_ef10_filter_vlan_sync_rx_mode(efx, vlan); |
|---|
| 5746 | | - |
|---|
| 5747 | | - efx_ef10_filter_remove_old(efx); |
|---|
| 5748 | | - table->mc_promisc_last = table->mc_promisc; |
|---|
| 5749 | | -} |
|---|
| 5750 | | - |
|---|
| 5751 | | -static struct efx_ef10_filter_vlan *efx_ef10_filter_find_vlan(struct efx_nic *efx, u16 vid) |
|---|
| 5752 | | -{ |
|---|
| 5753 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 5754 | | - struct efx_ef10_filter_vlan *vlan; |
|---|
| 5755 | | - |
|---|
| 5756 | | - WARN_ON(!rwsem_is_locked(&efx->filter_sem)); |
|---|
| 5757 | | - |
|---|
| 5758 | | - list_for_each_entry(vlan, &table->vlan_list, list) { |
|---|
| 5759 | | - if (vlan->vid == vid) |
|---|
| 5760 | | - return vlan; |
|---|
| 5761 | | - } |
|---|
| 5762 | | - |
|---|
| 5763 | | - return NULL; |
|---|
| 5764 | | -} |
|---|
| 5765 | | - |
|---|
| 5766 | | -static int efx_ef10_filter_add_vlan(struct efx_nic *efx, u16 vid) |
|---|
| 5767 | | -{ |
|---|
| 5768 | | - struct efx_ef10_filter_table *table = efx->filter_state; |
|---|
| 5769 | | - struct efx_ef10_filter_vlan *vlan; |
|---|
| 5770 | | - unsigned int i; |
|---|
| 5771 | | - |
|---|
| 5772 | | - if (!efx_rwsem_assert_write_locked(&efx->filter_sem)) |
|---|
| 5773 | | - return -EINVAL; |
|---|
| 5774 | | - |
|---|
| 5775 | | - vlan = efx_ef10_filter_find_vlan(efx, vid); |
|---|
| 5776 | | - if (WARN_ON(vlan)) { |
|---|
| 5777 | | - netif_err(efx, drv, efx->net_dev, |
|---|
| 5778 | | - "VLAN %u already added\n", vid); |
|---|
| 5779 | | - return -EALREADY; |
|---|
| 5780 | | - } |
|---|
| 5781 | | - |
|---|
| 5782 | | - vlan = kzalloc(sizeof(*vlan), GFP_KERNEL); |
|---|
| 5783 | | - if (!vlan) |
|---|
| 5784 | | - return -ENOMEM; |
|---|
| 5785 | | - |
|---|
| 5786 | | - vlan->vid = vid; |
|---|
| 5787 | | - |
|---|
| 5788 | | - for (i = 0; i < ARRAY_SIZE(vlan->uc); i++) |
|---|
| 5789 | | - vlan->uc[i] = EFX_EF10_FILTER_ID_INVALID; |
|---|
| 5790 | | - for (i = 0; i < ARRAY_SIZE(vlan->mc); i++) |
|---|
| 5791 | | - vlan->mc[i] = EFX_EF10_FILTER_ID_INVALID; |
|---|
| 5792 | | - for (i = 0; i < EFX_EF10_NUM_DEFAULT_FILTERS; i++) |
|---|
| 5793 | | - vlan->default_filters[i] = EFX_EF10_FILTER_ID_INVALID; |
|---|
| 5794 | | - |
|---|
| 5795 | | - list_add_tail(&vlan->list, &table->vlan_list); |
|---|
| 5796 | | - |
|---|
| 5797 | | - if (efx_dev_registered(efx)) |
|---|
| 5798 | | - efx_ef10_filter_vlan_sync_rx_mode(efx, vlan); |
|---|
| 5799 | | - |
|---|
| 5800 | | - return 0; |
|---|
| 5801 | | -} |
|---|
| 5802 | | - |
|---|
| 5803 | | -static void efx_ef10_filter_del_vlan_internal(struct efx_nic *efx, |
|---|
| 5804 | | - struct efx_ef10_filter_vlan *vlan) |
|---|
| 5805 | | -{ |
|---|
| 5806 | | - unsigned int i; |
|---|
| 5807 | | - |
|---|
| 5808 | | - /* See comment in efx_ef10_filter_table_remove() */ |
|---|
| 5809 | | - if (!efx_rwsem_assert_write_locked(&efx->filter_sem)) |
|---|
| 5810 | | - return; |
|---|
| 5811 | | - |
|---|
| 5812 | | - list_del(&vlan->list); |
|---|
| 5813 | | - |
|---|
| 5814 | | - for (i = 0; i < ARRAY_SIZE(vlan->uc); i++) |
|---|
| 5815 | | - efx_ef10_filter_remove_unsafe(efx, EFX_FILTER_PRI_AUTO, |
|---|
| 5816 | | - vlan->uc[i]); |
|---|
| 5817 | | - for (i = 0; i < ARRAY_SIZE(vlan->mc); i++) |
|---|
| 5818 | | - efx_ef10_filter_remove_unsafe(efx, EFX_FILTER_PRI_AUTO, |
|---|
| 5819 | | - vlan->mc[i]); |
|---|
| 5820 | | - for (i = 0; i < EFX_EF10_NUM_DEFAULT_FILTERS; i++) |
|---|
| 5821 | | - if (vlan->default_filters[i] != EFX_EF10_FILTER_ID_INVALID) |
|---|
| 5822 | | - efx_ef10_filter_remove_unsafe(efx, EFX_FILTER_PRI_AUTO, |
|---|
| 5823 | | - vlan->default_filters[i]); |
|---|
| 5824 | | - |
|---|
| 5825 | | - kfree(vlan); |
|---|
| 5826 | | -} |
|---|
| 5827 | | - |
|---|
| 5828 | | -static void efx_ef10_filter_del_vlan(struct efx_nic *efx, u16 vid) |
|---|
| 5829 | | -{ |
|---|
| 5830 | | - struct efx_ef10_filter_vlan *vlan; |
|---|
| 5831 | | - |
|---|
| 5832 | | - /* See comment in efx_ef10_filter_table_remove() */ |
|---|
| 5833 | | - if (!efx_rwsem_assert_write_locked(&efx->filter_sem)) |
|---|
| 5834 | | - return; |
|---|
| 5835 | | - |
|---|
| 5836 | | - vlan = efx_ef10_filter_find_vlan(efx, vid); |
|---|
| 5837 | | - if (!vlan) { |
|---|
| 5838 | | - netif_err(efx, drv, efx->net_dev, |
|---|
| 5839 | | - "VLAN %u not found in filter state\n", vid); |
|---|
| 5840 | | - return; |
|---|
| 5841 | | - } |
|---|
| 5842 | | - |
|---|
| 5843 | | - efx_ef10_filter_del_vlan_internal(efx, vlan); |
|---|
| 5844 | | -} |
|---|
| 5845 | | - |
|---|
| 5846 | 3252 | static int efx_ef10_set_mac_address(struct efx_nic *efx) |
|---|
| 5847 | 3253 | { |
|---|
| 5848 | 3254 | MCDI_DECLARE_BUF(inbuf, MC_CMD_VADAPTOR_SET_MAC_IN_LEN); |
|---|
| 5849 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 5850 | 3255 | bool was_enabled = efx->port_enabled; |
|---|
| 5851 | 3256 | int rc; |
|---|
| 3257 | + |
|---|
| 3258 | +#ifdef CONFIG_SFC_SRIOV |
|---|
| 3259 | + /* If this function is a VF and we have access to the parent PF, |
|---|
| 3260 | + * then use the PF control path to attempt to change the VF MAC address. |
|---|
| 3261 | + */ |
|---|
| 3262 | + if (efx->pci_dev->is_virtfn && efx->pci_dev->physfn) { |
|---|
| 3263 | + struct efx_nic *efx_pf = pci_get_drvdata(efx->pci_dev->physfn); |
|---|
| 3264 | + struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 3265 | + u8 mac[ETH_ALEN]; |
|---|
| 3266 | + |
|---|
| 3267 | + /* net_dev->dev_addr can be zeroed by efx_net_stop in |
|---|
| 3268 | + * efx_ef10_sriov_set_vf_mac, so pass in a copy. |
|---|
| 3269 | + */ |
|---|
| 3270 | + ether_addr_copy(mac, efx->net_dev->dev_addr); |
|---|
| 3271 | + |
|---|
| 3272 | + rc = efx_ef10_sriov_set_vf_mac(efx_pf, nic_data->vf_index, mac); |
|---|
| 3273 | + if (!rc) |
|---|
| 3274 | + return 0; |
|---|
| 3275 | + |
|---|
| 3276 | + netif_dbg(efx, drv, efx->net_dev, |
|---|
| 3277 | + "Updating VF mac via PF failed (%d), setting directly\n", |
|---|
| 3278 | + rc); |
|---|
| 3279 | + } |
|---|
| 3280 | +#endif |
|---|
| 5852 | 3281 | |
|---|
| 5853 | 3282 | efx_device_detach_sync(efx); |
|---|
| 5854 | 3283 | efx_net_stop(efx->net_dev); |
|---|
| 5855 | 3284 | |
|---|
| 5856 | 3285 | mutex_lock(&efx->mac_lock); |
|---|
| 5857 | 3286 | down_write(&efx->filter_sem); |
|---|
| 5858 | | - efx_ef10_filter_table_remove(efx); |
|---|
| 3287 | + efx_mcdi_filter_table_remove(efx); |
|---|
| 5859 | 3288 | |
|---|
| 5860 | 3289 | ether_addr_copy(MCDI_PTR(inbuf, VADAPTOR_SET_MAC_IN_MACADDR), |
|---|
| 5861 | 3290 | efx->net_dev->dev_addr); |
|---|
| 5862 | 3291 | MCDI_SET_DWORD(inbuf, VADAPTOR_SET_MAC_IN_UPSTREAM_PORT_ID, |
|---|
| 5863 | | - nic_data->vport_id); |
|---|
| 3292 | + efx->vport_id); |
|---|
| 5864 | 3293 | rc = efx_mcdi_rpc_quiet(efx, MC_CMD_VADAPTOR_SET_MAC, inbuf, |
|---|
| 5865 | 3294 | sizeof(inbuf), NULL, 0, NULL); |
|---|
| 5866 | 3295 | |
|---|
| .. | .. |
|---|
| 5872 | 3301 | efx_net_open(efx->net_dev); |
|---|
| 5873 | 3302 | efx_device_attach_if_not_resetting(efx); |
|---|
| 5874 | 3303 | |
|---|
| 5875 | | -#ifdef CONFIG_SFC_SRIOV |
|---|
| 5876 | | - if (efx->pci_dev->is_virtfn && efx->pci_dev->physfn) { |
|---|
| 5877 | | - struct pci_dev *pci_dev_pf = efx->pci_dev->physfn; |
|---|
| 5878 | | - |
|---|
| 5879 | | - if (rc == -EPERM) { |
|---|
| 5880 | | - struct efx_nic *efx_pf; |
|---|
| 5881 | | - |
|---|
| 5882 | | - /* Switch to PF and change MAC address on vport */ |
|---|
| 5883 | | - efx_pf = pci_get_drvdata(pci_dev_pf); |
|---|
| 5884 | | - |
|---|
| 5885 | | - rc = efx_ef10_sriov_set_vf_mac(efx_pf, |
|---|
| 5886 | | - nic_data->vf_index, |
|---|
| 5887 | | - efx->net_dev->dev_addr); |
|---|
| 5888 | | - } else if (!rc) { |
|---|
| 5889 | | - struct efx_nic *efx_pf = pci_get_drvdata(pci_dev_pf); |
|---|
| 5890 | | - struct efx_ef10_nic_data *nic_data = efx_pf->nic_data; |
|---|
| 5891 | | - unsigned int i; |
|---|
| 5892 | | - |
|---|
| 5893 | | - /* MAC address successfully changed by VF (with MAC |
|---|
| 5894 | | - * spoofing) so update the parent PF if possible. |
|---|
| 5895 | | - */ |
|---|
| 5896 | | - for (i = 0; i < efx_pf->vf_count; ++i) { |
|---|
| 5897 | | - struct ef10_vf *vf = nic_data->vf + i; |
|---|
| 5898 | | - |
|---|
| 5899 | | - if (vf->efx == efx) { |
|---|
| 5900 | | - ether_addr_copy(vf->mac, |
|---|
| 5901 | | - efx->net_dev->dev_addr); |
|---|
| 5902 | | - return 0; |
|---|
| 5903 | | - } |
|---|
| 5904 | | - } |
|---|
| 5905 | | - } |
|---|
| 5906 | | - } else |
|---|
| 5907 | | -#endif |
|---|
| 5908 | 3304 | if (rc == -EPERM) { |
|---|
| 5909 | 3305 | netif_err(efx, drv, efx->net_dev, |
|---|
| 5910 | 3306 | "Cannot change MAC address; use sfboot to enable" |
|---|
| .. | .. |
|---|
| 5924 | 3320 | return rc; |
|---|
| 5925 | 3321 | } |
|---|
| 5926 | 3322 | |
|---|
| 5927 | | -static int efx_ef10_mac_reconfigure(struct efx_nic *efx) |
|---|
| 3323 | +static int efx_ef10_mac_reconfigure(struct efx_nic *efx, bool mtu_only) |
|---|
| 5928 | 3324 | { |
|---|
| 5929 | | - efx_ef10_filter_sync_rx_mode(efx); |
|---|
| 3325 | + WARN_ON(!mutex_is_locked(&efx->mac_lock)); |
|---|
| 5930 | 3326 | |
|---|
| 3327 | + efx_mcdi_filter_sync_rx_mode(efx); |
|---|
| 3328 | + |
|---|
| 3329 | + if (mtu_only && efx_has_cap(efx, SET_MAC_ENHANCED)) |
|---|
| 3330 | + return efx_mcdi_set_mtu(efx); |
|---|
| 5931 | 3331 | return efx_mcdi_set_mac(efx); |
|---|
| 5932 | | -} |
|---|
| 5933 | | - |
|---|
| 5934 | | -static int efx_ef10_mac_reconfigure_vf(struct efx_nic *efx) |
|---|
| 5935 | | -{ |
|---|
| 5936 | | - efx_ef10_filter_sync_rx_mode(efx); |
|---|
| 5937 | | - |
|---|
| 5938 | | - return 0; |
|---|
| 5939 | 3332 | } |
|---|
| 5940 | 3333 | |
|---|
| 5941 | 3334 | static int efx_ef10_start_bist(struct efx_nic *efx, u32 bist_type) |
|---|
| .. | .. |
|---|
| 6041 | 3434 | { NVRAM_PARTITION_TYPE_EXPROM_CONFIG_PORT3, 0, 3, "sfc_exp_rom_cfg" }, |
|---|
| 6042 | 3435 | { NVRAM_PARTITION_TYPE_LICENSE, 0, 0, "sfc_license" }, |
|---|
| 6043 | 3436 | { NVRAM_PARTITION_TYPE_PHY_MIN, 0xff, 0, "sfc_phy_fw" }, |
|---|
| 3437 | + { NVRAM_PARTITION_TYPE_MUM_FIRMWARE, 0, 0, "sfc_mumfw" }, |
|---|
| 3438 | + { NVRAM_PARTITION_TYPE_EXPANSION_UEFI, 0, 0, "sfc_uefi" }, |
|---|
| 3439 | + { NVRAM_PARTITION_TYPE_DYNCONFIG_DEFAULTS, 0, 0, "sfc_dynamic_cfg_dflt" }, |
|---|
| 3440 | + { NVRAM_PARTITION_TYPE_ROMCONFIG_DEFAULTS, 0, 0, "sfc_exp_rom_cfg_dflt" }, |
|---|
| 3441 | + { NVRAM_PARTITION_TYPE_STATUS, 0, 0, "sfc_status" }, |
|---|
| 3442 | + { NVRAM_PARTITION_TYPE_BUNDLE, 0, 0, "sfc_bundle" }, |
|---|
| 3443 | + { NVRAM_PARTITION_TYPE_BUNDLE_METADATA, 0, 0, "sfc_bundle_metadata" }, |
|---|
| 6044 | 3444 | }; |
|---|
| 6045 | 3445 | #define EF10_NVRAM_PARTITION_COUNT ARRAY_SIZE(efx_ef10_nvram_types) |
|---|
| 6046 | 3446 | |
|---|
| .. | .. |
|---|
| 6070 | 3470 | rc = efx_mcdi_nvram_info(efx, type, &size, &erase_size, &protected); |
|---|
| 6071 | 3471 | if (rc) |
|---|
| 6072 | 3472 | return rc; |
|---|
| 3473 | + if (protected && |
|---|
| 3474 | + (type != NVRAM_PARTITION_TYPE_DYNCONFIG_DEFAULTS && |
|---|
| 3475 | + type != NVRAM_PARTITION_TYPE_ROMCONFIG_DEFAULTS)) |
|---|
| 3476 | + /* Hide protected partitions that don't provide defaults. */ |
|---|
| 3477 | + return -ENODEV; |
|---|
| 3478 | + |
|---|
| 6073 | 3479 | if (protected) |
|---|
| 6074 | | - return -ENODEV; /* hide it */ |
|---|
| 3480 | + /* Protected partitions are read only. */ |
|---|
| 3481 | + erase_size = 0; |
|---|
| 6075 | 3482 | |
|---|
| 6076 | 3483 | /* If we've already exposed a partition of this type, hide this |
|---|
| 6077 | 3484 | * duplicate. All operations on MTDs are keyed by the type anyway, |
|---|
| .. | .. |
|---|
| 6101 | 3508 | part->common.mtd.flags = MTD_CAP_NORFLASH; |
|---|
| 6102 | 3509 | part->common.mtd.size = size; |
|---|
| 6103 | 3510 | part->common.mtd.erasesize = erase_size; |
|---|
| 3511 | + /* sfc_status is read-only */ |
|---|
| 3512 | + if (!erase_size) |
|---|
| 3513 | + part->common.mtd.flags |= MTD_NO_ERASE; |
|---|
| 6104 | 3514 | |
|---|
| 6105 | 3515 | return 0; |
|---|
| 6106 | 3516 | } |
|---|
| .. | .. |
|---|
| 6144 | 3554 | if (rc) |
|---|
| 6145 | 3555 | goto fail; |
|---|
| 6146 | 3556 | n_parts++; |
|---|
| 3557 | + } |
|---|
| 3558 | + |
|---|
| 3559 | + if (!n_parts) { |
|---|
| 3560 | + kfree(parts); |
|---|
| 3561 | + return 0; |
|---|
| 6147 | 3562 | } |
|---|
| 6148 | 3563 | |
|---|
| 6149 | 3564 | rc = efx_mtd_add(efx, &parts[0].common, n_parts, sizeof(*parts)); |
|---|
| .. | .. |
|---|
| 6345 | 3760 | MC_CMD_SET_TUNNEL_ENCAP_UDP_PORTS_IN_ENTRIES_MAXNUM); |
|---|
| 6346 | 3761 | |
|---|
| 6347 | 3762 | for (i = 0; i < ARRAY_SIZE(nic_data->udp_tunnels); ++i) { |
|---|
| 6348 | | - if (nic_data->udp_tunnels[i].count && |
|---|
| 6349 | | - nic_data->udp_tunnels[i].port) { |
|---|
| 3763 | + if (nic_data->udp_tunnels[i].type != |
|---|
| 3764 | + TUNNEL_ENCAP_UDP_PORT_ENTRY_INVALID) { |
|---|
| 6350 | 3765 | efx_dword_t entry; |
|---|
| 6351 | 3766 | |
|---|
| 6352 | 3767 | EFX_POPULATE_DWORD_2(entry, |
|---|
| .. | .. |
|---|
| 6432 | 3847 | return rc; |
|---|
| 6433 | 3848 | } |
|---|
| 6434 | 3849 | |
|---|
| 6435 | | -static struct efx_udp_tunnel *__efx_ef10_udp_tnl_lookup_port(struct efx_nic *efx, |
|---|
| 6436 | | - __be16 port) |
|---|
| 3850 | +static int efx_ef10_udp_tnl_set_port(struct net_device *dev, |
|---|
| 3851 | + unsigned int table, unsigned int entry, |
|---|
| 3852 | + struct udp_tunnel_info *ti) |
|---|
| 6437 | 3853 | { |
|---|
| 6438 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 6439 | | - size_t i; |
|---|
| 3854 | + struct efx_nic *efx = netdev_priv(dev); |
|---|
| 3855 | + struct efx_ef10_nic_data *nic_data; |
|---|
| 3856 | + int efx_tunnel_type, rc; |
|---|
| 6440 | 3857 | |
|---|
| 6441 | | - for (i = 0; i < ARRAY_SIZE(nic_data->udp_tunnels); ++i) { |
|---|
| 6442 | | - if (!nic_data->udp_tunnels[i].count) |
|---|
| 6443 | | - continue; |
|---|
| 6444 | | - if (nic_data->udp_tunnels[i].port == port) |
|---|
| 6445 | | - return &nic_data->udp_tunnels[i]; |
|---|
| 6446 | | - } |
|---|
| 6447 | | - return NULL; |
|---|
| 6448 | | -} |
|---|
| 3858 | + if (ti->type == UDP_TUNNEL_TYPE_VXLAN) |
|---|
| 3859 | + efx_tunnel_type = TUNNEL_ENCAP_UDP_PORT_ENTRY_VXLAN; |
|---|
| 3860 | + else |
|---|
| 3861 | + efx_tunnel_type = TUNNEL_ENCAP_UDP_PORT_ENTRY_GENEVE; |
|---|
| 6449 | 3862 | |
|---|
| 6450 | | -static int efx_ef10_udp_tnl_add_port(struct efx_nic *efx, |
|---|
| 6451 | | - struct efx_udp_tunnel tnl) |
|---|
| 6452 | | -{ |
|---|
| 6453 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 6454 | | - struct efx_udp_tunnel *match; |
|---|
| 6455 | | - char typebuf[8]; |
|---|
| 6456 | | - size_t i; |
|---|
| 6457 | | - int rc; |
|---|
| 6458 | | - |
|---|
| 3863 | + nic_data = efx->nic_data; |
|---|
| 6459 | 3864 | if (!(nic_data->datapath_caps & |
|---|
| 6460 | 3865 | (1 << MC_CMD_GET_CAPABILITIES_OUT_VXLAN_NVGRE_LBN))) |
|---|
| 6461 | | - return 0; |
|---|
| 6462 | | - |
|---|
| 6463 | | - efx_get_udp_tunnel_type_name(tnl.type, typebuf, sizeof(typebuf)); |
|---|
| 6464 | | - netif_dbg(efx, drv, efx->net_dev, "Adding UDP tunnel (%s) port %d\n", |
|---|
| 6465 | | - typebuf, ntohs(tnl.port)); |
|---|
| 3866 | + return -EOPNOTSUPP; |
|---|
| 6466 | 3867 | |
|---|
| 6467 | 3868 | mutex_lock(&nic_data->udp_tunnels_lock); |
|---|
| 6468 | 3869 | /* Make sure all TX are stopped while we add to the table, else we |
|---|
| 6469 | 3870 | * might race against an efx_features_check(). |
|---|
| 6470 | 3871 | */ |
|---|
| 6471 | 3872 | efx_device_detach_sync(efx); |
|---|
| 6472 | | - |
|---|
| 6473 | | - match = __efx_ef10_udp_tnl_lookup_port(efx, tnl.port); |
|---|
| 6474 | | - if (match != NULL) { |
|---|
| 6475 | | - if (match->type == tnl.type) { |
|---|
| 6476 | | - netif_dbg(efx, drv, efx->net_dev, |
|---|
| 6477 | | - "Referencing existing tunnel entry\n"); |
|---|
| 6478 | | - match->count++; |
|---|
| 6479 | | - /* No need to cause an MCDI update */ |
|---|
| 6480 | | - rc = 0; |
|---|
| 6481 | | - goto unlock_out; |
|---|
| 6482 | | - } |
|---|
| 6483 | | - efx_get_udp_tunnel_type_name(match->type, |
|---|
| 6484 | | - typebuf, sizeof(typebuf)); |
|---|
| 6485 | | - netif_dbg(efx, drv, efx->net_dev, |
|---|
| 6486 | | - "UDP port %d is already in use by %s\n", |
|---|
| 6487 | | - ntohs(tnl.port), typebuf); |
|---|
| 6488 | | - rc = -EEXIST; |
|---|
| 6489 | | - goto unlock_out; |
|---|
| 6490 | | - } |
|---|
| 6491 | | - |
|---|
| 6492 | | - for (i = 0; i < ARRAY_SIZE(nic_data->udp_tunnels); ++i) |
|---|
| 6493 | | - if (!nic_data->udp_tunnels[i].count) { |
|---|
| 6494 | | - nic_data->udp_tunnels[i] = tnl; |
|---|
| 6495 | | - nic_data->udp_tunnels[i].count = 1; |
|---|
| 6496 | | - rc = efx_ef10_set_udp_tnl_ports(efx, false); |
|---|
| 6497 | | - goto unlock_out; |
|---|
| 6498 | | - } |
|---|
| 6499 | | - |
|---|
| 6500 | | - netif_dbg(efx, drv, efx->net_dev, |
|---|
| 6501 | | - "Unable to add UDP tunnel (%s) port %d; insufficient resources.\n", |
|---|
| 6502 | | - typebuf, ntohs(tnl.port)); |
|---|
| 6503 | | - |
|---|
| 6504 | | - rc = -ENOMEM; |
|---|
| 6505 | | - |
|---|
| 6506 | | -unlock_out: |
|---|
| 3873 | + nic_data->udp_tunnels[entry].type = efx_tunnel_type; |
|---|
| 3874 | + nic_data->udp_tunnels[entry].port = ti->port; |
|---|
| 3875 | + rc = efx_ef10_set_udp_tnl_ports(efx, false); |
|---|
| 6507 | 3876 | mutex_unlock(&nic_data->udp_tunnels_lock); |
|---|
| 3877 | + |
|---|
| 6508 | 3878 | return rc; |
|---|
| 6509 | 3879 | } |
|---|
| 6510 | 3880 | |
|---|
| .. | .. |
|---|
| 6516 | 3886 | static bool efx_ef10_udp_tnl_has_port(struct efx_nic *efx, __be16 port) |
|---|
| 6517 | 3887 | { |
|---|
| 6518 | 3888 | struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 3889 | + size_t i; |
|---|
| 6519 | 3890 | |
|---|
| 6520 | 3891 | if (!(nic_data->datapath_caps & |
|---|
| 6521 | 3892 | (1 << MC_CMD_GET_CAPABILITIES_OUT_VXLAN_NVGRE_LBN))) |
|---|
| .. | .. |
|---|
| 6527 | 3898 | */ |
|---|
| 6528 | 3899 | return false; |
|---|
| 6529 | 3900 | |
|---|
| 6530 | | - return __efx_ef10_udp_tnl_lookup_port(efx, port) != NULL; |
|---|
| 3901 | + for (i = 0; i < ARRAY_SIZE(nic_data->udp_tunnels); ++i) |
|---|
| 3902 | + if (nic_data->udp_tunnels[i].type != |
|---|
| 3903 | + TUNNEL_ENCAP_UDP_PORT_ENTRY_INVALID && |
|---|
| 3904 | + nic_data->udp_tunnels[i].port == port) |
|---|
| 3905 | + return true; |
|---|
| 3906 | + |
|---|
| 3907 | + return false; |
|---|
| 6531 | 3908 | } |
|---|
| 6532 | 3909 | |
|---|
| 6533 | | -static int efx_ef10_udp_tnl_del_port(struct efx_nic *efx, |
|---|
| 6534 | | - struct efx_udp_tunnel tnl) |
|---|
| 3910 | +static int efx_ef10_udp_tnl_unset_port(struct net_device *dev, |
|---|
| 3911 | + unsigned int table, unsigned int entry, |
|---|
| 3912 | + struct udp_tunnel_info *ti) |
|---|
| 6535 | 3913 | { |
|---|
| 6536 | | - struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 6537 | | - struct efx_udp_tunnel *match; |
|---|
| 6538 | | - char typebuf[8]; |
|---|
| 3914 | + struct efx_nic *efx = netdev_priv(dev); |
|---|
| 3915 | + struct efx_ef10_nic_data *nic_data; |
|---|
| 6539 | 3916 | int rc; |
|---|
| 6540 | 3917 | |
|---|
| 6541 | | - if (!(nic_data->datapath_caps & |
|---|
| 6542 | | - (1 << MC_CMD_GET_CAPABILITIES_OUT_VXLAN_NVGRE_LBN))) |
|---|
| 6543 | | - return 0; |
|---|
| 6544 | | - |
|---|
| 6545 | | - efx_get_udp_tunnel_type_name(tnl.type, typebuf, sizeof(typebuf)); |
|---|
| 6546 | | - netif_dbg(efx, drv, efx->net_dev, "Removing UDP tunnel (%s) port %d\n", |
|---|
| 6547 | | - typebuf, ntohs(tnl.port)); |
|---|
| 3918 | + nic_data = efx->nic_data; |
|---|
| 6548 | 3919 | |
|---|
| 6549 | 3920 | mutex_lock(&nic_data->udp_tunnels_lock); |
|---|
| 6550 | 3921 | /* Make sure all TX are stopped while we remove from the table, else we |
|---|
| 6551 | 3922 | * might race against an efx_features_check(). |
|---|
| 6552 | 3923 | */ |
|---|
| 6553 | 3924 | efx_device_detach_sync(efx); |
|---|
| 6554 | | - |
|---|
| 6555 | | - match = __efx_ef10_udp_tnl_lookup_port(efx, tnl.port); |
|---|
| 6556 | | - if (match != NULL) { |
|---|
| 6557 | | - if (match->type == tnl.type) { |
|---|
| 6558 | | - if (--match->count) { |
|---|
| 6559 | | - /* Port is still in use, so nothing to do */ |
|---|
| 6560 | | - netif_dbg(efx, drv, efx->net_dev, |
|---|
| 6561 | | - "UDP tunnel port %d remains active\n", |
|---|
| 6562 | | - ntohs(tnl.port)); |
|---|
| 6563 | | - rc = 0; |
|---|
| 6564 | | - goto out_unlock; |
|---|
| 6565 | | - } |
|---|
| 6566 | | - rc = efx_ef10_set_udp_tnl_ports(efx, false); |
|---|
| 6567 | | - goto out_unlock; |
|---|
| 6568 | | - } |
|---|
| 6569 | | - efx_get_udp_tunnel_type_name(match->type, |
|---|
| 6570 | | - typebuf, sizeof(typebuf)); |
|---|
| 6571 | | - netif_warn(efx, drv, efx->net_dev, |
|---|
| 6572 | | - "UDP port %d is actually in use by %s, not removing\n", |
|---|
| 6573 | | - ntohs(tnl.port), typebuf); |
|---|
| 6574 | | - } |
|---|
| 6575 | | - rc = -ENOENT; |
|---|
| 6576 | | - |
|---|
| 6577 | | -out_unlock: |
|---|
| 3925 | + nic_data->udp_tunnels[entry].type = TUNNEL_ENCAP_UDP_PORT_ENTRY_INVALID; |
|---|
| 3926 | + nic_data->udp_tunnels[entry].port = 0; |
|---|
| 3927 | + rc = efx_ef10_set_udp_tnl_ports(efx, false); |
|---|
| 6578 | 3928 | mutex_unlock(&nic_data->udp_tunnels_lock); |
|---|
| 3929 | + |
|---|
| 6579 | 3930 | return rc; |
|---|
| 3931 | +} |
|---|
| 3932 | + |
|---|
| 3933 | +static const struct udp_tunnel_nic_info efx_ef10_udp_tunnels = { |
|---|
| 3934 | + .set_port = efx_ef10_udp_tnl_set_port, |
|---|
| 3935 | + .unset_port = efx_ef10_udp_tnl_unset_port, |
|---|
| 3936 | + .flags = UDP_TUNNEL_NIC_INFO_MAY_SLEEP, |
|---|
| 3937 | + .tables = { |
|---|
| 3938 | + { |
|---|
| 3939 | + .n_entries = 16, |
|---|
| 3940 | + .tunnel_types = UDP_TUNNEL_TYPE_VXLAN | |
|---|
| 3941 | + UDP_TUNNEL_TYPE_GENEVE, |
|---|
| 3942 | + }, |
|---|
| 3943 | + }, |
|---|
| 3944 | +}; |
|---|
| 3945 | + |
|---|
| 3946 | +/* EF10 may have multiple datapath firmware variants within a |
|---|
| 3947 | + * single version. Report which variants are running. |
|---|
| 3948 | + */ |
|---|
| 3949 | +static size_t efx_ef10_print_additional_fwver(struct efx_nic *efx, char *buf, |
|---|
| 3950 | + size_t len) |
|---|
| 3951 | +{ |
|---|
| 3952 | + struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 3953 | + |
|---|
| 3954 | + return scnprintf(buf, len, " rx%x tx%x", |
|---|
| 3955 | + nic_data->rx_dpcpu_fw_id, |
|---|
| 3956 | + nic_data->tx_dpcpu_fw_id); |
|---|
| 3957 | +} |
|---|
| 3958 | + |
|---|
| 3959 | +static unsigned int ef10_check_caps(const struct efx_nic *efx, |
|---|
| 3960 | + u8 flag, |
|---|
| 3961 | + u32 offset) |
|---|
| 3962 | +{ |
|---|
| 3963 | + const struct efx_ef10_nic_data *nic_data = efx->nic_data; |
|---|
| 3964 | + |
|---|
| 3965 | + switch (offset) { |
|---|
| 3966 | + case(MC_CMD_GET_CAPABILITIES_V4_OUT_FLAGS1_OFST): |
|---|
| 3967 | + return nic_data->datapath_caps & BIT_ULL(flag); |
|---|
| 3968 | + case(MC_CMD_GET_CAPABILITIES_V4_OUT_FLAGS2_OFST): |
|---|
| 3969 | + return nic_data->datapath_caps2 & BIT_ULL(flag); |
|---|
| 3970 | + default: |
|---|
| 3971 | + return 0; |
|---|
| 3972 | + } |
|---|
| 6580 | 3973 | } |
|---|
| 6581 | 3974 | |
|---|
| 6582 | 3975 | #define EF10_OFFLOAD_FEATURES \ |
|---|
| .. | .. |
|---|
| 6594 | 3987 | .remove = efx_ef10_remove, |
|---|
| 6595 | 3988 | .dimension_resources = efx_ef10_dimension_resources, |
|---|
| 6596 | 3989 | .init = efx_ef10_init_nic, |
|---|
| 6597 | | - .fini = efx_port_dummy_op_void, |
|---|
| 3990 | + .fini = efx_ef10_fini_nic, |
|---|
| 6598 | 3991 | .map_reset_reason = efx_ef10_map_reset_reason, |
|---|
| 6599 | 3992 | .map_reset_flags = efx_ef10_map_reset_flags, |
|---|
| 6600 | 3993 | .reset = efx_ef10_reset, |
|---|
| 6601 | 3994 | .probe_port = efx_mcdi_port_probe, |
|---|
| 6602 | 3995 | .remove_port = efx_mcdi_port_remove, |
|---|
| 6603 | | - .fini_dmaq = efx_ef10_fini_dmaq, |
|---|
| 3996 | + .fini_dmaq = efx_fini_dmaq, |
|---|
| 6604 | 3997 | .prepare_flr = efx_ef10_prepare_flr, |
|---|
| 6605 | 3998 | .finish_flr = efx_port_dummy_op_void, |
|---|
| 6606 | 3999 | .describe_stats = efx_ef10_describe_stats, |
|---|
| 6607 | 4000 | .update_stats = efx_ef10_update_stats_vf, |
|---|
| 4001 | + .update_stats_atomic = efx_ef10_update_stats_atomic_vf, |
|---|
| 6608 | 4002 | .start_stats = efx_port_dummy_op_void, |
|---|
| 6609 | 4003 | .pull_stats = efx_port_dummy_op_void, |
|---|
| 6610 | 4004 | .stop_stats = efx_port_dummy_op_void, |
|---|
| 6611 | | - .set_id_led = efx_mcdi_set_id_led, |
|---|
| 6612 | 4005 | .push_irq_moderation = efx_ef10_push_irq_moderation, |
|---|
| 6613 | | - .reconfigure_mac = efx_ef10_mac_reconfigure_vf, |
|---|
| 4006 | + .reconfigure_mac = efx_ef10_mac_reconfigure, |
|---|
| 6614 | 4007 | .check_mac_fault = efx_mcdi_mac_check_fault, |
|---|
| 6615 | 4008 | .reconfigure_port = efx_mcdi_port_reconfigure, |
|---|
| 6616 | 4009 | .get_wol = efx_ef10_get_wol_vf, |
|---|
| .. | .. |
|---|
| 6628 | 4021 | .irq_handle_legacy = efx_ef10_legacy_interrupt, |
|---|
| 6629 | 4022 | .tx_probe = efx_ef10_tx_probe, |
|---|
| 6630 | 4023 | .tx_init = efx_ef10_tx_init, |
|---|
| 6631 | | - .tx_remove = efx_ef10_tx_remove, |
|---|
| 4024 | + .tx_remove = efx_mcdi_tx_remove, |
|---|
| 6632 | 4025 | .tx_write = efx_ef10_tx_write, |
|---|
| 6633 | 4026 | .tx_limit_len = efx_ef10_tx_limit_len, |
|---|
| 6634 | | - .rx_push_rss_config = efx_ef10_vf_rx_push_rss_config, |
|---|
| 6635 | | - .rx_pull_rss_config = efx_ef10_rx_pull_rss_config, |
|---|
| 6636 | | - .rx_probe = efx_ef10_rx_probe, |
|---|
| 6637 | | - .rx_init = efx_ef10_rx_init, |
|---|
| 6638 | | - .rx_remove = efx_ef10_rx_remove, |
|---|
| 4027 | + .tx_enqueue = __efx_enqueue_skb, |
|---|
| 4028 | + .rx_push_rss_config = efx_mcdi_vf_rx_push_rss_config, |
|---|
| 4029 | + .rx_pull_rss_config = efx_mcdi_rx_pull_rss_config, |
|---|
| 4030 | + .rx_probe = efx_mcdi_rx_probe, |
|---|
| 4031 | + .rx_init = efx_mcdi_rx_init, |
|---|
| 4032 | + .rx_remove = efx_mcdi_rx_remove, |
|---|
| 6639 | 4033 | .rx_write = efx_ef10_rx_write, |
|---|
| 6640 | 4034 | .rx_defer_refill = efx_ef10_rx_defer_refill, |
|---|
| 6641 | | - .ev_probe = efx_ef10_ev_probe, |
|---|
| 4035 | + .rx_packet = __efx_rx_packet, |
|---|
| 4036 | + .ev_probe = efx_mcdi_ev_probe, |
|---|
| 6642 | 4037 | .ev_init = efx_ef10_ev_init, |
|---|
| 6643 | | - .ev_fini = efx_ef10_ev_fini, |
|---|
| 6644 | | - .ev_remove = efx_ef10_ev_remove, |
|---|
| 4038 | + .ev_fini = efx_mcdi_ev_fini, |
|---|
| 4039 | + .ev_remove = efx_mcdi_ev_remove, |
|---|
| 6645 | 4040 | .ev_process = efx_ef10_ev_process, |
|---|
| 6646 | 4041 | .ev_read_ack = efx_ef10_ev_read_ack, |
|---|
| 6647 | 4042 | .ev_test_generate = efx_ef10_ev_test_generate, |
|---|
| 6648 | 4043 | .filter_table_probe = efx_ef10_filter_table_probe, |
|---|
| 6649 | | - .filter_table_restore = efx_ef10_filter_table_restore, |
|---|
| 6650 | | - .filter_table_remove = efx_ef10_filter_table_remove, |
|---|
| 6651 | | - .filter_update_rx_scatter = efx_ef10_filter_update_rx_scatter, |
|---|
| 6652 | | - .filter_insert = efx_ef10_filter_insert, |
|---|
| 6653 | | - .filter_remove_safe = efx_ef10_filter_remove_safe, |
|---|
| 6654 | | - .filter_get_safe = efx_ef10_filter_get_safe, |
|---|
| 6655 | | - .filter_clear_rx = efx_ef10_filter_clear_rx, |
|---|
| 6656 | | - .filter_count_rx_used = efx_ef10_filter_count_rx_used, |
|---|
| 6657 | | - .filter_get_rx_id_limit = efx_ef10_filter_get_rx_id_limit, |
|---|
| 6658 | | - .filter_get_rx_ids = efx_ef10_filter_get_rx_ids, |
|---|
| 4044 | + .filter_table_restore = efx_mcdi_filter_table_restore, |
|---|
| 4045 | + .filter_table_remove = efx_mcdi_filter_table_remove, |
|---|
| 4046 | + .filter_update_rx_scatter = efx_mcdi_update_rx_scatter, |
|---|
| 4047 | + .filter_insert = efx_mcdi_filter_insert, |
|---|
| 4048 | + .filter_remove_safe = efx_mcdi_filter_remove_safe, |
|---|
| 4049 | + .filter_get_safe = efx_mcdi_filter_get_safe, |
|---|
| 4050 | + .filter_clear_rx = efx_mcdi_filter_clear_rx, |
|---|
| 4051 | + .filter_count_rx_used = efx_mcdi_filter_count_rx_used, |
|---|
| 4052 | + .filter_get_rx_id_limit = efx_mcdi_filter_get_rx_id_limit, |
|---|
| 4053 | + .filter_get_rx_ids = efx_mcdi_filter_get_rx_ids, |
|---|
| 6659 | 4054 | #ifdef CONFIG_RFS_ACCEL |
|---|
| 6660 | | - .filter_rfs_expire_one = efx_ef10_filter_rfs_expire_one, |
|---|
| 4055 | + .filter_rfs_expire_one = efx_mcdi_filter_rfs_expire_one, |
|---|
| 6661 | 4056 | #endif |
|---|
| 6662 | 4057 | #ifdef CONFIG_SFC_MTD |
|---|
| 6663 | 4058 | .mtd_probe = efx_port_dummy_op_int, |
|---|
| .. | .. |
|---|
| 6683 | 4078 | .can_rx_scatter = true, |
|---|
| 6684 | 4079 | .always_rx_scatter = true, |
|---|
| 6685 | 4080 | .min_interrupt_mode = EFX_INT_MODE_MSIX, |
|---|
| 6686 | | - .max_interrupt_mode = EFX_INT_MODE_MSIX, |
|---|
| 6687 | 4081 | .timer_period_max = 1 << ERF_DD_EVQ_IND_TIMER_VAL_WIDTH, |
|---|
| 6688 | 4082 | .offload_features = EF10_OFFLOAD_FEATURES, |
|---|
| 6689 | 4083 | .mcdi_max_ver = 2, |
|---|
| 6690 | | - .max_rx_ip_filters = HUNT_FILTER_TBL_ROWS, |
|---|
| 4084 | + .max_rx_ip_filters = EFX_MCDI_FILTER_TBL_ROWS, |
|---|
| 6691 | 4085 | .hwtstamp_filters = 1 << HWTSTAMP_FILTER_NONE | |
|---|
| 6692 | 4086 | 1 << HWTSTAMP_FILTER_ALL, |
|---|
| 6693 | 4087 | .rx_hash_key_size = 40, |
|---|
| 4088 | + .check_caps = ef10_check_caps, |
|---|
| 4089 | + .print_additional_fwver = efx_ef10_print_additional_fwver, |
|---|
| 4090 | + .sensor_event = efx_mcdi_sensor_event, |
|---|
| 6694 | 4091 | }; |
|---|
| 6695 | 4092 | |
|---|
| 6696 | 4093 | const struct efx_nic_type efx_hunt_a0_nic_type = { |
|---|
| .. | .. |
|---|
| 6701 | 4098 | .remove = efx_ef10_remove, |
|---|
| 6702 | 4099 | .dimension_resources = efx_ef10_dimension_resources, |
|---|
| 6703 | 4100 | .init = efx_ef10_init_nic, |
|---|
| 6704 | | - .fini = efx_port_dummy_op_void, |
|---|
| 4101 | + .fini = efx_ef10_fini_nic, |
|---|
| 6705 | 4102 | .map_reset_reason = efx_ef10_map_reset_reason, |
|---|
| 6706 | 4103 | .map_reset_flags = efx_ef10_map_reset_flags, |
|---|
| 6707 | 4104 | .reset = efx_ef10_reset, |
|---|
| 6708 | 4105 | .probe_port = efx_mcdi_port_probe, |
|---|
| 6709 | 4106 | .remove_port = efx_mcdi_port_remove, |
|---|
| 6710 | | - .fini_dmaq = efx_ef10_fini_dmaq, |
|---|
| 4107 | + .fini_dmaq = efx_fini_dmaq, |
|---|
| 6711 | 4108 | .prepare_flr = efx_ef10_prepare_flr, |
|---|
| 6712 | 4109 | .finish_flr = efx_port_dummy_op_void, |
|---|
| 6713 | 4110 | .describe_stats = efx_ef10_describe_stats, |
|---|
| .. | .. |
|---|
| 6715 | 4112 | .start_stats = efx_mcdi_mac_start_stats, |
|---|
| 6716 | 4113 | .pull_stats = efx_mcdi_mac_pull_stats, |
|---|
| 6717 | 4114 | .stop_stats = efx_mcdi_mac_stop_stats, |
|---|
| 6718 | | - .set_id_led = efx_mcdi_set_id_led, |
|---|
| 6719 | 4115 | .push_irq_moderation = efx_ef10_push_irq_moderation, |
|---|
| 6720 | 4116 | .reconfigure_mac = efx_ef10_mac_reconfigure, |
|---|
| 6721 | 4117 | .check_mac_fault = efx_mcdi_mac_check_fault, |
|---|
| .. | .. |
|---|
| 6737 | 4133 | .irq_handle_legacy = efx_ef10_legacy_interrupt, |
|---|
| 6738 | 4134 | .tx_probe = efx_ef10_tx_probe, |
|---|
| 6739 | 4135 | .tx_init = efx_ef10_tx_init, |
|---|
| 6740 | | - .tx_remove = efx_ef10_tx_remove, |
|---|
| 4136 | + .tx_remove = efx_mcdi_tx_remove, |
|---|
| 6741 | 4137 | .tx_write = efx_ef10_tx_write, |
|---|
| 6742 | 4138 | .tx_limit_len = efx_ef10_tx_limit_len, |
|---|
| 6743 | | - .rx_push_rss_config = efx_ef10_pf_rx_push_rss_config, |
|---|
| 6744 | | - .rx_pull_rss_config = efx_ef10_rx_pull_rss_config, |
|---|
| 6745 | | - .rx_push_rss_context_config = efx_ef10_rx_push_rss_context_config, |
|---|
| 6746 | | - .rx_pull_rss_context_config = efx_ef10_rx_pull_rss_context_config, |
|---|
| 6747 | | - .rx_restore_rss_contexts = efx_ef10_rx_restore_rss_contexts, |
|---|
| 6748 | | - .rx_probe = efx_ef10_rx_probe, |
|---|
| 6749 | | - .rx_init = efx_ef10_rx_init, |
|---|
| 6750 | | - .rx_remove = efx_ef10_rx_remove, |
|---|
| 4139 | + .tx_enqueue = __efx_enqueue_skb, |
|---|
| 4140 | + .rx_push_rss_config = efx_mcdi_pf_rx_push_rss_config, |
|---|
| 4141 | + .rx_pull_rss_config = efx_mcdi_rx_pull_rss_config, |
|---|
| 4142 | + .rx_push_rss_context_config = efx_mcdi_rx_push_rss_context_config, |
|---|
| 4143 | + .rx_pull_rss_context_config = efx_mcdi_rx_pull_rss_context_config, |
|---|
| 4144 | + .rx_restore_rss_contexts = efx_mcdi_rx_restore_rss_contexts, |
|---|
| 4145 | + .rx_probe = efx_mcdi_rx_probe, |
|---|
| 4146 | + .rx_init = efx_mcdi_rx_init, |
|---|
| 4147 | + .rx_remove = efx_mcdi_rx_remove, |
|---|
| 6751 | 4148 | .rx_write = efx_ef10_rx_write, |
|---|
| 6752 | 4149 | .rx_defer_refill = efx_ef10_rx_defer_refill, |
|---|
| 6753 | | - .ev_probe = efx_ef10_ev_probe, |
|---|
| 4150 | + .rx_packet = __efx_rx_packet, |
|---|
| 4151 | + .ev_probe = efx_mcdi_ev_probe, |
|---|
| 6754 | 4152 | .ev_init = efx_ef10_ev_init, |
|---|
| 6755 | | - .ev_fini = efx_ef10_ev_fini, |
|---|
| 6756 | | - .ev_remove = efx_ef10_ev_remove, |
|---|
| 4153 | + .ev_fini = efx_mcdi_ev_fini, |
|---|
| 4154 | + .ev_remove = efx_mcdi_ev_remove, |
|---|
| 6757 | 4155 | .ev_process = efx_ef10_ev_process, |
|---|
| 6758 | 4156 | .ev_read_ack = efx_ef10_ev_read_ack, |
|---|
| 6759 | 4157 | .ev_test_generate = efx_ef10_ev_test_generate, |
|---|
| 6760 | 4158 | .filter_table_probe = efx_ef10_filter_table_probe, |
|---|
| 6761 | | - .filter_table_restore = efx_ef10_filter_table_restore, |
|---|
| 6762 | | - .filter_table_remove = efx_ef10_filter_table_remove, |
|---|
| 6763 | | - .filter_update_rx_scatter = efx_ef10_filter_update_rx_scatter, |
|---|
| 6764 | | - .filter_insert = efx_ef10_filter_insert, |
|---|
| 6765 | | - .filter_remove_safe = efx_ef10_filter_remove_safe, |
|---|
| 6766 | | - .filter_get_safe = efx_ef10_filter_get_safe, |
|---|
| 6767 | | - .filter_clear_rx = efx_ef10_filter_clear_rx, |
|---|
| 6768 | | - .filter_count_rx_used = efx_ef10_filter_count_rx_used, |
|---|
| 6769 | | - .filter_get_rx_id_limit = efx_ef10_filter_get_rx_id_limit, |
|---|
| 6770 | | - .filter_get_rx_ids = efx_ef10_filter_get_rx_ids, |
|---|
| 4159 | + .filter_table_restore = efx_mcdi_filter_table_restore, |
|---|
| 4160 | + .filter_table_remove = efx_mcdi_filter_table_remove, |
|---|
| 4161 | + .filter_update_rx_scatter = efx_mcdi_update_rx_scatter, |
|---|
| 4162 | + .filter_insert = efx_mcdi_filter_insert, |
|---|
| 4163 | + .filter_remove_safe = efx_mcdi_filter_remove_safe, |
|---|
| 4164 | + .filter_get_safe = efx_mcdi_filter_get_safe, |
|---|
| 4165 | + .filter_clear_rx = efx_mcdi_filter_clear_rx, |
|---|
| 4166 | + .filter_count_rx_used = efx_mcdi_filter_count_rx_used, |
|---|
| 4167 | + .filter_get_rx_id_limit = efx_mcdi_filter_get_rx_id_limit, |
|---|
| 4168 | + .filter_get_rx_ids = efx_mcdi_filter_get_rx_ids, |
|---|
| 6771 | 4169 | #ifdef CONFIG_RFS_ACCEL |
|---|
| 6772 | | - .filter_rfs_expire_one = efx_ef10_filter_rfs_expire_one, |
|---|
| 4170 | + .filter_rfs_expire_one = efx_mcdi_filter_rfs_expire_one, |
|---|
| 6773 | 4171 | #endif |
|---|
| 6774 | 4172 | #ifdef CONFIG_SFC_MTD |
|---|
| 6775 | 4173 | .mtd_probe = efx_ef10_mtd_probe, |
|---|
| .. | .. |
|---|
| 6785 | 4183 | .vlan_rx_add_vid = efx_ef10_vlan_rx_add_vid, |
|---|
| 6786 | 4184 | .vlan_rx_kill_vid = efx_ef10_vlan_rx_kill_vid, |
|---|
| 6787 | 4185 | .udp_tnl_push_ports = efx_ef10_udp_tnl_push_ports, |
|---|
| 6788 | | - .udp_tnl_add_port = efx_ef10_udp_tnl_add_port, |
|---|
| 6789 | 4186 | .udp_tnl_has_port = efx_ef10_udp_tnl_has_port, |
|---|
| 6790 | | - .udp_tnl_del_port = efx_ef10_udp_tnl_del_port, |
|---|
| 6791 | 4187 | #ifdef CONFIG_SFC_SRIOV |
|---|
| 6792 | 4188 | .sriov_configure = efx_ef10_sriov_configure, |
|---|
| 6793 | 4189 | .sriov_init = efx_ef10_sriov_init, |
|---|
| .. | .. |
|---|
| 6818 | 4214 | .always_rx_scatter = true, |
|---|
| 6819 | 4215 | .option_descriptors = true, |
|---|
| 6820 | 4216 | .min_interrupt_mode = EFX_INT_MODE_LEGACY, |
|---|
| 6821 | | - .max_interrupt_mode = EFX_INT_MODE_MSIX, |
|---|
| 6822 | 4217 | .timer_period_max = 1 << ERF_DD_EVQ_IND_TIMER_VAL_WIDTH, |
|---|
| 6823 | 4218 | .offload_features = EF10_OFFLOAD_FEATURES, |
|---|
| 6824 | 4219 | .mcdi_max_ver = 2, |
|---|
| 6825 | | - .max_rx_ip_filters = HUNT_FILTER_TBL_ROWS, |
|---|
| 4220 | + .max_rx_ip_filters = EFX_MCDI_FILTER_TBL_ROWS, |
|---|
| 6826 | 4221 | .hwtstamp_filters = 1 << HWTSTAMP_FILTER_NONE | |
|---|
| 6827 | 4222 | 1 << HWTSTAMP_FILTER_ALL, |
|---|
| 6828 | 4223 | .rx_hash_key_size = 40, |
|---|
| 4224 | + .check_caps = ef10_check_caps, |
|---|
| 4225 | + .print_additional_fwver = efx_ef10_print_additional_fwver, |
|---|
| 4226 | + .sensor_event = efx_mcdi_sensor_event, |
|---|
| 6829 | 4227 | }; |
|---|