Commit 31500e90 authored by Mike Christie's avatar Mike Christie Committed by Martin K. Petersen

scsi: iscsi: Fix session removal on shutdown

When the system is shutting down, iscsid is not running so we will not get
a response to the ISCSI_ERR_INVALID_HOST error event. The system shutdown
will then hang waiting on userspace to remove the session.

This has libiscsi force the destruction of the session from the kernel when
iscsi_host_remove() is called from a driver's shutdown callout.

This fixes a regression added in qedi boot with commit d1f2ce77 ("scsi:
qedi: Fix host removal with running sessions") which made qedi use the
common session removal function that waits on userspace instead of rolling
its own kernel based removal.

Link: https://lore.kernel.org/r/20220616222738.5722-7-michael.christie@oracle.com
Fixes: d1f2ce77 ("scsi: qedi: Fix host removal with running sessions")
Tested-by: default avatarNilesh Javali <njavali@marvell.com>
Reviewed-by: default avatarLee Duncan <lduncan@suse.com>
Reviewed-by: default avatarNilesh Javali <njavali@marvell.com>
Signed-off-by: default avatarMike Christie <michael.christie@oracle.com>
Signed-off-by: default avatarMartin K. Petersen <martin.petersen@oracle.com>
parent 7bf01eb0
...@@ -568,7 +568,7 @@ static void iscsi_iser_session_destroy(struct iscsi_cls_session *cls_session) ...@@ -568,7 +568,7 @@ static void iscsi_iser_session_destroy(struct iscsi_cls_session *cls_session)
struct Scsi_Host *shost = iscsi_session_to_shost(cls_session); struct Scsi_Host *shost = iscsi_session_to_shost(cls_session);
iscsi_session_teardown(cls_session); iscsi_session_teardown(cls_session);
iscsi_host_remove(shost); iscsi_host_remove(shost, false);
iscsi_host_free(shost); iscsi_host_free(shost);
} }
...@@ -685,7 +685,7 @@ iscsi_iser_session_create(struct iscsi_endpoint *ep, ...@@ -685,7 +685,7 @@ iscsi_iser_session_create(struct iscsi_endpoint *ep,
return cls_session; return cls_session;
remove_host: remove_host:
iscsi_host_remove(shost); iscsi_host_remove(shost, false);
free_host: free_host:
iscsi_host_free(shost); iscsi_host_free(shost);
return NULL; return NULL;
......
...@@ -5745,7 +5745,7 @@ static void beiscsi_remove(struct pci_dev *pcidev) ...@@ -5745,7 +5745,7 @@ static void beiscsi_remove(struct pci_dev *pcidev)
cancel_work_sync(&phba->sess_work); cancel_work_sync(&phba->sess_work);
beiscsi_iface_destroy_default(phba); beiscsi_iface_destroy_default(phba);
iscsi_host_remove(phba->shost); iscsi_host_remove(phba->shost, false);
beiscsi_disable_port(phba, 1); beiscsi_disable_port(phba, 1);
/* after cancelling boot_work */ /* after cancelling boot_work */
......
...@@ -909,7 +909,7 @@ void bnx2i_free_hba(struct bnx2i_hba *hba) ...@@ -909,7 +909,7 @@ void bnx2i_free_hba(struct bnx2i_hba *hba)
{ {
struct Scsi_Host *shost = hba->shost; struct Scsi_Host *shost = hba->shost;
iscsi_host_remove(shost); iscsi_host_remove(shost, false);
INIT_LIST_HEAD(&hba->ep_ofld_list); INIT_LIST_HEAD(&hba->ep_ofld_list);
INIT_LIST_HEAD(&hba->ep_active_list); INIT_LIST_HEAD(&hba->ep_active_list);
INIT_LIST_HEAD(&hba->ep_destroy_list); INIT_LIST_HEAD(&hba->ep_destroy_list);
......
...@@ -328,7 +328,7 @@ void cxgbi_hbas_remove(struct cxgbi_device *cdev) ...@@ -328,7 +328,7 @@ void cxgbi_hbas_remove(struct cxgbi_device *cdev)
chba = cdev->hbas[i]; chba = cdev->hbas[i];
if (chba) { if (chba) {
cdev->hbas[i] = NULL; cdev->hbas[i] = NULL;
iscsi_host_remove(chba->shost); iscsi_host_remove(chba->shost, false);
pci_dev_put(cdev->pdev); pci_dev_put(cdev->pdev);
iscsi_host_free(chba->shost); iscsi_host_free(chba->shost);
} }
......
...@@ -898,7 +898,7 @@ iscsi_sw_tcp_session_create(struct iscsi_endpoint *ep, uint16_t cmds_max, ...@@ -898,7 +898,7 @@ iscsi_sw_tcp_session_create(struct iscsi_endpoint *ep, uint16_t cmds_max,
remove_session: remove_session:
iscsi_session_teardown(cls_session); iscsi_session_teardown(cls_session);
remove_host: remove_host:
iscsi_host_remove(shost); iscsi_host_remove(shost, false);
free_host: free_host:
iscsi_host_free(shost); iscsi_host_free(shost);
return NULL; return NULL;
...@@ -915,7 +915,7 @@ static void iscsi_sw_tcp_session_destroy(struct iscsi_cls_session *cls_session) ...@@ -915,7 +915,7 @@ static void iscsi_sw_tcp_session_destroy(struct iscsi_cls_session *cls_session)
iscsi_tcp_r2tpool_free(cls_session->dd_data); iscsi_tcp_r2tpool_free(cls_session->dd_data);
iscsi_session_teardown(cls_session); iscsi_session_teardown(cls_session);
iscsi_host_remove(shost); iscsi_host_remove(shost, false);
iscsi_host_free(shost); iscsi_host_free(shost);
} }
......
...@@ -2832,11 +2832,12 @@ static void iscsi_notify_host_removed(struct iscsi_cls_session *cls_session) ...@@ -2832,11 +2832,12 @@ static void iscsi_notify_host_removed(struct iscsi_cls_session *cls_session)
/** /**
* iscsi_host_remove - remove host and sessions * iscsi_host_remove - remove host and sessions
* @shost: scsi host * @shost: scsi host
* @is_shutdown: true if called from a driver shutdown callout
* *
* If there are any sessions left, this will initiate the removal and wait * If there are any sessions left, this will initiate the removal and wait
* for the completion. * for the completion.
*/ */
void iscsi_host_remove(struct Scsi_Host *shost) void iscsi_host_remove(struct Scsi_Host *shost, bool is_shutdown)
{ {
struct iscsi_host *ihost = shost_priv(shost); struct iscsi_host *ihost = shost_priv(shost);
unsigned long flags; unsigned long flags;
...@@ -2845,7 +2846,11 @@ void iscsi_host_remove(struct Scsi_Host *shost) ...@@ -2845,7 +2846,11 @@ void iscsi_host_remove(struct Scsi_Host *shost)
ihost->state = ISCSI_HOST_REMOVED; ihost->state = ISCSI_HOST_REMOVED;
spin_unlock_irqrestore(&ihost->lock, flags); spin_unlock_irqrestore(&ihost->lock, flags);
iscsi_host_for_each_session(shost, iscsi_notify_host_removed); if (!is_shutdown)
iscsi_host_for_each_session(shost, iscsi_notify_host_removed);
else
iscsi_host_for_each_session(shost, iscsi_force_destroy_session);
wait_event_interruptible(ihost->session_removal_wq, wait_event_interruptible(ihost->session_removal_wq,
ihost->num_sessions == 0); ihost->num_sessions == 0);
if (signal_pending(current)) if (signal_pending(current))
......
...@@ -2414,9 +2414,12 @@ static void __qedi_remove(struct pci_dev *pdev, int mode) ...@@ -2414,9 +2414,12 @@ static void __qedi_remove(struct pci_dev *pdev, int mode)
int rval; int rval;
u16 retry = 10; u16 retry = 10;
if (mode == QEDI_MODE_NORMAL || mode == QEDI_MODE_SHUTDOWN) { if (mode == QEDI_MODE_NORMAL)
iscsi_host_remove(qedi->shost); iscsi_host_remove(qedi->shost, false);
else if (mode == QEDI_MODE_SHUTDOWN)
iscsi_host_remove(qedi->shost, true);
if (mode == QEDI_MODE_NORMAL || mode == QEDI_MODE_SHUTDOWN) {
if (qedi->tmf_thread) { if (qedi->tmf_thread) {
destroy_workqueue(qedi->tmf_thread); destroy_workqueue(qedi->tmf_thread);
qedi->tmf_thread = NULL; qedi->tmf_thread = NULL;
...@@ -2791,7 +2794,7 @@ static int __qedi_probe(struct pci_dev *pdev, int mode) ...@@ -2791,7 +2794,7 @@ static int __qedi_probe(struct pci_dev *pdev, int mode)
#ifdef CONFIG_DEBUG_FS #ifdef CONFIG_DEBUG_FS
qedi_dbg_host_exit(&qedi->dbg_ctx); qedi_dbg_host_exit(&qedi->dbg_ctx);
#endif #endif
iscsi_host_remove(qedi->shost); iscsi_host_remove(qedi->shost, false);
stop_iscsi_func: stop_iscsi_func:
qedi_ops->stop(qedi->cdev); qedi_ops->stop(qedi->cdev);
stop_slowpath: stop_slowpath:
......
...@@ -411,7 +411,7 @@ extern int iscsi_host_add(struct Scsi_Host *shost, struct device *pdev); ...@@ -411,7 +411,7 @@ extern int iscsi_host_add(struct Scsi_Host *shost, struct device *pdev);
extern struct Scsi_Host *iscsi_host_alloc(struct scsi_host_template *sht, extern struct Scsi_Host *iscsi_host_alloc(struct scsi_host_template *sht,
int dd_data_size, int dd_data_size,
bool xmit_can_sleep); bool xmit_can_sleep);
extern void iscsi_host_remove(struct Scsi_Host *shost); extern void iscsi_host_remove(struct Scsi_Host *shost, bool is_shutdown);
extern void iscsi_host_free(struct Scsi_Host *shost); extern void iscsi_host_free(struct Scsi_Host *shost);
extern int iscsi_target_alloc(struct scsi_target *starget); extern int iscsi_target_alloc(struct scsi_target *starget);
extern int iscsi_host_get_max_scsi_cmds(struct Scsi_Host *shost, extern int iscsi_host_get_max_scsi_cmds(struct Scsi_Host *shost,
......
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