aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/infiniband/hw/nes/nes_cm.c
diff options
context:
space:
mode:
authorDon Wood <donald.e.wood@intel.com>2009-09-05 23:36:37 -0400
committerRoland Dreier <rolandd@cisco.com>2009-09-05 23:36:37 -0400
commit873fcdd4bfc75880888a7d148a71d70ed87c9ebf (patch)
treed64569366205a1c42be20806bb56e6b7885fe196 /drivers/infiniband/hw/nes/nes_cm.c
parentc4c3f279cd8e9cc1d3e2f364a27beadb2e69cda8 (diff)
RDMA/nes: Allocate work item for disconnect event handling
The code currently has a work structure in the QP. This requires a lock and a pending flag to ensure there is never more than one request active. When two events happen quickly (such as FIN and LLP CLOSE), it causes unnecessary timeouts since the second one is dropped. This fix allocates memory for the work request so the second one can be queued. A lock is removed since it is no longer needed. Signed-off-by: Don Wood <donald.e.wood@intel.com> Signed-off-by: Roland Dreier <rolandd@cisco.com>
Diffstat (limited to 'drivers/infiniband/hw/nes/nes_cm.c')
-rw-r--r--drivers/infiniband/hw/nes/nes_cm.c26
1 files changed, 11 insertions, 15 deletions
diff --git a/drivers/infiniband/hw/nes/nes_cm.c b/drivers/infiniband/hw/nes/nes_cm.c
index fe08eb57c99f..d89bdee0cf55 100644
--- a/drivers/infiniband/hw/nes/nes_cm.c
+++ b/drivers/infiniband/hw/nes/nes_cm.c
@@ -2450,20 +2450,16 @@ static int nes_cm_init_tsa_conn(struct nes_qp *nesqp, struct nes_cm_node *cm_nod
2450 */ 2450 */
2451int nes_cm_disconn(struct nes_qp *nesqp) 2451int nes_cm_disconn(struct nes_qp *nesqp)
2452{ 2452{
2453 unsigned long flags; 2453 struct disconn_work *work;
2454 2454
2455 spin_lock_irqsave(&nesqp->lock, flags); 2455 work = kzalloc(sizeof *work, GFP_ATOMIC);
2456 if (nesqp->disconn_pending == 0) { 2456 if (!work)
2457 nesqp->disconn_pending++; 2457 return -ENOMEM; /* Timer will clean up */
2458 spin_unlock_irqrestore(&nesqp->lock, flags);
2459 nes_add_ref(&nesqp->ibqp);
2460 /* init our disconnect work element, to */
2461 INIT_WORK(&nesqp->disconn_work, nes_disconnect_worker);
2462
2463 queue_work(g_cm_core->disconn_wq, &nesqp->disconn_work);
2464 } else
2465 spin_unlock_irqrestore(&nesqp->lock, flags);
2466 2458
2459 nes_add_ref(&nesqp->ibqp);
2460 work->nesqp = nesqp;
2461 INIT_WORK(&work->work, nes_disconnect_worker);
2462 queue_work(g_cm_core->disconn_wq, &work->work);
2467 return 0; 2463 return 0;
2468} 2464}
2469 2465
@@ -2473,8 +2469,10 @@ int nes_cm_disconn(struct nes_qp *nesqp)
2473 */ 2469 */
2474static void nes_disconnect_worker(struct work_struct *work) 2470static void nes_disconnect_worker(struct work_struct *work)
2475{ 2471{
2476 struct nes_qp *nesqp = container_of(work, struct nes_qp, disconn_work); 2472 struct disconn_work *dwork = container_of(work, struct disconn_work, work);
2473 struct nes_qp *nesqp = dwork->nesqp;
2477 2474
2475 kfree(dwork);
2478 nes_debug(NES_DBG_CM, "processing AEQE id 0x%04X for QP%u.\n", 2476 nes_debug(NES_DBG_CM, "processing AEQE id 0x%04X for QP%u.\n",
2479 nesqp->last_aeq, nesqp->hwqp.qp_id); 2477 nesqp->last_aeq, nesqp->hwqp.qp_id);
2480 nes_cm_disconn_true(nesqp); 2478 nes_cm_disconn_true(nesqp);
@@ -2557,7 +2555,6 @@ static int nes_cm_disconn_true(struct nes_qp *nesqp)
2557 spin_lock_irqsave(&nesqp->lock, flags); 2555 spin_lock_irqsave(&nesqp->lock, flags);
2558 } 2556 }
2559 2557
2560 nesqp->disconn_pending = 0;
2561 /* There might have been another AE while the lock was released */ 2558 /* There might have been another AE while the lock was released */
2562 original_hw_tcp_state = nesqp->hw_tcp_state; 2559 original_hw_tcp_state = nesqp->hw_tcp_state;
2563 original_ibqp_state = nesqp->ibqp_state; 2560 original_ibqp_state = nesqp->ibqp_state;
@@ -2610,7 +2607,6 @@ static int nes_cm_disconn_true(struct nes_qp *nesqp)
2610 } 2607 }
2611 } 2608 }
2612 } else { 2609 } else {
2613 nesqp->disconn_pending = 0;
2614 spin_unlock_irqrestore(&nesqp->lock, flags); 2610 spin_unlock_irqrestore(&nesqp->lock, flags);
2615 } 2611 }
2616 2612