int     (*update_rx)(struct mlx5e_priv *priv);
        void    (*update_stats)(struct mlx5e_priv *priv);
        void    (*update_carrier)(struct mlx5e_priv *priv);
+       unsigned int (*stats_grps_num)(struct mlx5e_priv *priv);
+       const struct mlx5e_stats_grp *stats_grps;
        struct {
                mlx5e_fp_handle_rx_cqe handle_rx_cqe;
                mlx5e_fp_handle_rx_cqe handle_rx_cqe_mpwqe;
 mlx5e_skb_from_cqe_nonlinear(struct mlx5e_rq *rq, struct mlx5_cqe64 *cqe,
                             struct mlx5e_wqe_frag_info *wi, u32 cqe_bcnt);
 
-void mlx5e_update_stats(struct mlx5e_priv *priv);
 void mlx5e_get_stats(struct net_device *dev, struct rtnl_link_stats64 *stats);
 void mlx5e_fold_sw_stats64(struct mlx5e_priv *priv, struct rtnl_link_stats64 *s);
 
 
 
 int mlx5e_ethtool_get_sset_count(struct mlx5e_priv *priv, int sset)
 {
-       int i, num_stats = 0;
-
        switch (sset) {
        case ETH_SS_STATS:
-               for (i = 0; i < mlx5e_num_stats_grps; i++)
-                       num_stats += mlx5e_stats_grps[i].get_num_stats(priv);
-               return num_stats;
+               return mlx5e_stats_total_num(priv);
        case ETH_SS_PRIV_FLAGS:
                return MLX5E_NUM_PFLAGS;
        case ETH_SS_TEST:
        return mlx5e_ethtool_get_sset_count(priv, sset);
 }
 
-static void mlx5e_fill_stats_strings(struct mlx5e_priv *priv, u8 *data)
-{
-       int i, idx = 0;
-
-       for (i = 0; i < mlx5e_num_stats_grps; i++)
-               idx = mlx5e_stats_grps[i].fill_strings(priv, data, idx);
-}
-
 void mlx5e_ethtool_get_strings(struct mlx5e_priv *priv, u32 stringset, u8 *data)
 {
        int i;
                break;
 
        case ETH_SS_STATS:
-               mlx5e_fill_stats_strings(priv, data);
+               mlx5e_stats_fill_strings(priv, data);
                break;
        }
 }
 void mlx5e_ethtool_get_ethtool_stats(struct mlx5e_priv *priv,
                                     struct ethtool_stats *stats, u64 *data)
 {
-       int i, idx = 0;
+       int idx = 0;
 
        mutex_lock(&priv->state_lock);
-       mlx5e_update_stats(priv);
+       mlx5e_stats_update(priv);
        mutex_unlock(&priv->state_lock);
 
-       for (i = 0; i < mlx5e_num_stats_grps; i++)
-               idx = mlx5e_stats_grps[i].fill_stats(priv, data, idx);
+       mlx5e_stats_fill(priv, data, idx);
 }
 
 static void mlx5e_get_ethtool_stats(struct net_device *dev,
 
        mutex_unlock(&priv->state_lock);
 }
 
-void mlx5e_update_stats(struct mlx5e_priv *priv)
-{
-       int i;
-
-       for (i = mlx5e_num_stats_grps - 1; i >= 0; i--)
-               if (mlx5e_stats_grps[i].update_stats)
-                       mlx5e_stats_grps[i].update_stats(priv);
-}
-
 void mlx5e_update_ndo_stats(struct mlx5e_priv *priv)
 {
        int i;
 
-       for (i = mlx5e_num_stats_grps - 1; i >= 0; i--)
-               if (mlx5e_stats_grps[i].update_stats_mask &
+       for (i = mlx5e_nic_stats_grps_num(priv) - 1; i >= 0; i--)
+               if (mlx5e_nic_stats_grps[i].update_stats_mask &
                    MLX5E_NDO_UPDATE_STATS)
-                       mlx5e_stats_grps[i].update_stats(priv);
+                       mlx5e_nic_stats_grps[i].update_stats(priv);
 }
 
 static void mlx5e_update_stats_work(struct work_struct *work)
        .rx_handlers.handle_rx_cqe_mpwqe = mlx5e_handle_rx_cqe_mpwrq,
        .max_tc            = MLX5E_MAX_NUM_TC,
        .rq_groups         = MLX5E_NUM_RQ_GROUPS(XSK),
+       .stats_grps        = mlx5e_nic_stats_grps,
+       .stats_grps_num    = mlx5e_nic_stats_grps_num,
 };
 
 /* mlx5e generic netdev management API (move to en_common.c) */
 
 #include "en_accel/ipsec.h"
 #include "en_accel/tls.h"
 
+static unsigned int stats_grps_num(struct mlx5e_priv *priv)
+{
+       return !priv->profile->stats_grps_num ? 0 :
+               priv->profile->stats_grps_num(priv);
+}
+
+unsigned int mlx5e_stats_total_num(struct mlx5e_priv *priv)
+{
+       const struct mlx5e_stats_grp *stats_grps = priv->profile->stats_grps;
+       const unsigned int num_stats_grps = stats_grps_num(priv);
+       unsigned int total = 0;
+       int i;
+
+       for (i = 0; i < num_stats_grps; i++)
+               total += stats_grps[i].get_num_stats(priv);
+
+       return total;
+}
+
+void mlx5e_stats_update(struct mlx5e_priv *priv)
+{
+       const struct mlx5e_stats_grp *stats_grps = priv->profile->stats_grps;
+       const unsigned int num_stats_grps = stats_grps_num(priv);
+       int i;
+
+       for (i = num_stats_grps - 1; i >= 0; i--)
+               if (stats_grps[i].update_stats)
+                       stats_grps[i].update_stats(priv);
+}
+
+void mlx5e_stats_fill(struct mlx5e_priv *priv, u64 *data, int idx)
+{
+       const struct mlx5e_stats_grp *stats_grps = priv->profile->stats_grps;
+       const unsigned int num_stats_grps = stats_grps_num(priv);
+       int i;
+
+       for (i = 0; i < num_stats_grps; i++)
+               idx = stats_grps[i].fill_stats(priv, data, idx);
+}
+
+void mlx5e_stats_fill_strings(struct mlx5e_priv *priv, u8 *data)
+{
+       const struct mlx5e_stats_grp *stats_grps = priv->profile->stats_grps;
+       const unsigned int num_stats_grps = stats_grps_num(priv);
+       int i, idx = 0;
+
+       for (i = 0; i < num_stats_grps; i++)
+               idx = stats_grps[i].fill_strings(priv, data, idx);
+}
+
+/* Concrete NIC Stats */
+
 static const struct counter_desc sw_stats_desc[] = {
        { MLX5E_DECLARE_STAT(struct mlx5e_sw_stats, rx_packets) },
        { MLX5E_DECLARE_STAT(struct mlx5e_sw_stats, rx_bytes) },
 }
 
 /* The stats groups order is opposite to the update_stats() order calls */
-const struct mlx5e_stats_grp mlx5e_stats_grps[] = {
+const struct mlx5e_stats_grp mlx5e_nic_stats_grps[] = {
        {
                .get_num_stats = mlx5e_grp_sw_get_num_stats,
                .fill_strings = mlx5e_grp_sw_fill_strings,
        },
 };
 
-const int mlx5e_num_stats_grps = ARRAY_SIZE(mlx5e_stats_grps);
+unsigned int mlx5e_nic_stats_grps_num(struct mlx5e_priv *priv)
+{
+       return ARRAY_SIZE(mlx5e_nic_stats_grps);
+}
 
        size_t          offset; /* Byte offset */
 };
 
+enum {
+       MLX5E_NDO_UPDATE_STATS = BIT(0x1),
+};
+
+struct mlx5e_priv;
+struct mlx5e_stats_grp {
+       u16 update_stats_mask;
+       int (*get_num_stats)(struct mlx5e_priv *priv);
+       int (*fill_strings)(struct mlx5e_priv *priv, u8 *data, int idx);
+       int (*fill_stats)(struct mlx5e_priv *priv, u64 *data, int idx);
+       void (*update_stats)(struct mlx5e_priv *priv);
+};
+
+unsigned int mlx5e_stats_total_num(struct mlx5e_priv *priv);
+void mlx5e_stats_update(struct mlx5e_priv *priv);
+void mlx5e_stats_fill(struct mlx5e_priv *priv, u64 *data, int idx);
+void mlx5e_stats_fill_strings(struct mlx5e_priv *priv, u8 *data);
+
+/* Concrete NIC Stats */
+
 struct mlx5e_sw_stats {
        u64 rx_packets;
        u64 rx_bytes;
        struct mlx5e_pcie_stats pcie;
 };
 
-enum {
-       MLX5E_NDO_UPDATE_STATS = BIT(0x1),
-};
-
-struct mlx5e_priv;
-struct mlx5e_stats_grp {
-       u16 update_stats_mask;
-       int (*get_num_stats)(struct mlx5e_priv *priv);
-       int (*fill_strings)(struct mlx5e_priv *priv, u8 *data, int idx);
-       int (*fill_stats)(struct mlx5e_priv *priv, u64 *data, int idx);
-       void (*update_stats)(struct mlx5e_priv *priv);
-};
-
-extern const struct mlx5e_stats_grp mlx5e_stats_grps[];
-extern const int mlx5e_num_stats_grps;
+extern const struct mlx5e_stats_grp mlx5e_nic_stats_grps[];
+unsigned int mlx5e_nic_stats_grps_num(struct mlx5e_priv *priv);
 
 void mlx5e_grp_802_3_update_stats(struct mlx5e_priv *priv);
 
 
        .rx_handlers.handle_rx_cqe_mpwqe = NULL, /* Not supported */
        .max_tc            = MLX5I_MAX_NUM_TC,
        .rq_groups         = MLX5E_NUM_RQ_GROUPS(REGULAR),
+       .stats_grps        = mlx5e_nic_stats_grps,
+       .stats_grps_num    = mlx5e_nic_stats_grps_num,
 };
 
 /* mlx5i netdev NDos */