--- /dev/null
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * DPAA2 Ethernet Switch ethtool support
+ *
+ * Copyright 2014-2016 Freescale Semiconductor Inc.
+ * Copyright 2017-2018 NXP
+ *
+ */
+
+#include "ethsw.h"
+
+static struct {
+       enum dpsw_counter id;
+       char name[ETH_GSTRING_LEN];
+} ethsw_ethtool_counters[] =  {
+       {DPSW_CNT_ING_FRAME,            "rx frames"},
+       {DPSW_CNT_ING_BYTE,             "rx bytes"},
+       {DPSW_CNT_ING_FLTR_FRAME,       "rx filtered frames"},
+       {DPSW_CNT_ING_FRAME_DISCARD,    "rx discarded frames"},
+       {DPSW_CNT_ING_BCAST_FRAME,      "rx b-cast frames"},
+       {DPSW_CNT_ING_BCAST_BYTES,      "rx b-cast bytes"},
+       {DPSW_CNT_ING_MCAST_FRAME,      "rx m-cast frames"},
+       {DPSW_CNT_ING_MCAST_BYTE,       "rx m-cast bytes"},
+       {DPSW_CNT_EGR_FRAME,            "tx frames"},
+       {DPSW_CNT_EGR_BYTE,             "tx bytes"},
+       {DPSW_CNT_EGR_FRAME_DISCARD,    "tx discarded frames"},
+
+};
+
+#define ETHSW_NUM_COUNTERS     ARRAY_SIZE(ethsw_ethtool_counters)
+
+static void ethsw_get_drvinfo(struct net_device *netdev,
+                             struct ethtool_drvinfo *drvinfo)
+{
+       struct ethsw_port_priv *port_priv = netdev_priv(netdev);
+       u16 version_major, version_minor;
+       int err;
+
+       strlcpy(drvinfo->driver, KBUILD_MODNAME, sizeof(drvinfo->driver));
+
+       err = dpsw_get_api_version(port_priv->ethsw_data->mc_io, 0,
+                                  &version_major,
+                                  &version_minor);
+       if (err)
+               strlcpy(drvinfo->fw_version, "N/A",
+                       sizeof(drvinfo->fw_version));
+       else
+               snprintf(drvinfo->fw_version, sizeof(drvinfo->fw_version),
+                        "%u.%u", version_major, version_minor);
+
+       strlcpy(drvinfo->bus_info, dev_name(netdev->dev.parent->parent),
+               sizeof(drvinfo->bus_info));
+}
+
+static int
+ethsw_get_link_ksettings(struct net_device *netdev,
+                        struct ethtool_link_ksettings *link_ksettings)
+{
+       struct ethsw_port_priv *port_priv = netdev_priv(netdev);
+       struct dpsw_link_state state = {0};
+       int err = 0;
+
+       err = dpsw_if_get_link_state(port_priv->ethsw_data->mc_io, 0,
+                                    port_priv->ethsw_data->dpsw_handle,
+                                    port_priv->idx,
+                                    &state);
+       if (err) {
+               netdev_err(netdev, "ERROR %d getting link state", err);
+               goto out;
+       }
+
+       /* At the moment, we have no way of interrogating the DPMAC
+        * from the DPSW side or there may not exist a DPMAC at all.
+        * Report only autoneg state, duplexity and speed.
+        */
+       if (state.options & DPSW_LINK_OPT_AUTONEG)
+               link_ksettings->base.autoneg = AUTONEG_ENABLE;
+       if (!(state.options & DPSW_LINK_OPT_HALF_DUPLEX))
+               link_ksettings->base.duplex = DUPLEX_FULL;
+       link_ksettings->base.speed = state.rate;
+
+out:
+       return err;
+}
+
+static int
+ethsw_set_link_ksettings(struct net_device *netdev,
+                        const struct ethtool_link_ksettings *link_ksettings)
+{
+       struct ethsw_port_priv *port_priv = netdev_priv(netdev);
+       struct dpsw_link_cfg cfg = {0};
+       int err = 0;
+
+       netdev_dbg(netdev, "Setting link parameters...");
+
+       /* Due to a temporary MC limitation, the DPSW port must be down
+        * in order to be able to change link settings. Taking steps to let
+        * the user know that.
+        */
+       if (netif_running(netdev)) {
+               netdev_info(netdev, "Sorry, interface must be brought down first.\n");
+               return -EACCES;
+       }
+
+       cfg.rate = link_ksettings->base.speed;
+       if (link_ksettings->base.autoneg == AUTONEG_ENABLE)
+               cfg.options |= DPSW_LINK_OPT_AUTONEG;
+       else
+               cfg.options &= ~DPSW_LINK_OPT_AUTONEG;
+       if (link_ksettings->base.duplex  == DUPLEX_HALF)
+               cfg.options |= DPSW_LINK_OPT_HALF_DUPLEX;
+       else
+               cfg.options &= ~DPSW_LINK_OPT_HALF_DUPLEX;
+
+       err = dpsw_if_set_link_cfg(port_priv->ethsw_data->mc_io, 0,
+                                  port_priv->ethsw_data->dpsw_handle,
+                                  port_priv->idx,
+                                  &cfg);
+       if (err)
+               /* ethtool will be loud enough if we return an error; no point
+                * in putting our own error message on the console by default
+                */
+               netdev_dbg(netdev, "ERROR %d setting link cfg", err);
+
+       return err;
+}
+
+static int ethsw_ethtool_get_sset_count(struct net_device *dev, int sset)
+{
+       switch (sset) {
+       case ETH_SS_STATS:
+               return ETHSW_NUM_COUNTERS;
+       default:
+               return -EOPNOTSUPP;
+       }
+}
+
+static void ethsw_ethtool_get_strings(struct net_device *netdev,
+                                     u32 stringset, u8 *data)
+{
+       int i;
+
+       switch (stringset) {
+       case ETH_SS_STATS:
+               for (i = 0; i < ETHSW_NUM_COUNTERS; i++)
+                       memcpy(data + i * ETH_GSTRING_LEN,
+                              ethsw_ethtool_counters[i].name, ETH_GSTRING_LEN);
+               break;
+       }
+}
+
+static void ethsw_ethtool_get_stats(struct net_device *netdev,
+                                   struct ethtool_stats *stats,
+                                   u64 *data)
+{
+       struct ethsw_port_priv *port_priv = netdev_priv(netdev);
+       int i, err;
+
+       memset(data, 0,
+              sizeof(u64) * ETHSW_NUM_COUNTERS);
+
+       for (i = 0; i < ETHSW_NUM_COUNTERS; i++) {
+               err = dpsw_if_get_counter(port_priv->ethsw_data->mc_io, 0,
+                                         port_priv->ethsw_data->dpsw_handle,
+                                         port_priv->idx,
+                                         ethsw_ethtool_counters[i].id,
+                                         &data[i]);
+               if (err)
+                       netdev_err(netdev, "dpsw_if_get_counter[%s] err %d\n",
+                                  ethsw_ethtool_counters[i].name, err);
+       }
+}
+
+const struct ethtool_ops ethsw_port_ethtool_ops = {
+       .get_drvinfo            = ethsw_get_drvinfo,
+       .get_link               = ethtool_op_get_link,
+       .get_link_ksettings     = ethsw_get_link_ksettings,
+       .set_link_ksettings     = ethsw_set_link_ksettings,
+       .get_strings            = ethsw_ethtool_get_strings,
+       .get_ethtool_stats      = ethsw_ethtool_get_stats,
+       .get_sset_count         = ethsw_ethtool_get_sset_count,
+};