net/mlx5: Hairpin pair core object setup
authorOr Gerlitz <ogerlitz@mellanox.com>
Sun, 12 Nov 2017 13:15:10 +0000 (15:15 +0200)
committerSaeed Mahameed <saeedm@mellanox.com>
Tue, 9 Jan 2018 05:40:48 +0000 (07:40 +0200)
Low level code to setup hairpin pair core object, deals with:
 - create hairpin RQs/SQs
 - destroy hairpin RQs/SQs
 - modifying hairpin RQs/SQs - pairing (rst2rdy) and unpairing (rdy2rst)

Unlike conventional RQs/SQs, the memory used for the packet and descriptor
buffers is allocated by the firmware and not the driver. The driver sets
the overall data size (log).

Signed-off-by: Or Gerlitz <ogerlitz@mellanox.com>
Signed-off-by: Saeed Mahameed <saeedm@mellanox.com>
drivers/net/ethernet/mellanox/mlx5/core/transobj.c
include/linux/mlx5/transobj.h

index 5e128d7a9ffdfddd2549f972065f57c780c62c11..a09ebbaf3b68b186240e54d54828db7cec9498ae 100644 (file)
@@ -398,3 +398,187 @@ void mlx5_core_destroy_rqt(struct mlx5_core_dev *dev, u32 rqtn)
        mlx5_cmd_exec(dev, in, sizeof(in), out, sizeof(out));
 }
 EXPORT_SYMBOL(mlx5_core_destroy_rqt);
