// SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB // Copyright (c) 2018 Mellanox Technologies #include "en.h" #include "en/hv_vhca_stats.h" #include "lib/hv_vhca.h" #include "lib/hv.h" struct mlx5e_hv_vhca_per_ring_stats { u64 rx_packets; u64 rx_bytes; u64 tx_packets; u64 tx_bytes; }; static void mlx5e_hv_vhca_fill_ring_stats(struct mlx5e_priv *priv, int ch, struct mlx5e_hv_vhca_per_ring_stats *data) { struct mlx5e_channel_stats *stats; int tc; stats = &priv->channel_stats[ch]; data->rx_packets = stats->rq.packets; data->rx_bytes = stats->rq.bytes; for (tc = 0; tc < priv->max_opened_tc; tc++) { data->tx_packets += stats->sq[tc].packets; data->tx_bytes += stats->sq[tc].bytes; } } static void mlx5e_hv_vhca_fill_stats(struct mlx5e_priv *priv, void *data, int buf_len) { int ch, i = 0; for (ch = 0; ch < priv->max_nch; ch++) { void *buf = data + i; if (WARN_ON_ONCE(buf + sizeof(struct mlx5e_hv_vhca_per_ring_stats) > data + buf_len)) return; mlx5e_hv_vhca_fill_ring_stats(priv, ch, buf); i += sizeof(struct mlx5e_hv_vhca_per_ring_stats); } } static int mlx5e_hv_vhca_stats_buf_size(struct mlx5e_priv *priv) { return (sizeof(struct mlx5e_hv_vhca_per_ring_stats) * priv->max_nch); } static void mlx5e_hv_vhca_stats_work(struct work_struct *work) { struct mlx5e_hv_vhca_stats_agent *sagent; struct mlx5_hv_vhca_agent *agent; struct delayed_work *dwork; struct mlx5e_priv *priv; int buf_len, rc; void *buf; dwork = to_delayed_work(work); sagent = container_of(dwork, struct mlx5e_hv_vhca_stats_agent, work); priv = container_of(sagent, struct mlx5e_priv, stats_agent); buf_len = mlx5e_hv_vhca_stats_buf_size(priv); agent = sagent->agent; buf = sagent->buf; memset(buf, 0, buf_len); mlx5e_hv_vhca_fill_stats(priv, buf, buf_len); rc = mlx5_hv_vhca_agent_write(agent, buf, buf_len); if (rc) { mlx5_core_err(priv->mdev, "%s: Failed to write stats, err = %d\n", __func__, rc); return; } if (sagent->delay) queue_delayed_work(priv->wq, &sagent->work, sagent->delay); } enum { MLX5_HV_VHCA_STATS_VERSION = 1, MLX5_HV_VHCA_STATS_UPDATE_ONCE = 0xFFFF, }; static void mlx5e_hv_vhca_stats_control(struct mlx5_hv_vhca_agent *agent, struct mlx5_hv_vhca_control_block *block) { struct mlx5e_hv_vhca_stats_agent *sagent; struct mlx5e_priv *priv; priv = mlx5_hv_vhca_agent_priv(agent); sagent = &priv->stats_agent; block->version = MLX5_HV_VHCA_STATS_VERSION; block->rings = priv->max_nch; if (!block->command) { cancel_delayed_work_sync(&priv->stats_agent.work); return; } sagent->delay = block->command == MLX5_HV_VHCA_STATS_UPDATE_ONCE ? 0 : msecs_to_jiffies(block->command * 100); queue_delayed_work(priv->wq, &sagent->work, sagent->delay); } static void mlx5e_hv_vhca_stats_cleanup(struct mlx5_hv_vhca_agent *agent) { struct mlx5e_priv *priv = mlx5_hv_vhca_agent_priv(agent); cancel_delayed_work_sync(&priv->stats_agent.work); } int mlx5e_hv_vhca_stats_create(struct mlx5e_priv *priv) { int buf_len = mlx5e_hv_vhca_stats_buf_size(priv); struct mlx5_hv_vhca_agent *agent; priv->stats_agent.buf = kvzalloc(buf_len, GFP_KERNEL); if (!priv->stats_agent.buf) return -ENOMEM; agent = mlx5_hv_vhca_agent_create(priv->mdev->hv_vhca, MLX5_HV_VHCA_AGENT_STATS, mlx5e_hv_vhca_stats_control, NULL, mlx5e_hv_vhca_stats_cleanup, priv); if (IS_ERR_OR_NULL(agent)) { if (IS_ERR(agent)) netdev_warn(priv->netdev, "Failed to create hv vhca stats agent, err = %ld\n", PTR_ERR(agent)); kvfree(priv->stats_agent.buf); return IS_ERR_OR_NULL(agent); } priv->stats_agent.agent = agent; INIT_DELAYED_WORK(&priv->stats_agent.work, mlx5e_hv_vhca_stats_work); return 0; } void mlx5e_hv_vhca_stats_destroy(struct mlx5e_priv *priv) { if (IS_ERR_OR_NULL(priv->stats_agent.agent)) return; mlx5_hv_vhca_agent_destroy(priv->stats_agent.agent); kvfree(priv->stats_agent.buf); }