diff options
author | Bryan O'Sullivan <bos@pathscale.com> | 2006-07-01 07:36:10 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@g5.osdl.org> | 2006-07-01 12:56:01 -0400 |
commit | 12eef41f8b72b6e11e36b48c78849c17e49781c8 (patch) | |
tree | 3a8bd77d77c7542e19d200d3abf25fc07f0d8f51 /drivers/infiniband/hw/ipath/ipath_qp.c | |
parent | fba75200ad92892bf32d8d6f1443c6f1e4f48676 (diff) |
[PATCH] IB/ipath: rC receive interrupt performance changes
This patch separates QP state used for sending and receiving RC packets so the
processing in the receive interrupt handler can be done mostly without locks
being held. ACK packets are now sent without requiring synchronization with
the send tasklet.
Signed-off-by: Ralph Campbell <ralph.campbell@qlogic.com>
Signed-off-by: Bryan O'Sullivan <bryan.osullivan@qlogic.com>
Cc: "Michael S. Tsirkin" <mst@mellanox.co.il>
Cc: Roland Dreier <rolandd@cisco.com>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Diffstat (limited to 'drivers/infiniband/hw/ipath/ipath_qp.c')
-rw-r--r-- | drivers/infiniband/hw/ipath/ipath_qp.c | 43 |
1 files changed, 15 insertions, 28 deletions
diff --git a/drivers/infiniband/hw/ipath/ipath_qp.c b/drivers/infiniband/hw/ipath/ipath_qp.c index 83b9a6a5e2c6..68f01513214f 100644 --- a/drivers/infiniband/hw/ipath/ipath_qp.c +++ b/drivers/infiniband/hw/ipath/ipath_qp.c | |||
@@ -333,10 +333,11 @@ static void ipath_reset_qp(struct ipath_qp *qp) | |||
333 | qp->remote_qpn = 0; | 333 | qp->remote_qpn = 0; |
334 | qp->qkey = 0; | 334 | qp->qkey = 0; |
335 | qp->qp_access_flags = 0; | 335 | qp->qp_access_flags = 0; |
336 | clear_bit(IPATH_S_BUSY, &qp->s_flags); | ||
336 | qp->s_hdrwords = 0; | 337 | qp->s_hdrwords = 0; |
337 | qp->s_psn = 0; | 338 | qp->s_psn = 0; |
338 | qp->r_psn = 0; | 339 | qp->r_psn = 0; |
339 | atomic_set(&qp->msn, 0); | 340 | qp->r_msn = 0; |
340 | if (qp->ibqp.qp_type == IB_QPT_RC) { | 341 | if (qp->ibqp.qp_type == IB_QPT_RC) { |
341 | qp->s_state = IB_OPCODE_RC_SEND_LAST; | 342 | qp->s_state = IB_OPCODE_RC_SEND_LAST; |
342 | qp->r_state = IB_OPCODE_RC_SEND_LAST; | 343 | qp->r_state = IB_OPCODE_RC_SEND_LAST; |
@@ -345,7 +346,8 @@ static void ipath_reset_qp(struct ipath_qp *qp) | |||
345 | qp->r_state = IB_OPCODE_UC_SEND_LAST; | 346 | qp->r_state = IB_OPCODE_UC_SEND_LAST; |
346 | } | 347 | } |
347 | qp->s_ack_state = IB_OPCODE_RC_ACKNOWLEDGE; | 348 | qp->s_ack_state = IB_OPCODE_RC_ACKNOWLEDGE; |
348 | qp->s_nak_state = 0; | 349 | qp->r_ack_state = IB_OPCODE_RC_ACKNOWLEDGE; |
350 | qp->r_nak_state = 0; | ||
349 | qp->s_rnr_timeout = 0; | 351 | qp->s_rnr_timeout = 0; |
350 | qp->s_head = 0; | 352 | qp->s_head = 0; |
351 | qp->s_tail = 0; | 353 | qp->s_tail = 0; |
@@ -363,10 +365,10 @@ static void ipath_reset_qp(struct ipath_qp *qp) | |||
363 | * @qp: the QP to put into an error state | 365 | * @qp: the QP to put into an error state |
364 | * | 366 | * |
365 | * Flushes both send and receive work queues. | 367 | * Flushes both send and receive work queues. |
366 | * QP r_rq.lock and s_lock should be held. | 368 | * QP s_lock should be held and interrupts disabled. |
367 | */ | 369 | */ |
368 | 370 | ||
369 | static void ipath_error_qp(struct ipath_qp *qp) | 371 | void ipath_error_qp(struct ipath_qp *qp) |
370 | { | 372 | { |
371 | struct ipath_ibdev *dev = to_idev(qp->ibqp.device); | 373 | struct ipath_ibdev *dev = to_idev(qp->ibqp.device); |
372 | struct ib_wc wc; | 374 | struct ib_wc wc; |
@@ -409,12 +411,14 @@ static void ipath_error_qp(struct ipath_qp *qp) | |||
409 | qp->s_ack_state = IB_OPCODE_RC_ACKNOWLEDGE; | 411 | qp->s_ack_state = IB_OPCODE_RC_ACKNOWLEDGE; |
410 | 412 | ||
411 | wc.opcode = IB_WC_RECV; | 413 | wc.opcode = IB_WC_RECV; |
414 | spin_lock(&qp->r_rq.lock); | ||
412 | while (qp->r_rq.tail != qp->r_rq.head) { | 415 | while (qp->r_rq.tail != qp->r_rq.head) { |
413 | wc.wr_id = get_rwqe_ptr(&qp->r_rq, qp->r_rq.tail)->wr_id; | 416 | wc.wr_id = get_rwqe_ptr(&qp->r_rq, qp->r_rq.tail)->wr_id; |
414 | if (++qp->r_rq.tail >= qp->r_rq.size) | 417 | if (++qp->r_rq.tail >= qp->r_rq.size) |
415 | qp->r_rq.tail = 0; | 418 | qp->r_rq.tail = 0; |
416 | ipath_cq_enter(to_icq(qp->ibqp.recv_cq), &wc, 1); | 419 | ipath_cq_enter(to_icq(qp->ibqp.recv_cq), &wc, 1); |
417 | } | 420 | } |
421 | spin_unlock(&qp->r_rq.lock); | ||
418 | } | 422 | } |
419 | 423 | ||
420 | /** | 424 | /** |
@@ -434,8 +438,7 @@ int ipath_modify_qp(struct ib_qp *ibqp, struct ib_qp_attr *attr, | |||
434 | unsigned long flags; | 438 | unsigned long flags; |
435 | int ret; | 439 | int ret; |
436 | 440 | ||
437 | spin_lock_irqsave(&qp->r_rq.lock, flags); | 441 | spin_lock_irqsave(&qp->s_lock, flags); |
438 | spin_lock(&qp->s_lock); | ||
439 | 442 | ||
440 | cur_state = attr_mask & IB_QP_CUR_STATE ? | 443 | cur_state = attr_mask & IB_QP_CUR_STATE ? |
441 | attr->cur_qp_state : qp->state; | 444 | attr->cur_qp_state : qp->state; |
@@ -506,31 +509,19 @@ int ipath_modify_qp(struct ib_qp *ibqp, struct ib_qp_attr *attr, | |||
506 | } | 509 | } |
507 | 510 | ||
508 | if (attr_mask & IB_QP_MIN_RNR_TIMER) | 511 | if (attr_mask & IB_QP_MIN_RNR_TIMER) |
509 | qp->s_min_rnr_timer = attr->min_rnr_timer; | 512 | qp->r_min_rnr_timer = attr->min_rnr_timer; |
510 | 513 | ||
511 | if (attr_mask & IB_QP_QKEY) | 514 | if (attr_mask & IB_QP_QKEY) |
512 | qp->qkey = attr->qkey; | 515 | qp->qkey = attr->qkey; |
513 | 516 | ||
514 | qp->state = new_state; | 517 | qp->state = new_state; |
515 | spin_unlock(&qp->s_lock); | 518 | spin_unlock_irqrestore(&qp->s_lock, flags); |
516 | spin_unlock_irqrestore(&qp->r_rq.lock, flags); | ||
517 | |||
518 | /* | ||
519 | * If QP1 changed to the RTS state, try to move to the link to INIT | ||
520 | * even if it was ACTIVE so the SM will reinitialize the SMA's | ||
521 | * state. | ||
522 | */ | ||
523 | if (qp->ibqp.qp_num == 1 && new_state == IB_QPS_RTS) { | ||
524 | struct ipath_ibdev *dev = to_idev(ibqp->device); | ||
525 | 519 | ||
526 | ipath_layer_set_linkstate(dev->dd, IPATH_IB_LINKDOWN); | ||
527 | } | ||
528 | ret = 0; | 520 | ret = 0; |
529 | goto bail; | 521 | goto bail; |
530 | 522 | ||
531 | inval: | 523 | inval: |
532 | spin_unlock(&qp->s_lock); | 524 | spin_unlock_irqrestore(&qp->s_lock, flags); |
533 | spin_unlock_irqrestore(&qp->r_rq.lock, flags); | ||
534 | ret = -EINVAL; | 525 | ret = -EINVAL; |
535 | 526 | ||
536 | bail: | 527 | bail: |
@@ -564,7 +555,7 @@ int ipath_query_qp(struct ib_qp *ibqp, struct ib_qp_attr *attr, | |||
564 | attr->sq_draining = 0; | 555 | attr->sq_draining = 0; |
565 | attr->max_rd_atomic = 1; | 556 | attr->max_rd_atomic = 1; |
566 | attr->max_dest_rd_atomic = 1; | 557 | attr->max_dest_rd_atomic = 1; |
567 | attr->min_rnr_timer = qp->s_min_rnr_timer; | 558 | attr->min_rnr_timer = qp->r_min_rnr_timer; |
568 | attr->port_num = 1; | 559 | attr->port_num = 1; |
569 | attr->timeout = 0; | 560 | attr->timeout = 0; |
570 | attr->retry_cnt = qp->s_retry_cnt; | 561 | attr->retry_cnt = qp->s_retry_cnt; |
@@ -591,16 +582,12 @@ int ipath_query_qp(struct ib_qp *ibqp, struct ib_qp_attr *attr, | |||
591 | * @qp: the queue pair to compute the AETH for | 582 | * @qp: the queue pair to compute the AETH for |
592 | * | 583 | * |
593 | * Returns the AETH. | 584 | * Returns the AETH. |
594 | * | ||
595 | * The QP s_lock should be held. | ||
596 | */ | 585 | */ |
597 | __be32 ipath_compute_aeth(struct ipath_qp *qp) | 586 | __be32 ipath_compute_aeth(struct ipath_qp *qp) |
598 | { | 587 | { |
599 | u32 aeth = atomic_read(&qp->msn) & IPS_MSN_MASK; | 588 | u32 aeth = qp->r_msn & IPS_MSN_MASK; |
600 | 589 | ||
601 | if (qp->s_nak_state) { | 590 | if (qp->ibqp.srq) { |
602 | aeth |= qp->s_nak_state << IPS_AETH_CREDIT_SHIFT; | ||
603 | } else if (qp->ibqp.srq) { | ||
604 | /* | 591 | /* |
605 | * Shared receive queues don't generate credits. | 592 | * Shared receive queues don't generate credits. |
606 | * Set the credit field to the invalid value. | 593 | * Set the credit field to the invalid value. |