Move the duplicated code from FC LLDs to SCSI FC transport class.
Acked-by: James Smart <james.smart@emulex.com>
Acked-by: Giridhar Malavali <giridhar.malavali@qlogic.com>
Acked-by: Abhijeet Joglekar <abjoglek@cisco.com>
Signed-off-by: Christof Schmitt <christof.schmitt@de.ibm.com>
Signed-off-by: James Bottomley <James.Bottomley@suse.de>
 
 }
 
-static void fnic_block_error_handler(struct scsi_cmnd *sc)
-{
-       struct Scsi_Host *shost = sc->device->host;
-       struct fc_rport *rport = starget_to_rport(scsi_target(sc->device));
-       unsigned long flags;
-
-       spin_lock_irqsave(shost->host_lock, flags);
-       while (rport->port_state == FC_PORTSTATE_BLOCKED) {
-               spin_unlock_irqrestore(shost->host_lock, flags);
-               msleep(1000);
-               spin_lock_irqsave(shost->host_lock, flags);
-       }
-       spin_unlock_irqrestore(shost->host_lock, flags);
-
-}
-
 /*
  * This function is exported to SCSI for sending abort cmnds.
  * A SCSI IO is represented by a io_req in the driver.
        DECLARE_COMPLETION_ONSTACK(tm_done);
 
        /* Wait for rport to unblock */
-       fnic_block_error_handler(sc);
+       fc_block_scsi_eh(sc);
 
        /* Get local-port, check ready and link up */
        lp = shost_priv(sc->device->host);
        DECLARE_COMPLETION_ONSTACK(tm_done);
 
        /* Wait for rport to unblock */
-       fnic_block_error_handler(sc);
+       fc_block_scsi_eh(sc);
 
        /* Get local-port, check ready and link up */
        lp = shost_priv(sc->device->host);
 
        return 0;
 }
 
-/**
- * lpfc_block_error_handler - Routine to block error  handler
- * @cmnd: Pointer to scsi_cmnd data structure.
- *
- *  This routine blocks execution till fc_rport state is not FC_PORSTAT_BLCOEKD.
- **/
-static void
-lpfc_block_error_handler(struct scsi_cmnd *cmnd)
-{
-       struct Scsi_Host *shost = cmnd->device->host;
-       struct fc_rport *rport = starget_to_rport(scsi_target(cmnd->device));
-
-       spin_lock_irq(shost->host_lock);
-       while (rport->port_state == FC_PORTSTATE_BLOCKED) {
-               spin_unlock_irq(shost->host_lock);
-               msleep(1000);
-               spin_lock_irq(shost->host_lock);
-       }
-       spin_unlock_irq(shost->host_lock);
-       return;
-}
-
 /**
  * lpfc_abort_handler - scsi_host_template eh_abort_handler entry point
  * @cmnd: Pointer to scsi_cmnd data structure.
        int ret = SUCCESS;
        DECLARE_WAIT_QUEUE_HEAD_ONSTACK(waitq);
 
-       lpfc_block_error_handler(cmnd);
+       fc_block_scsi_eh(cmnd);
        lpfc_cmd = (struct lpfc_scsi_buf *)cmnd->host_scribble;
        BUG_ON(!lpfc_cmd);
 
        struct lpfc_scsi_event_header scsi_event;
        int status;
 
-       lpfc_block_error_handler(cmnd);
+       fc_block_scsi_eh(cmnd);
 
        status = lpfc_chk_tgt_mapped(vport, cmnd);
        if (status == FAILED) {
        struct lpfc_scsi_event_header scsi_event;
        int status;
 
-       lpfc_block_error_handler(cmnd);
+       fc_block_scsi_eh(cmnd);
 
        status = lpfc_chk_tgt_mapped(vport, cmnd);
        if (status == FAILED) {
        fc_host_post_vendor_event(shost, fc_get_event_number(),
                sizeof(scsi_event), (char *)&scsi_event, LPFC_NL_VENDOR_ID);
 
-       lpfc_block_error_handler(cmnd);
+       fc_block_scsi_eh(cmnd);
 
        /*
         * Since the driver manages a single bus device, reset all
 
        spin_unlock_irqrestore(&ha->hardware_lock, flags);
 }
 
-static void
-qla2x00_block_error_handler(struct scsi_cmnd *cmnd)
-{
-       struct Scsi_Host *shost = cmnd->device->host;
-       struct fc_rport *rport = starget_to_rport(scsi_target(cmnd->device));
-       unsigned long flags;
-
-       spin_lock_irqsave(shost->host_lock, flags);
-       while (rport->port_state == FC_PORTSTATE_BLOCKED) {
-               spin_unlock_irqrestore(shost->host_lock, flags);
-               msleep(1000);
-               spin_lock_irqsave(shost->host_lock, flags);
-       }
-       spin_unlock_irqrestore(shost->host_lock, flags);
-       return;
-}
-
 /**************************************************************************
 * qla2xxx_eh_abort
 *
        struct req_que *req = vha->req;
        srb_t *spt;
 
-       qla2x00_block_error_handler(cmd);
+       fc_block_scsi_eh(cmd);
 
        if (!CMD_SP(cmd))
                return SUCCESS;
        fc_port_t *fcport = (struct fc_port *) cmd->device->hostdata;
        int err;
 
-       qla2x00_block_error_handler(cmd);
+       fc_block_scsi_eh(cmd);
 
        if (!fcport)
                return FAILED;
        unsigned long serial;
        srb_t *sp = (srb_t *) CMD_SP(cmd);
 
-       qla2x00_block_error_handler(cmd);
+       fc_block_scsi_eh(cmd);
 
        id = cmd->device->id;
        lun = cmd->device->lun;
        srb_t *sp = (srb_t *) CMD_SP(cmd);
        scsi_qla_host_t *base_vha = pci_get_drvdata(ha->pdev);
 
-       qla2x00_block_error_handler(cmd);
+       fc_block_scsi_eh(cmd);
 
        id = cmd->device->id;
        lun = cmd->device->lun;
 
  */
 #include <linux/module.h>
 #include <linux/init.h>
