If a sequence should be aborted the exchange might already
be completed (eg if the response is still queued in the rx
queue), so this shouldn't considered as an error.

Signed-off-by: Hannes Reinecke <h...@suse.com>
Acked-by: Johannes Thumshirn <j...@kernel.org>
---
 drivers/scsi/libfc/fc_fcp.c | 40 ++++++++++++++++++++++++++++++----------
 1 file changed, 30 insertions(+), 10 deletions(-)

diff --git a/drivers/scsi/libfc/fc_fcp.c b/drivers/scsi/libfc/fc_fcp.c
index 780d9f0..d43c925 100644
--- a/drivers/scsi/libfc/fc_fcp.c
+++ b/drivers/scsi/libfc/fc_fcp.c
@@ -258,6 +258,17 @@ static void fc_fcp_timer_set(struct fc_fcp_pkt *fsp, 
unsigned long delay)
                mod_timer(&fsp->timer, jiffies + delay);
 }
 
+static void fc_fcp_abort_done(struct fc_fcp_pkt *fsp)
+{
+       fsp->state |= FC_SRB_ABORTED;
+       fsp->state &= ~FC_SRB_ABORT_PENDING;
+
+       if (fsp->wait_for_comp)
+               complete(&fsp->tm_done);
+       else
+               fc_fcp_complete_locked(fsp);
+}
+
 /**
  * fc_fcp_send_abort() - Send an abort for exchanges associated with a
  *                      fcp_pkt
@@ -265,6 +276,8 @@ static void fc_fcp_timer_set(struct fc_fcp_pkt *fsp, 
unsigned long delay)
  */
 static int fc_fcp_send_abort(struct fc_fcp_pkt *fsp)
 {
+       int rc;
+
        if (!fsp->seq_ptr)
                return -EINVAL;
 
@@ -272,7 +285,16 @@ static int fc_fcp_send_abort(struct fc_fcp_pkt *fsp)
        put_cpu();
 
        fsp->state |= FC_SRB_ABORT_PENDING;
-       return fsp->lp->tt.seq_exch_abort(fsp->seq_ptr, 0);
+       rc = fsp->lp->tt.seq_exch_abort(fsp->seq_ptr, 0);
+       /*
+        * ->seq_exch_abort() might return -ENXIO if
+        * the sequence is already completed
+        */
+       if (rc == -ENXIO) {
+               fc_fcp_abort_done(fsp);
+               rc = 0;
+       }
+       return rc;
 }
 
 /**
@@ -729,15 +751,8 @@ static void fc_fcp_abts_resp(struct fc_fcp_pkt *fsp, 
struct fc_frame *fp)
                ba_done = 0;
        }
 
-       if (ba_done) {
-               fsp->state |= FC_SRB_ABORTED;
-               fsp->state &= ~FC_SRB_ABORT_PENDING;
-
-               if (fsp->wait_for_comp)
-                       complete(&fsp->tm_done);
-               else
-                       fc_fcp_complete_locked(fsp);
-       }
+       if (ba_done)
+               fc_fcp_abort_done(fsp);
 }
 
 /**
@@ -1245,6 +1260,11 @@ static int fc_fcp_pkt_abort(struct fc_fcp_pkt *fsp)
                return FAILED;
        }
 
+       if (fsp->state & FC_SRB_ABORTED) {
+               FC_FCP_DBG(fsp, "target abort cmd  completed\n");
+               return SUCCESS;
+       }
+
        init_completion(&fsp->tm_done);
        fsp->wait_for_comp = 1;
 
-- 
1.8.5.6

--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to