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
  */
 static int fc_fcp_send_abort(struct fc_fcp_pkt *fsp)
 {
+       int rc;
+
        if (!fsp->seq_ptr)
                return -EINVAL;
 
        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;
 }
 
 /**
                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);
 }
 
 /**
                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;