[PATCH v2 02/14] drivers: net: ldpaa: add support for probing based on the DTS
Priyanka Jain (OSS)
priyanka.jain at oss.nxp.com
Tue Apr 7 08:28:40 CEST 2020
>-----Original Message-----
>From: U-Boot <u-boot-bounces at lists.denx.de> On Behalf Of Ioana Ciornei
>Sent: Wednesday, March 18, 2020 8:18 PM
>To: Priyanka Jain <priyanka.jain at nxp.com>; joe.hershberger at ni.com; u-
>boot at lists.denx.de
>Cc: Florin Laurentiu Chiculita <florinlaurentiu.chiculita at nxp.com>; Ioana
>Ciornei <ioana.ciornei at 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.chiculita at nxp.com>
>Signed-off-by: Ioana Ciornei <ioana.ciornei at 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
More information about the U-Boot
mailing list