net/sfc: support multicast addresses list controls
[dpdk.git] / drivers / net / sfc / sfc_ethdev.c
index aa02f73..4766ae9 100644 (file)
@@ -705,6 +705,97 @@ fail_inval:
        SFC_ASSERT(rc > 0);
        return -rc;
 }
+static void
+sfc_mac_addr_set(struct rte_eth_dev *dev, struct ether_addr *mac_addr)
+{
+       struct sfc_adapter *sa = dev->data->dev_private;
+       const efx_nic_cfg_t *encp = efx_nic_cfg_get(sa->nic);
+       int rc;
+
+       sfc_adapter_lock(sa);
+
+       if (sa->state != SFC_ADAPTER_STARTED) {
+               sfc_info(sa, "the port is not started");
+               sfc_info(sa, "the new MAC address will be set on port start");
+
+               goto unlock;
+       }
+
+       if (encp->enc_allow_set_mac_with_installed_filters) {
+               rc = efx_mac_addr_set(sa->nic, mac_addr->addr_bytes);
+               if (rc != 0) {
+                       sfc_err(sa, "cannot set MAC address (rc = %u)", rc);
+                       goto unlock;
+               }
+
+               /*
+                * Changing the MAC address by means of MCDI request
+                * has no effect on received traffic, therefore
+                * we also need to update unicast filters
+                */
+               rc = sfc_set_rx_mode(sa);
+               if (rc != 0)
+                       sfc_err(sa, "cannot set filter (rc = %u)", rc);
+       } else {
+               sfc_warn(sa, "cannot set MAC address with filters installed");
+               sfc_warn(sa, "adapter will be restarted to pick the new MAC");
+               sfc_warn(sa, "(some traffic may be dropped)");
+
+               /*
+                * Since setting MAC address with filters installed is not
+                * allowed on the adapter, one needs to simply restart adapter
+                * so that the new MAC address will be taken from an outer
+                * storage and set flawlessly by means of sfc_start() call
+                */
+               sfc_stop(sa);
+               rc = sfc_start(sa);
+               if (rc != 0)
+                       sfc_err(sa, "cannot restart adapter (rc = %u)", rc);
+       }
+
+unlock:
+       sfc_adapter_unlock(sa);
+}
+
+
+static int
+sfc_set_mc_addr_list(struct rte_eth_dev *dev, struct ether_addr *mc_addr_set,
+                    uint32_t nb_mc_addr)
+{
+       struct sfc_adapter *sa = dev->data->dev_private;
+       uint8_t *mc_addrs_p;
+       uint8_t *mc_addrs;
+       int rc;
+       unsigned int i;
+
+       if (nb_mc_addr > EFX_MAC_MULTICAST_LIST_MAX) {
+               sfc_err(sa, "too many multicast addresses: %u > %u",
+                        nb_mc_addr, EFX_MAC_MULTICAST_LIST_MAX);
+               return -EINVAL;
+       }
+
+       mc_addrs_p = rte_calloc("mc-addrs", nb_mc_addr, EFX_MAC_ADDR_LEN, 0);
+       if (mc_addrs_p == NULL)
+               return -ENOMEM;
+
+       mc_addrs = mc_addrs_p;
+
+       for (i = 0; i < nb_mc_addr; ++i) {
+               (void)rte_memcpy(mc_addrs, mc_addr_set[i].addr_bytes,
+                                EFX_MAC_ADDR_LEN);
+               mc_addrs += EFX_MAC_ADDR_LEN;
+       }
+
+       rc = efx_mac_multicast_list_set(sa->nic, mc_addrs_p, nb_mc_addr);
+
+       rte_free(mc_addrs_p);
+
+       if (rc != 0)
+               sfc_err(sa, "cannot set multicast address list (rc = %u)", rc);
+
+       SFC_ASSERT(rc > 0);
+       return -rc;
+}
 
 static const struct eth_dev_ops sfc_eth_dev_ops = {
        .dev_configure                  = sfc_dev_configure,
@@ -729,6 +820,8 @@ static const struct eth_dev_ops sfc_eth_dev_ops = {
        .tx_queue_release               = sfc_tx_queue_release,
        .flow_ctrl_get                  = sfc_flow_ctrl_get,
        .flow_ctrl_set                  = sfc_flow_ctrl_set,
+       .mac_addr_set                   = sfc_mac_addr_set,
+       .set_mc_addr_list               = sfc_set_mc_addr_list,
 };
 
 static int