7dd7be29b4
In the current code, on some cases the representor ethdev is using the
PF interface to query some link status information or pause parameters.
It was done because in previous kernel versions there was no support
from the kernel for the representor info.
Using the PF i/f for such ioctl is error prone and not always working
because:
* On some cases there is no PF at all, only representors (e.g Bluefield
with host representors)
* Query the up/down status from representor and link status from PF
is in-consist
* PF link is down doesn't necessarily means representor is down.
* setting different pause configuration for the PF and the
representors will result on undefined behaviour
Making the code cleaner and more robust by using only the representor
i/f for the ioctl. whatever the kernel will provide on this query will
be used. No need to do W.A. for kernel missing functionality.
Note:
1. Setting pause parameters will obviously won't work on representors
2. Old kernel will not report all the possible representor info
Fixes: 2b73026388
("net/mlx5: probe all port representors")
Cc: stable@dpdk.org
Signed-off-by: Shahaf Shuler <shahafs@mellanox.com>
233 lines
5.2 KiB
C
233 lines
5.2 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright 2015 6WIND S.A.
|
|
* Copyright 2015 Mellanox Technologies, Ltd
|
|
*/
|
|
|
|
#include <stddef.h>
|
|
#include <assert.h>
|
|
#include <stdint.h>
|
|
#include <string.h>
|
|
#include <inttypes.h>
|
|
#include <errno.h>
|
|
#include <netinet/in.h>
|
|
#include <sys/ioctl.h>
|
|
#include <arpa/inet.h>
|
|
|
|
/* Verbs header. */
|
|
/* ISO C doesn't support unnamed structs/unions, disabling -pedantic. */
|
|
#ifdef PEDANTIC
|
|
#pragma GCC diagnostic ignored "-Wpedantic"
|
|
#endif
|
|
#include <infiniband/verbs.h>
|
|
#ifdef PEDANTIC
|
|
#pragma GCC diagnostic error "-Wpedantic"
|
|
#endif
|
|
|
|
#include <rte_ether.h>
|
|
#include <rte_ethdev_driver.h>
|
|
#include <rte_common.h>
|
|
|
|
#include "mlx5.h"
|
|
#include "mlx5_utils.h"
|
|
#include "mlx5_rxtx.h"
|
|
#include "mlx5_defs.h"
|
|
|
|
/**
|
|
* Get MAC address by querying netdevice.
|
|
*
|
|
* @param[in] dev
|
|
* Pointer to Ethernet device.
|
|
* @param[out] mac
|
|
* MAC address output buffer.
|
|
*
|
|
* @return
|
|
* 0 on success, a negative errno value otherwise and rte_errno is set.
|
|
*/
|
|
int
|
|
mlx5_get_mac(struct rte_eth_dev *dev, uint8_t (*mac)[ETHER_ADDR_LEN])
|
|
{
|
|
struct ifreq request;
|
|
int ret;
|
|
|
|
ret = mlx5_ifreq(dev, SIOCGIFHWADDR, &request);
|
|
if (ret)
|
|
return ret;
|
|
memcpy(mac, request.ifr_hwaddr.sa_data, ETHER_ADDR_LEN);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Remove a MAC address from the internal array.
|
|
*
|
|
* @param dev
|
|
* Pointer to Ethernet device structure.
|
|
* @param index
|
|
* MAC address index.
|
|
*/
|
|
static void
|
|
mlx5_internal_mac_addr_remove(struct rte_eth_dev *dev, uint32_t index)
|
|
{
|
|
struct priv *priv = dev->data->dev_private;
|
|
const int vf = priv->config.vf;
|
|
|
|
assert(index < MLX5_MAX_MAC_ADDRESSES);
|
|
if (is_zero_ether_addr(&dev->data->mac_addrs[index]))
|
|
return;
|
|
if (vf)
|
|
mlx5_nl_mac_addr_remove(dev, &dev->data->mac_addrs[index],
|
|
index);
|
|
memset(&dev->data->mac_addrs[index], 0, sizeof(struct ether_addr));
|
|
}
|
|
|
|
/**
|
|
* Adds a MAC address to the internal array.
|
|
*
|
|
* @param dev
|
|
* Pointer to Ethernet device structure.
|
|
* @param mac_addr
|
|
* MAC address to register.
|
|
* @param index
|
|
* MAC address index.
|
|
*
|
|
* @return
|
|
* 0 on success, a negative errno value otherwise and rte_errno is set.
|
|
*/
|
|
static int
|
|
mlx5_internal_mac_addr_add(struct rte_eth_dev *dev, struct ether_addr *mac,
|
|
uint32_t index)
|
|
{
|
|
struct priv *priv = dev->data->dev_private;
|
|
const int vf = priv->config.vf;
|
|
unsigned int i;
|
|
|
|
assert(index < MLX5_MAX_MAC_ADDRESSES);
|
|
if (is_zero_ether_addr(mac)) {
|
|
rte_errno = EINVAL;
|
|
return -rte_errno;
|
|
}
|
|
/* First, make sure this address isn't already configured. */
|
|
for (i = 0; (i != MLX5_MAX_MAC_ADDRESSES); ++i) {
|
|
/* Skip this index, it's going to be reconfigured. */
|
|
if (i == index)
|
|
continue;
|
|
if (memcmp(&dev->data->mac_addrs[i], mac, sizeof(*mac)))
|
|
continue;
|
|
/* Address already configured elsewhere, return with error. */
|
|
rte_errno = EADDRINUSE;
|
|
return -rte_errno;
|
|
}
|
|
if (vf) {
|
|
int ret = mlx5_nl_mac_addr_add(dev, mac, index);
|
|
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
dev->data->mac_addrs[index] = *mac;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* DPDK callback to remove a MAC address.
|
|
*
|
|
* @param dev
|
|
* Pointer to Ethernet device structure.
|
|
* @param index
|
|
* MAC address index.
|
|
*/
|
|
void
|
|
mlx5_mac_addr_remove(struct rte_eth_dev *dev, uint32_t index)
|
|
{
|
|
int ret;
|
|
|
|
if (index >= MLX5_MAX_UC_MAC_ADDRESSES)
|
|
return;
|
|
mlx5_internal_mac_addr_remove(dev, index);
|
|
if (!dev->data->promiscuous) {
|
|
ret = mlx5_traffic_restart(dev);
|
|
if (ret)
|
|
DRV_LOG(ERR, "port %u cannot restart traffic: %s",
|
|
dev->data->port_id, strerror(rte_errno));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* DPDK callback to add a MAC address.
|
|
*
|
|
* @param dev
|
|
* Pointer to Ethernet device structure.
|
|
* @param mac_addr
|
|
* MAC address to register.
|
|
* @param index
|
|
* MAC address index.
|
|
* @param vmdq
|
|
* VMDq pool index to associate address with (ignored).
|
|
*
|
|
* @return
|
|
* 0 on success, a negative errno value otherwise and rte_errno is set.
|
|
*/
|
|
int
|
|
mlx5_mac_addr_add(struct rte_eth_dev *dev, struct ether_addr *mac,
|
|
uint32_t index, uint32_t vmdq __rte_unused)
|
|
{
|
|
int ret;
|
|
|
|
if (index >= MLX5_MAX_UC_MAC_ADDRESSES) {
|
|
rte_errno = EINVAL;
|
|
return -rte_errno;
|
|
}
|
|
ret = mlx5_internal_mac_addr_add(dev, mac, index);
|
|
if (ret < 0)
|
|
return ret;
|
|
if (!dev->data->promiscuous)
|
|
return mlx5_traffic_restart(dev);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* DPDK callback to set primary MAC address.
|
|
*
|
|
* @param dev
|
|
* Pointer to Ethernet device structure.
|
|
* @param mac_addr
|
|
* MAC address to register.
|
|
*
|
|
* @return
|
|
* 0 on success, a negative errno value otherwise and rte_errno is set.
|
|
*/
|
|
int
|
|
mlx5_mac_addr_set(struct rte_eth_dev *dev, struct ether_addr *mac_addr)
|
|
{
|
|
DRV_LOG(DEBUG, "port %u setting primary MAC address",
|
|
dev->data->port_id);
|
|
return mlx5_mac_addr_add(dev, mac_addr, 0, 0);
|
|
}
|
|
|
|
/**
|
|
* DPDK callback to set multicast addresses list.
|
|
*
|
|
* @see rte_eth_dev_set_mc_addr_list()
|
|
*/
|
|
int
|
|
mlx5_set_mc_addr_list(struct rte_eth_dev *dev,
|
|
struct ether_addr *mc_addr_set, uint32_t nb_mc_addr)
|
|
{
|
|
uint32_t i;
|
|
int ret;
|
|
|
|
if (nb_mc_addr >= MLX5_MAX_MC_MAC_ADDRESSES) {
|
|
rte_errno = ENOSPC;
|
|
return -rte_errno;
|
|
}
|
|
for (i = MLX5_MAX_UC_MAC_ADDRESSES; i != MLX5_MAX_MAC_ADDRESSES; ++i)
|
|
mlx5_internal_mac_addr_remove(dev, i);
|
|
i = MLX5_MAX_UC_MAC_ADDRESSES;
|
|
while (nb_mc_addr--) {
|
|
ret = mlx5_internal_mac_addr_add(dev, mc_addr_set++, i++);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
if (!dev->data->promiscuous)
|
|
return mlx5_traffic_restart(dev);
|
|
return 0;
|
|
}
|