callback to retrieve physical link information

Signed-off-by: Srikanth Kaka <srikant...@oneconvergence.com>
Signed-off-by: Vag Singh <vag.si...@oneconvergence.com>
Signed-off-by: Anand Thulasiram <av...@juniper.net>
---
 drivers/net/mlx5/freebsd/mlx5_ethdev_os.c | 175 ++++++++++++++++++++++
 1 file changed, 175 insertions(+)

diff --git a/drivers/net/mlx5/freebsd/mlx5_ethdev_os.c 
b/drivers/net/mlx5/freebsd/mlx5_ethdev_os.c
index 8b2ea1c594..d100addd51 100644
--- a/drivers/net/mlx5/freebsd/mlx5_ethdev_os.c
+++ b/drivers/net/mlx5/freebsd/mlx5_ethdev_os.c
@@ -7,6 +7,7 @@
 #include <stdint.h>
 #include <net/if_media.h>
 #include <sys/ioctl.h>
+#include <time.h>
 
 #include <ethdev_driver.h>
 
@@ -164,6 +165,180 @@ mlx5_set_flags(struct rte_eth_dev *dev, unsigned int 
keep, unsigned int flags)
        return mlx5_ifreq(dev, SIOCSIFFLAGS, &request);
 }
 
+static const struct ifmedia_baudrate ifmedia_baudrate_desc[] =
+       IFM_BAUDRATE_DESCRIPTIONS;
+
+static uint64_t
+mlx5_ifmedia_baudrate(int mword)
+{
+       int i;
+
+       for (i = 0; ifmedia_baudrate_desc[i].ifmb_word != 0; i++) {
+               if (IFM_TYPE_MATCH(mword, ifmedia_baudrate_desc[i].ifmb_word))
+                       return (ifmedia_baudrate_desc[i].ifmb_baudrate);
+       }
+
+       return (0);
+}
+
+static int
+mlx5_link_update_bsd(struct rte_eth_dev *dev,
+                    struct rte_eth_link *link)
+{
+       struct mlx5_priv *priv = dev->data->dev_private;
+       struct rte_eth_link dev_link;
+       int link_speed = 0, sock;
+       struct ifmediareq ifmr;
+       char ifname[IF_NAMESIZE];
+       int *media_list;
+
+       sock = socket(PF_INET, SOCK_DGRAM, IPPROTO_IP);
+       if (sock == -1) {
+               DRV_LOG(ERR,
+                       "port %u failed to open socket for link update: %s",
+                       dev->data->port_id, strerror(rte_errno));
+               return sock;
+       }
+
+       mlx5_get_ifname(dev, &ifname);
+       memset(&ifmr, 0, sizeof(struct ifmediareq));
+       strlcpy(ifmr.ifm_name, ifname, sizeof(ifmr.ifm_name));
+
+       if (ioctl(sock, SIOCGIFXMEDIA, (caddr_t)&ifmr) < 0) {
+               DRV_LOG(ERR,
+                       "ioctl(SIOCGIFMEDIA) on %s: %s",
+                       ifname, strerror(errno));
+               close(sock);
+               return errno;
+       }
+
+       media_list = (int *)malloc(ifmr.ifm_count * sizeof(int));
+       ifmr.ifm_ulist = media_list;
+
+       if (ioctl(sock, SIOCGIFXMEDIA, (caddr_t)&ifmr) < 0) {
+               DRV_LOG(ERR,
+                       "ioctl(SIOCGIFMEDIA) on %s: %s",
+                       ifname, strerror(errno));
+               close(sock);
+               return errno;
+       }
+
+       if (ifmr.ifm_status == (IFM_AVALID | IFM_ACTIVE))
+               dev_link.link_status = ETH_LINK_UP;
+       else
+               dev_link.link_status = ETH_LINK_DOWN;
+
+       link_speed = ifmr.ifm_status & IFM_AVALID ?
+                  mlx5_ifmedia_baudrate(ifmr.ifm_active)/(1000*1000) : 0;
+
+       if (link_speed == 0)
+               dev_link.link_speed = ETH_SPEED_NUM_NONE;
+       else
+               dev_link.link_speed = link_speed;
+
+       priv->link_speed_capa = 0;
+       /* Add support for duplex types */
+       dev_link.link_duplex = ETH_LINK_FULL_DUPLEX;
+       /* FreeBSD automatically negotiates speed,
+        * so it is displayed in its capabilities.
+        */
+       priv->link_speed_capa |= ETH_LINK_SPEED_AUTONEG;
+
+       for (int i = 1; i < ifmr.ifm_count; i += 2) {
+               switch (mlx5_ifmedia_baudrate(media_list[i]) / (1000 * 1000)) {
+               case 100000:
+                       priv->link_speed_capa |= ETH_LINK_SPEED_100G;
+                       break;
+               case 56000:
+                       priv->link_speed_capa |= ETH_LINK_SPEED_56G;
+                       break;
+               case 50000:
+                       priv->link_speed_capa |= ETH_LINK_SPEED_50G;
+                       break;
+               case 40000:
+                       priv->link_speed_capa |= ETH_LINK_SPEED_40G;
+                       break;
+               case 25000:
+                       priv->link_speed_capa |= ETH_LINK_SPEED_25G;
+                       break;
+               case 10000:
+                       priv->link_speed_capa |= ETH_LINK_SPEED_10G;
+                       break;
+               case 2500:
+                       priv->link_speed_capa |= ETH_LINK_SPEED_2_5G;
+                       break;
+               case 1000:
+                       priv->link_speed_capa |= ETH_LINK_SPEED_1G;
+                       break;
+               case 100:
+                       priv->link_speed_capa |= (dev_link.link_duplex ==
+                                               ETH_LINK_FULL_DUPLEX) ?
+                                               ETH_LINK_SPEED_100M :
+                                               ETH_LINK_SPEED_100M_HD;
+                       break;
+               case 10:
+                       priv->link_speed_capa |= (dev_link.link_duplex ==
+                                               ETH_LINK_FULL_DUPLEX) ?
+                                               ETH_LINK_SPEED_10M :
+                                               ETH_LINK_SPEED_10M_HD;
+                       break;
+               case 0:
+               default:
+                       break;
+               }
+       }
+       dev_link.link_autoneg = !(dev->data->dev_conf.link_speeds &
+                               ETH_LINK_SPEED_FIXED);
+       free(media_list);
+       *link = dev_link;
+       close(sock);
+       return 0;
+}
+
+/**
+ * DPDK callback to retrieve physical link information.
+ *
+ * @param dev
+ *   Pointer to Ethernet device structure.
+ * @param wait_to_complete
+ *   Wait for request completion.
+ *
+ * @return
+ *   0 if link status was not updated, positive if it was, a negative errno
+ *   value otherwise and rte_errno is set.
+ */
+int
+mlx5_link_update(struct rte_eth_dev *dev, int wait_to_complete)
+{
+       int ret;
+       struct rte_eth_link dev_link;
+       time_t start_time = time(NULL);
+       int retry = MLX5_GET_LINK_STATUS_RETRY_COUNT;
+
+       do {
+               ret = mlx5_link_update_bsd(dev, &dev_link);
+               if (ret == 0)
+                       break;
+               /* Handle wait to complete situation. */
+               if ((wait_to_complete || retry) && ret == -EAGAIN) {
+                       if (abs((int)difftime(time(NULL), start_time)) <
+                           MLX5_LINK_STATUS_TIMEOUT) {
+                               usleep(0);
+                               continue;
+                       } else {
+                               rte_errno = EBUSY;
+                               return -rte_errno;
+                       }
+               } else if (ret < 0) {
+                       return ret;
+               }
+       } while (wait_to_complete || retry-- > 0);
+       ret = !!memcmp(&dev->data->dev_link, &dev_link,
+                      sizeof(struct rte_eth_link));
+       dev->data->dev_link = dev_link;
+       return ret;
+}
+
 /**
  * DPDK callback to bring the link DOWN.
  *
-- 
2.30.2

Reply via email to