mirror of
https://github.com/adulau/aha.git
synced 2025-01-03 22:53:18 +00:00
IB/ipath: Duplicate RDMA reads can cause responder to NAK inappropriately
A duplicate RDMA read request can fool the responder into NAKing a new RDMA read request because the responder wasn't keeping track of whether the queue of RDMA read requests had been sent at least once. For example, requester sends 4 2K byte RDMA read requests, times out, and resends the first, then sees the 4 responses, then sends a 5th RDMA read or atomic operation. The responder sees the 4 requests, sends 4 responses, sees the resent 1st request, rewinds the queue, then sees the 5th request but thinks the queue is full and that the requester is invalidly sending a 5th new request. Signed-off-by: Ralph Campbell <ralph.campbell@qlogic.com> Signed-off-by: Roland Dreier <rolandd@cisco.com>
This commit is contained in:
parent
30d149ab58
commit
d781b129f1
2 changed files with 34 additions and 5 deletions
|
@ -125,8 +125,10 @@ static int ipath_make_rc_ack(struct ipath_qp *qp,
|
||||||
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);
|
||||||
} else
|
} else {
|
||||||
qp->s_ack_state = OP(RDMA_READ_RESPONSE_ONLY);
|
qp->s_ack_state = OP(RDMA_READ_RESPONSE_ONLY);
|
||||||
|
e->sent = 1;
|
||||||
|
}
|
||||||
ohdr->u.aeth = ipath_compute_aeth(qp);
|
ohdr->u.aeth = ipath_compute_aeth(qp);
|
||||||
hwords++;
|
hwords++;
|
||||||
qp->s_ack_rdma_psn = e->psn;
|
qp->s_ack_rdma_psn = e->psn;
|
||||||
|
@ -143,6 +145,7 @@ static int ipath_make_rc_ack(struct ipath_qp *qp,
|
||||||
cpu_to_be32(e->atomic_data);
|
cpu_to_be32(e->atomic_data);
|
||||||
hwords += sizeof(ohdr->u.at) / sizeof(u32);
|
hwords += sizeof(ohdr->u.at) / sizeof(u32);
|
||||||
bth2 = e->psn;
|
bth2 = e->psn;
|
||||||
|
e->sent = 1;
|
||||||
}
|
}
|
||||||
bth0 = qp->s_ack_state << 24;
|
bth0 = qp->s_ack_state << 24;
|
||||||
break;
|
break;
|
||||||
|
@ -158,6 +161,7 @@ static int ipath_make_rc_ack(struct ipath_qp *qp,
|
||||||
ohdr->u.aeth = ipath_compute_aeth(qp);
|
ohdr->u.aeth = ipath_compute_aeth(qp);
|
||||||
hwords++;
|
hwords++;
|
||||||
qp->s_ack_state = OP(RDMA_READ_RESPONSE_LAST);
|
qp->s_ack_state = OP(RDMA_READ_RESPONSE_LAST);
|
||||||
|
qp->s_ack_queue[qp->s_tail_ack_queue].sent = 1;
|
||||||
}
|
}
|
||||||
bth0 = qp->s_ack_state << 24;
|
bth0 = qp->s_ack_state << 24;
|
||||||
bth2 = qp->s_ack_rdma_psn++ & IPATH_PSN_MASK;
|
bth2 = qp->s_ack_rdma_psn++ & IPATH_PSN_MASK;
|
||||||
|
@ -1479,6 +1483,22 @@ static void ipath_rc_error(struct ipath_qp *qp, enum ib_wc_status err)
|
||||||
spin_unlock_irqrestore(&qp->s_lock, flags);
|
spin_unlock_irqrestore(&qp->s_lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline void ipath_update_ack_queue(struct ipath_qp *qp, unsigned n)
|
||||||
|
{
|
||||||
|
unsigned long flags;
|
||||||
|
unsigned next;
|
||||||
|
|
||||||
|
next = n + 1;
|
||||||
|
if (next > IPATH_MAX_RDMA_ATOMIC)
|
||||||
|
next = 0;
|
||||||
|
spin_lock_irqsave(&qp->s_lock, flags);
|
||||||
|
if (n == qp->s_tail_ack_queue) {
|
||||||
|
qp->s_tail_ack_queue = next;
|
||||||
|
qp->s_ack_state = OP(ACKNOWLEDGE);
|
||||||
|
}
|
||||||
|
spin_unlock_irqrestore(&qp->s_lock, flags);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ipath_rc_rcv - process an incoming RC packet
|
* ipath_rc_rcv - process an incoming RC packet
|
||||||
* @dev: the device this packet came in on
|
* @dev: the device this packet came in on
|
||||||
|
@ -1741,8 +1761,11 @@ void ipath_rc_rcv(struct ipath_ibdev *dev, struct ipath_ib_header *hdr,
|
||||||
next = qp->r_head_ack_queue + 1;
|
next = qp->r_head_ack_queue + 1;
|
||||||
if (next > IPATH_MAX_RDMA_ATOMIC)
|
if (next > IPATH_MAX_RDMA_ATOMIC)
|
||||||
next = 0;
|
next = 0;
|
||||||
if (unlikely(next == qp->s_tail_ack_queue))
|
if (unlikely(next == qp->s_tail_ack_queue)) {
|
||||||
goto nack_inv;
|
if (!qp->s_ack_queue[next].sent)
|
||||||
|
goto nack_inv;
|
||||||
|
ipath_update_ack_queue(qp, next);
|
||||||
|
}
|
||||||
e = &qp->s_ack_queue[qp->r_head_ack_queue];
|
e = &qp->s_ack_queue[qp->r_head_ack_queue];
|
||||||
/* RETH comes after BTH */
|
/* RETH comes after BTH */
|
||||||
if (!header_in_data)
|
if (!header_in_data)
|
||||||
|
@ -1777,6 +1800,7 @@ void ipath_rc_rcv(struct ipath_ibdev *dev, struct ipath_ib_header *hdr,
|
||||||
e->rdma_sge.sge.sge_length = 0;
|
e->rdma_sge.sge.sge_length = 0;
|
||||||
}
|
}
|
||||||
e->opcode = opcode;
|
e->opcode = opcode;
|
||||||
|
e->sent = 0;
|
||||||
e->psn = psn;
|
e->psn = psn;
|
||||||
/*
|
/*
|
||||||
* We need to increment the MSN here instead of when we
|
* We need to increment the MSN here instead of when we
|
||||||
|
@ -1812,8 +1836,11 @@ void ipath_rc_rcv(struct ipath_ibdev *dev, struct ipath_ib_header *hdr,
|
||||||
next = qp->r_head_ack_queue + 1;
|
next = qp->r_head_ack_queue + 1;
|
||||||
if (next > IPATH_MAX_RDMA_ATOMIC)
|
if (next > IPATH_MAX_RDMA_ATOMIC)
|
||||||
next = 0;
|
next = 0;
|
||||||
if (unlikely(next == qp->s_tail_ack_queue))
|
if (unlikely(next == qp->s_tail_ack_queue)) {
|
||||||
goto nack_inv;
|
if (!qp->s_ack_queue[next].sent)
|
||||||
|
goto nack_inv;
|
||||||
|
ipath_update_ack_queue(qp, next);
|
||||||
|
}
|
||||||
if (!header_in_data)
|
if (!header_in_data)
|
||||||
ateth = &ohdr->u.atomic_eth;
|
ateth = &ohdr->u.atomic_eth;
|
||||||
else
|
else
|
||||||
|
@ -1838,6 +1865,7 @@ void ipath_rc_rcv(struct ipath_ibdev *dev, struct ipath_ib_header *hdr,
|
||||||
be64_to_cpu(ateth->compare_data),
|
be64_to_cpu(ateth->compare_data),
|
||||||
sdata);
|
sdata);
|
||||||
e->opcode = opcode;
|
e->opcode = opcode;
|
||||||
|
e->sent = 0;
|
||||||
e->psn = psn & IPATH_PSN_MASK;
|
e->psn = psn & IPATH_PSN_MASK;
|
||||||
qp->r_msn++;
|
qp->r_msn++;
|
||||||
qp->r_psn++;
|
qp->r_psn++;
|
||||||
|
|
|
@ -321,6 +321,7 @@ struct ipath_sge_state {
|
||||||
*/
|
*/
|
||||||
struct ipath_ack_entry {
|
struct ipath_ack_entry {
|
||||||
u8 opcode;
|
u8 opcode;
|
||||||
|
u8 sent;
|
||||||
u32 psn;
|
u32 psn;
|
||||||
union {
|
union {
|
||||||
struct ipath_sge_state rdma_sge;
|
struct ipath_sge_state rdma_sge;
|
||||||
|
|
Loading…
Reference in a new issue