+#include <linux/delay.h>
 #include <scsi/scsi_device.h>
 #include <scsi/scsi_host.h>
 #include <scsi/scsi_transport.h>
        spin_unlock_irqrestore(shost->host_lock, flags);
 }
 
+/**
+ * fc_block_scsi_eh - Block SCSI eh thread for blocked fc_rport
+ * @cmnd: SCSI command that scsi_eh is trying to recover
+ *
+ * This routine can be called from a FC LLD scsi_eh callback. It
+ * blocks the scsi_eh thread until the fc_rport leaves the
+ * FC_PORTSTATE_BLOCKED. This is necessary to avoid the scsi_eh
+ * failing recovery actions for blocked rports which would lead to
+ * offlined SCSI devices.
+ */
+void fc_block_scsi_eh(struct scsi_cmnd *cmnd)
+{
+       struct Scsi_Host *shost = cmnd->device->host;
+       struct fc_rport *rport = starget_to_rport(scsi_target(cmnd->device));
+       unsigned long flags;
+
+       spin_lock_irqsave(shost->host_lock, flags);
+       while (rport->port_state == FC_PORTSTATE_BLOCKED) {
+               spin_unlock_irqrestore(shost->host_lock, flags);
+               msleep(1000);
+               spin_lock_irqsave(shost->host_lock, flags);
+       }
+       spin_unlock_irqrestore(shost->host_lock, flags);
+}
+EXPORT_SYMBOL(fc_block_scsi_eh);
 
 /**
  * fc_vport_setup - allocates and creates a FC virtual port.
 
 struct fc_vport *fc_vport_create(struct Scsi_Host *shost, int channel,
                struct fc_vport_identifiers *);
 int fc_vport_terminate(struct fc_vport *vport);
+void fc_block_scsi_eh(struct scsi_cmnd *cmnd);
 
 #endif /* SCSI_TRANSPORT_FC_H */