Commit 0cf89d1d authored by Dan Williams's avatar Dan Williams

isci: cleanup "starting" state handling

The lldd actively disallows requests in the "starting" state.  Retrying
or holding off commands in this state is sub-optimal:
1/ it adds another state check to the fast path
2/ retrying can cause libsas to give up

However, isci's ->lldd_dev_found() routine already waits for controller
start to complete before allowing further progress.  Checking the
"starting" state in isci_task_execute_task and the isr is redundant and
misleading.  Clean this up and introduce a controller-wide event queue
to start reeling in "completion" proliferation in the driver.

The "stopping" state cleanups are in a similar vein, rely on the the isr
and other paths being precluded from occurring rather than implementing
state checking logic.
Reported-by: default avatarChristoph Hellwig <hch@infradead.org>
Cc: Jeff Garzik <jeff@garzik.org>
Signed-off-by: default avatarEdmund Nadolski <edmund.nadolski@intel.com>
Signed-off-by: default avatarDan Williams <dan.j.williams@intel.com>
parent c7ef4031
...@@ -67,15 +67,8 @@ irqreturn_t isci_msix_isr(int vec, void *data) ...@@ -67,15 +67,8 @@ irqreturn_t isci_msix_isr(int vec, void *data)
struct isci_host *ihost = data; struct isci_host *ihost = data;
struct scic_sds_controller *scic = ihost->core_controller; struct scic_sds_controller *scic = ihost->core_controller;
if (isci_host_get_state(ihost) != isci_starting) { if (scic_sds_controller_isr(scic))
if (scic_sds_controller_isr(scic)) {
if (isci_host_get_state(ihost) != isci_stopped)
tasklet_schedule(&ihost->completion_tasklet); tasklet_schedule(&ihost->completion_tasklet);
else
dev_dbg(&ihost->pdev->dev,
"%s: controller stopped\n", __func__);
}
}
return IRQ_HANDLED; return IRQ_HANDLED;
} }
...@@ -89,22 +82,10 @@ irqreturn_t isci_intx_isr(int vec, void *data) ...@@ -89,22 +82,10 @@ irqreturn_t isci_intx_isr(int vec, void *data)
for_each_isci_host(ihost, pdev) { for_each_isci_host(ihost, pdev) {
struct scic_sds_controller *scic = ihost->core_controller; struct scic_sds_controller *scic = ihost->core_controller;
if (isci_host_get_state(ihost) != isci_starting) {
if (scic_sds_controller_isr(scic)) { if (scic_sds_controller_isr(scic)) {
if (isci_host_get_state(ihost) != isci_stopped)
tasklet_schedule(&ihost->completion_tasklet); tasklet_schedule(&ihost->completion_tasklet);
else
dev_dbg(&ihost->pdev->dev,
"%s: controller stopped\n",
__func__);
ret = IRQ_HANDLED; ret = IRQ_HANDLED;
} }
} else
dev_warn(&ihost->pdev->dev,
"%s: get_handler_methods failed, "
"ihost->status = 0x%x\n",
__func__,
isci_host_get_state(ihost));
} }
return ret; return ret;
} }
...@@ -118,26 +99,19 @@ irqreturn_t isci_intx_isr(int vec, void *data) ...@@ -118,26 +99,19 @@ irqreturn_t isci_intx_isr(int vec, void *data)
* core library. * core library.
* *
*/ */
void isci_host_start_complete( void isci_host_start_complete(struct isci_host *ihost, enum sci_status completion_status)
struct isci_host *isci_host,
enum sci_status completion_status)
{ {
if (completion_status == SCI_SUCCESS) { if (completion_status != SCI_SUCCESS)
dev_dbg(&isci_host->pdev->dev, dev_info(&ihost->pdev->dev,
"%s: completion_status: SCI_SUCCESS\n", __func__); "controller start timed out, continuing...\n");
isci_host_change_state(isci_host, isci_ready); isci_host_change_state(ihost, isci_ready);
complete_all(&isci_host->start_complete); clear_bit(IHOST_START_PENDING, &ihost->flags);
} else wake_up(&ihost->eventq);
dev_err(&isci_host->pdev->dev,
"controller start failed with "
"completion_status = 0x%x;",
completion_status);
} }
int isci_host_scan_finished(struct Scsi_Host *shost, unsigned long time) int isci_host_scan_finished(struct Scsi_Host *shost, unsigned long time)
{ {
struct isci_host *isci_host = isci_host_from_sas_ha(SHOST_TO_SAS_HA(shost)); struct isci_host *ihost = isci_host_from_sas_ha(SHOST_TO_SAS_HA(shost));
/** /**
* check interrupt_handler's status and call completion_handler if true, * check interrupt_handler's status and call completion_handler if true,
...@@ -148,61 +122,44 @@ int isci_host_scan_finished(struct Scsi_Host *shost, unsigned long time) ...@@ -148,61 +122,44 @@ int isci_host_scan_finished(struct Scsi_Host *shost, unsigned long time)
* continue to return zero from thee scan_finished routine until * continue to return zero from thee scan_finished routine until
* the scic_cb_controller_start_complete() call comes from the core. * the scic_cb_controller_start_complete() call comes from the core.
**/ **/
if (scic_sds_controller_isr(isci_host->core_controller)) if (scic_sds_controller_isr(ihost->core_controller))
scic_sds_controller_completion_handler(isci_host->core_controller); scic_sds_controller_completion_handler(ihost->core_controller);
if (isci_starting == isci_host_get_state(isci_host) if (test_bit(IHOST_START_PENDING, &ihost->flags) && time < HZ*10) {
&& time < (HZ * 10)) { dev_dbg(&ihost->pdev->dev,
dev_dbg(&isci_host->pdev->dev, "%s: ihost->status = %d, time = %ld\n",
"%s: isci_host->status = %d, time = %ld\n", __func__, isci_host_get_state(ihost), time);
__func__, isci_host_get_state(isci_host), time);
return 0; return 0;
} }
dev_dbg(&isci_host->pdev->dev, dev_dbg(&ihost->pdev->dev,
"%s: isci_host->status = %d, time = %ld\n", "%s: ihost->status = %d, time = %ld\n",
__func__, isci_host_get_state(isci_host), time); __func__, isci_host_get_state(ihost), time);
scic_controller_enable_interrupts(isci_host->core_controller); scic_controller_enable_interrupts(ihost->core_controller);
return 1; return 1;
} }
/**
* isci_host_scan_start() - This function is one of the SCSI Host Template
* function, called by the SCSI mid layer berfore a target scan begins. The
* core library controller start routine is called from here.
* @shost: This parameter specifies the SCSI host to be scanned
*
*/
void isci_host_scan_start(struct Scsi_Host *shost) void isci_host_scan_start(struct Scsi_Host *shost)
{ {
struct isci_host *isci_host; struct isci_host *ihost = isci_host_from_sas_ha(SHOST_TO_SAS_HA(shost));
struct scic_sds_controller *scic = ihost->core_controller;
isci_host = isci_host_from_sas_ha(SHOST_TO_SAS_HA(shost)); unsigned long tmo = scic_controller_get_suggested_start_timeout(scic);
isci_host_change_state(isci_host, isci_starting);
scic_controller_disable_interrupts(isci_host->core_controller); set_bit(IHOST_START_PENDING, &ihost->flags);
init_completion(&isci_host->start_complete); scic_controller_disable_interrupts(ihost->core_controller);
scic_controller_start( scic_controller_start(scic, tmo);
isci_host->core_controller,
scic_controller_get_suggested_start_timeout(
isci_host->core_controller)
);
} }
void isci_host_stop_complete( void isci_host_stop_complete(struct isci_host *ihost, enum sci_status completion_status)
struct isci_host *isci_host,
enum sci_status completion_status)
{ {
isci_host_change_state(isci_host, isci_stopped); isci_host_change_state(ihost, isci_stopped);
scic_controller_disable_interrupts( scic_controller_disable_interrupts(ihost->core_controller);
isci_host->core_controller clear_bit(IHOST_STOP_PENDING, &ihost->flags);
); wake_up(&ihost->eventq);
complete(&isci_host->stop_complete);
} }
static struct coherent_memory_info *isci_host_alloc_mdl_struct( static struct coherent_memory_info *isci_host_alloc_mdl_struct(
...@@ -370,31 +327,26 @@ static void isci_host_completion_routine(unsigned long data) ...@@ -370,31 +327,26 @@ static void isci_host_completion_routine(unsigned long data)
} }
void isci_host_deinit( void isci_host_deinit(struct isci_host *ihost)
struct isci_host *isci_host)
{ {
struct scic_sds_controller *scic = ihost->core_controller;
int i; int i;
isci_host_change_state(isci_host, isci_stopping); isci_host_change_state(ihost, isci_stopping);
for (i = 0; i < SCI_MAX_PORTS; i++) { for (i = 0; i < SCI_MAX_PORTS; i++) {
struct isci_port *port = &isci_host->isci_ports[i]; struct isci_port *port = &ihost->isci_ports[i];
struct isci_remote_device *device, *tmpdev; struct isci_remote_device *idev, *d;
list_for_each_entry_safe(device, tmpdev,
&port->remote_dev_list, node) { list_for_each_entry_safe(idev, d, &port->remote_dev_list, node) {
isci_remote_device_change_state(device, isci_stopping); isci_remote_device_change_state(idev, isci_stopping);
isci_remote_device_stop(device); isci_remote_device_stop(idev);
} }
} }
/* stop the comtroller and wait for completion. */ set_bit(IHOST_STOP_PENDING, &ihost->flags);
init_completion(&isci_host->stop_complete); scic_controller_stop(scic, SCIC_CONTROLLER_STOP_TIMEOUT);
scic_controller_stop( wait_for_stop(ihost);
isci_host->core_controller, scic_controller_reset(scic);
SCIC_CONTROLLER_STOP_TIMEOUT
);
wait_for_completion(&isci_host->stop_complete);
/* next, reset the controller. */
scic_controller_reset(isci_host->core_controller);
} }
static int isci_verify_firmware(const struct firmware *fw, static int isci_verify_firmware(const struct firmware *fw,
...@@ -506,6 +458,7 @@ int isci_host_init(struct isci_host *isci_host) ...@@ -506,6 +458,7 @@ int isci_host_init(struct isci_host *isci_host)
spin_lock_init(&isci_host->state_lock); spin_lock_init(&isci_host->state_lock);
spin_lock_init(&isci_host->scic_lock); spin_lock_init(&isci_host->scic_lock);
spin_lock_init(&isci_host->queue_lock); spin_lock_init(&isci_host->queue_lock);
init_waitqueue_head(&isci_host->eventq);
isci_host_change_state(isci_host, isci_starting); isci_host_change_state(isci_host, isci_starting);
isci_host->can_queue = ISCI_CAN_QUEUE_VAL; isci_host->can_queue = ISCI_CAN_QUEUE_VAL;
......
...@@ -109,13 +109,15 @@ struct isci_host { ...@@ -109,13 +109,15 @@ struct isci_host {
u8 sas_addr[SAS_ADDR_SIZE]; u8 sas_addr[SAS_ADDR_SIZE];
enum isci_status status; enum isci_status status;
#define IHOST_START_PENDING 0
#define IHOST_STOP_PENDING 1
unsigned long flags;
wait_queue_head_t eventq;
struct Scsi_Host *shost; struct Scsi_Host *shost;
struct tasklet_struct completion_tasklet; struct tasklet_struct completion_tasklet;
struct list_head mdl_struct_list; struct list_head mdl_struct_list;
struct list_head requests_to_complete; struct list_head requests_to_complete;
struct list_head requests_to_abort; struct list_head requests_to_abort;
struct completion stop_complete;
struct completion start_complete;
spinlock_t scic_lock; spinlock_t scic_lock;
struct isci_host *next; struct isci_host *next;
}; };
...@@ -202,6 +204,17 @@ static inline void isci_host_can_dequeue( ...@@ -202,6 +204,17 @@ static inline void isci_host_can_dequeue(
spin_unlock_irqrestore(&isci_host->queue_lock, flags); spin_unlock_irqrestore(&isci_host->queue_lock, flags);
} }
static inline void wait_for_start(struct isci_host *ihost)
{
wait_event(ihost->eventq, !test_bit(IHOST_START_PENDING, &ihost->flags));
}
static inline void wait_for_stop(struct isci_host *ihost)
{
wait_event(ihost->eventq, !test_bit(IHOST_STOP_PENDING, &ihost->flags));
}
/** /**
* isci_host_from_sas_ha() - This accessor retrieves the isci_host object * isci_host_from_sas_ha() - This accessor retrieves the isci_host object
* reference from the Linux sas_ha_struct reference. * reference from the Linux sas_ha_struct reference.
......
...@@ -507,6 +507,8 @@ int isci_remote_device_found(struct domain_device *domain_dev) ...@@ -507,6 +507,8 @@ int isci_remote_device_found(struct domain_device *domain_dev)
dev_dbg(&isci_host->pdev->dev, dev_dbg(&isci_host->pdev->dev,
"%s: domain_device = %p\n", __func__, domain_dev); "%s: domain_device = %p\n", __func__, domain_dev);
wait_for_start(isci_host);
sas_port = domain_dev->port; sas_port = domain_dev->port;
sas_phy = list_first_entry(&sas_port->phy_list, struct asd_sas_phy, sas_phy = list_first_entry(&sas_port->phy_list, struct asd_sas_phy,
port_phy_el); port_phy_el);
...@@ -560,8 +562,6 @@ int isci_remote_device_found(struct domain_device *domain_dev) ...@@ -560,8 +562,6 @@ int isci_remote_device_found(struct domain_device *domain_dev)
return -ENODEV; return -ENODEV;
} }
wait_for_completion(&isci_host->start_complete);
return 0; return 0;
} }
/** /**
......
...@@ -164,8 +164,7 @@ int isci_task_execute_task(struct sas_task *task, int num, gfp_t gfp_flags) ...@@ -164,8 +164,7 @@ int isci_task_execute_task(struct sas_task *task, int num, gfp_t gfp_flags)
* for the quiesce spinlock. * for the quiesce spinlock.
*/ */
if (isci_host_get_state(isci_host) == isci_starting || if ((device && ((isci_remote_device_get_state(device) == isci_ready) ||
(device && ((isci_remote_device_get_state(device) == isci_ready) ||
(isci_remote_device_get_state(device) == isci_host_quiesce)))) { (isci_remote_device_get_state(device) == isci_host_quiesce)))) {
/* Forces a retry from scsi mid layer. */ /* Forces a retry from scsi mid layer. */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment