]> git.baikalelectronics.ru Git - kernel.git/commitdiff
dpaa2-eth: Keep congestion group taildrop enabled when PFC on
authorIoana Ciornei <ioana.ciornei@nxp.com>
Sat, 30 May 2020 21:08:14 +0000 (00:08 +0300)
committerDavid S. Miller <davem@davemloft.net>
Mon, 1 Jun 2020 19:04:32 +0000 (12:04 -0700)
Leave congestion group taildrop enabled for all traffic classes
when PFC is enabled. Notification threshold is low enough such
that it will be hit first and this also ensures that FQs on
traffic classes which are not PFC enabled won't drain the buffer
pool.

FQ taildrop threshold is kept disabled as long as any form of
flow control is on. Since FQ taildrop works with bytes, not number
of frames, we can't guarantee it will not interfere with the
congestion notification mechanism for all frame sizes.

Signed-off-by: Ioana Ciornei <ioana.ciornei@nxp.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/ethernet/freescale/dpaa2/dpaa2-eth-dcb.c
drivers/net/ethernet/freescale/dpaa2/dpaa2-eth.c
drivers/net/ethernet/freescale/dpaa2/dpaa2-eth.h

index 7ee07872af4dce51762524788739d14827397aa2..83dee575c2faf8590e0839935dc474c0a3b415d4 100644 (file)
@@ -63,6 +63,7 @@ static int dpaa2_eth_dcbnl_ieee_setpfc(struct net_device *net_dev,
 {
        struct dpaa2_eth_priv *priv = netdev_priv(net_dev);
        struct dpni_link_cfg link_cfg = {0};
+       bool tx_pause;
        int err;
 
        if (pfc->mbc || pfc->delay)
@@ -75,8 +76,8 @@ static int dpaa2_eth_dcbnl_ieee_setpfc(struct net_device *net_dev,
        /* We allow PFC configuration even if it won't have any effect until
         * general pause frames are enabled
         */
-       if (!dpaa2_eth_rx_pause_enabled(priv->link_state.options) ||
-           !dpaa2_eth_tx_pause_enabled(priv->link_state.options))
+       tx_pause = dpaa2_eth_tx_pause_enabled(priv->link_state.options);
+       if (!dpaa2_eth_rx_pause_enabled(priv->link_state.options) || !tx_pause)
                netdev_warn(net_dev, "Pause support must be enabled in order for PFC to work!\n");
 
        link_cfg.rate = priv->link_state.rate;
@@ -97,6 +98,9 @@ static int dpaa2_eth_dcbnl_ieee_setpfc(struct net_device *net_dev,
                return err;
 
        memcpy(&priv->pfc, pfc, sizeof(priv->pfc));
+       priv->pfc_enabled = !!pfc->pfc_en;
+
+       dpaa2_eth_set_rx_taildrop(priv, tx_pause, priv->pfc_enabled);
 
        return 0;
 }
index cde9d0e2dd6d84f59123111a0d8ae7f1ea6066d7..8fb48de5d18cf3909a37d3242396a3949f45bc10 100644 (file)
@@ -1287,18 +1287,22 @@ static void disable_ch_napi(struct dpaa2_eth_priv *priv)
        }
 }
 
-static void dpaa2_eth_set_rx_taildrop(struct dpaa2_eth_priv *priv,
-                                     bool tx_pause)
+void dpaa2_eth_set_rx_taildrop(struct dpaa2_eth_priv *priv,
+                              bool tx_pause, bool pfc)
 {
        struct dpni_taildrop td = {0};
        struct dpaa2_eth_fq *fq;
        int i, err;
 
+       /* FQ taildrop: threshold is in bytes, per frame queue. Enabled if
+        * flow control is disabled (as it might interfere with either the
+        * buffer pool depletion trigger for pause frames or with the group
+        * congestion trigger for PFC frames)
+        */
        td.enable = !tx_pause;
-       if (priv->rx_td_enabled == td.enable)
-               return;
+       if (priv->rx_fqtd_enabled == td.enable)
+               goto set_cgtd;
 
-       /* FQ taildrop: threshold is in bytes, per frame queue */
        td.threshold = DPAA2_ETH_FQ_TAILDROP_THRESH;
        td.units = DPNI_CONGESTION_UNIT_BYTES;
 
@@ -1316,9 +1320,20 @@ static void dpaa2_eth_set_rx_taildrop(struct dpaa2_eth_priv *priv,
                }
        }
 
+       priv->rx_fqtd_enabled = td.enable;
+
+set_cgtd:
        /* Congestion group taildrop: threshold is in frames, per group
         * of FQs belonging to the same traffic class
+        * Enabled if general Tx pause disabled or if PFCs are enabled
+        * (congestion group threhsold for PFC generation is lower than the
+        * CG taildrop threshold, so it won't interfere with it; we also
+        * want frames in non-PFC enabled traffic classes to be kept in check)
         */
+       td.enable = !tx_pause || (tx_pause && pfc);
+       if (priv->rx_cgtd_enabled == td.enable)
+               return;
+
        td.threshold = DPAA2_ETH_CG_TAILDROP_THRESH(priv);
        td.units = DPNI_CONGESTION_UNIT_FRAMES;
        for (i = 0; i < dpaa2_eth_tc_count(priv); i++) {
@@ -1332,7 +1347,7 @@ static void dpaa2_eth_set_rx_taildrop(struct dpaa2_eth_priv *priv,
                }
        }
 
-       priv->rx_td_enabled = td.enable;
+       priv->rx_cgtd_enabled = td.enable;
 }
 
 static int link_state_update(struct dpaa2_eth_priv *priv)
@@ -1353,7 +1368,7 @@ static int link_state_update(struct dpaa2_eth_priv *priv)
         * only when pause frame generation is disabled.
         */
        tx_pause = dpaa2_eth_tx_pause_enabled(state.options);
-       dpaa2_eth_set_rx_taildrop(priv, tx_pause);
+       dpaa2_eth_set_rx_taildrop(priv, tx_pause, priv->pfc_enabled);
 
        /* When we manage the MAC/PHY using phylink there is no need
         * to manually update the netif_carrier.
index 31b7b9b52da060f935fdd359db89bbeb2e353f4c..2d7ada0f0dbdbfb0510399706514573c6022f324 100644 (file)
@@ -436,7 +436,8 @@ struct dpaa2_eth_priv {
        struct dpaa2_eth_drv_stats __percpu *percpu_extras;
 
        u16 mc_token;
-       u8 rx_td_enabled;
+       u8 rx_fqtd_enabled;
+       u8 rx_cgtd_enabled;
 
        struct dpni_link_state link_state;
        bool do_link_poll;
@@ -448,6 +449,7 @@ struct dpaa2_eth_priv {
        struct dpaa2_eth_cls_rule *cls_rules;
        u8 rx_cls_enabled;
        u8 vlan_cls_enabled;
+       u8 pfc_enabled;
 #ifdef CONFIG_FSL_DPAA2_ETH_DCB
        u8 dcbx_mode;
        struct ieee_pfc pfc;
@@ -584,6 +586,9 @@ int dpaa2_eth_cls_key_size(u64 key);
 int dpaa2_eth_cls_fld_off(int prot, int field);
 void dpaa2_eth_cls_trim_rule(void *key_mem, u64 fields);
 
+void dpaa2_eth_set_rx_taildrop(struct dpaa2_eth_priv *priv,
+                              bool tx_pause, bool pfc);
+
 extern const struct dcbnl_rtnl_ops dpaa2_eth_dcbnl_ops;
 
 #endif /* __DPAA2_H */