IB/mlx5: Support set qp counter
authorMark Zhang <markz@mellanox.com>
Tue, 2 Jul 2019 10:02:36 +0000 (13:02 +0300)
committerJason Gunthorpe <jgg@mellanox.com>
Fri, 5 Jul 2019 13:22:55 +0000 (10:22 -0300)
Support bind a qp with counter. If counter is null then bind the qp to the
default counter. Different QP state has different operation:

- RESET: Set the counter field so that it will take effective during
  RST2INIT change;
- RTS: Issue an RTS2RTS change to update the QP counter;
- Other: Set the counter field and mark the counter_pending flag, when QP
  is moved to RTS state and this flag is set, then issue an RTS2RTS
  modification to update the counter.

Signed-off-by: Mark Zhang <markz@mellanox.com>
Reviewed-by: Majd Dibbiny <majd@mellanox.com>
Acked-by: Saeed Mahameed <saeedm@mellanox.com>
Signed-off-by: Leon Romanovsky <leonro@mellanox.com>
Signed-off-by: Jason Gunthorpe <jgg@mellanox.com>
drivers/infiniband/hw/mlx5/mlx5_ib.h
drivers/infiniband/hw/mlx5/qp.c
include/linux/mlx5/qp.h

index 7373e9da0919b562863e5e3cde31db7a697077cb..c482f19958b39754555e8c74cb31906a95a1992a 100644 (file)
@@ -439,6 +439,10 @@ struct mlx5_ib_qp {
        u32                     flags_en;
        /* storage for qp sub type when core qp type is IB_QPT_DRIVER */
        enum ib_qp_type         qp_sub_type;
+       /* A flag to indicate if there's a new counter is configured
+        * but not take effective
+        */
+       u32                     counter_pending;
 };
 
 struct mlx5_ib_cq_buf {
@@ -1468,4 +1472,6 @@ void mlx5_ib_put_xlt_emergency_page(void);
 int bfregn_to_uar_index(struct mlx5_ib_dev *dev,
                        struct mlx5_bfreg_info *bfregi, u32 bfregn,
                        bool dyn_bfreg);
+
+int mlx5_ib_qp_set_counter(struct ib_qp *qp, struct rdma_counter *counter);
 #endif /* MLX5_IB_H */
index 8b7a60ada92c076172cfd2a02d56fa0f7c9309bd..2a97619ed6034d13e4091659dfdd4cb7b844db46 100644 (file)
@@ -34,6 +34,7 @@
 #include <rdma/ib_umem.h>
 #include <rdma/ib_cache.h>
 #include <rdma/ib_user_verbs.h>
+#include <rdma/rdma_counter.h>
 #include <linux/mlx5/fs.h>
 #include "mlx5_ib.h"
 #include "ib_rep.h"
@@ -3380,6 +3381,35 @@ static unsigned int get_tx_affinity(struct mlx5_ib_dev *dev,
        return tx_port_affinity;
 }
 
+static int __mlx5_ib_qp_set_counter(struct ib_qp *qp,
+                                   struct rdma_counter *counter)
+{
+       struct mlx5_ib_dev *dev = to_mdev(qp->device);
+       struct mlx5_ib_qp *mqp = to_mqp(qp);
+       struct mlx5_qp_context context = {};
+       struct mlx5_ib_port *mibport = NULL;
+       struct mlx5_ib_qp_base *base;
+       u32 set_id;
+
+       if (!MLX5_CAP_GEN(dev->mdev, rts2rts_qp_counters_set_id))
+               return 0;
+
+       if (counter) {
+               set_id = counter->id;
+       } else {
+               mibport = &dev->port[mqp->port - 1];
+               set_id = mibport->cnts.set_id;
+       }
+
+       base = &mqp->trans_qp.base;
+       context.qp_counter_set_usr_page &= cpu_to_be32(0xffffff);
+       context.qp_counter_set_usr_page |= cpu_to_be32(set_id << 24);
+       return mlx5_core_qp_modify(dev->mdev,
+                                  MLX5_CMD_OP_RTS2RTS_QP,
+                                  MLX5_QP_OPTPAR_COUNTER_SET_ID,
+                                  &context, &base->mqp);
+}
+
 static int __mlx5_ib_modify_qp(struct ib_qp *ibqp,
                               const struct ib_qp_attr *attr, int attr_mask,
                               enum ib_qp_state cur_state,
@@ -3433,6 +3463,7 @@ static int __mlx5_ib_modify_qp(struct ib_qp *ibqp,
        struct mlx5_ib_port *mibport = NULL;
        enum mlx5_qp_state mlx5_cur, mlx5_new;
        enum mlx5_qp_optpar optpar;
+       u32 set_id = 0;
        int mlx5_st;
        int err;
        u16 op;
@@ -3595,8 +3626,12 @@ static int __mlx5_ib_modify_qp(struct ib_qp *ibqp,
                        port_num = 0;
 
                mibport = &dev->port[port_num];
+               if (ibqp->counter)
+                       set_id = ibqp->counter->id;
+               else
+                       set_id = mibport->cnts.set_id;
                context->qp_counter_set_usr_page |=
-                       cpu_to_be32((u32)(mibport->cnts.set_id) << 24);
+                       cpu_to_be32(set_id << 24);
        }
 
        if (!ibqp->uobject && cur_state == IB_QPS_RESET && new_state == IB_QPS_INIT)
@@ -3624,7 +3659,7 @@ static int __mlx5_ib_modify_qp(struct ib_qp *ibqp,
 
                raw_qp_param.operation = op;
                if (cur_state == IB_QPS_RESET && new_state == IB_QPS_INIT) {
-                       raw_qp_param.rq_q_ctr_id = mibport->cnts.set_id;
+                       raw_qp_param.rq_q_ctr_id = set_id;
                        raw_qp_param.set_mask |= MLX5_RAW_QP_MOD_SET_RQ_Q_CTR_ID;
                }
 
@@ -3701,6 +3736,12 @@ static int __mlx5_ib_modify_qp(struct ib_qp *ibqp,
                qp->db.db[MLX5_SND_DBR] = 0;
        }
 
+       if ((new_state == IB_QPS_RTS) && qp->counter_pending) {
+               err = __mlx5_ib_qp_set_counter(ibqp, ibqp->counter);
+               if (!err)
+                       qp->counter_pending = 0;
+       }
+
 out:
        kfree(context);
        return err;
@@ -6435,3 +6476,34 @@ void mlx5_ib_drain_rq(struct ib_qp *qp)
 
        handle_drain_completion(cq, &rdrain, dev);
 }
+
+/**
+ * Bind a qp to a counter. If @counter is NULL then bind the qp to
+ * the default counter
+ */
+int mlx5_ib_qp_set_counter(struct ib_qp *qp, struct rdma_counter *counter)
+{
+       struct mlx5_ib_qp *mqp = to_mqp(qp);
+       int err = 0;
+
+       mutex_lock(&mqp->mutex);
+       if (mqp->state == IB_QPS_RESET) {
+               qp->counter = counter;
+               goto out;
+       }
+
+       if (mqp->state == IB_QPS_RTS) {
+               err = __mlx5_ib_qp_set_counter(qp, counter);
+               if (!err)
+                       qp->counter = counter;
+
+               goto out;
+       }
+
+       mqp->counter_pending = 1;
+       qp->counter = counter;
+
+out:
+       mutex_unlock(&mqp->mutex);
+       return err;
+}
index 93704110150483a429b99c96c0ea148230a87496..ae63b1ae90048cb84e35df4c509f2e6dfcd34abc 100644 (file)
@@ -71,6 +71,7 @@ enum mlx5_qp_optpar {
        MLX5_QP_OPTPAR_CQN_RCV                  = 1 << 19,
        MLX5_QP_OPTPAR_DC_HS                    = 1 << 20,
        MLX5_QP_OPTPAR_DC_KEY                   = 1 << 21,
+       MLX5_QP_OPTPAR_COUNTER_SET_ID           = 1 << 25,
 };
 
 enum mlx5_qp_state {