diff options
author | Vu Pham <vuhuong@mellanox.com> | 2013-10-10 07:50:29 -0400 |
---|---|---|
committer | Roland Dreier <roland@purestorage.com> | 2013-11-08 17:43:17 -0500 |
commit | 65d7dd2f3479ef5aec1d9ddd1481cb7851c11af6 (patch) | |
tree | efdbf96930f18494284b8d90f95ad6db4be9d18a /drivers/infiniband/ulp/srp/ib_srp.c | |
parent | 71444b97816ae08a1cc435f044f9f1f4cbd51e6a (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.c | 9 |
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 | ||
686 | static void srp_rport_delete(struct srp_rport *rport) | 687 | static void srp_rport_delete(struct srp_rport *rport) |