>-----Original Message-----
>From: U-Boot <u-boot-boun...@lists.denx.de> On Behalf Of Ioana Ciornei
>Sent: Wednesday, March 18, 2020 8:18 PM
>To: Priyanka Jain <priyanka.j...@nxp.com>; joe.hershber...@ni.com; u-
>b...@lists.denx.de
>Cc: Florin Laurentiu Chiculita <florinlaurentiu.chicul...@nxp.com>; Ioana
>Ciornei <ioana.cior...@nxp.com>
>Subject: [PATCH v2 02/14] drivers: net: ldpaa: add support for probing based
>on the DTS
>
>When CONFIG_DM_ETH is enabled DPAA2 network interfaces will now probe
>based on DTS nodes with the "fsl,qoriq-mc-dpmac" compatible.
>In this case, transform the ldpaa_eth driver into a UCLASS_ETH driver and
>reuse the _open()/_tx()/_stop() functions already inplemented.
>
>For the moment, the ldpaa_eth driver will support both configurations:
>with or without CONFIG_DM_ETH enabled. Any 'struct eth_device' occurrence
>now has a matching 'struct udevice' made mutually exclusive based on the
>state of CONFIG_DM_ETH.
>
>Signed-off-by: Florin Laurentiu Chiculita <florinlaurentiu.chicul...@nxp.com>
>Signed-off-by: Ioana Ciornei <ioana.cior...@nxp.com>
>---
>Changes in v2:
> - used the positive logic (ifdef CONFIG_DM_ETH instead of ifndef)
>
> drivers/net/ldpaa_eth/ldpaa_eth.c | 231 +++++++++++++++++++++++++-----
> drivers/net/ldpaa_eth/ldpaa_eth.h |   6 +
> 2 files changed, 205 insertions(+), 32 deletions(-)
>
>diff --git a/drivers/net/ldpaa_eth/ldpaa_eth.c
>b/drivers/net/ldpaa_eth/ldpaa_eth.c
>index a3b9c152b256..f589a482b734 100644
>--- a/drivers/net/ldpaa_eth/ldpaa_eth.c
>+++ b/drivers/net/ldpaa_eth/ldpaa_eth.c
>@@ -12,6 +12,7 @@
> #include <net.h>
> #include <hwconfig.h>
> #include <phy.h>
>+#include <miiphy.h>
> #include <linux/compat.h>
> #include <fsl-mc/fsl_dpmac.h>
>
>@@ -19,6 +20,19 @@
> #include "ldpaa_eth.h"
>
> #ifdef CONFIG_PHYLIB
>+#ifdef CONFIG_DM_ETH
>+static void init_phy(struct udevice *dev) {
>+      struct ldpaa_eth_priv *priv = dev_get_priv(dev);
>+
>+      priv->phy = dm_eth_phy_connect(dev);
>+
>+      if (!priv->phy)
>+              return;
>+
>+      phy_config(priv->phy);
>+}
>+#else
> static int init_phy(struct eth_device *dev)  {
>       struct ldpaa_eth_priv *priv = (struct ldpaa_eth_priv *)dev->priv; @@ -
>63,6 +77,7 @@ static int init_phy(struct eth_device *dev)
>       return ret;
> }
> #endif
>+#endif
>
> #ifdef DEBUG
>
>@@ -128,9 +143,15 @@ static void ldpaa_eth_get_dpni_counter(void)
>       }
> }
>
>+#ifdef CONFIG_DM_ETH
>+static void ldpaa_eth_get_dpmac_counter(struct udevice *dev) {
>+      struct ldpaa_eth_priv *priv = dev_get_priv(dev); #else
> static void ldpaa_eth_get_dpmac_counter(struct eth_device *net_dev)  {
>       struct ldpaa_eth_priv *priv = (struct ldpaa_eth_priv *)net_dev->priv;
>+#endif
>       int err = 0;
>       u64 value;
>
>@@ -263,9 +284,16 @@ error:
>       return;
> }
>
>+#ifdef CONFIG_DM_ETH
>+static int ldpaa_eth_pull_dequeue_rx(struct udevice *dev,
>+                                   int flags, uchar **packetp)
>+{
>+      struct ldpaa_eth_priv *priv = dev_get_priv(dev); #else
> static int ldpaa_eth_pull_dequeue_rx(struct eth_device *dev)  {
>       struct ldpaa_eth_priv *priv = (struct ldpaa_eth_priv *)dev->priv;
>+#endif
>       const struct ldpaa_dq *dq;
>       const struct dpaa_fd *fd;
>       int i = 5, err = 0, status;
>@@ -322,9 +350,15 @@ static int ldpaa_eth_pull_dequeue_rx(struct
>eth_device *dev)
>       return err;
> }
>
>+#ifdef CONFIG_DM_ETH
>+static int ldpaa_eth_tx(struct udevice *dev, void *buf, int len) {
>+      struct ldpaa_eth_priv *priv = dev_get_priv(dev); #else
> static int ldpaa_eth_tx(struct eth_device *net_dev, void *buf, int len)  {
>       struct ldpaa_eth_priv *priv = (struct ldpaa_eth_priv *)net_dev->priv;
>+#endif
>       struct dpaa_fd fd;
>       u64 buffer_start;
>       int data_offset, err;
>@@ -400,15 +434,32 @@ error:
>       return err;
> }
>
>+static struct phy_device *ldpaa_get_phydev(struct ldpaa_eth_priv *priv)
>+{ #ifdef CONFIG_DM_ETH
>+      return priv->phy;
>+#else
>+#ifdef CONFIG_PHYLIB
>+      struct phy_device *phydev = NULL;
>+      int phy_num;
>+
>+      /* start the phy devices one by one and update the dpmac state */
>+      for (phy_num = 0; phy_num < WRIOP_MAX_PHY_NUM; phy_num++) {
>+              phydev = wriop_get_phy_dev(priv->dpmac_id, phy_num);
>+              if (phydev)
>+                      return phydev;
>+      }
>+      return NULL;
>+#endif
>+#endif
>+}
>+
> static int ldpaa_get_dpmac_state(struct ldpaa_eth_priv *priv,
>                                struct dpmac_link_state *state)
> {
>       phy_interface_t enet_if;
>-      int phys_detected;
>-#ifdef CONFIG_PHYLIB
>       struct phy_device *phydev = NULL;
>-      int err, phy_num;
>-#endif
>+      int err;
>
>       /* let's start off with maximum capabilities */
>       enet_if = wriop_get_enet_if(priv->dpmac_id);
>@@ -420,39 +471,28 @@ static int ldpaa_get_dpmac_state(struct
>ldpaa_eth_priv *priv,
>               state->rate = SPEED_1000;
>               break;
>       }
>-      state->up = 1;
>
>-      phys_detected = 0;
>-#ifdef CONFIG_PHYLIB
>+      state->up = 1;
>       state->options |= DPMAC_LINK_OPT_AUTONEG;
>+      phydev = ldpaa_get_phydev(priv);
>
>-      /* start the phy devices one by one and update the dpmac state */
>-      for (phy_num = 0; phy_num < WRIOP_MAX_PHY_NUM; phy_num++) {
>-              phydev = wriop_get_phy_dev(priv->dpmac_id, phy_num);
>-              if (!phydev)
>-                      continue;
>-
>-              phys_detected++;
>+      if (phydev) {
>               err = phy_startup(phydev);
>               if (err) {
>                       printf("%s: Could not initialize\n", phydev->dev-
>>name);
>                       state->up = 0;
>-                      break;
>-              }
>-              if (phydev->link) {
>+              } else if (phydev->link) {
>                       state->rate = min(state->rate, (uint32_t)phydev-
>>speed);
>                       if (!phydev->duplex)
>                               state->options |=
>DPMAC_LINK_OPT_HALF_DUPLEX;
>                       if (!phydev->autoneg)
>                               state->options &=
>~DPMAC_LINK_OPT_AUTONEG;
>               } else {
>-                      /* break out of loop even if one phy is down */
>                       state->up = 0;
>-                      break;
>               }
>       }
>-#endif
>-      if (!phys_detected)
>+
>+      if (!phydev)
>               state->options &= ~DPMAC_LINK_OPT_AUTONEG;
>
>       if (!state->up) {
>@@ -464,9 +504,16 @@ static int ldpaa_get_dpmac_state(struct
>ldpaa_eth_priv *priv,
>       return 0;
> }
>
>+#ifdef CONFIG_DM_ETH
>+static int ldpaa_eth_open(struct udevice *dev) {
>+      struct eth_pdata *plat = dev_get_platdata(dev);
>+      struct ldpaa_eth_priv *priv = dev_get_priv(dev); #else
> static int ldpaa_eth_open(struct eth_device *net_dev, bd_t *bd)  {
>       struct ldpaa_eth_priv *priv = (struct ldpaa_eth_priv *)net_dev->priv;
>+#endif
>       struct dpmac_link_state dpmac_link_state = { 0 };
> #ifdef DEBUG
>       struct dpni_link_state link_state;
>@@ -474,8 +521,13 @@ static int ldpaa_eth_open(struct eth_device
>*net_dev, bd_t *bd)
>       int err = 0;
>       struct dpni_queue d_queue;
>
>+#ifdef CONFIG_DM_ETH
>+      if (eth_is_active(dev))
>+              return 0;
>+#else
>       if (net_dev->state == ETH_STATE_ACTIVE)
>               return 0;
>+#endif
>
>       if (get_mc_boot_status() != 0) {
>               printf("ERROR (MC is not booted)\n"); @@ -515,8 +567,13
>@@ static int ldpaa_eth_open(struct eth_device *net_dev, bd_t *bd)
>       if (err)
>               goto err_dpni_bind;
>
>+#ifdef CONFIG_DM_ETH
>+      err = dpni_add_mac_addr(dflt_mc_io, MC_CMD_NO_FLAGS,
>+                              dflt_dpni->dpni_handle, plat->enetaddr);
>#else
>       err = dpni_add_mac_addr(dflt_mc_io, MC_CMD_NO_FLAGS,
>                               dflt_dpni->dpni_handle, net_dev->enetaddr);
>+#endif
>       if (err) {
>               printf("dpni_add_mac_addr() failed\n");
>               return err;
>@@ -589,22 +646,34 @@ err_dpmac_setup:
>       return err;
> }
>
>+#ifdef CONFIG_DM_ETH
>+static void ldpaa_eth_stop(struct udevice *dev) {
>+      struct ldpaa_eth_priv *priv = dev_get_priv(dev); #else
> static void ldpaa_eth_stop(struct eth_device *net_dev)  {
>       struct ldpaa_eth_priv *priv = (struct ldpaa_eth_priv *)net_dev->priv;
>-      int err = 0;
>-#ifdef CONFIG_PHYLIB
>-      struct phy_device *phydev = NULL;
>-      int phy_num;
> #endif
>+      struct phy_device *phydev = NULL;
>+      int err = 0;
>
>+#ifdef CONFIG_DM_ETH
>+      if (!eth_is_active(dev))
>+              return;
>+#else
>       if ((net_dev->state == ETH_STATE_PASSIVE) ||
>           (net_dev->state == ETH_STATE_INIT))
>               return;
>+#endif
>
> #ifdef DEBUG
>       ldpaa_eth_get_dpni_counter();
>+#ifdef CONFIG_DM_ETH
>+      ldpaa_eth_get_dpmac_counter(dev);
>+#else
>       ldpaa_eth_get_dpmac_counter(net_dev);
>+#endif
> #endif
>
>       err = dprc_disconnect(dflt_mc_io, MC_CMD_NO_FLAGS, @@ -628,13
>+697,9 @@ static void ldpaa_eth_stop(struct eth_device *net_dev)
>       if (err < 0)
>               printf("dpni_disable() failed\n");
>
>-#ifdef CONFIG_PHYLIB
>-      for (phy_num = 0; phy_num < WRIOP_MAX_PHY_NUM; phy_num++) {
>-              phydev = wriop_get_phy_dev(priv->dpmac_id, phy_num);
>-              if (phydev)
>-                      phy_shutdown(phydev);
>-      }
>-#endif
>+      phydev = ldpaa_get_phydev(priv);
>+      if (phydev)
>+              phy_shutdown(phydev);
>
>       /* Free DPBP handle and reset. */
>       ldpaa_dpbp_free();
>@@ -1027,6 +1092,107 @@ static int ldpaa_dpni_bind(struct ldpaa_eth_priv
>*priv)
>       return 0;
> }
>
>+#ifdef CONFIG_DM_ETH
>+static int ldpaa_eth_probe(struct udevice *dev) {
>+      struct ofnode_phandle_args phandle;
>+
>+      /* Nothing to do if there is no "phy-handle" in the DTS node */
>+      if (dev_read_phandle_with_args(dev, "phy-handle", NULL,
>+                                     0, 0, &phandle)) {
>+              return 0;
>+      }
>+
>+      init_phy(dev);
>+
>+      return 0;
>+}
>+
>+static uint32_t ldpaa_eth_get_dpmac_id(struct udevice *dev) {
>+      int port_node = dev_of_offset(dev);
>+
>+      return fdtdec_get_uint(gd->fdt_blob, port_node, "reg", -1); }
>+
>+static const char *ldpaa_eth_get_phy_mode_str(struct udevice *dev) {
>+      int port_node = dev_of_offset(dev);
>+      const char *phy_mode_str;
>+
>+      phy_mode_str = fdt_getprop(gd->fdt_blob, port_node,
>+                                 "phy-connection-type", NULL);
>+      if (phy_mode_str)
>+              return phy_mode_str;
>+
>+      phy_mode_str = fdt_getprop(gd->fdt_blob, port_node, "phy-mode",
>NULL);
>+      return phy_mode_str;
>+}
>+
>+static int ldpaa_eth_bind(struct udevice *dev) {
>+      const char *phy_mode_str = NULL;
>+      uint32_t dpmac_id;
>+      char eth_name[16];
>+      int phy_mode = -1;
>+
>+      phy_mode_str = ldpaa_eth_get_phy_mode_str(dev);
>+      if (phy_mode_str)
>+              phy_mode = phy_get_interface_by_name(phy_mode_str);
>+      if (phy_mode == -1) {
>+              dev_err(dev, "incorrect phy mode\n");
>+              return -EINVAL;
>+      }
>+
>+      dpmac_id = ldpaa_eth_get_dpmac_id(dev);
>+      if (dpmac_id == -1) {
>+              dev_err(dev, "missing reg field from the dpmac node\n");
>+              return -EINVAL;
>+      }
>+
>+      sprintf(eth_name, "DPMAC%d@%s", dpmac_id, phy_mode_str);
>+      device_set_name(dev, eth_name);
>+
>+      return 0;
>+}
>+
>+static int ldpaa_eth_ofdata_to_platdata(struct udevice *dev) {
>+      struct ldpaa_eth_priv *priv = dev_get_priv(dev);
>+      const char *phy_mode_str;
>+
>+      priv->dpmac_id = ldpaa_eth_get_dpmac_id(dev);
>+      phy_mode_str = ldpaa_eth_get_phy_mode_str(dev);
>+      priv->phy_mode = phy_get_interface_by_name(phy_mode_str);
>+
>+      return 0;
>+}
>+
>+static const struct eth_ops ldpaa_eth_ops = {
>+      .start  = ldpaa_eth_open,
>+      .send   = ldpaa_eth_tx,
>+      .recv   = ldpaa_eth_pull_dequeue_rx,
>+      .stop   = ldpaa_eth_stop,
>+};
>+
>+static const struct udevice_id ldpaa_eth_of_ids[] = {
>+      { .compatible = "fsl,qoriq-mc-dpmac" }, };
>+
>+U_BOOT_DRIVER(ldpaa_eth) = {
>+      .name = "ldpaa_eth",
>+      .id = UCLASS_ETH,
>+      .of_match = ldpaa_eth_of_ids,
>+      .ofdata_to_platdata = ldpaa_eth_ofdata_to_platdata,
>+      .bind = ldpaa_eth_bind,
>+      .probe = ldpaa_eth_probe,
>+      .ops = &ldpaa_eth_ops,
>+      .priv_auto_alloc_size = sizeof(struct ldpaa_eth_priv),
>+      .platdata_auto_alloc_size = sizeof(struct eth_pdata), };
>+
>+#else
>+
> static int ldpaa_eth_netdev_init(struct eth_device *net_dev,
>                                phy_interface_t enet_if)
> {
>@@ -1099,3 +1265,4 @@ err_netdev_init:
>
>       return err;
> }
>+#endif
>diff --git a/drivers/net/ldpaa_eth/ldpaa_eth.h
>b/drivers/net/ldpaa_eth/ldpaa_eth.h
>index 3f9154b5bbcd..e90513e56f9a 100644
>--- a/drivers/net/ldpaa_eth/ldpaa_eth.h
>+++ b/drivers/net/ldpaa_eth/ldpaa_eth.h
>@@ -116,7 +116,13 @@ struct ldpaa_fas {
>                                        LDPAA_ETH_FAS_TIDE)
>
> struct ldpaa_eth_priv {
>+#ifdef CONFIG_DM_ETH
>+      struct phy_device *phy;
>+      int phy_mode;
>+      bool started;
>+#else
>       struct eth_device *net_dev;
>+#endif
>       uint32_t dpmac_id;
>       uint16_t dpmac_handle;
>
>--
>2.17.1
Joe,

Kindly help to review/ACK this patch

Thanks
Priyanka

Reply via email to