diff options
Diffstat (limited to 'drivers/scsi/mpt3sas/mpt3sas_scsih.c')
| -rw-r--r-- | drivers/scsi/mpt3sas/mpt3sas_scsih.c | 23 | 
1 files changed, 17 insertions, 6 deletions
| diff --git a/drivers/scsi/mpt3sas/mpt3sas_scsih.c b/drivers/scsi/mpt3sas/mpt3sas_scsih.c index 74fca184dba9..a1cb0236c550 100644 --- a/drivers/scsi/mpt3sas/mpt3sas_scsih.c +++ b/drivers/scsi/mpt3sas/mpt3sas_scsih.c @@ -2835,7 +2835,8 @@ scsih_abort(struct scsi_cmnd *scmd)  	_scsih_tm_display_info(ioc, scmd);  	sas_device_priv_data = scmd->device->hostdata; -	if (!sas_device_priv_data || !sas_device_priv_data->sas_target) { +	if (!sas_device_priv_data || !sas_device_priv_data->sas_target || +	    ioc->remove_host) {  		sdev_printk(KERN_INFO, scmd->device,  			"device been deleted! scmd(%p)\n", scmd);  		scmd->result = DID_NO_CONNECT << 16; @@ -2898,7 +2899,8 @@ scsih_dev_reset(struct scsi_cmnd *scmd)  	_scsih_tm_display_info(ioc, scmd);  	sas_device_priv_data = scmd->device->hostdata; -	if (!sas_device_priv_data || !sas_device_priv_data->sas_target) { +	if (!sas_device_priv_data || !sas_device_priv_data->sas_target || +	    ioc->remove_host) {  		sdev_printk(KERN_INFO, scmd->device,  			"device been deleted! scmd(%p)\n", scmd);  		scmd->result = DID_NO_CONNECT << 16; @@ -2961,7 +2963,8 @@ scsih_target_reset(struct scsi_cmnd *scmd)  	_scsih_tm_display_info(ioc, scmd);  	sas_device_priv_data = scmd->device->hostdata; -	if (!sas_device_priv_data || !sas_device_priv_data->sas_target) { +	if (!sas_device_priv_data || !sas_device_priv_data->sas_target || +	    ioc->remove_host) {  		starget_printk(KERN_INFO, starget, "target been deleted! scmd(%p)\n",  			scmd);  		scmd->result = DID_NO_CONNECT << 16; @@ -3019,7 +3022,7 @@ scsih_host_reset(struct scsi_cmnd *scmd)  	    ioc->name, scmd);  	scsi_print_command(scmd); -	if (ioc->is_driver_loading) { +	if (ioc->is_driver_loading || ioc->remove_host) {  		pr_info(MPT3SAS_FMT "Blocking the host reset\n",  		    ioc->name);  		r = FAILED; @@ -4453,7 +4456,7 @@ _scsih_flush_running_cmds(struct MPT3SAS_ADAPTER *ioc)  		st = scsi_cmd_priv(scmd);  		mpt3sas_base_clear_st(ioc, st);  		scsi_dma_unmap(scmd); -		if (ioc->pci_error_recovery) +		if (ioc->pci_error_recovery || ioc->remove_host)  			scmd->result = DID_NO_CONNECT << 16;  		else  			scmd->result = DID_RESET << 16; @@ -9739,6 +9742,10 @@ static void scsih_remove(struct pci_dev *pdev)  	unsigned long flags;  	ioc->remove_host = 1; + +	mpt3sas_wait_for_commands_to_complete(ioc); +	_scsih_flush_running_cmds(ioc); +  	_scsih_fw_event_cleanup_queue(ioc);  	spin_lock_irqsave(&ioc->fw_event_lock, flags); @@ -9815,6 +9822,10 @@ scsih_shutdown(struct pci_dev *pdev)  	unsigned long flags;  	ioc->remove_host = 1; + +	mpt3sas_wait_for_commands_to_complete(ioc); +	_scsih_flush_running_cmds(ioc); +  	_scsih_fw_event_cleanup_queue(ioc);  	spin_lock_irqsave(&ioc->fw_event_lock, flags); @@ -10547,7 +10558,7 @@ _scsih_probe(struct pci_dev *pdev, const struct pci_device_id *id)  	snprintf(ioc->firmware_event_name, sizeof(ioc->firmware_event_name),  	    "fw_event_%s%d", ioc->driver_name, ioc->id);  	ioc->firmware_event_thread = alloc_ordered_workqueue( -	    ioc->firmware_event_name, WQ_MEM_RECLAIM); +	    ioc->firmware_event_name, 0);  	if (!ioc->firmware_event_thread) {  		pr_err(MPT3SAS_FMT "failure at %s:%d/%s()!\n",  		    ioc->name, __FILE__, __LINE__, __func__); | 
