aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorBart Van Assche <bvanassche@acm.org>2010-11-26 15:08:38 -0500
committerDavid Dillow <dillowda@ornl.gov>2011-01-10 15:44:50 -0500
commite9684678221441f886b4d7c74f8770bb0981737a (patch)
tree76ed83fb7dd41c84c2b8e08b785a802e99c6e790
parent94a9174c630c8465ed9e97ecd242993429930c05 (diff)
IB/srp: stop sharing the host lock with SCSI
We don't need protection against the SCSI stack, so use our own lock to allow parallel progress on separate CPUs. Signed-off-by: Bart Van Assche <bvanassche@acm.org> [ broken out and small cleanups by David Dillow ] Signed-off-by: David Dillow <dillowda@ornl.gov>
-rw-r--r--drivers/infiniband/ulp/srp/ib_srp.c46
-rw-r--r--drivers/infiniband/ulp/srp/ib_srp.h2
2 files changed, 25 insertions, 23 deletions
diff --git a/drivers/infiniband/ulp/srp/ib_srp.c b/drivers/infiniband/ulp/srp/ib_srp.c
index e76fe54faeea..8691fc83f70b 100644
--- a/drivers/infiniband/ulp/srp/ib_srp.c
+++ b/drivers/infiniband/ulp/srp/ib_srp.c
@@ -447,12 +447,12 @@ static bool srp_change_state(struct srp_target_port *target,
447{ 447{
448 bool changed = false; 448 bool changed = false;
449 449
450 spin_lock_irq(target->scsi_host->host_lock); 450 spin_lock_irq(&target->lock);
451 if (target->state == old) { 451 if (target->state == old) {
452 target->state = new; 452 target->state = new;
453 changed = true; 453 changed = true;
454 } 454 }
455 spin_unlock_irq(target->scsi_host->host_lock); 455 spin_unlock_irq(&target->lock);
456 return changed; 456 return changed;
457} 457}
458 458
@@ -555,11 +555,11 @@ static void srp_remove_req(struct srp_target_port *target,
555 unsigned long flags; 555 unsigned long flags;
556 556
557 srp_unmap_data(req->scmnd, target, req); 557 srp_unmap_data(req->scmnd, target, req);
558 spin_lock_irqsave(target->scsi_host->host_lock, flags); 558 spin_lock_irqsave(&target->lock, flags);
559 target->req_lim += req_lim_delta; 559 target->req_lim += req_lim_delta;
560 req->scmnd = NULL; 560 req->scmnd = NULL;
561 list_add_tail(&req->list, &target->free_reqs); 561 list_add_tail(&req->list, &target->free_reqs);
562 spin_unlock_irqrestore(target->scsi_host->host_lock, flags); 562 spin_unlock_irqrestore(&target->lock, flags);
563} 563}
564 564
565static void srp_reset_req(struct srp_target_port *target, struct srp_request *req) 565static void srp_reset_req(struct srp_target_port *target, struct srp_request *req)
@@ -634,13 +634,13 @@ err:
634 * Schedule our work inside the lock to avoid a race with 634 * Schedule our work inside the lock to avoid a race with
635 * the flush_scheduled_work() in srp_remove_one(). 635 * the flush_scheduled_work() in srp_remove_one().
636 */ 636 */
637 spin_lock_irq(target->scsi_host->host_lock); 637 spin_lock_irq(&target->lock);
638 if (target->state == SRP_TARGET_CONNECTING) { 638 if (target->state == SRP_TARGET_CONNECTING) {
639 target->state = SRP_TARGET_DEAD; 639 target->state = SRP_TARGET_DEAD;
640 INIT_WORK(&target->work, srp_remove_work); 640 INIT_WORK(&target->work, srp_remove_work);
641 schedule_work(&target->work); 641 schedule_work(&target->work);
642 } 642 }
643 spin_unlock_irq(target->scsi_host->host_lock); 643 spin_unlock_irq(&target->lock);
644 644
645 return ret; 645 return ret;
646} 646}
@@ -829,17 +829,16 @@ static void srp_put_tx_iu(struct srp_target_port *target, struct srp_iu *iu,
829{ 829{
830 unsigned long flags; 830 unsigned long flags;
831 831
832 spin_lock_irqsave(target->scsi_host->host_lock, flags); 832 spin_lock_irqsave(&target->lock, flags);
833 list_add(&iu->list, &target->free_tx); 833 list_add(&iu->list, &target->free_tx);
834 if (iu_type != SRP_IU_RSP) 834 if (iu_type != SRP_IU_RSP)
835 ++target->req_lim; 835 ++target->req_lim;
836 spin_unlock_irqrestore(target->scsi_host->host_lock, flags); 836 spin_unlock_irqrestore(&target->lock, flags);
837} 837}
838 838
839/* 839/*
840 * Must be called with target->scsi_host->host_lock held to protect 840 * Must be called with target->lock held to protect req_lim and free_tx.
841 * req_lim and free_tx. If IU is not sent, it must be returned using 841 * If IU is not sent, it must be returned using srp_put_tx_iu().
842 * srp_put_tx_iu().
843 * 842 *
844 * Note: 843 * Note:
845 * An upper limit for the number of allocated information units for each 844 * An upper limit for the number of allocated information units for each
@@ -920,9 +919,9 @@ static void srp_process_rsp(struct srp_target_port *target, struct srp_rsp *rsp)
920 unsigned long flags; 919 unsigned long flags;
921 920
922 if (unlikely(rsp->tag & SRP_TAG_TSK_MGMT)) { 921 if (unlikely(rsp->tag & SRP_TAG_TSK_MGMT)) {
923 spin_lock_irqsave(target->scsi_host->host_lock, flags); 922 spin_lock_irqsave(&target->lock, flags);
924 target->req_lim += be32_to_cpu(rsp->req_lim_delta); 923 target->req_lim += be32_to_cpu(rsp->req_lim_delta);
925 spin_unlock_irqrestore(target->scsi_host->host_lock, flags); 924 spin_unlock_irqrestore(&target->lock, flags);
926 925
927 target->tsk_mgmt_status = -1; 926 target->tsk_mgmt_status = -1;
928 if (be32_to_cpu(rsp->resp_data_len) >= 4) 927 if (be32_to_cpu(rsp->resp_data_len) >= 4)
@@ -963,10 +962,10 @@ static int srp_response_common(struct srp_target_port *target, s32 req_delta,
963 struct srp_iu *iu; 962 struct srp_iu *iu;
964 int err; 963 int err;
965 964
966 spin_lock_irqsave(target->scsi_host->host_lock, flags); 965 spin_lock_irqsave(&target->lock, flags);
967 target->req_lim += req_delta; 966 target->req_lim += req_delta;
968 iu = __srp_get_tx_iu(target, SRP_IU_RSP); 967 iu = __srp_get_tx_iu(target, SRP_IU_RSP);
969 spin_unlock_irqrestore(target->scsi_host->host_lock, flags); 968 spin_unlock_irqrestore(&target->lock, flags);
970 969
971 if (!iu) { 970 if (!iu) {
972 shost_printk(KERN_ERR, target->scsi_host, PFX 971 shost_printk(KERN_ERR, target->scsi_host, PFX
@@ -1131,14 +1130,14 @@ static int srp_queuecommand(struct Scsi_Host *shost, struct scsi_cmnd *scmnd)
1131 return 0; 1130 return 0;
1132 } 1131 }
1133 1132
1134 spin_lock_irqsave(shost->host_lock, flags); 1133 spin_lock_irqsave(&target->lock, flags);
1135 iu = __srp_get_tx_iu(target, SRP_IU_CMD); 1134 iu = __srp_get_tx_iu(target, SRP_IU_CMD);
1136 if (iu) { 1135 if (iu) {
1137 req = list_first_entry(&target->free_reqs, struct srp_request, 1136 req = list_first_entry(&target->free_reqs, struct srp_request,
1138 list); 1137 list);
1139 list_del(&req->list); 1138 list_del(&req->list);
1140 } 1139 }
1141 spin_unlock_irqrestore(shost->host_lock, flags); 1140 spin_unlock_irqrestore(&target->lock, flags);
1142 1141
1143 if (!iu) 1142 if (!iu)
1144 goto err; 1143 goto err;
@@ -1184,9 +1183,9 @@ err_unmap:
1184err_iu: 1183err_iu:
1185 srp_put_tx_iu(target, iu, SRP_IU_CMD); 1184 srp_put_tx_iu(target, iu, SRP_IU_CMD);
1186 1185
1187 spin_lock_irqsave(shost->host_lock, flags); 1186 spin_lock_irqsave(&target->lock, flags);
1188 list_add(&req->list, &target->free_reqs); 1187 list_add(&req->list, &target->free_reqs);
1189 spin_unlock_irqrestore(shost->host_lock, flags); 1188 spin_unlock_irqrestore(&target->lock, flags);
1190 1189
1191err: 1190err:
1192 return SCSI_MLQUEUE_HOST_BUSY; 1191 return SCSI_MLQUEUE_HOST_BUSY;
@@ -1451,9 +1450,9 @@ static int srp_send_tsk_mgmt(struct srp_target_port *target,
1451 1450
1452 init_completion(&target->tsk_mgmt_done); 1451 init_completion(&target->tsk_mgmt_done);
1453 1452
1454 spin_lock_irq(target->scsi_host->host_lock); 1453 spin_lock_irq(&target->lock);
1455 iu = __srp_get_tx_iu(target, SRP_IU_TSK_MGMT); 1454 iu = __srp_get_tx_iu(target, SRP_IU_TSK_MGMT);
1456 spin_unlock_irq(target->scsi_host->host_lock); 1455 spin_unlock_irq(&target->lock);
1457 1456
1458 if (!iu) 1457 if (!iu)
1459 return -1; 1458 return -1;
@@ -1957,6 +1956,7 @@ static ssize_t srp_create_target(struct device *dev,
1957 target->scsi_host = target_host; 1956 target->scsi_host = target_host;
1958 target->srp_host = host; 1957 target->srp_host = host;
1959 1958
1959 spin_lock_init(&target->lock);
1960 INIT_LIST_HEAD(&target->free_tx); 1960 INIT_LIST_HEAD(&target->free_tx);
1961 INIT_LIST_HEAD(&target->free_reqs); 1961 INIT_LIST_HEAD(&target->free_reqs);
1962 for (i = 0; i < SRP_CMD_SQ_SIZE; ++i) { 1962 for (i = 0; i < SRP_CMD_SQ_SIZE; ++i) {
@@ -2186,9 +2186,9 @@ static void srp_remove_one(struct ib_device *device)
2186 */ 2186 */
2187 spin_lock(&host->target_lock); 2187 spin_lock(&host->target_lock);
2188 list_for_each_entry(target, &host->target_list, list) { 2188 list_for_each_entry(target, &host->target_list, list) {
2189 spin_lock_irq(target->scsi_host->host_lock); 2189 spin_lock_irq(&target->lock);
2190 target->state = SRP_TARGET_REMOVED; 2190 target->state = SRP_TARGET_REMOVED;
2191 spin_unlock_irq(target->scsi_host->host_lock); 2191 spin_unlock_irq(&target->lock);
2192 } 2192 }
2193 spin_unlock(&host->target_lock); 2193 spin_unlock(&host->target_lock);
2194 2194
diff --git a/drivers/infiniband/ulp/srp/ib_srp.h b/drivers/infiniband/ulp/srp/ib_srp.h
index 81686eee7e62..acb435d3c1e3 100644
--- a/drivers/infiniband/ulp/srp/ib_srp.h
+++ b/drivers/infiniband/ulp/srp/ib_srp.h
@@ -144,6 +144,8 @@ struct srp_target_port {
144 144
145 struct srp_iu *rx_ring[SRP_RQ_SIZE]; 145 struct srp_iu *rx_ring[SRP_RQ_SIZE];
146 146
147 spinlock_t lock;
148
147 struct list_head free_tx; 149 struct list_head free_tx;
148 struct srp_iu *tx_ring[SRP_SQ_SIZE]; 150 struct srp_iu *tx_ring[SRP_SQ_SIZE];
149 151