提交 b777f154 编写于 作者: M Mitko Haralanov 提交者: Doug Ledford

IB/hfi1: Remove usage of qp->s_cur_sge

The s_cur_sge field in the qp structure holds a pointer to the
SGE of the currently processed WQE. It assumes the protection
of the RVT_S_BUSY flag to prevent the changing of this field
while the send engine is using it. This scheme works as long
as there is only one instance of the send engine running at a
time.

Scaling of the send engine to multiple cores would break this
assumption as there could be multiple instances of the send engine
running on different CPUs. This opens a window where the QP's
RVT_S_BUSY flag is not set but the send engine is still running.

To prevent accidental changing of the s_cur_sge pointer, the QP's
dependence on it is removed. The SGE pointer is now stored in the
verbs_txreq, which is a per-packet data structure. This ensures
that each individual packet has it's own pointer, which is setup
while the RVT_S_BUSY flag is set.
Reviewed-by: NMike Marciniszyn <mike.marciniszyn@intel.com>
Reviewed-by: NDennis Dalessandro <dennis.dalessandro@intel.com>
Signed-off-by: NMitko Haralanov <mitko.haralanov@intel.com>
Signed-off-by: NDennis Dalessandro <dennis.dalessandro@intel.com>
Signed-off-by: NDoug Ledford <dledford@redhat.com>
上级 fcb29a66
...@@ -276,7 +276,7 @@ static int make_rc_ack(struct hfi1_ibdev *dev, struct rvt_qp *qp, ...@@ -276,7 +276,7 @@ static int make_rc_ack(struct hfi1_ibdev *dev, struct rvt_qp *qp,
rvt_get_mr(ps->s_txreq->mr); rvt_get_mr(ps->s_txreq->mr);
qp->s_ack_rdma_sge.sge = e->rdma_sge; qp->s_ack_rdma_sge.sge = e->rdma_sge;
qp->s_ack_rdma_sge.num_sge = 1; qp->s_ack_rdma_sge.num_sge = 1;
qp->s_cur_sge = &qp->s_ack_rdma_sge; ps->s_txreq->ss = &qp->s_ack_rdma_sge;
if (len > pmtu) { if (len > pmtu) {
len = pmtu; len = pmtu;
qp->s_ack_state = OP(RDMA_READ_RESPONSE_FIRST); qp->s_ack_state = OP(RDMA_READ_RESPONSE_FIRST);
...@@ -290,7 +290,7 @@ static int make_rc_ack(struct hfi1_ibdev *dev, struct rvt_qp *qp, ...@@ -290,7 +290,7 @@ static int make_rc_ack(struct hfi1_ibdev *dev, struct rvt_qp *qp,
bth2 = mask_psn(qp->s_ack_rdma_psn++); bth2 = mask_psn(qp->s_ack_rdma_psn++);
} else { } else {
/* COMPARE_SWAP or FETCH_ADD */ /* COMPARE_SWAP or FETCH_ADD */
qp->s_cur_sge = NULL; ps->s_txreq->ss = NULL;
len = 0; len = 0;
qp->s_ack_state = OP(ATOMIC_ACKNOWLEDGE); qp->s_ack_state = OP(ATOMIC_ACKNOWLEDGE);
ohdr->u.at.aeth = hfi1_compute_aeth(qp); ohdr->u.at.aeth = hfi1_compute_aeth(qp);
...@@ -306,7 +306,7 @@ static int make_rc_ack(struct hfi1_ibdev *dev, struct rvt_qp *qp, ...@@ -306,7 +306,7 @@ static int make_rc_ack(struct hfi1_ibdev *dev, struct rvt_qp *qp,
qp->s_ack_state = OP(RDMA_READ_RESPONSE_MIDDLE); qp->s_ack_state = OP(RDMA_READ_RESPONSE_MIDDLE);
/* FALLTHROUGH */ /* FALLTHROUGH */
case OP(RDMA_READ_RESPONSE_MIDDLE): case OP(RDMA_READ_RESPONSE_MIDDLE):
qp->s_cur_sge = &qp->s_ack_rdma_sge; ps->s_txreq->ss = &qp->s_ack_rdma_sge;
ps->s_txreq->mr = qp->s_ack_rdma_sge.sge.mr; ps->s_txreq->mr = qp->s_ack_rdma_sge.sge.mr;
if (ps->s_txreq->mr) if (ps->s_txreq->mr)
rvt_get_mr(ps->s_txreq->mr); rvt_get_mr(ps->s_txreq->mr);
...@@ -335,7 +335,7 @@ static int make_rc_ack(struct hfi1_ibdev *dev, struct rvt_qp *qp, ...@@ -335,7 +335,7 @@ static int make_rc_ack(struct hfi1_ibdev *dev, struct rvt_qp *qp,
*/ */
qp->s_ack_state = OP(SEND_ONLY); qp->s_ack_state = OP(SEND_ONLY);
qp->s_flags &= ~RVT_S_ACK_PENDING; qp->s_flags &= ~RVT_S_ACK_PENDING;
qp->s_cur_sge = NULL; ps->s_txreq->ss = NULL;
if (qp->s_nak_state) if (qp->s_nak_state)
ohdr->u.aeth = ohdr->u.aeth =
cpu_to_be32((qp->r_msn & HFI1_MSN_MASK) | cpu_to_be32((qp->r_msn & HFI1_MSN_MASK) |
...@@ -801,7 +801,7 @@ int hfi1_make_rc_req(struct rvt_qp *qp, struct hfi1_pkt_state *ps) ...@@ -801,7 +801,7 @@ int hfi1_make_rc_req(struct rvt_qp *qp, struct hfi1_pkt_state *ps)
qp->s_len -= len; qp->s_len -= len;
qp->s_hdrwords = hwords; qp->s_hdrwords = hwords;
ps->s_txreq->sde = priv->s_sde; ps->s_txreq->sde = priv->s_sde;
qp->s_cur_sge = ss; ps->s_txreq->ss = ss;
ps->s_txreq->s_cur_size = len; ps->s_txreq->s_cur_size = len;
hfi1_make_ruc_header( hfi1_make_ruc_header(
qp, qp,
......
...@@ -258,7 +258,7 @@ int hfi1_make_uc_req(struct rvt_qp *qp, struct hfi1_pkt_state *ps) ...@@ -258,7 +258,7 @@ int hfi1_make_uc_req(struct rvt_qp *qp, struct hfi1_pkt_state *ps)
qp->s_len -= len; qp->s_len -= len;
qp->s_hdrwords = hwords; qp->s_hdrwords = hwords;
ps->s_txreq->sde = priv->s_sde; ps->s_txreq->sde = priv->s_sde;
qp->s_cur_sge = &qp->s_sge; ps->s_txreq->ss = &qp->s_sge;
ps->s_txreq->s_cur_size = len; ps->s_txreq->s_cur_size = len;
hfi1_make_ruc_header(qp, ohdr, bth0 | (qp->s_state << 24), hfi1_make_ruc_header(qp, ohdr, bth0 | (qp->s_state << 24),
mask_psn(qp->s_psn++), middle, ps); mask_psn(qp->s_psn++), middle, ps);
......
...@@ -355,7 +355,7 @@ int hfi1_make_ud_req(struct rvt_qp *qp, struct hfi1_pkt_state *ps) ...@@ -355,7 +355,7 @@ int hfi1_make_ud_req(struct rvt_qp *qp, struct hfi1_pkt_state *ps)
/* header size in 32-bit words LRH+BTH+DETH = (8+12+8)/4. */ /* header size in 32-bit words LRH+BTH+DETH = (8+12+8)/4. */
qp->s_hdrwords = 7; qp->s_hdrwords = 7;
ps->s_txreq->s_cur_size = wqe->length; ps->s_txreq->s_cur_size = wqe->length;
qp->s_cur_sge = &qp->s_sge; ps->s_txreq->ss = &qp->s_sge;
qp->s_srate = ah_attr->static_rate; qp->s_srate = ah_attr->static_rate;
qp->srate_mbps = ib_rate_to_mbps(qp->s_srate); qp->srate_mbps = ib_rate_to_mbps(qp->s_srate);
qp->s_wqe = wqe; qp->s_wqe = wqe;
......
...@@ -790,10 +790,10 @@ static int wait_kmem(struct hfi1_ibdev *dev, ...@@ -790,10 +790,10 @@ static int wait_kmem(struct hfi1_ibdev *dev,
*/ */
static noinline int build_verbs_ulp_payload( static noinline int build_verbs_ulp_payload(
struct sdma_engine *sde, struct sdma_engine *sde,
struct rvt_sge_state *ss,
u32 length, u32 length,
struct verbs_txreq *tx) struct verbs_txreq *tx)
{ {
struct rvt_sge_state *ss = tx->ss;
struct rvt_sge *sg_list = ss->sg_list; struct rvt_sge *sg_list = ss->sg_list;
struct rvt_sge sge = ss->sge; struct rvt_sge sge = ss->sge;
u8 num_sge = ss->num_sge; u8 num_sge = ss->num_sge;
...@@ -837,7 +837,6 @@ static noinline int build_verbs_ulp_payload( ...@@ -837,7 +837,6 @@ static noinline int build_verbs_ulp_payload(
/* New API */ /* New API */
static int build_verbs_tx_desc( static int build_verbs_tx_desc(
struct sdma_engine *sde, struct sdma_engine *sde,
struct rvt_sge_state *ss,
u32 length, u32 length,
struct verbs_txreq *tx, struct verbs_txreq *tx,
struct hfi1_ahg_info *ahg_info, struct hfi1_ahg_info *ahg_info,
...@@ -881,9 +880,9 @@ static int build_verbs_tx_desc( ...@@ -881,9 +880,9 @@ static int build_verbs_tx_desc(
goto bail_txadd; goto bail_txadd;
} }
/* add the ulp payload - if any. ss can be NULL for acks */ /* add the ulp payload - if any. tx->ss can be NULL for acks */
if (ss) if (tx->ss)
ret = build_verbs_ulp_payload(sde, ss, length, tx); ret = build_verbs_ulp_payload(sde, length, tx);
bail_txadd: bail_txadd:
return ret; return ret;
} }
...@@ -894,7 +893,6 @@ int hfi1_verbs_send_dma(struct rvt_qp *qp, struct hfi1_pkt_state *ps, ...@@ -894,7 +893,6 @@ int hfi1_verbs_send_dma(struct rvt_qp *qp, struct hfi1_pkt_state *ps,
struct hfi1_qp_priv *priv = qp->priv; struct hfi1_qp_priv *priv = qp->priv;
struct hfi1_ahg_info *ahg_info = priv->s_ahg; struct hfi1_ahg_info *ahg_info = priv->s_ahg;
u32 hdrwords = qp->s_hdrwords; u32 hdrwords = qp->s_hdrwords;
struct rvt_sge_state *ss = qp->s_cur_sge;
u32 len = ps->s_txreq->s_cur_size; u32 len = ps->s_txreq->s_cur_size;
u32 plen = hdrwords + ((len + 3) >> 2) + 2; /* includes pbc */ u32 plen = hdrwords + ((len + 3) >> 2) + 2; /* includes pbc */
struct hfi1_ibdev *dev = ps->dev; struct hfi1_ibdev *dev = ps->dev;
...@@ -920,7 +918,7 @@ int hfi1_verbs_send_dma(struct rvt_qp *qp, struct hfi1_pkt_state *ps, ...@@ -920,7 +918,7 @@ int hfi1_verbs_send_dma(struct rvt_qp *qp, struct hfi1_pkt_state *ps,
plen); plen);
} }
tx->wqe = qp->s_wqe; tx->wqe = qp->s_wqe;
ret = build_verbs_tx_desc(tx->sde, ss, len, tx, ahg_info, pbc); ret = build_verbs_tx_desc(tx->sde, len, tx, ahg_info, pbc);
if (unlikely(ret)) if (unlikely(ret))
goto bail_build; goto bail_build;
} }
...@@ -1011,7 +1009,7 @@ int hfi1_verbs_send_pio(struct rvt_qp *qp, struct hfi1_pkt_state *ps, ...@@ -1011,7 +1009,7 @@ int hfi1_verbs_send_pio(struct rvt_qp *qp, struct hfi1_pkt_state *ps,
{ {
struct hfi1_qp_priv *priv = qp->priv; struct hfi1_qp_priv *priv = qp->priv;
u32 hdrwords = qp->s_hdrwords; u32 hdrwords = qp->s_hdrwords;
struct rvt_sge_state *ss = qp->s_cur_sge; struct rvt_sge_state *ss = ps->s_txreq->ss;
u32 len = ps->s_txreq->s_cur_size; u32 len = ps->s_txreq->s_cur_size;
u32 dwords = (len + 3) >> 2; u32 dwords = (len + 3) >> 2;
u32 plen = hdrwords + dwords + 2; /* includes pbc */ u32 plen = hdrwords + dwords + 2; /* includes pbc */
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册