Skip to content

Commit b368d7c

Browse files
matanb10dledford
authored andcommitted
IB/mlx5: Add hca_core_clock_offset to udata in init_ucontext
Pass hca_core_clock_offset to user-space is mandatory in order to let the user-space read the free-running clock register from the right offset in the memory mapped page. Passing this value is done by changing the vendor's command and response of init_ucontext to be in extensible form. Signed-off-by: Matan Barak <[email protected]> Reviewed-by: Moshe Lazer <[email protected]> Signed-off-by: Doug Ledford <[email protected]>
1 parent 7c60bcb commit b368d7c

File tree

4 files changed

+47
-12
lines changed

4 files changed

+47
-12
lines changed

drivers/infiniband/hw/mlx5/main.c

Lines changed: 29 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -795,8 +795,8 @@ static struct ib_ucontext *mlx5_ib_alloc_ucontext(struct ib_device *ibdev,
795795
struct ib_udata *udata)
796796
{
797797
struct mlx5_ib_dev *dev = to_mdev(ibdev);
798-
struct mlx5_ib_alloc_ucontext_req_v2 req;
799-
struct mlx5_ib_alloc_ucontext_resp resp;
798+
struct mlx5_ib_alloc_ucontext_req_v2 req = {};
799+
struct mlx5_ib_alloc_ucontext_resp resp = {};
800800
struct mlx5_ib_ucontext *context;
801801
struct mlx5_uuar_info *uuari;
802802
struct mlx5_uar *uars;
@@ -811,20 +811,19 @@ static struct ib_ucontext *mlx5_ib_alloc_ucontext(struct ib_device *ibdev,
811811
if (!dev->ib_active)
812812
return ERR_PTR(-EAGAIN);
813813

814-
memset(&req, 0, sizeof(req));
815814
reqlen = udata->inlen - sizeof(struct ib_uverbs_cmd_hdr);
816815
if (reqlen == sizeof(struct mlx5_ib_alloc_ucontext_req))
817816
ver = 0;
818-
else if (reqlen == sizeof(struct mlx5_ib_alloc_ucontext_req_v2))
817+
else if (reqlen >= sizeof(struct mlx5_ib_alloc_ucontext_req_v2))
819818
ver = 2;
820819
else
821820
return ERR_PTR(-EINVAL);
822821

823-
err = ib_copy_from_udata(&req, udata, reqlen);
822+
err = ib_copy_from_udata(&req, udata, min(reqlen, sizeof(req)));
824823
if (err)
825824
return ERR_PTR(err);
826825

827-
if (req.flags || req.reserved)
826+
if (req.flags)
828827
return ERR_PTR(-EINVAL);
829828

830829
if (req.total_num_uuars > MLX5_MAX_UUARS)
@@ -833,6 +832,14 @@ static struct ib_ucontext *mlx5_ib_alloc_ucontext(struct ib_device *ibdev,
833832
if (req.total_num_uuars == 0)
834833
return ERR_PTR(-EINVAL);
835834

835+
if (req.comp_mask)
836+
return ERR_PTR(-EOPNOTSUPP);
837+
838+
if (reqlen > sizeof(req) &&
839+
!ib_is_udata_cleared(udata, sizeof(req),
840+
udata->inlen - sizeof(req)))
841+
return ERR_PTR(-EOPNOTSUPP);
842+
836843
req.total_num_uuars = ALIGN(req.total_num_uuars,
837844
MLX5_NON_FP_BF_REGS_PER_PAGE);
838845
if (req.num_low_latency_uuars > req.total_num_uuars - 1)
@@ -848,6 +855,8 @@ static struct ib_ucontext *mlx5_ib_alloc_ucontext(struct ib_device *ibdev,
848855
resp.max_send_wqebb = 1 << MLX5_CAP_GEN(dev->mdev, log_max_qp_sz);
849856
resp.max_recv_wr = 1 << MLX5_CAP_GEN(dev->mdev, log_max_qp_sz);
850857
resp.max_srq_recv_wr = 1 << MLX5_CAP_GEN(dev->mdev, log_max_srq_sz);
858+
resp.response_length = min(offsetof(typeof(resp), response_length) +
859+
sizeof(resp.response_length), udata->outlen);
851860

852861
context = kzalloc(sizeof(*context), GFP_KERNEL);
853862
if (!context)
@@ -898,8 +907,20 @@ static struct ib_ucontext *mlx5_ib_alloc_ucontext(struct ib_device *ibdev,
898907

899908
resp.tot_uuars = req.total_num_uuars;
900909
resp.num_ports = MLX5_CAP_GEN(dev->mdev, num_ports);
901-
err = ib_copy_to_udata(udata, &resp,
902-
sizeof(resp) - sizeof(resp.reserved));
910+
911+
if (field_avail(typeof(resp), reserved2, udata->outlen))
912+
resp.response_length += sizeof(resp.reserved2);
913+
914+
if (field_avail(typeof(resp), hca_core_clock_offset, udata->outlen)) {
915+
resp.comp_mask |=
916+
MLX5_IB_ALLOC_UCONTEXT_RESP_MASK_CORE_CLOCK_OFFSET;
917+
resp.hca_core_clock_offset =
918+
offsetof(struct mlx5_init_seg, internal_timer_h) %
919+
PAGE_SIZE;
920+
resp.response_length += sizeof(resp.hca_core_clock_offset);
921+
}
922+
923+
err = ib_copy_to_udata(udata, &resp, resp.response_length);
903924
if (err)
904925
goto out_uars;
905926

drivers/infiniband/hw/mlx5/mlx5_ib.h

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,9 @@ pr_err("%s:%s:%d:(pid %d): " format, (dev)->ib_dev.name, __func__, \
5555
pr_warn("%s:%s:%d:(pid %d): " format, (dev)->ib_dev.name, __func__, \
5656
__LINE__, current->pid, ##arg)
5757

58+
#define field_avail(type, fld, sz) (offsetof(type, fld) + \
59+
sizeof(((type *)0)->fld) <= (sz))
60+
5861
enum {
5962
MLX5_IB_MMAP_CMD_SHIFT = 8,
6063
MLX5_IB_MMAP_CMD_MASK = 0xff,

drivers/infiniband/hw/mlx5/user.h

Lines changed: 10 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -66,7 +66,11 @@ struct mlx5_ib_alloc_ucontext_req_v2 {
6666
__u32 total_num_uuars;
6767
__u32 num_low_latency_uuars;
6868
__u32 flags;
69-
__u32 reserved;
69+
__u32 comp_mask;
70+
};
71+
72+
enum mlx5_ib_alloc_ucontext_resp_mask {
73+
MLX5_IB_ALLOC_UCONTEXT_RESP_MASK_CORE_CLOCK_OFFSET = 1UL << 0,
7074
};
7175

7276
struct mlx5_ib_alloc_ucontext_resp {
@@ -80,7 +84,11 @@ struct mlx5_ib_alloc_ucontext_resp {
8084
__u32 max_recv_wr;
8185
__u32 max_srq_recv_wr;
8286
__u16 num_ports;
83-
__u16 reserved;
87+
__u16 reserved1;
88+
__u32 comp_mask;
89+
__u32 response_length;
90+
__u32 reserved2;
91+
__u64 hca_core_clock_offset;
8492
};
8593

8694
struct mlx5_ib_alloc_pd_resp {

include/linux/mlx5/device.h

Lines changed: 5 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -462,9 +462,12 @@ struct mlx5_init_seg {
462462
__be32 rsvd1[120];
463463
__be32 initializing;
464464
struct health_buffer health;
465-
__be32 rsvd2[884];
465+
__be32 rsvd2[880];
466+
__be32 internal_timer_h;
467+
__be32 internal_timer_l;
468+
__be32 rsvd3[2];
466469
__be32 health_counter;
467-
__be32 rsvd3[1019];
470+
__be32 rsvd4[1019];
468471
__be64 ieee1588_clk;
469472
__be32 ieee1588_clk_type;
470473
__be32 clr_intx;

0 commit comments

Comments
 (0)