IB/srp: Avoid that I/O hangs due to a cable pull during LUN scanning

If a cable is pulled during LUN scanning it can happen that the
SRP rport and the SCSI host have been created but no LUNs have been
added to the SCSI host. Since multipathd only sends SCSI commands
to a SCSI target if one or more SCSI devices are present and since
there is no keepalive mechanism for IB queue pairs this means that
after a LUN scan failed and after a reconnect has succeeded no
data will be sent over the QP and hence that a subsequent cable
pull will not be detected. Avoid this by not creating an rport or
SCSI host if a cable is pulled during a SCSI LUN scan.

Note: so far the above behavior has only been observed with the
kernel module parameter ch_count set to a value >= 2.

Signed-off-by: Bart Van Assche <bvanassche@acm.org>
Reviewed-by: Sagi Grimberg <sagig@mellanox.com>
Signed-off-by: Christoph Hellwig <hch@lst.de>
This commit is contained in:
Bart Van Assche 2014-10-30 14:47:22 +01:00 committed by Christoph Hellwig
parent 205619f2f8
commit 34aa654ecb
2 changed files with 52 additions and 9 deletions

View File

@ -1111,6 +1111,10 @@ static int srp_rport_reconnect(struct srp_rport *rport)
int i, ret; int i, ret;
srp_disconnect_target(target); srp_disconnect_target(target);
if (target->state == SRP_TARGET_SCANNING)
return -ENODEV;
/* /*
* Now get a new local CM ID so that we avoid confusing the target in * Now get a new local CM ID so that we avoid confusing the target in
* case things are really fouled up. Doing so also ensures that all CM * case things are really fouled up. Doing so also ensures that all CM
@ -2585,11 +2589,23 @@ static struct scsi_host_template srp_template = {
.shost_attrs = srp_host_attrs .shost_attrs = srp_host_attrs
}; };
static int srp_sdev_count(struct Scsi_Host *host)
{
struct scsi_device *sdev;
int c = 0;
shost_for_each_device(sdev, host)
c++;
return c;
}
static int srp_add_target(struct srp_host *host, struct srp_target_port *target) static int srp_add_target(struct srp_host *host, struct srp_target_port *target)
{ {
struct srp_rport_identifiers ids; struct srp_rport_identifiers ids;
struct srp_rport *rport; struct srp_rport *rport;
target->state = SRP_TARGET_SCANNING;
sprintf(target->target_name, "SRP.T10:%016llX", sprintf(target->target_name, "SRP.T10:%016llX",
(unsigned long long) be64_to_cpu(target->id_ext)); (unsigned long long) be64_to_cpu(target->id_ext));
@ -2612,11 +2628,26 @@ static int srp_add_target(struct srp_host *host, struct srp_target_port *target)
list_add_tail(&target->list, &host->target_list); list_add_tail(&target->list, &host->target_list);
spin_unlock(&host->target_lock); spin_unlock(&host->target_lock);
target->state = SRP_TARGET_LIVE;
scsi_scan_target(&target->scsi_host->shost_gendev, scsi_scan_target(&target->scsi_host->shost_gendev,
0, target->scsi_id, SCAN_WILD_CARD, 0); 0, target->scsi_id, SCAN_WILD_CARD, 0);
if (!target->connected || target->qp_in_error) {
shost_printk(KERN_INFO, target->scsi_host,
PFX "SCSI scan failed - removing SCSI host\n");
srp_queue_remove_work(target);
goto out;
}
pr_debug(PFX "%s: SCSI scan succeeded - detected %d LUNs\n",
dev_name(&target->scsi_host->shost_gendev),
srp_sdev_count(target->scsi_host));
spin_lock_irq(&target->lock);
if (target->state == SRP_TARGET_SCANNING)
target->state = SRP_TARGET_LIVE;
spin_unlock_irq(&target->lock);
out:
return 0; return 0;
} }
@ -2960,6 +2991,12 @@ static ssize_t srp_create_target(struct device *dev,
target->tl_retry_count = 7; target->tl_retry_count = 7;
target->queue_size = SRP_DEFAULT_QUEUE_SIZE; target->queue_size = SRP_DEFAULT_QUEUE_SIZE;
/*
* Avoid that the SCSI host can be removed by srp_remove_target()
* before this function returns.
*/
scsi_host_get(target->scsi_host);
mutex_lock(&host->add_target_mutex); mutex_lock(&host->add_target_mutex);
ret = srp_parse_options(buf, target); ret = srp_parse_options(buf, target);
@ -3022,18 +3059,23 @@ static ssize_t srp_create_target(struct device *dev,
if (ret) if (ret)
goto err_disconnect; goto err_disconnect;
shost_printk(KERN_DEBUG, target->scsi_host, PFX if (target->state != SRP_TARGET_REMOVED) {
"new target: id_ext %016llx ioc_guid %016llx pkey %04x service_id %016llx sgid %pI6 dgid %pI6\n", shost_printk(KERN_DEBUG, target->scsi_host, PFX
be64_to_cpu(target->id_ext), "new target: id_ext %016llx ioc_guid %016llx pkey %04x service_id %016llx sgid %pI6 dgid %pI6\n",
be64_to_cpu(target->ioc_guid), be64_to_cpu(target->id_ext),
be16_to_cpu(target->path.pkey), be64_to_cpu(target->ioc_guid),
be64_to_cpu(target->service_id), be16_to_cpu(target->path.pkey),
target->path.sgid.raw, target->path.dgid.raw); be64_to_cpu(target->service_id),
target->path.sgid.raw, target->orig_dgid);
}
ret = count; ret = count;
out: out:
mutex_unlock(&host->add_target_mutex); mutex_unlock(&host->add_target_mutex);
scsi_host_put(target->scsi_host);
return ret; return ret;
err_disconnect: err_disconnect:

View File

@ -73,6 +73,7 @@ enum {
}; };
enum srp_target_state { enum srp_target_state {
SRP_TARGET_SCANNING,
SRP_TARGET_LIVE, SRP_TARGET_LIVE,
SRP_TARGET_REMOVED, SRP_TARGET_REMOVED,
}; };