aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/infiniband/ulp/srp/ib_srp.c
diff options
context:
space:
mode:
authorVu Pham <vuhuong@mellanox.com>2013-10-10 07:50:29 -0400
committerRoland Dreier <roland@purestorage.com>2013-11-08 17:43:17 -0500
commit65d7dd2f3479ef5aec1d9ddd1481cb7851c11af6 (patch)
treeefdbf96930f18494284b8d90f95ad6db4be9d18a /drivers/infiniband/ulp/srp/ib_srp.c
parent71444b97816ae08a1cc435f044f9f1f4cbd51e6a (diff)
IB/srp: Remove target from list before freeing Scsi_Host structure
Remove an SRP target from the SRP target list before invoking the last scsi_host_put() call. This change is necessary because that last put frees the memory that holds the srp_target_port structure. This patch prevents the following kernel oops: RIP: 0010:[<ffffffff810b00d0>] __lock_acquire+0x500/0x1570 Call Trace: [<ffffffff810b11e4>] lock_acquire+0xa4/0x120 [<ffffffff81531206>] _spin_lock+0x36/0x70 [<ffffffffa01b6d8f>] srp_remove_work+0xef/0x180 [ib_srp] [<ffffffff8109125c>] worker_thread+0x21c/0x3d0 [<ffffffff81096e86>] kthread+0x96/0xa0 [<ffffffff8100c20a>] child_rip+0xa/0x20 Signed-off-by: Vu Pham <vuhuong@mellanox.com> [ bvanassche - Modified path description and CC'ed stable. ] Signed-off-by: Bart Van Assche <bvanassche@acm.org> Cc: <stable@vger.kernel.org> Signed-off-by: Roland Dreier <roland@purestorage.com>
Diffstat (limited to 'drivers/infiniband/ulp/srp/ib_srp.c')
-rw-r--r--drivers/infiniband/ulp/srp/ib_srp.c9
1 files changed, 5 insertions, 4 deletions
diff --git a/drivers/infiniband/ulp/srp/ib_srp.c b/drivers/infiniband/ulp/srp/ib_srp.c
index a8f8d0b8a777..a8c06c4451c0 100644
--- a/drivers/infiniband/ulp/srp/ib_srp.c
+++ b/drivers/infiniband/ulp/srp/ib_srp.c
@@ -666,6 +666,11 @@ static void srp_remove_target(struct srp_target_port *target)
666 cancel_work_sync(&target->tl_err_work); 666 cancel_work_sync(&target->tl_err_work);
667 srp_rport_put(target->rport); 667 srp_rport_put(target->rport);
668 srp_free_req_data(target); 668 srp_free_req_data(target);
669
670 spin_lock(&target->srp_host->target_lock);
671 list_del(&target->list);
672 spin_unlock(&target->srp_host->target_lock);
673
669 scsi_host_put(target->scsi_host); 674 scsi_host_put(target->scsi_host);
670} 675}
671 676
@@ -677,10 +682,6 @@ static void srp_remove_work(struct work_struct *work)
677 WARN_ON_ONCE(target->state != SRP_TARGET_REMOVED); 682 WARN_ON_ONCE(target->state != SRP_TARGET_REMOVED);
678 683
679 srp_remove_target(target); 684 srp_remove_target(target);
680
681 spin_lock(&target->srp_host->target_lock);
682 list_del(&target->list);
683 spin_unlock(&target->srp_host->target_lock);
684} 685}
685 686
686static void srp_rport_delete(struct srp_rport *rport) 687static void srp_rport_delete(struct srp_rport *rport)