]> git.karo-electronics.de Git - karo-tx-linux.git/commitdiff
scsi: libfc: don't fail sequence abort for completed exchanges
authorHannes Reinecke <hare@suse.de>
Thu, 13 Oct 2016 13:10:51 +0000 (15:10 +0200)
committerMartin K. Petersen <martin.petersen@oracle.com>
Tue, 8 Nov 2016 22:29:52 +0000 (17:29 -0500)
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 <hare@suse.com>
Acked-by: Johannes Thumshirn <jth@kernel.org>
Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
drivers/scsi/libfc/fc_fcp.c

index 780d9f09a267cc0ec04b52f6d65a16723762ab54..d43c925bace3ad33973ff5c9640a36e390fec921 100644 (file)
@@ -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;