dpaa2-mac: split up initializing the MAC object from connecting to it
authorIoana Ciornei <ioana.ciornei@nxp.com>
Fri, 8 Jan 2021 09:07:22 +0000 (11:07 +0200)
committerJakub Kicinski <kuba@kernel.org>
Sun, 10 Jan 2021 00:21:29 +0000 (16:21 -0800)
Split up the initialization phase of the dpmac object from actually
configuring the phylink instance, connecting to it and configuring the
MAC. This is done so that even though the dpni object is connected to a
dpmac which has link management handled by the firmware we are still
able to export the MAC counters.

Signed-off-by: Ioana Ciornei <ioana.ciornei@nxp.com>
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
drivers/net/ethernet/freescale/dpaa2/dpaa2-eth.c
drivers/net/ethernet/freescale/dpaa2/dpaa2-mac.c
drivers/net/ethernet/freescale/dpaa2/dpaa2-mac.h

index fb0bcd18ec0c1c6e82ae89a4b63a9872bb9dab4d..61385894e8c7b793aaf8d08b92ef8603b72b6d8f 100644 (file)
@@ -4056,15 +4056,24 @@ static int dpaa2_eth_connect_mac(struct dpaa2_eth_priv *priv)
        mac->mc_io = priv->mc_io;
        mac->net_dev = priv->net_dev;
 
+       err = dpaa2_mac_open(mac);
+       if (err)
+               goto err_free_mac;
+
        err = dpaa2_mac_connect(mac);
        if (err) {
                netdev_err(priv->net_dev, "Error connecting to the MAC endpoint\n");
-               kfree(mac);
-               return err;
+               goto err_close_mac;
        }
        priv->mac = mac;
 
        return 0;
+
+err_close_mac:
+       dpaa2_mac_close(mac);
+err_free_mac:
+       kfree(mac);
+       return err;
 }
 
 static void dpaa2_eth_disconnect_mac(struct dpaa2_eth_priv *priv)
@@ -4073,6 +4082,7 @@ static void dpaa2_eth_disconnect_mac(struct dpaa2_eth_priv *priv)
                return;
 
        dpaa2_mac_disconnect(priv->mac);
+       dpaa2_mac_close(priv->mac);
        kfree(priv->mac);
        priv->mac = NULL;
 }
