[SCSI] aacraid: Better handling of in-flight events on thread stop
authorBen Collins <bcollins@ubuntu.com>
Mon, 11 Jun 2012 18:16:36 +0000 (14:16 -0400)
committerJames Bottomley <JBottomley@Parallels.com>
Fri, 20 Jul 2012 07:58:43 +0000 (08:58 +0100)
When an error occured that would shut down the driver, some in-flight
events were getting caught up, deadlocking a CPU or two.

Signed-off-by: Ben Collins <bcollins@ubuntu.com>
Acked-by: Achim Leubner <Achim_Leubner@pmc-sierra.com>
Signed-off-by: James Bottomley <JBottomley@Parallels.com>
drivers/scsi/aacraid/linit.c

index fdfc4be..2be612b 100644 (file)
@@ -1089,8 +1089,17 @@ static struct scsi_host_template aac_driver_template = {
 
 static void __aac_shutdown(struct aac_dev * aac)
 {
-       if (aac->aif_thread)
+       if (aac->aif_thread) {
+               int i;
+               /* Clear out events first */
+               for (i = 0; i < (aac->scsi_host_ptr->can_queue + AAC_NUM_MGT_FIB); i++) {
+                       struct fib *fib = &aac->fibs[i];
+                       if (!(fib->hw_fib_va->header.XferState & cpu_to_le32(NoResponseExpected | Async)) &&
+                           (fib->hw_fib_va->header.XferState & cpu_to_le32(ResponseExpected)))
+                               up(&fib->event_wait);
+               }
                kthread_stop(aac->thread);
+       }
        aac_send_shutdown(aac);
        aac_adapter_disable_int(aac);
        free_irq(aac->pdev->irq, aac);
@@ -1191,6 +1200,7 @@ static int __devinit aac_probe_one(struct pci_dev *pdev,
        if (IS_ERR(aac->thread)) {
                printk(KERN_ERR "aacraid: Unable to create command thread.\n");
                error = PTR_ERR(aac->thread);
+               aac->thread = NULL;
                goto out_deinit;
        }