+
+static int mlx5_hairpin_create_rq(struct mlx5_core_dev *mdev,
+                                 struct mlx5_hairpin_params *params, u32 *rqn)
+{
+       u32 in[MLX5_ST_SZ_DW(create_rq_in)] = {0};
+       void *rqc, *wq;
+
+       rqc = MLX5_ADDR_OF(create_rq_in, in, ctx);
+       wq  = MLX5_ADDR_OF(rqc, rqc, wq);
+
+       MLX5_SET(rqc, rqc, hairpin, 1);
+       MLX5_SET(rqc, rqc, state, MLX5_RQC_STATE_RST);
+       MLX5_SET(rqc, rqc, counter_set_id, params->q_counter);
+
+       MLX5_SET(wq, wq, log_hairpin_data_sz, params->log_data_size);
+
+       return mlx5_core_create_rq(mdev, in, MLX5_ST_SZ_BYTES(create_rq_in), rqn);
+}
+
+static int mlx5_hairpin_create_sq(struct mlx5_core_dev *mdev,
+                                 struct mlx5_hairpin_params *params, u32 *sqn)
+{
+       u32 in[MLX5_ST_SZ_DW(create_sq_in)] = {0};
+       void *sqc, *wq;
+
+       sqc = MLX5_ADDR_OF(create_sq_in, in, ctx);
+       wq  = MLX5_ADDR_OF(sqc, sqc, wq);
+
+       MLX5_SET(sqc, sqc, hairpin, 1);
+       MLX5_SET(sqc, sqc, state, MLX5_SQC_STATE_RST);
+
+       MLX5_SET(wq, wq, log_hairpin_data_sz, params->log_data_size);
+
+       return mlx5_core_create_sq(mdev, in, MLX5_ST_SZ_BYTES(create_sq_in), sqn);
+}
+
+static int mlx5_hairpin_create_queues(struct mlx5_hairpin *hp,
+                                     struct mlx5_hairpin_params *params)
+{
+       int err;
+
+       err = mlx5_hairpin_create_rq(hp->func_mdev, params, &hp->rqn);
+       if (err)
+               goto out_err_rq;
+
+       err = mlx5_hairpin_create_sq(hp->peer_mdev, params, &hp->sqn);
+       if (err)
+               goto out_err_sq;
+
+       return 0;
+
+out_err_sq:
+       mlx5_core_destroy_rq(hp->func_mdev, hp->rqn);
+out_err_rq:
+       return err;
+}
+
+static void mlx5_hairpin_destroy_queues(struct mlx5_hairpin *hp)
+{
+       mlx5_core_destroy_rq(hp->func_mdev, hp->rqn);
+       mlx5_core_destroy_sq(hp->peer_mdev, hp->sqn);
+}
+
+static int mlx5_hairpin_modify_rq(struct mlx5_core_dev *func_mdev, u32 rqn,
+                                 int curr_state, int next_state,
+                                 u16 peer_vhca, u32 peer_sq)
+{
+       u32 in[MLX5_ST_SZ_DW(modify_rq_in)] = {0};
+       void *rqc;
+
+       rqc = MLX5_ADDR_OF(modify_rq_in, in, ctx);
+
+       if (next_state == MLX5_RQC_STATE_RDY) {
+               MLX5_SET(rqc, rqc, hairpin_peer_sq, peer_sq);
+               MLX5_SET(rqc, rqc, hairpin_peer_vhca, peer_vhca);
+       }
+
+       MLX5_SET(modify_rq_in, in, rq_state, curr_state);
+       MLX5_SET(rqc, rqc, state, next_state);
+
+       return mlx5_core_modify_rq(func_mdev, rqn,
+                                  in, MLX5_ST_SZ_BYTES(modify_rq_in));
+}
+
+static int mlx5_hairpin_modify_sq(struct mlx5_core_dev *peer_mdev, u32 sqn,
+                                 int curr_state, int next_state,
+                                 u16 peer_vhca, u32 peer_rq)
+{
+       u32 in[MLX5_ST_SZ_DW(modify_sq_in)] = {0};
+       void *sqc;
+
+       sqc = MLX5_ADDR_OF(modify_sq_in, in, ctx);
+
+       if (next_state == MLX5_RQC_STATE_RDY) {
+               MLX5_SET(sqc, sqc, hairpin_peer_rq, peer_rq);
+               MLX5_SET(sqc, sqc, hairpin_peer_vhca, peer_vhca);
+       }
+
+       MLX5_SET(modify_sq_in, in, sq_state, curr_state);
+       MLX5_SET(sqc, sqc, state, next_state);
+
+       return mlx5_core_modify_sq(peer_mdev, sqn,
+                                  in, MLX5_ST_SZ_BYTES(modify_sq_in));
+}
+
+static int mlx5_hairpin_pair_queues(struct mlx5_hairpin *hp)
+{
+       int err;
+
+       /* set peer SQ */
+       err = mlx5_hairpin_modify_sq(hp->peer_mdev, hp->sqn,
+                                    MLX5_SQC_STATE_RST, MLX5_SQC_STATE_RDY,
+                                    MLX5_CAP_GEN(hp->func_mdev, vhca_id), hp->rqn);
+       if (err)
+               goto err_modify_sq;
+
+       /* set func RQ */
+       err = mlx5_hairpin_modify_rq(hp->func_mdev, hp->rqn,
+                                    MLX5_RQC_STATE_RST, MLX5_RQC_STATE_RDY,
+                                    MLX5_CAP_GEN(hp->peer_mdev, vhca_id), hp->sqn);
+
+       if (err)
+               goto err_modify_rq;
+
+       return 0;
+
+err_modify_rq:
+       mlx5_hairpin_modify_sq(hp->peer_mdev, hp->sqn, MLX5_SQC_STATE_RDY,
+                              MLX5_SQC_STATE_RST, 0, 0);
+err_modify_sq:
+       return err;
+}
+
+static void mlx5_hairpin_unpair_queues(struct mlx5_hairpin *hp)
+{
+       /* unset func RQ */
+       mlx5_hairpin_modify_rq(hp->func_mdev, hp->rqn, MLX5_RQC_STATE_RDY,
+                              MLX5_RQC_STATE_RST, 0, 0);
+
+       /* unset peer SQ */
+       mlx5_hairpin_modify_sq(hp->peer_mdev, hp->sqn, MLX5_SQC_STATE_RDY,
+                              MLX5_SQC_STATE_RST, 0, 0);
+}
+
+struct mlx5_hairpin *
+mlx5_core_hairpin_create(struct mlx5_core_dev *func_mdev,
+                        struct mlx5_core_dev *peer_mdev,
+                        struct mlx5_hairpin_params *params)
+{
+       struct mlx5_hairpin *hp;
+       int size, err;
+
+       size = sizeof(*hp);
+       hp = kzalloc(size, GFP_KERNEL);
+       if (!hp)
+               return ERR_PTR(-ENOMEM);
+
+       hp->func_mdev = func_mdev;
+       hp->peer_mdev = peer_mdev;
+
+       /* alloc and pair func --> peer hairpin */
+       err = mlx5_hairpin_create_queues(hp, params);
+       if (err)
+               goto err_create_queues;
+
+       err = mlx5_hairpin_pair_queues(hp);
+       if (err)
+               goto err_pair_queues;
+
+       return hp;
+
+err_pair_queues:
+       mlx5_hairpin_destroy_queues(hp);
+err_create_queues:
+       kfree(hp);
+       return ERR_PTR(err);
+}
+
+void mlx5_core_hairpin_destroy(struct mlx5_hairpin *hp)
+{
+       mlx5_hairpin_unpair_queues(hp);
+       mlx5_hairpin_destroy_queues(hp);
+       kfree(hp);
+}
index 88441f5ece25f9dbe6f45676789b23a32a8d53d4..a228310c1968a700710fc808806778ab33dccd7b 100644 (file)
@@ -75,4 +75,23 @@ int mlx5_core_modify_rqt(struct mlx5_core_dev *dev, u32 rqtn, u32 *in,
                         int inlen);
 void mlx5_core_destroy_rqt(struct mlx5_core_dev *dev, u32 rqtn);
 
+struct mlx5_hairpin_params {
+       u8  log_data_size;
+       u16 q_counter;
+};
+
+struct mlx5_hairpin {
+       struct mlx5_core_dev *func_mdev;
+       struct mlx5_core_dev *peer_mdev;
+
+       u32 rqn;
+       u32 sqn;
+};
+
+struct mlx5_hairpin *
+mlx5_core_hairpin_create(struct mlx5_core_dev *func_mdev,
+                        struct mlx5_core_dev *peer_mdev,
+                        struct mlx5_hairpin_params *params);
+
+void mlx5_core_hairpin_destroy(struct mlx5_hairpin *pair);
 #endif /* __TRANSOBJ_H__ */