index 828c177df03d5915d2faeb81fb9ad6b74496f032..50dd302abcf48adab86e4d2b78d179fa9d4c85dc 100644 (file)
@@ -302,36 +302,20 @@ static void dpaa2_pcs_destroy(struct dpaa2_mac *mac)
 
 int dpaa2_mac_connect(struct dpaa2_mac *mac)
 {
-       struct fsl_mc_device *dpmac_dev = mac->mc_dev;
        struct net_device *net_dev = mac->net_dev;
        struct device_node *dpmac_node;
        struct phylink *phylink;
-       struct dpmac_attr attr;
        int err;
 
-       err = dpmac_open(mac->mc_io, 0, dpmac_dev->obj_desc.id,
-                        &dpmac_dev->mc_handle);
-       if (err || !dpmac_dev->mc_handle) {
-               netdev_err(net_dev, "dpmac_open() = %d\n", err);
-               return -ENODEV;
-       }
-
-       err = dpmac_get_attributes(mac->mc_io, 0, dpmac_dev->mc_handle, &attr);
-       if (err) {
-               netdev_err(net_dev, "dpmac_get_attributes() = %d\n", err);
-               goto err_close_dpmac;
-       }
-
-       mac->if_link_type = attr.link_type;
+       mac->if_link_type = mac->attr.link_type;
 
-       dpmac_node = dpaa2_mac_get_node(attr.id);
+       dpmac_node = dpaa2_mac_get_node(mac->attr.id);
        if (!dpmac_node) {
-               netdev_err(net_dev, "No dpmac@%d node found.\n", attr.id);
-               err = -ENODEV;
-               goto err_close_dpmac;
+               netdev_err(net_dev, "No dpmac@%d node found.\n", mac->attr.id);
+               return -ENODEV;
        }
 
-       err = dpaa2_mac_get_if_mode(dpmac_node, attr);
+       err = dpaa2_mac_get_if_mode(dpmac_node, mac->attr);
        if (err < 0) {
                err = -EINVAL;
                goto err_put_node;
@@ -351,9 +335,9 @@ int dpaa2_mac_connect(struct dpaa2_mac *mac)
                goto err_put_node;
        }
 
-       if (attr.link_type == DPMAC_LINK_TYPE_PHY &&
-           attr.eth_if != DPMAC_ETH_IF_RGMII) {
-               err = dpaa2_pcs_create(mac, dpmac_node, attr.id);
+       if (mac->attr.link_type == DPMAC_LINK_TYPE_PHY &&
+           mac->attr.eth_if != DPMAC_ETH_IF_RGMII) {
+               err = dpaa2_pcs_create(mac, dpmac_node, mac->attr.id);
                if (err)
                        goto err_put_node;
        }
@@ -389,8 +373,7 @@ err_pcs_destroy:
        dpaa2_pcs_destroy(mac);
 err_put_node:
        of_node_put(dpmac_node);
-err_close_dpmac:
-       dpmac_close(mac->mc_io, 0, dpmac_dev->mc_handle);
+
        return err;
 }
 
@@ -402,8 +385,40 @@ void dpaa2_mac_disconnect(struct dpaa2_mac *mac)
        phylink_disconnect_phy(mac->phylink);
        phylink_destroy(mac->phylink);
        dpaa2_pcs_destroy(mac);
+}
+
+int dpaa2_mac_open(struct dpaa2_mac *mac)
+{
+       struct fsl_mc_device *dpmac_dev = mac->mc_dev;
+       struct net_device *net_dev = mac->net_dev;
+       int err;
+
+       err = dpmac_open(mac->mc_io, 0, dpmac_dev->obj_desc.id,
+                        &dpmac_dev->mc_handle);
+       if (err || !dpmac_dev->mc_handle) {
+               netdev_err(net_dev, "dpmac_open() = %d\n", err);
+               return -ENODEV;
+       }
+
+       err = dpmac_get_attributes(mac->mc_io, 0, dpmac_dev->mc_handle,
+                                  &mac->attr);
+       if (err) {
+               netdev_err(net_dev, "dpmac_get_attributes() = %d\n", err);
+               goto err_close_dpmac;
+       }
+
+       return 0;
+
+err_close_dpmac:
+       dpmac_close(mac->mc_io, 0, dpmac_dev->mc_handle);
+       return err;
+}
+
+void dpaa2_mac_close(struct dpaa2_mac *mac)
+{
+       struct fsl_mc_device *dpmac_dev = mac->mc_dev;
 
-       dpmac_close(mac->mc_io, 0, mac->mc_dev->mc_handle);
+       dpmac_close(mac->mc_io, 0, dpmac_dev->mc_handle);
 }
 
 static char dpaa2_mac_ethtool_stats[][ETH_GSTRING_LEN] = {
index 955a52856210f4fd995ead047ecd70c62f26934e..13d42dd58ec90af3d072bbf1cb53008c36f3d204 100644 (file)
@@ -17,6 +17,7 @@ struct dpaa2_mac {
        struct dpmac_link_state state;
        struct net_device *net_dev;
        struct fsl_mc_io *mc_io;
+       struct dpmac_attr attr;
 
        struct phylink_config phylink_config;
        struct phylink *phylink;
@@ -28,6 +29,10 @@ struct dpaa2_mac {
 bool dpaa2_mac_is_type_fixed(struct fsl_mc_device *dpmac_dev,
                             struct fsl_mc_io *mc_io);
 
+int dpaa2_mac_open(struct dpaa2_mac *mac);
+
+void dpaa2_mac_close(struct dpaa2_mac *mac);
+
 int dpaa2_mac_connect(struct dpaa2_mac *mac);
 
 void dpaa2_mac_disconnect(struct dpaa2_mac *mac);