Commit 2bb3c32c authored by Devesh Sharma's avatar Devesh Sharma Committed by Jason Gunthorpe

RDMA/bnxt_re: Change wr posting logic to accommodate variable wqes

Modifying the post-send and post-recv to initialize the wqes slot by slot
dynamically depending on the number of max sges requested by consumer at
the time of QP creation.

Changed the QP creation logic to determine the size of SQ and RQ in 16B
slots based on the number of wqe and number of SGEs requested by consumer

Link: https://lore.kernel.org/r/1594822619-4098-6-git-send-email-devesh.sharma@broadcom.comSigned-off-by: default avatarDevesh Sharma <devesh.sharma@broadcom.com>
Signed-off-by: default avatarJason Gunthorpe <jgg@nvidia.com>
parent 54ace984
...@@ -842,16 +842,79 @@ static u8 __from_ib_qp_type(enum ib_qp_type type) ...@@ -842,16 +842,79 @@ static u8 __from_ib_qp_type(enum ib_qp_type type)
} }
} }
static u16 bnxt_re_setup_rwqe_size(struct bnxt_qplib_qp *qplqp,
int rsge, int max)
{
if (qplqp->wqe_mode == BNXT_QPLIB_WQE_MODE_STATIC)
rsge = max;
return bnxt_re_get_rwqe_size(rsge);
}
static u16 bnxt_re_get_wqe_size(int ilsize, int nsge)
{
u16 wqe_size, calc_ils;
wqe_size = bnxt_re_get_swqe_size(nsge);
if (ilsize) {
calc_ils = sizeof(struct sq_send_hdr) + ilsize;
wqe_size = max_t(u16, calc_ils, wqe_size);
wqe_size = ALIGN(wqe_size, sizeof(struct sq_send_hdr));
}
return wqe_size;
}
static int bnxt_re_setup_swqe_size(struct bnxt_re_qp *qp,
struct ib_qp_init_attr *init_attr)
{
struct bnxt_qplib_dev_attr *dev_attr;
struct bnxt_qplib_qp *qplqp;
struct bnxt_re_dev *rdev;
struct bnxt_qplib_q *sq;
int align, ilsize;
rdev = qp->rdev;
qplqp = &qp->qplib_qp;
sq = &qplqp->sq;
dev_attr = &rdev->dev_attr;
align = sizeof(struct sq_send_hdr);
ilsize = ALIGN(init_attr->cap.max_inline_data, align);
sq->wqe_size = bnxt_re_get_wqe_size(ilsize, sq->max_sge);
if (sq->wqe_size > bnxt_re_get_swqe_size(dev_attr->max_qp_sges))
return -EINVAL;
/* For gen p4 and gen p5 backward compatibility mode
* wqe size is fixed to 128 bytes
*/
if (sq->wqe_size < bnxt_re_get_swqe_size(dev_attr->max_qp_sges) &&
qplqp->wqe_mode == BNXT_QPLIB_WQE_MODE_STATIC)
sq->wqe_size = bnxt_re_get_swqe_size(dev_attr->max_qp_sges);
if (init_attr->cap.max_inline_data) {
qplqp->max_inline_data = sq->wqe_size -
sizeof(struct sq_send_hdr);
init_attr->cap.max_inline_data = qplqp->max_inline_data;
if (qplqp->wqe_mode == BNXT_QPLIB_WQE_MODE_STATIC)
sq->max_sge = qplqp->max_inline_data /
sizeof(struct sq_sge);
}
return 0;
}
static int bnxt_re_init_user_qp(struct bnxt_re_dev *rdev, struct bnxt_re_pd *pd, static int bnxt_re_init_user_qp(struct bnxt_re_dev *rdev, struct bnxt_re_pd *pd,
struct bnxt_re_qp *qp, struct ib_udata *udata) struct bnxt_re_qp *qp, struct ib_udata *udata)
{ {
struct bnxt_qplib_qp *qplib_qp;
struct bnxt_re_ucontext *cntx;
struct bnxt_re_qp_req ureq; struct bnxt_re_qp_req ureq;
struct bnxt_qplib_qp *qplib_qp = &qp->qplib_qp;
struct ib_umem *umem;
int bytes = 0, psn_sz; int bytes = 0, psn_sz;
struct bnxt_re_ucontext *cntx = rdma_udata_to_drv_context( struct ib_umem *umem;
udata, struct bnxt_re_ucontext, ib_uctx); int psn_nume;
qplib_qp = &qp->qplib_qp;
cntx = rdma_udata_to_drv_context(udata, struct bnxt_re_ucontext,
ib_uctx);
if (ib_copy_from_udata(&ureq, udata, sizeof(ureq))) if (ib_copy_from_udata(&ureq, udata, sizeof(ureq)))
return -EFAULT; return -EFAULT;
...@@ -859,10 +922,15 @@ static int bnxt_re_init_user_qp(struct bnxt_re_dev *rdev, struct bnxt_re_pd *pd, ...@@ -859,10 +922,15 @@ static int bnxt_re_init_user_qp(struct bnxt_re_dev *rdev, struct bnxt_re_pd *pd,
/* Consider mapping PSN search memory only for RC QPs. */ /* Consider mapping PSN search memory only for RC QPs. */
if (qplib_qp->type == CMDQ_CREATE_QP_TYPE_RC) { if (qplib_qp->type == CMDQ_CREATE_QP_TYPE_RC) {
psn_sz = bnxt_qplib_is_chip_gen_p5(rdev->chip_ctx) ? psn_sz = bnxt_qplib_is_chip_gen_p5(rdev->chip_ctx) ?
sizeof(struct sq_psn_search_ext) : sizeof(struct sq_psn_search_ext) :
sizeof(struct sq_psn_search); sizeof(struct sq_psn_search);
bytes += (qplib_qp->sq.max_wqe * psn_sz); psn_nume = (qplib_qp->wqe_mode == BNXT_QPLIB_WQE_MODE_STATIC) ?
qplib_qp->sq.max_wqe :
((qplib_qp->sq.max_wqe * qplib_qp->sq.wqe_size) /
sizeof(struct bnxt_qplib_sge));
bytes += (psn_nume * psn_sz);
} }
bytes = PAGE_ALIGN(bytes); bytes = PAGE_ALIGN(bytes);
umem = ib_umem_get(&rdev->ibdev, ureq.qpsva, bytes, umem = ib_umem_get(&rdev->ibdev, ureq.qpsva, bytes,
IB_ACCESS_LOCAL_WRITE); IB_ACCESS_LOCAL_WRITE);
...@@ -975,7 +1043,7 @@ static struct bnxt_re_qp *bnxt_re_create_shadow_qp ...@@ -975,7 +1043,7 @@ static struct bnxt_re_qp *bnxt_re_create_shadow_qp
qp->qplib_qp.sig_type = true; qp->qplib_qp.sig_type = true;
/* Shadow QP SQ depth should be same as QP1 RQ depth */ /* Shadow QP SQ depth should be same as QP1 RQ depth */
qp->qplib_qp.sq.wqe_size = bnxt_re_get_swqe_size(); qp->qplib_qp.sq.wqe_size = bnxt_re_get_wqe_size(0, 6);
qp->qplib_qp.sq.max_wqe = qp1_qp->rq.max_wqe; qp->qplib_qp.sq.max_wqe = qp1_qp->rq.max_wqe;
qp->qplib_qp.sq.max_sge = 2; qp->qplib_qp.sq.max_sge = 2;
/* Q full delta can be 1 since it is internal QP */ /* Q full delta can be 1 since it is internal QP */
...@@ -986,7 +1054,7 @@ static struct bnxt_re_qp *bnxt_re_create_shadow_qp ...@@ -986,7 +1054,7 @@ static struct bnxt_re_qp *bnxt_re_create_shadow_qp
qp->qplib_qp.scq = qp1_qp->scq; qp->qplib_qp.scq = qp1_qp->scq;
qp->qplib_qp.rcq = qp1_qp->rcq; qp->qplib_qp.rcq = qp1_qp->rcq;
qp->qplib_qp.rq.wqe_size = bnxt_re_get_rwqe_size(); qp->qplib_qp.rq.wqe_size = bnxt_re_get_rwqe_size(6);
qp->qplib_qp.rq.max_wqe = qp1_qp->rq.max_wqe; qp->qplib_qp.rq.max_wqe = qp1_qp->rq.max_wqe;
qp->qplib_qp.rq.max_sge = qp1_qp->rq.max_sge; qp->qplib_qp.rq.max_sge = qp1_qp->rq.max_sge;
/* Q full delta can be 1 since it is internal QP */ /* Q full delta can be 1 since it is internal QP */
...@@ -1041,19 +1109,21 @@ static int bnxt_re_init_rq_attr(struct bnxt_re_qp *qp, ...@@ -1041,19 +1109,21 @@ static int bnxt_re_init_rq_attr(struct bnxt_re_qp *qp,
qplqp->srq = &srq->qplib_srq; qplqp->srq = &srq->qplib_srq;
rq->max_wqe = 0; rq->max_wqe = 0;
} else { } else {
rq->wqe_size = bnxt_re_get_rwqe_size(); rq->max_sge = init_attr->cap.max_recv_sge;
if (rq->max_sge > dev_attr->max_qp_sges)
rq->max_sge = dev_attr->max_qp_sges;
init_attr->cap.max_recv_sge = rq->max_sge;
rq->wqe_size = bnxt_re_setup_rwqe_size(qplqp, rq->max_sge,
dev_attr->max_qp_sges);
/* Allocate 1 more than what's provided so posting max doesn't /* Allocate 1 more than what's provided so posting max doesn't
* mean empty. * mean empty.
*/ */
entries = roundup_pow_of_two(init_attr->cap.max_recv_wr + 1); entries = roundup_pow_of_two(init_attr->cap.max_recv_wr + 1);
rq->max_wqe = min_t(u32, entries, dev_attr->max_qp_wqes + 1); rq->max_wqe = min_t(u32, entries, dev_attr->max_qp_wqes + 1);
rq->q_full_delta = rq->max_wqe - init_attr->cap.max_recv_wr; rq->q_full_delta = 0;
rq->max_sge = init_attr->cap.max_recv_sge; rq->sg_info.pgsize = PAGE_SIZE;
if (rq->max_sge > dev_attr->max_qp_sges) rq->sg_info.pgshft = PAGE_SHIFT;
rq->max_sge = dev_attr->max_qp_sges;
} }
rq->sg_info.pgsize = PAGE_SIZE;
rq->sg_info.pgshft = PAGE_SHIFT;
return 0; return 0;
} }
...@@ -1068,41 +1138,48 @@ static void bnxt_re_adjust_gsi_rq_attr(struct bnxt_re_qp *qp) ...@@ -1068,41 +1138,48 @@ static void bnxt_re_adjust_gsi_rq_attr(struct bnxt_re_qp *qp)
qplqp = &qp->qplib_qp; qplqp = &qp->qplib_qp;
dev_attr = &rdev->dev_attr; dev_attr = &rdev->dev_attr;
qplqp->rq.max_sge = dev_attr->max_qp_sges; if (!bnxt_qplib_is_chip_gen_p5(rdev->chip_ctx)) {
if (qplqp->rq.max_sge > dev_attr->max_qp_sges)
qplqp->rq.max_sge = dev_attr->max_qp_sges; qplqp->rq.max_sge = dev_attr->max_qp_sges;
qplqp->rq.max_sge = 6; if (qplqp->rq.max_sge > dev_attr->max_qp_sges)
qplqp->rq.max_sge = dev_attr->max_qp_sges;
qplqp->rq.max_sge = 6;
}
} }
static void bnxt_re_init_sq_attr(struct bnxt_re_qp *qp, static int bnxt_re_init_sq_attr(struct bnxt_re_qp *qp,
struct ib_qp_init_attr *init_attr, struct ib_qp_init_attr *init_attr,
struct ib_udata *udata) struct ib_udata *udata)
{ {
struct bnxt_qplib_dev_attr *dev_attr; struct bnxt_qplib_dev_attr *dev_attr;
struct bnxt_qplib_qp *qplqp; struct bnxt_qplib_qp *qplqp;
struct bnxt_re_dev *rdev; struct bnxt_re_dev *rdev;
struct bnxt_qplib_q *sq; struct bnxt_qplib_q *sq;
int entries; int entries;
int diff;
int rc;
rdev = qp->rdev; rdev = qp->rdev;
qplqp = &qp->qplib_qp; qplqp = &qp->qplib_qp;
sq = &qplqp->sq; sq = &qplqp->sq;
dev_attr = &rdev->dev_attr; dev_attr = &rdev->dev_attr;
sq->wqe_size = bnxt_re_get_swqe_size();
sq->max_sge = init_attr->cap.max_send_sge; sq->max_sge = init_attr->cap.max_send_sge;
if (sq->max_sge > dev_attr->max_qp_sges) if (sq->max_sge > dev_attr->max_qp_sges) {
sq->max_sge = dev_attr->max_qp_sges; sq->max_sge = dev_attr->max_qp_sges;
/* init_attr->cap.max_send_sge = sq->max_sge;
* Change the SQ depth if user has requested minimum using }
* configfs. Only supported for kernel consumers
*/ rc = bnxt_re_setup_swqe_size(qp, init_attr);
if (rc)
return rc;
entries = init_attr->cap.max_send_wr; entries = init_attr->cap.max_send_wr;
/* Allocate 128 + 1 more than what's provided */ /* Allocate 128 + 1 more than what's provided */
entries = roundup_pow_of_two(entries + BNXT_QPLIB_RESERVED_QP_WRS + 1); diff = (qplqp->wqe_mode == BNXT_QPLIB_WQE_MODE_VARIABLE) ?
sq->max_wqe = min_t(u32, entries, dev_attr->max_qp_wqes + 0 : BNXT_QPLIB_RESERVED_QP_WRS;
BNXT_QPLIB_RESERVED_QP_WRS + 1); entries = roundup_pow_of_two(entries + diff + 1);
sq->q_full_delta = BNXT_QPLIB_RESERVED_QP_WRS + 1; sq->max_wqe = min_t(u32, entries, dev_attr->max_qp_wqes + diff + 1);
sq->q_full_delta = diff + 1;
/* /*
* Reserving one slot for Phantom WQE. Application can * Reserving one slot for Phantom WQE. Application can
* post one extra entry in this case. But allowing this to avoid * post one extra entry in this case. But allowing this to avoid
...@@ -1111,6 +1188,8 @@ static void bnxt_re_init_sq_attr(struct bnxt_re_qp *qp, ...@@ -1111,6 +1188,8 @@ static void bnxt_re_init_sq_attr(struct bnxt_re_qp *qp,
qplqp->sq.q_full_delta -= 1; qplqp->sq.q_full_delta -= 1;
qplqp->sq.sg_info.pgsize = PAGE_SIZE; qplqp->sq.sg_info.pgsize = PAGE_SIZE;
qplqp->sq.sg_info.pgshft = PAGE_SHIFT; qplqp->sq.sg_info.pgshft = PAGE_SHIFT;
return 0;
} }
static void bnxt_re_adjust_gsi_sq_attr(struct bnxt_re_qp *qp, static void bnxt_re_adjust_gsi_sq_attr(struct bnxt_re_qp *qp,
...@@ -1125,13 +1204,16 @@ static void bnxt_re_adjust_gsi_sq_attr(struct bnxt_re_qp *qp, ...@@ -1125,13 +1204,16 @@ static void bnxt_re_adjust_gsi_sq_attr(struct bnxt_re_qp *qp,
qplqp = &qp->qplib_qp; qplqp = &qp->qplib_qp;
dev_attr = &rdev->dev_attr; dev_attr = &rdev->dev_attr;
entries = roundup_pow_of_two(init_attr->cap.max_send_wr + 1); if (!bnxt_qplib_is_chip_gen_p5(rdev->chip_ctx)) {
qplqp->sq.max_wqe = min_t(u32, entries, dev_attr->max_qp_wqes + 1); entries = roundup_pow_of_two(init_attr->cap.max_send_wr + 1);
qplqp->sq.q_full_delta = qplqp->sq.max_wqe - qplqp->sq.max_wqe = min_t(u32, entries,
init_attr->cap.max_send_wr; dev_attr->max_qp_wqes + 1);
qplqp->sq.max_sge++; /* Need one extra sge to put UD header */ qplqp->sq.q_full_delta = qplqp->sq.max_wqe -
if (qplqp->sq.max_sge > dev_attr->max_qp_sges) init_attr->cap.max_send_wr;
qplqp->sq.max_sge = dev_attr->max_qp_sges; qplqp->sq.max_sge++; /* Need one extra sge to put UD header */
if (qplqp->sq.max_sge > dev_attr->max_qp_sges)
qplqp->sq.max_sge = dev_attr->max_qp_sges;
}
} }
static int bnxt_re_init_qp_type(struct bnxt_re_dev *rdev, static int bnxt_re_init_qp_type(struct bnxt_re_dev *rdev,
...@@ -1227,7 +1309,9 @@ static int bnxt_re_init_qp_attr(struct bnxt_re_qp *qp, struct bnxt_re_pd *pd, ...@@ -1227,7 +1309,9 @@ static int bnxt_re_init_qp_attr(struct bnxt_re_qp *qp, struct bnxt_re_pd *pd,
bnxt_re_adjust_gsi_rq_attr(qp); bnxt_re_adjust_gsi_rq_attr(qp);
/* Setup SQ */ /* Setup SQ */
bnxt_re_init_sq_attr(qp, init_attr, udata); rc = bnxt_re_init_sq_attr(qp, init_attr, udata);
if (rc)
goto out;
if (init_attr->qp_type == IB_QPT_GSI) if (init_attr->qp_type == IB_QPT_GSI)
bnxt_re_adjust_gsi_sq_attr(qp, init_attr); bnxt_re_adjust_gsi_sq_attr(qp, init_attr);
...@@ -1575,8 +1659,9 @@ int bnxt_re_create_srq(struct ib_srq *ib_srq, ...@@ -1575,8 +1659,9 @@ int bnxt_re_create_srq(struct ib_srq *ib_srq,
entries = dev_attr->max_srq_wqes + 1; entries = dev_attr->max_srq_wqes + 1;
srq->qplib_srq.max_wqe = entries; srq->qplib_srq.max_wqe = entries;
srq->qplib_srq.wqe_size = bnxt_re_get_rwqe_size();
srq->qplib_srq.max_sge = srq_init_attr->attr.max_sge; srq->qplib_srq.max_sge = srq_init_attr->attr.max_sge;
srq->qplib_srq.wqe_size =
bnxt_re_get_rwqe_size(srq->qplib_srq.max_sge);
srq->qplib_srq.threshold = srq_init_attr->attr.srq_limit; srq->qplib_srq.threshold = srq_init_attr->attr.srq_limit;
srq->srq_limit = srq_init_attr->attr.srq_limit; srq->srq_limit = srq_init_attr->attr.srq_limit;
srq->qplib_srq.eventq_hw_ring_id = rdev->nq[0].ring_id; srq->qplib_srq.eventq_hw_ring_id = rdev->nq[0].ring_id;
......
...@@ -136,14 +136,14 @@ struct bnxt_re_ucontext { ...@@ -136,14 +136,14 @@ struct bnxt_re_ucontext {
spinlock_t sh_lock; /* protect shpg */ spinlock_t sh_lock; /* protect shpg */
}; };
static inline u16 bnxt_re_get_swqe_size(void) static inline u16 bnxt_re_get_swqe_size(int nsge)
{ {
return sizeof(struct sq_send); return sizeof(struct sq_send_hdr) + nsge * sizeof(struct sq_sge);
} }
static inline u16 bnxt_re_get_rwqe_size(void) static inline u16 bnxt_re_get_rwqe_size(int nsge)
{ {
return sizeof(struct rq_wqe); return sizeof(struct rq_wqe_hdr) + (nsge * sizeof(struct sq_sge));
} }
int bnxt_re_query_device(struct ib_device *ibdev, int bnxt_re_query_device(struct ib_device *ibdev,
......
...@@ -660,6 +660,7 @@ int bnxt_qplib_create_srq(struct bnxt_qplib_res *res, ...@@ -660,6 +660,7 @@ int bnxt_qplib_create_srq(struct bnxt_qplib_res *res,
srq->dbinfo.hwq = &srq->hwq; srq->dbinfo.hwq = &srq->hwq;
srq->dbinfo.xid = srq->id; srq->dbinfo.xid = srq->id;
srq->dbinfo.db = srq->dpi->dbr; srq->dbinfo.db = srq->dpi->dbr;
srq->dbinfo.max_slot = 1;
srq->dbinfo.priv_db = res->dpi_tbl.dbr_bar_reg_iomem; srq->dbinfo.priv_db = res->dpi_tbl.dbr_bar_reg_iomem;
if (srq->threshold) if (srq->threshold)
bnxt_qplib_armen_db(&srq->dbinfo, DBC_DBC_TYPE_SRQ_ARMENA); bnxt_qplib_armen_db(&srq->dbinfo, DBC_DBC_TYPE_SRQ_ARMENA);
...@@ -797,10 +798,8 @@ static int bnxt_qplib_alloc_init_swq(struct bnxt_qplib_q *que) ...@@ -797,10 +798,8 @@ static int bnxt_qplib_alloc_init_swq(struct bnxt_qplib_q *que)
que->swq_start = 0; que->swq_start = 0;
que->swq_last = que->max_wqe - 1; que->swq_last = que->max_wqe - 1;
for (indx = 0; indx < que->max_wqe; indx++) { for (indx = 0; indx < que->max_wqe; indx++)
que->swq[indx].slots = 1;
que->swq[indx].next_idx = indx + 1; que->swq[indx].next_idx = indx + 1;
}
que->swq[que->swq_last].next_idx = 0; /* Make it circular */ que->swq[que->swq_last].next_idx = 0; /* Make it circular */
que->swq_last = 0; que->swq_last = 0;
out: out:
...@@ -831,8 +830,8 @@ int bnxt_qplib_create_qp1(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp) ...@@ -831,8 +830,8 @@ int bnxt_qplib_create_qp1(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp)
/* SQ */ /* SQ */
hwq_attr.res = res; hwq_attr.res = res;
hwq_attr.sginfo = &sq->sg_info; hwq_attr.sginfo = &sq->sg_info;
hwq_attr.depth = sq->max_wqe; hwq_attr.stride = sizeof(struct sq_sge);
hwq_attr.stride = sq->wqe_size; hwq_attr.depth = bnxt_qplib_get_depth(sq);
hwq_attr.type = HWQ_TYPE_QUEUE; hwq_attr.type = HWQ_TYPE_QUEUE;
rc = bnxt_qplib_alloc_init_hwq(&sq->hwq, &hwq_attr); rc = bnxt_qplib_alloc_init_hwq(&sq->hwq, &hwq_attr);
if (rc) if (rc)
...@@ -842,7 +841,7 @@ int bnxt_qplib_create_qp1(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp) ...@@ -842,7 +841,7 @@ int bnxt_qplib_create_qp1(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp)
if (rc) if (rc)
goto fail_sq; goto fail_sq;
req.sq_size = cpu_to_le32(sq->max_wqe); req.sq_size = cpu_to_le32(bnxt_qplib_set_sq_size(sq, qp->wqe_mode));
pbl = &sq->hwq.pbl[PBL_LVL_0]; pbl = &sq->hwq.pbl[PBL_LVL_0];
req.sq_pbl = cpu_to_le64(pbl->pg_map_arr[0]); req.sq_pbl = cpu_to_le64(pbl->pg_map_arr[0]);
pg_sz_lvl = (bnxt_qplib_base_pg_size(&sq->hwq) << pg_sz_lvl = (bnxt_qplib_base_pg_size(&sq->hwq) <<
...@@ -858,8 +857,8 @@ int bnxt_qplib_create_qp1(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp) ...@@ -858,8 +857,8 @@ int bnxt_qplib_create_qp1(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp)
if (rq->max_wqe) { if (rq->max_wqe) {
hwq_attr.res = res; hwq_attr.res = res;
hwq_attr.sginfo = &rq->sg_info; hwq_attr.sginfo = &rq->sg_info;
hwq_attr.stride = rq->wqe_size; hwq_attr.stride = sizeof(struct sq_sge);
hwq_attr.depth = qp->rq.max_wqe; hwq_attr.depth = bnxt_qplib_get_depth(rq);
hwq_attr.type = HWQ_TYPE_QUEUE; hwq_attr.type = HWQ_TYPE_QUEUE;
rc = bnxt_qplib_alloc_init_hwq(&rq->hwq, &hwq_attr); rc = bnxt_qplib_alloc_init_hwq(&rq->hwq, &hwq_attr);
if (rc) if (rc)
...@@ -901,10 +900,12 @@ int bnxt_qplib_create_qp1(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp) ...@@ -901,10 +900,12 @@ int bnxt_qplib_create_qp1(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp)
sq->dbinfo.hwq = &sq->hwq; sq->dbinfo.hwq = &sq->hwq;
sq->dbinfo.xid = qp->id; sq->dbinfo.xid = qp->id;
sq->dbinfo.db = qp->dpi->dbr; sq->dbinfo.db = qp->dpi->dbr;
sq->dbinfo.max_slot = bnxt_qplib_set_sq_max_slot(qp->wqe_mode);
if (rq->max_wqe) { if (rq->max_wqe) {
rq->dbinfo.hwq = &rq->hwq; rq->dbinfo.hwq = &rq->hwq;
rq->dbinfo.xid = qp->id; rq->dbinfo.xid = qp->id;
rq->dbinfo.db = qp->dpi->dbr; rq->dbinfo.db = qp->dpi->dbr;
rq->dbinfo.max_slot = bnxt_qplib_set_rq_max_slot(rq->wqe_size);
} }
rcfw->qp_tbl[qp->id].qp_id = qp->id; rcfw->qp_tbl[qp->id].qp_id = qp->id;
rcfw->qp_tbl[qp->id].qp_handle = (void *)qp; rcfw->qp_tbl[qp->id].qp_handle = (void *)qp;
...@@ -976,10 +977,10 @@ int bnxt_qplib_create_qp(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp) ...@@ -976,10 +977,10 @@ int bnxt_qplib_create_qp(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp)
hwq_attr.res = res; hwq_attr.res = res;
hwq_attr.sginfo = &sq->sg_info; hwq_attr.sginfo = &sq->sg_info;
hwq_attr.stride = sq->wqe_size; hwq_attr.stride = sizeof(struct sq_sge);
hwq_attr.depth = sq->max_wqe; hwq_attr.depth = bnxt_qplib_get_depth(sq);
hwq_attr.aux_stride = psn_sz; hwq_attr.aux_stride = psn_sz;
hwq_attr.aux_depth = hwq_attr.depth; hwq_attr.aux_depth = bnxt_qplib_set_sq_size(sq, qp->wqe_mode);
hwq_attr.type = HWQ_TYPE_QUEUE; hwq_attr.type = HWQ_TYPE_QUEUE;
rc = bnxt_qplib_alloc_init_hwq(&sq->hwq, &hwq_attr); rc = bnxt_qplib_alloc_init_hwq(&sq->hwq, &hwq_attr);
if (rc) if (rc)
...@@ -992,7 +993,7 @@ int bnxt_qplib_create_qp(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp) ...@@ -992,7 +993,7 @@ int bnxt_qplib_create_qp(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp)
if (psn_sz) if (psn_sz)
bnxt_qplib_init_psn_ptr(qp, psn_sz); bnxt_qplib_init_psn_ptr(qp, psn_sz);
req.sq_size = cpu_to_le32(sq->max_wqe); req.sq_size = cpu_to_le32(bnxt_qplib_set_sq_size(sq, qp->wqe_mode));
pbl = &sq->hwq.pbl[PBL_LVL_0]; pbl = &sq->hwq.pbl[PBL_LVL_0];
req.sq_pbl = cpu_to_le64(pbl->pg_map_arr[0]); req.sq_pbl = cpu_to_le64(pbl->pg_map_arr[0]);
pg_sz_lvl = (bnxt_qplib_base_pg_size(&sq->hwq) << pg_sz_lvl = (bnxt_qplib_base_pg_size(&sq->hwq) <<
...@@ -1008,8 +1009,8 @@ int bnxt_qplib_create_qp(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp) ...@@ -1008,8 +1009,8 @@ int bnxt_qplib_create_qp(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp)
if (!qp->srq) { if (!qp->srq) {
hwq_attr.res = res; hwq_attr.res = res;
hwq_attr.sginfo = &rq->sg_info; hwq_attr.sginfo = &rq->sg_info;
hwq_attr.stride = rq->wqe_size; hwq_attr.stride = sizeof(struct sq_sge);
hwq_attr.depth = rq->max_wqe; hwq_attr.depth = bnxt_qplib_get_depth(rq);
hwq_attr.aux_stride = 0; hwq_attr.aux_stride = 0;
hwq_attr.aux_depth = 0; hwq_attr.aux_depth = 0;
hwq_attr.type = HWQ_TYPE_QUEUE; hwq_attr.type = HWQ_TYPE_QUEUE;
...@@ -1044,6 +1045,8 @@ int bnxt_qplib_create_qp(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp) ...@@ -1044,6 +1045,8 @@ int bnxt_qplib_create_qp(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp)
qp_flags |= CMDQ_CREATE_QP_QP_FLAGS_FR_PMR_ENABLED; qp_flags |= CMDQ_CREATE_QP_QP_FLAGS_FR_PMR_ENABLED;
if (qp->sig_type) if (qp->sig_type)
qp_flags |= CMDQ_CREATE_QP_QP_FLAGS_FORCE_COMPLETION; qp_flags |= CMDQ_CREATE_QP_QP_FLAGS_FORCE_COMPLETION;
if (qp->wqe_mode == BNXT_QPLIB_WQE_MODE_VARIABLE)
qp_flags |= CMDQ_CREATE_QP_QP_FLAGS_VARIABLE_SIZED_WQE_ENABLED;
req.qp_flags = cpu_to_le32(qp_flags); req.qp_flags = cpu_to_le32(qp_flags);
/* ORRQ and IRRQ */ /* ORRQ and IRRQ */
...@@ -1101,10 +1104,12 @@ int bnxt_qplib_create_qp(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp) ...@@ -1101,10 +1104,12 @@ int bnxt_qplib_create_qp(struct bnxt_qplib_res *res, struct bnxt_qplib_qp *qp)
sq->dbinfo.hwq = &sq->hwq; sq->dbinfo.hwq = &sq->hwq;
sq->dbinfo.xid = qp->id; sq->dbinfo.xid = qp->id;
sq->dbinfo.db = qp->dpi->dbr; sq->dbinfo.db = qp->dpi->dbr;
sq->dbinfo.max_slot = bnxt_qplib_set_sq_max_slot(qp->wqe_mode);
if (rq->max_wqe) { if (rq->max_wqe) {
rq->dbinfo.hwq = &rq->hwq; rq->dbinfo.hwq = &rq->hwq;
rq->dbinfo.xid = qp->id; rq->dbinfo.xid = qp->id;
rq->dbinfo.db = qp->dpi->dbr; rq->dbinfo.db = qp->dpi->dbr;
rq->dbinfo.max_slot = bnxt_qplib_set_rq_max_slot(rq->wqe_size);
} }
rcfw->qp_tbl[qp->id].qp_id = qp->id; rcfw->qp_tbl[qp->id].qp_id = qp->id;
rcfw->qp_tbl[qp->id].qp_handle = (void *)qp; rcfw->qp_tbl[qp->id].qp_handle = (void *)qp;
...@@ -1562,22 +1567,115 @@ static void bnxt_qplib_fill_psn_search(struct bnxt_qplib_qp *qp, ...@@ -1562,22 +1567,115 @@ static void bnxt_qplib_fill_psn_search(struct bnxt_qplib_qp *qp,
if (bnxt_qplib_is_chip_gen_p5(qp->cctx)) { if (bnxt_qplib_is_chip_gen_p5(qp->cctx)) {
psns_ext->opcode_start_psn = cpu_to_le32(op_spsn); psns_ext->opcode_start_psn = cpu_to_le32(op_spsn);
psns_ext->flags_next_psn = cpu_to_le32(flg_npsn); psns_ext->flags_next_psn = cpu_to_le32(flg_npsn);
psns_ext->start_slot_idx = cpu_to_le16(swq->slot_idx);
} else { } else {
psns->opcode_start_psn = cpu_to_le32(op_spsn); psns->opcode_start_psn = cpu_to_le32(op_spsn);
psns->flags_next_psn = cpu_to_le32(flg_npsn); psns->flags_next_psn = cpu_to_le32(flg_npsn);
} }
} }
static int bnxt_qplib_put_inline(struct bnxt_qplib_qp *qp,
struct bnxt_qplib_swqe *wqe,
u16 *idx)
{
struct bnxt_qplib_hwq *hwq;
int len, t_len, offt;
bool pull_dst = true;
void *il_dst = NULL;
void *il_src = NULL;
int t_cplen, cplen;
int indx;
hwq = &qp->sq.hwq;
t_len = 0;
for (indx = 0; indx < wqe->num_sge; indx++) {
len = wqe->sg_list[indx].size;
il_src = (void *)wqe->sg_list[indx].addr;
t_len += len;
if (t_len > qp->max_inline_data)
goto bad;
while (len) {
if (pull_dst) {
pull_dst = false;
il_dst = bnxt_qplib_get_prod_qe(hwq, *idx);
(*idx)++;
t_cplen = 0;
offt = 0;
}
cplen = min_t(int, len, sizeof(struct sq_sge));
cplen = min_t(int, cplen,
(sizeof(struct sq_sge) - offt));
memcpy(il_dst, il_src, cplen);
t_cplen += cplen;
il_src += cplen;
il_dst += cplen;
offt += cplen;
len -= cplen;
if (t_cplen == sizeof(struct sq_sge))
pull_dst = true;
}
}
return t_len;
bad:
return -ENOMEM;
}
static u32 bnxt_qplib_put_sges(struct bnxt_qplib_hwq *hwq,
struct bnxt_qplib_sge *ssge,
u16 nsge, u16 *idx)
{
struct sq_sge *dsge;
int indx, len = 0;
for (indx = 0; indx < nsge; indx++, (*idx)++) {
dsge = bnxt_qplib_get_prod_qe(hwq, *idx);
dsge->va_or_pa = cpu_to_le64(ssge[indx].addr);
dsge->l_key = cpu_to_le32(ssge[indx].lkey);
dsge->size = cpu_to_le32(ssge[indx].size);
len += ssge[indx].size;
}
return len;
}
static u16 bnxt_qplib_required_slots(struct bnxt_qplib_qp *qp,
struct bnxt_qplib_swqe *wqe,
u16 *wqe_sz, u16 *qdf, u8 mode)
{
u32 ilsize, bytes;
u16 nsge;
u16 slot;
nsge = wqe->num_sge;
/* Adding sq_send_hdr is a misnomer, for rq also hdr size is same. */
bytes = sizeof(struct sq_send_hdr) + nsge * sizeof(struct sq_sge);
if (wqe->flags & BNXT_QPLIB_SWQE_FLAGS_INLINE) {
ilsize = bnxt_qplib_calc_ilsize(wqe, qp->max_inline_data);
bytes = ALIGN(ilsize, sizeof(struct sq_sge));
bytes += sizeof(struct sq_send_hdr);
}
*qdf = __xlate_qfd(qp->sq.q_full_delta, bytes);
slot = bytes >> 4;
*wqe_sz = slot;
if (mode == BNXT_QPLIB_WQE_MODE_STATIC)
slot = 8;
return slot;
}
static void bnxt_qplib_pull_psn_buff(struct bnxt_qplib_q *sq, static void bnxt_qplib_pull_psn_buff(struct bnxt_qplib_q *sq,
struct bnxt_qplib_swq *swq, u32 tail) struct bnxt_qplib_swq *swq)
{ {
struct bnxt_qplib_hwq *hwq; struct bnxt_qplib_hwq *hwq;
u32 pg_num, pg_indx; u32 pg_num, pg_indx;
void *buff; void *buff;
u32 tail;
hwq = &sq->hwq; hwq = &sq->hwq;
if (!hwq->pad_pg) if (!hwq->pad_pg)
return; return;
tail = swq->slot_idx / sq->dbinfo.max_slot;
pg_num = (tail + hwq->pad_pgofft) / (PAGE_SIZE / hwq->pad_stride); pg_num = (tail + hwq->pad_pgofft) / (PAGE_SIZE / hwq->pad_stride);
pg_indx = (tail + hwq->pad_pgofft) % (PAGE_SIZE / hwq->pad_stride); pg_indx = (tail + hwq->pad_pgofft) % (PAGE_SIZE / hwq->pad_stride);
buff = (void *)(hwq->pad_pg[pg_num] + pg_indx * hwq->pad_stride); buff = (void *)(hwq->pad_pg[pg_num] + pg_indx * hwq->pad_stride);
...@@ -1598,14 +1696,16 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp, ...@@ -1598,14 +1696,16 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp,
struct bnxt_qplib_nq_work *nq_work = NULL; struct bnxt_qplib_nq_work *nq_work = NULL;
int i, rc = 0, data_len = 0, pkt_num = 0; int i, rc = 0, data_len = 0, pkt_num = 0;
struct bnxt_qplib_q *sq = &qp->sq; struct bnxt_qplib_q *sq = &qp->sq;
struct sq_send *hw_sq_send_hdr;
struct bnxt_qplib_hwq *hwq; struct bnxt_qplib_hwq *hwq;
struct bnxt_qplib_swq *swq; struct bnxt_qplib_swq *swq;
bool sch_handler = false; bool sch_handler = false;
struct sq_sge *hw_sge; u16 wqe_sz, qdf = 0;
u8 wqe_size16; void *base_hdr;
void *ext_hdr;
__le32 temp32; __le32 temp32;
u32 sw_prod; u32 wqe_idx;
u32 slots;
u16 idx;
hwq = &sq->hwq; hwq = &sq->hwq;
if (qp->state != CMDQ_MODIFY_QP_NEW_STATE_RTS && if (qp->state != CMDQ_MODIFY_QP_NEW_STATE_RTS &&
...@@ -1617,18 +1717,21 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp, ...@@ -1617,18 +1717,21 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp,
goto done; goto done;
} }
if (bnxt_qplib_queue_full(sq)) { slots = bnxt_qplib_required_slots(qp, wqe, &wqe_sz, &qdf, qp->wqe_mode);
if (bnxt_qplib_queue_full(sq, slots + qdf)) {
dev_err(&hwq->pdev->dev, dev_err(&hwq->pdev->dev,
"prod = %#x cons = %#x qdepth = %#x delta = %#x\n", "prod = %#x cons = %#x qdepth = %#x delta = %#x\n",
hwq->prod, hwq->cons, hwq->max_elements, hwq->prod, hwq->cons, hwq->depth, sq->q_full_delta);
sq->q_full_delta);
rc = -ENOMEM; rc = -ENOMEM;
goto done; goto done;
} }
sw_prod = sq->hwq.prod; swq = bnxt_qplib_get_swqe(sq, &wqe_idx);
swq = bnxt_qplib_get_swqe(sq, NULL); bnxt_qplib_pull_psn_buff(sq, swq);
bnxt_qplib_pull_psn_buff(sq, swq, sw_prod);
idx = 0;
swq->slot_idx = hwq->prod;
swq->slots = slots;
swq->wr_id = wqe->wr_id; swq->wr_id = wqe->wr_id;
swq->type = wqe->type; swq->type = wqe->type;
swq->flags = wqe->flags; swq->flags = wqe->flags;
...@@ -1636,8 +1739,6 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp, ...@@ -1636,8 +1739,6 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp,
if (qp->sig_type) if (qp->sig_type)
swq->flags |= SQ_SEND_FLAGS_SIGNAL_COMP; swq->flags |= SQ_SEND_FLAGS_SIGNAL_COMP;
hw_sq_send_hdr = bnxt_qplib_get_qe(hwq, sw_prod, NULL);
memset(hw_sq_send_hdr, 0, sq->wqe_size);
if (qp->state == CMDQ_MODIFY_QP_NEW_STATE_ERR) { if (qp->state == CMDQ_MODIFY_QP_NEW_STATE_ERR) {
sch_handler = true; sch_handler = true;
dev_dbg(&hwq->pdev->dev, dev_dbg(&hwq->pdev->dev,
...@@ -1645,50 +1746,34 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp, ...@@ -1645,50 +1746,34 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp,
goto queue_err; goto queue_err;
} }
if (wqe->flags & BNXT_QPLIB_SWQE_FLAGS_INLINE) { base_hdr = bnxt_qplib_get_prod_qe(hwq, idx++);
/* Copy the inline data */ ext_hdr = bnxt_qplib_get_prod_qe(hwq, idx++);
if (wqe->inline_len > BNXT_QPLIB_SWQE_MAX_INLINE_LENGTH) { memset(base_hdr, 0, sizeof(struct sq_sge));
dev_warn(&hwq->pdev->dev, memset(ext_hdr, 0, sizeof(struct sq_sge));
"Inline data length > 96 detected\n");
data_len = BNXT_QPLIB_SWQE_MAX_INLINE_LENGTH;
} else {
data_len = wqe->inline_len;
}
memcpy(hw_sq_send_hdr->data, wqe->inline_data, data_len);
wqe_size16 = (data_len + 15) >> 4;
} else {
for (i = 0, hw_sge = (struct sq_sge *)hw_sq_send_hdr->data;
i < wqe->num_sge; i++, hw_sge++) {
hw_sge->va_or_pa = cpu_to_le64(wqe->sg_list[i].addr);
hw_sge->l_key = cpu_to_le32(wqe->sg_list[i].lkey);
hw_sge->size = cpu_to_le32(wqe->sg_list[i].size);
data_len += wqe->sg_list[i].size;
}
/* Each SGE entry = 1 WQE size16 */
wqe_size16 = wqe->num_sge;
/* HW requires wqe size has room for atleast one SGE even if
* none was supplied by ULP
*/
if (!wqe->num_sge)
wqe_size16++;
}
if (wqe->flags & BNXT_QPLIB_SWQE_FLAGS_INLINE)
/* Copy the inline data */
data_len = bnxt_qplib_put_inline(qp, wqe, &idx);
else
data_len = bnxt_qplib_put_sges(hwq, wqe->sg_list, wqe->num_sge,
&idx);
if (data_len < 0)
goto queue_err;
/* Specifics */ /* Specifics */
switch (wqe->type) { switch (wqe->type) {
case BNXT_QPLIB_SWQE_TYPE_SEND: case BNXT_QPLIB_SWQE_TYPE_SEND:
if (qp->type == CMDQ_CREATE_QP1_TYPE_GSI) { if (qp->type == CMDQ_CREATE_QP1_TYPE_GSI) {
struct sq_send_raweth_qp1_hdr *sqe = base_hdr;
struct sq_raw_ext_hdr *ext_sqe = ext_hdr;
/* Assemble info for Raw Ethertype QPs */ /* Assemble info for Raw Ethertype QPs */
struct sq_send_raweth_qp1 *sqe =
(struct sq_send_raweth_qp1 *)hw_sq_send_hdr;
sqe->wqe_type = wqe->type; sqe->wqe_type = wqe->type;
sqe->flags = wqe->flags; sqe->flags = wqe->flags;
sqe->wqe_size = wqe_size16 + sqe->wqe_size = wqe_sz;
((offsetof(typeof(*sqe), data) + 15) >> 4);
sqe->cfa_action = cpu_to_le16(wqe->rawqp1.cfa_action); sqe->cfa_action = cpu_to_le16(wqe->rawqp1.cfa_action);
sqe->lflags = cpu_to_le16(wqe->rawqp1.lflags); sqe->lflags = cpu_to_le16(wqe->rawqp1.lflags);
sqe->length = cpu_to_le32(data_len); sqe->length = cpu_to_le32(data_len);
sqe->cfa_meta = cpu_to_le32((wqe->rawqp1.cfa_meta & ext_sqe->cfa_meta = cpu_to_le32((wqe->rawqp1.cfa_meta &
SQ_SEND_RAWETH_QP1_CFA_META_VLAN_VID_MASK) << SQ_SEND_RAWETH_QP1_CFA_META_VLAN_VID_MASK) <<
SQ_SEND_RAWETH_QP1_CFA_META_VLAN_VID_SFT); SQ_SEND_RAWETH_QP1_CFA_META_VLAN_VID_SFT);
...@@ -1698,27 +1783,24 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp, ...@@ -1698,27 +1783,24 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp,
case BNXT_QPLIB_SWQE_TYPE_SEND_WITH_IMM: case BNXT_QPLIB_SWQE_TYPE_SEND_WITH_IMM:
case BNXT_QPLIB_SWQE_TYPE_SEND_WITH_INV: case BNXT_QPLIB_SWQE_TYPE_SEND_WITH_INV:
{ {
struct sq_send *sqe = (struct sq_send *)hw_sq_send_hdr; struct sq_ud_ext_hdr *ext_sqe = ext_hdr;
struct sq_send_hdr *sqe = base_hdr;
sqe->wqe_type = wqe->type; sqe->wqe_type = wqe->type;
sqe->flags = wqe->flags; sqe->flags = wqe->flags;
sqe->wqe_size = wqe_size16 + sqe->wqe_size = wqe_sz;
((offsetof(typeof(*sqe), data) + 15) >> 4); sqe->inv_key_or_imm_data = cpu_to_le32(wqe->send.inv_key);
sqe->inv_key_or_imm_data = cpu_to_le32(
wqe->send.inv_key);
if (qp->type == CMDQ_CREATE_QP_TYPE_UD || if (qp->type == CMDQ_CREATE_QP_TYPE_UD ||
qp->type == CMDQ_CREATE_QP_TYPE_GSI) { qp->type == CMDQ_CREATE_QP_TYPE_GSI) {
sqe->q_key = cpu_to_le32(wqe->send.q_key); sqe->q_key = cpu_to_le32(wqe->send.q_key);
sqe->dst_qp = cpu_to_le32(
wqe->send.dst_qp & SQ_SEND_DST_QP_MASK);
sqe->length = cpu_to_le32(data_len); sqe->length = cpu_to_le32(data_len);
sqe->avid = cpu_to_le32(wqe->send.avid &
SQ_SEND_AVID_MASK);
sq->psn = (sq->psn + 1) & BTH_PSN_MASK; sq->psn = (sq->psn + 1) & BTH_PSN_MASK;
ext_sqe->dst_qp = cpu_to_le32(wqe->send.dst_qp &
SQ_SEND_DST_QP_MASK);
ext_sqe->avid = cpu_to_le32(wqe->send.avid &
SQ_SEND_AVID_MASK);
} else { } else {
sqe->length = cpu_to_le32(data_len); sqe->length = cpu_to_le32(data_len);
sqe->dst_qp = 0;
sqe->avid = 0;
if (qp->mtu) if (qp->mtu)
pkt_num = (data_len + qp->mtu - 1) / qp->mtu; pkt_num = (data_len + qp->mtu - 1) / qp->mtu;
if (!pkt_num) if (!pkt_num)
...@@ -1731,16 +1813,16 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp, ...@@ -1731,16 +1813,16 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp,
case BNXT_QPLIB_SWQE_TYPE_RDMA_WRITE_WITH_IMM: case BNXT_QPLIB_SWQE_TYPE_RDMA_WRITE_WITH_IMM:
case BNXT_QPLIB_SWQE_TYPE_RDMA_READ: case BNXT_QPLIB_SWQE_TYPE_RDMA_READ:
{ {
struct sq_rdma *sqe = (struct sq_rdma *)hw_sq_send_hdr; struct sq_rdma_ext_hdr *ext_sqe = ext_hdr;
struct sq_rdma_hdr *sqe = base_hdr;
sqe->wqe_type = wqe->type; sqe->wqe_type = wqe->type;
sqe->flags = wqe->flags; sqe->flags = wqe->flags;
sqe->wqe_size = wqe_size16 + sqe->wqe_size = wqe_sz;
((offsetof(typeof(*sqe), data) + 15) >> 4);
sqe->imm_data = cpu_to_le32(wqe->rdma.inv_key); sqe->imm_data = cpu_to_le32(wqe->rdma.inv_key);
sqe->length = cpu_to_le32((u32)data_len); sqe->length = cpu_to_le32((u32)data_len);
sqe->remote_va = cpu_to_le64(wqe->rdma.remote_va); ext_sqe->remote_va = cpu_to_le64(wqe->rdma.remote_va);
sqe->remote_key = cpu_to_le32(wqe->rdma.r_key); ext_sqe->remote_key = cpu_to_le32(wqe->rdma.r_key);
if (qp->mtu) if (qp->mtu)
pkt_num = (data_len + qp->mtu - 1) / qp->mtu; pkt_num = (data_len + qp->mtu - 1) / qp->mtu;
if (!pkt_num) if (!pkt_num)
...@@ -1751,14 +1833,15 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp, ...@@ -1751,14 +1833,15 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp,
case BNXT_QPLIB_SWQE_TYPE_ATOMIC_CMP_AND_SWP: case BNXT_QPLIB_SWQE_TYPE_ATOMIC_CMP_AND_SWP:
case BNXT_QPLIB_SWQE_TYPE_ATOMIC_FETCH_AND_ADD: case BNXT_QPLIB_SWQE_TYPE_ATOMIC_FETCH_AND_ADD:
{ {
struct sq_atomic *sqe = (struct sq_atomic *)hw_sq_send_hdr; struct sq_atomic_ext_hdr *ext_sqe = ext_hdr;
struct sq_atomic_hdr *sqe = base_hdr;
sqe->wqe_type = wqe->type; sqe->wqe_type = wqe->type;
sqe->flags = wqe->flags; sqe->flags = wqe->flags;
sqe->remote_key = cpu_to_le32(wqe->atomic.r_key); sqe->remote_key = cpu_to_le32(wqe->atomic.r_key);
sqe->remote_va = cpu_to_le64(wqe->atomic.remote_va); sqe->remote_va = cpu_to_le64(wqe->atomic.remote_va);
sqe->swap_data = cpu_to_le64(wqe->atomic.swap_data); ext_sqe->swap_data = cpu_to_le64(wqe->atomic.swap_data);
sqe->cmp_data = cpu_to_le64(wqe->atomic.cmp_data); ext_sqe->cmp_data = cpu_to_le64(wqe->atomic.cmp_data);
if (qp->mtu) if (qp->mtu)
pkt_num = (data_len + qp->mtu - 1) / qp->mtu; pkt_num = (data_len + qp->mtu - 1) / qp->mtu;
if (!pkt_num) if (!pkt_num)
...@@ -1768,8 +1851,7 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp, ...@@ -1768,8 +1851,7 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp,
} }
case BNXT_QPLIB_SWQE_TYPE_LOCAL_INV: case BNXT_QPLIB_SWQE_TYPE_LOCAL_INV:
{ {
struct sq_localinvalidate *sqe = struct sq_localinvalidate *sqe = base_hdr;
(struct sq_localinvalidate *)hw_sq_send_hdr;
sqe->wqe_type = wqe->type; sqe->wqe_type = wqe->type;
sqe->flags = wqe->flags; sqe->flags = wqe->flags;
...@@ -1779,7 +1861,8 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp, ...@@ -1779,7 +1861,8 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp,
} }
case BNXT_QPLIB_SWQE_TYPE_FAST_REG_MR: case BNXT_QPLIB_SWQE_TYPE_FAST_REG_MR:
{ {
struct sq_fr_pmr *sqe = (struct sq_fr_pmr *)hw_sq_send_hdr; struct sq_fr_pmr_ext_hdr *ext_sqe = ext_hdr;
struct sq_fr_pmr_hdr *sqe = base_hdr;
sqe->wqe_type = wqe->type; sqe->wqe_type = wqe->type;
sqe->flags = wqe->flags; sqe->flags = wqe->flags;
...@@ -1803,14 +1886,15 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp, ...@@ -1803,14 +1886,15 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp,
wqe->frmr.pbl_ptr[i] = cpu_to_le64( wqe->frmr.pbl_ptr[i] = cpu_to_le64(
wqe->frmr.page_list[i] | wqe->frmr.page_list[i] |
PTU_PTE_VALID); PTU_PTE_VALID);
sqe->pblptr = cpu_to_le64(wqe->frmr.pbl_dma_ptr); ext_sqe->pblptr = cpu_to_le64(wqe->frmr.pbl_dma_ptr);
sqe->va = cpu_to_le64(wqe->frmr.va); ext_sqe->va = cpu_to_le64(wqe->frmr.va);
break; break;
} }
case BNXT_QPLIB_SWQE_TYPE_BIND_MW: case BNXT_QPLIB_SWQE_TYPE_BIND_MW:
{ {
struct sq_bind *sqe = (struct sq_bind *)hw_sq_send_hdr; struct sq_bind_ext_hdr *ext_sqe = ext_hdr;
struct sq_bind_hdr *sqe = base_hdr;
sqe->wqe_type = wqe->type; sqe->wqe_type = wqe->type;
sqe->flags = wqe->flags; sqe->flags = wqe->flags;
...@@ -1819,9 +1903,8 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp, ...@@ -1819,9 +1903,8 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp,
(wqe->bind.zero_based ? SQ_BIND_ZERO_BASED : 0); (wqe->bind.zero_based ? SQ_BIND_ZERO_BASED : 0);
sqe->parent_l_key = cpu_to_le32(wqe->bind.parent_l_key); sqe->parent_l_key = cpu_to_le32(wqe->bind.parent_l_key);
sqe->l_key = cpu_to_le32(wqe->bind.r_key); sqe->l_key = cpu_to_le32(wqe->bind.r_key);
sqe->va = cpu_to_le64(wqe->bind.va); ext_sqe->va = cpu_to_le64(wqe->bind.va);
temp32 = cpu_to_le32(wqe->bind.length); ext_sqe->length_lo = cpu_to_le32(wqe->bind.length);
memcpy(&sqe->length, &temp32, sizeof(wqe->bind.length));
break; break;
} }
default: default:
...@@ -1832,8 +1915,8 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp, ...@@ -1832,8 +1915,8 @@ int bnxt_qplib_post_send(struct bnxt_qplib_qp *qp,
swq->next_psn = sq->psn & BTH_PSN_MASK; swq->next_psn = sq->psn & BTH_PSN_MASK;
bnxt_qplib_fill_psn_search(qp, wqe, swq); bnxt_qplib_fill_psn_search(qp, wqe, swq);
queue_err: queue_err:
bnxt_qplib_swq_mod_start(sq, sw_prod); bnxt_qplib_swq_mod_start(sq, wqe_idx);
bnxt_qplib_hwq_incr_prod(&sq->hwq, 1); bnxt_qplib_hwq_incr_prod(hwq, swq->slots);
qp->wqe_cnt++; qp->wqe_cnt++;
done: done:
if (sch_handler) { if (sch_handler) {
...@@ -1864,13 +1947,14 @@ int bnxt_qplib_post_recv(struct bnxt_qplib_qp *qp, ...@@ -1864,13 +1947,14 @@ int bnxt_qplib_post_recv(struct bnxt_qplib_qp *qp,
{ {
struct bnxt_qplib_nq_work *nq_work = NULL; struct bnxt_qplib_nq_work *nq_work = NULL;
struct bnxt_qplib_q *rq = &qp->rq; struct bnxt_qplib_q *rq = &qp->rq;
struct bnxt_qplib_swq *swq; struct rq_wqe_hdr *base_hdr;
struct rq_ext_hdr *ext_hdr;
struct bnxt_qplib_hwq *hwq; struct bnxt_qplib_hwq *hwq;
struct bnxt_qplib_swq *swq;
bool sch_handler = false; bool sch_handler = false;
struct sq_sge *hw_sge; u16 wqe_sz, idx;
struct rq_wqe *rqe; u32 wqe_idx;
int i, rc = 0; int rc = 0;
u32 sw_prod;
hwq = &rq->hwq; hwq = &rq->hwq;
if (qp->state == CMDQ_MODIFY_QP_NEW_STATE_RESET) { if (qp->state == CMDQ_MODIFY_QP_NEW_STATE_RESET) {
...@@ -1881,16 +1965,16 @@ int bnxt_qplib_post_recv(struct bnxt_qplib_qp *qp, ...@@ -1881,16 +1965,16 @@ int bnxt_qplib_post_recv(struct bnxt_qplib_qp *qp,
goto done; goto done;
} }
if (bnxt_qplib_queue_full(rq)) { if (bnxt_qplib_queue_full(rq, rq->dbinfo.max_slot)) {
dev_err(&hwq->pdev->dev, dev_err(&hwq->pdev->dev,
"FP: QP (0x%x) RQ is full!\n", qp->id); "FP: QP (0x%x) RQ is full!\n", qp->id);
rc = -EINVAL; rc = -EINVAL;
goto done; goto done;
} }
sw_prod = rq->hwq.prod; swq = bnxt_qplib_get_swqe(rq, &wqe_idx);
swq = bnxt_qplib_get_swqe(rq, NULL);
swq->wr_id = wqe->wr_id; swq->wr_id = wqe->wr_id;
swq->slots = rq->dbinfo.max_slot;
if (qp->state == CMDQ_MODIFY_QP_NEW_STATE_ERR) { if (qp->state == CMDQ_MODIFY_QP_NEW_STATE_ERR) {
sch_handler = true; sch_handler = true;
...@@ -1899,32 +1983,28 @@ int bnxt_qplib_post_recv(struct bnxt_qplib_qp *qp, ...@@ -1899,32 +1983,28 @@ int bnxt_qplib_post_recv(struct bnxt_qplib_qp *qp,
goto queue_err; goto queue_err;
} }
rqe = bnxt_qplib_get_qe(hwq, sw_prod, NULL); idx = 0;
memset(rqe, 0, rq->wqe_size); base_hdr = bnxt_qplib_get_prod_qe(hwq, idx++);
ext_hdr = bnxt_qplib_get_prod_qe(hwq, idx++);
/* Calculate wqe_size16 and data_len */ memset(base_hdr, 0, sizeof(struct sq_sge));
for (i = 0, hw_sge = (struct sq_sge *)rqe->data; memset(ext_hdr, 0, sizeof(struct sq_sge));
i < wqe->num_sge; i++, hw_sge++) { wqe_sz = (sizeof(struct rq_wqe_hdr) +
hw_sge->va_or_pa = cpu_to_le64(wqe->sg_list[i].addr); wqe->num_sge * sizeof(struct sq_sge)) >> 4;
hw_sge->l_key = cpu_to_le32(wqe->sg_list[i].lkey); bnxt_qplib_put_sges(hwq, wqe->sg_list, wqe->num_sge, &idx);
hw_sge->size = cpu_to_le32(wqe->sg_list[i].size); if (!wqe->num_sge) {
} struct sq_sge *sge;
rqe->wqe_type = wqe->type;
rqe->flags = wqe->flags; sge = bnxt_qplib_get_prod_qe(hwq, idx++);
rqe->wqe_size = wqe->num_sge + sge->size = 0;
((offsetof(typeof(*rqe), data) + 15) >> 4); wqe_sz++;
/* HW requires wqe size has room for atleast one SGE even if none }
* was supplied by ULP base_hdr->wqe_type = wqe->type;
*/ base_hdr->flags = wqe->flags;
if (!wqe->num_sge) base_hdr->wqe_size = wqe_sz;
rqe->wqe_size++; base_hdr->wr_id[0] = cpu_to_le32(wqe_idx);
/* Supply the rqe->wr_id index to the wr_id_tbl for now */
rqe->wr_id[0] = cpu_to_le32(sw_prod);
queue_err: queue_err:
bnxt_qplib_swq_mod_start(rq, sw_prod); bnxt_qplib_swq_mod_start(rq, wqe_idx);
bnxt_qplib_hwq_incr_prod(&rq->hwq, 1); bnxt_qplib_hwq_incr_prod(hwq, swq->slots);
done: done:
if (sch_handler) { if (sch_handler) {
nq_work = kzalloc(sizeof(*nq_work), GFP_ATOMIC); nq_work = kzalloc(sizeof(*nq_work), GFP_ATOMIC);
......
...@@ -119,6 +119,7 @@ struct bnxt_qplib_swq { ...@@ -119,6 +119,7 @@ struct bnxt_qplib_swq {
u8 flags; u8 flags;
u32 start_psn; u32 start_psn;
u32 next_psn; u32 next_psn;
u32 slot_idx;
u8 slots; u8 slots;
struct sq_psn_search *psn_search; struct sq_psn_search *psn_search;
struct sq_psn_search_ext *psn_ext; struct sq_psn_search_ext *psn_ext;
...@@ -349,11 +350,18 @@ struct bnxt_qplib_qp { ...@@ -349,11 +350,18 @@ struct bnxt_qplib_qp {
(!!((hdr)->cqe_type_toggle & CQ_BASE_TOGGLE) == \ (!!((hdr)->cqe_type_toggle & CQ_BASE_TOGGLE) == \
!((raw_cons) & (cp_bit))) !((raw_cons) & (cp_bit)))
static inline bool bnxt_qplib_queue_full(struct bnxt_qplib_q *qplib_q) static inline bool bnxt_qplib_queue_full(struct bnxt_qplib_q *que,
u8 slots)
{ {
return HWQ_CMP((qplib_q->hwq.prod + qplib_q->q_full_delta), struct bnxt_qplib_hwq *hwq;
&qplib_q->hwq) == HWQ_CMP(qplib_q->hwq.cons, int avail;
&qplib_q->hwq);
hwq = &que->hwq;
/* False full is possible, retrying post-send makes sense */
avail = hwq->cons - hwq->prod;
if (hwq->cons <= hwq->prod)
avail += hwq->depth;
return avail <= slots;
} }
struct bnxt_qplib_cqe { struct bnxt_qplib_cqe {
...@@ -554,4 +562,48 @@ static inline void bnxt_qplib_swq_mod_start(struct bnxt_qplib_q *que, u32 idx) ...@@ -554,4 +562,48 @@ static inline void bnxt_qplib_swq_mod_start(struct bnxt_qplib_q *que, u32 idx)
que->swq_start = que->swq[idx].next_idx; que->swq_start = que->swq[idx].next_idx;
} }
static inline u32 bnxt_qplib_get_depth(struct bnxt_qplib_q *que)
{
return (que->wqe_size * que->max_wqe) / sizeof(struct sq_sge);
}
static inline u32 bnxt_qplib_set_sq_size(struct bnxt_qplib_q *que, u8 wqe_mode)
{
return (wqe_mode == BNXT_QPLIB_WQE_MODE_STATIC) ?
que->max_wqe : bnxt_qplib_get_depth(que);
}
static inline u32 bnxt_qplib_set_sq_max_slot(u8 wqe_mode)
{
return (wqe_mode == BNXT_QPLIB_WQE_MODE_STATIC) ?
sizeof(struct sq_send) / sizeof(struct sq_sge) : 1;
}
static inline u32 bnxt_qplib_set_rq_max_slot(u32 wqe_size)
{
return (wqe_size / sizeof(struct sq_sge));
}
static inline u16 __xlate_qfd(u16 delta, u16 wqe_bytes)
{
/* For Cu/Wh delta = 128, stride = 16, wqe_bytes = 128
* For Gen-p5 B/C mode delta = 0, stride = 16, wqe_bytes = 128.
* For Gen-p5 delta = 0, stride = 16, 32 <= wqe_bytes <= 512.
* when 8916 is disabled.
*/
return (delta * wqe_bytes) / sizeof(struct sq_sge);
}
static inline u16 bnxt_qplib_calc_ilsize(struct bnxt_qplib_swqe *wqe, u16 max)
{
u16 size = 0;
int indx;
for (indx = 0; indx < wqe->num_sge; indx++)
size += wqe->sg_list[indx].size;
if (size > max)
size = max;
return size;
}
#endif /* __BNXT_QPLIB_FP_H__ */ #endif /* __BNXT_QPLIB_FP_H__ */
...@@ -173,6 +173,7 @@ struct bnxt_qplib_db_info { ...@@ -173,6 +173,7 @@ struct bnxt_qplib_db_info {
void __iomem *priv_db; void __iomem *priv_db;
struct bnxt_qplib_hwq *hwq; struct bnxt_qplib_hwq *hwq;
u32 xid; u32 xid;
u32 max_slot;
}; };
/* Tables */ /* Tables */
...@@ -332,6 +333,14 @@ static inline void *bnxt_qplib_get_qe(struct bnxt_qplib_hwq *hwq, ...@@ -332,6 +333,14 @@ static inline void *bnxt_qplib_get_qe(struct bnxt_qplib_hwq *hwq,
return (void *)(hwq->pbl_ptr[pg_num] + hwq->element_size * pg_idx); return (void *)(hwq->pbl_ptr[pg_num] + hwq->element_size * pg_idx);
} }
static inline void *bnxt_qplib_get_prod_qe(struct bnxt_qplib_hwq *hwq, u32 idx)
{
idx += hwq->prod;
if (idx >= hwq->depth)
idx -= hwq->depth;
return bnxt_qplib_get_qe(hwq, idx, NULL);
}
#define to_bnxt_qplib(ptr, type, member) \ #define to_bnxt_qplib(ptr, type, member) \
container_of(ptr, type, member) container_of(ptr, type, member)
...@@ -409,8 +418,7 @@ static inline void bnxt_qplib_ring_prod_db(struct bnxt_qplib_db_info *info, ...@@ -409,8 +418,7 @@ static inline void bnxt_qplib_ring_prod_db(struct bnxt_qplib_db_info *info,
key = (info->xid & DBC_DBC_XID_MASK) | DBC_DBC_PATH_ROCE | type; key = (info->xid & DBC_DBC_XID_MASK) | DBC_DBC_PATH_ROCE | type;
key <<= 32; key <<= 32;
key |= (info->hwq->prod & (info->hwq->max_elements - 1)) & key |= ((info->hwq->prod / info->max_slot)) & DBC_DBC_INDEX_MASK;
DBC_DBC_INDEX_MASK;
writeq(key, info->db); writeq(key, info->db);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment