aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorHiral Shah <hishah@cisco.com>2014-11-10 15:54:36 -0500
committerChristoph Hellwig <hch@lst.de>2014-11-20 03:11:00 -0500
commit41df7b02db82cf6c14f094757bac3830d10a827f (patch)
tree4d32c8ce1aa9fe73dc269129a4a4c3b5626e90f0
parent0ee7b8714dca511a68826a300cc4966f2e83a6c3 (diff)
Fnic: Fnic Driver crashed with NULL pointer reference
When issuing I/O request, if the I/O completes before returning from fnic_queuecommand(), we may be referencing scsi_cmnd structure that may be freed by interrupt handler. Acquring IO lock would synchronize fnic_queuecommand and interrupt handler. - Increment fnic version from 1.6.0.15 to 1.6.0.16 Signed-off-by: Hiral Shah <hishah@cisco.com> Signed-off-by: Sesidhar Baddela <sebaddel@cisco.com> Signed-off-by: Anil Chintalapati <achintal@cisco.com> Signed-off-by: Christoph Hellwig <hch@lst.de>
-rw-r--r--drivers/scsi/fnic/fnic.h2
-rw-r--r--drivers/scsi/fnic/fnic_scsi.c21
2 files changed, 19 insertions, 4 deletions
diff --git a/drivers/scsi/fnic/fnic.h b/drivers/scsi/fnic/fnic.h
index 977960b03c9f..3b73b96619e2 100644
--- a/drivers/scsi/fnic/fnic.h
+++ b/drivers/scsi/fnic/fnic.h
@@ -39,7 +39,7 @@
39 39
40#define DRV_NAME "fnic" 40#define DRV_NAME "fnic"
41#define DRV_DESCRIPTION "Cisco FCoE HBA Driver" 41#define DRV_DESCRIPTION "Cisco FCoE HBA Driver"
42#define DRV_VERSION "1.6.0.15" 42#define DRV_VERSION "1.6.0.16"
43#define PFX DRV_NAME ": " 43#define PFX DRV_NAME ": "
44#define DFX DRV_NAME "%d: " 44#define DFX DRV_NAME "%d: "
45 45
diff --git a/drivers/scsi/fnic/fnic_scsi.c b/drivers/scsi/fnic/fnic_scsi.c
index 29d23a3c0686..2097de42a147 100644
--- a/drivers/scsi/fnic/fnic_scsi.c
+++ b/drivers/scsi/fnic/fnic_scsi.c
@@ -421,9 +421,10 @@ static int fnic_queuecommand_lck(struct scsi_cmnd *sc, void (*done)(struct scsi_
421 int ret; 421 int ret;
422 u64 cmd_trace; 422 u64 cmd_trace;
423 int sg_count = 0; 423 int sg_count = 0;
424 unsigned long flags; 424 unsigned long flags = 0;
425 unsigned long ptr; 425 unsigned long ptr;
426 struct fc_rport_priv *rdata; 426 struct fc_rport_priv *rdata;
427 spinlock_t *io_lock = NULL;
427 428
428 if (unlikely(fnic_chk_state_flags_locked(fnic, FNIC_FLAGS_IO_BLOCKED))) 429 if (unlikely(fnic_chk_state_flags_locked(fnic, FNIC_FLAGS_IO_BLOCKED)))
429 return SCSI_MLQUEUE_HOST_BUSY; 430 return SCSI_MLQUEUE_HOST_BUSY;
@@ -509,6 +510,13 @@ static int fnic_queuecommand_lck(struct scsi_cmnd *sc, void (*done)(struct scsi_
509 } 510 }
510 } 511 }
511 512
513 /*
514 * Will acquire lock defore setting to IO initialized.
515 */
516
517 io_lock = fnic_io_lock_hash(fnic, sc);
518 spin_lock_irqsave(io_lock, flags);
519
512 /* initialize rest of io_req */ 520 /* initialize rest of io_req */
513 io_req->port_id = rport->port_id; 521 io_req->port_id = rport->port_id;
514 io_req->start_time = jiffies; 522 io_req->start_time = jiffies;
@@ -525,11 +533,9 @@ static int fnic_queuecommand_lck(struct scsi_cmnd *sc, void (*done)(struct scsi_
525 * In case another thread cancelled the request, 533 * In case another thread cancelled the request,
526 * refetch the pointer under the lock. 534 * refetch the pointer under the lock.
527 */ 535 */
528 spinlock_t *io_lock = fnic_io_lock_hash(fnic, sc);
529 FNIC_TRACE(fnic_queuecommand, sc->device->host->host_no, 536 FNIC_TRACE(fnic_queuecommand, sc->device->host->host_no,
530 sc->request->tag, sc, 0, 0, 0, 537 sc->request->tag, sc, 0, 0, 0,
531 (((u64)CMD_FLAGS(sc) << 32) | CMD_STATE(sc))); 538 (((u64)CMD_FLAGS(sc) << 32) | CMD_STATE(sc)));
532 spin_lock_irqsave(io_lock, flags);
533 io_req = (struct fnic_io_req *)CMD_SP(sc); 539 io_req = (struct fnic_io_req *)CMD_SP(sc);
534 CMD_SP(sc) = NULL; 540 CMD_SP(sc) = NULL;
535 CMD_STATE(sc) = FNIC_IOREQ_CMD_COMPLETE; 541 CMD_STATE(sc) = FNIC_IOREQ_CMD_COMPLETE;
@@ -538,6 +544,10 @@ static int fnic_queuecommand_lck(struct scsi_cmnd *sc, void (*done)(struct scsi_
538 fnic_release_ioreq_buf(fnic, io_req, sc); 544 fnic_release_ioreq_buf(fnic, io_req, sc);
539 mempool_free(io_req, fnic->io_req_pool); 545 mempool_free(io_req, fnic->io_req_pool);
540 } 546 }
547 atomic_dec(&fnic->in_flight);
548 /* acquire host lock before returning to SCSI */
549 spin_lock(lp->host->host_lock);
550 return ret;
541 } else { 551 } else {
542 atomic64_inc(&fnic_stats->io_stats.active_ios); 552 atomic64_inc(&fnic_stats->io_stats.active_ios);
543 atomic64_inc(&fnic_stats->io_stats.num_ios); 553 atomic64_inc(&fnic_stats->io_stats.num_ios);
@@ -559,6 +569,11 @@ out:
559 sc->request->tag, sc, io_req, 569 sc->request->tag, sc, io_req,
560 sg_count, cmd_trace, 570 sg_count, cmd_trace,
561 (((u64)CMD_FLAGS(sc) >> 32) | CMD_STATE(sc))); 571 (((u64)CMD_FLAGS(sc) >> 32) | CMD_STATE(sc)));
572
573 /* if only we issued IO, will we have the io lock */
574 if (CMD_FLAGS(sc) & FNIC_IO_INITIALIZED)
575 spin_unlock_irqrestore(io_lock, flags);
576
562 atomic_dec(&fnic->in_flight); 577 atomic_dec(&fnic->in_flight);
563 /* acquire host lock before returning to SCSI */ 578 /* acquire host lock before returning to SCSI */
564 spin_lock(lp->host->host_lock); 579 spin_lock(lp->host->host_lock);