scsi: cxlflash: Remove commmands from pending list on timeout
authorUma Krishnan <ukrishn@linux.vnet.ibm.com>
Mon, 26 Mar 2018 16:35:34 +0000 (11:35 -0500)
committerMartin K. Petersen <martin.petersen@oracle.com>
Wed, 18 Apr 2018 23:32:50 +0000 (19:32 -0400)
The following Oops can occur if an internal command sent to the AFU does not
complete within the timeout:

[c000000ff101b810c008000016020d94 term_mc+0xfc/0x1b0 [cxlflash]
[c000000ff101b8a0c008000016020fb0 term_afu+0x168/0x280 [cxlflash]
[c000000ff101b930c0080000160232ec cxlflash_pci_error_detected+0x184/0x230
                                       [cxlflash]
[c000000ff101b9e0c00800000d95d468 cxl_vphb_error_detected+0x90/0x150[cxl]
[c000000ff101ba20c00800000d95f27c cxl_pci_error_detected+0xa4/0x240 [cxl]
[c000000ff101bac0c00000000003eaf8 eeh_report_error+0xd8/0x1b0
[c000000ff101bb20c00000000003d0b8 eeh_pe_dev_traverse+0x98/0x170
[c000000ff101bbb0c00000000003f438 eeh_handle_normal_event+0x198/0x580
[c000000ff101bc60c00000000003fba4 eeh_handle_event+0x2a4/0x338
[c000000ff101bd10c0000000000400b8 eeh_event_handler+0x1f8/0x200
[c000000ff101bdc0c00000000013da48 kthread+0x1a8/0x1b0
[c000000ff101be30c00000000000b528 ret_from_kernel_thread+0x5c/0xb4

When an internal command times out, the command buffer is freed while it is
still in the pending commands list of the context. This corrupts the list and
when the context is cleaned up, a crash is encountered.

To resolve this issue, when an AFU command or TMF command times out, the
command should be deleted from the hardware queue pending command list before
freeing the buffer.

Signed-off-by: Uma Krishnan <ukrishn@linux.vnet.ibm.com>
Acked-by: Matthew R. Ochs <mrochs@linux.vnet.ibm.com>
Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
drivers/scsi/cxlflash/main.c

index dfe76485aab680660d9e4baea8fa79517628ff60..c9203282d943621b0d770ee46e40cb768cbc0a3e 100644 (file)
@@ -473,6 +473,7 @@ static int send_tmf(struct cxlflash_cfg *cfg, struct scsi_device *sdev,
        struct afu_cmd *cmd = NULL;
        struct device *dev = &cfg->dev->dev;
        struct hwq *hwq = get_hwq(afu, PRIMARY_HWQ);
+       bool needs_deletion = false;
        char *buf = NULL;
        ulong lock_flags;
        int rc = 0;
@@ -527,6 +528,7 @@ static int send_tmf(struct cxlflash_cfg *cfg, struct scsi_device *sdev,
        if (!to) {
                dev_err(dev, "%s: TMF timed out\n", __func__);
                rc = -ETIMEDOUT;
+               needs_deletion = true;
        } else if (cmd->cmd_aborted) {
                dev_err(dev, "%s: TMF aborted\n", __func__);
                rc = -EAGAIN;
@@ -537,6 +539,12 @@ static int send_tmf(struct cxlflash_cfg *cfg, struct scsi_device *sdev,
        }
        cfg->tmf_active = false;
        spin_unlock_irqrestore(&cfg->tmf_slock, lock_flags);
+
+       if (needs_deletion) {
+               spin_lock_irqsave(&hwq->hsq_slock, lock_flags);
+               list_del(&cmd->list);
+               spin_unlock_irqrestore(&hwq->hsq_slock, lock_flags);
+       }
 out:
        kfree(buf);
        return rc;
@@ -2284,6 +2292,7 @@ static int send_afu_cmd(struct afu *afu, struct sisl_ioarcb *rcb)
        struct device *dev = &cfg->dev->dev;
        struct afu_cmd *cmd = NULL;
        struct hwq *hwq = get_hwq(afu, PRIMARY_HWQ);
+       ulong lock_flags;
        char *buf = NULL;
        int rc = 0;
        int nretry = 0;
@@ -2329,6 +2338,11 @@ retry:
        case -ETIMEDOUT:
                rc = afu->context_reset(hwq);
                if (rc) {
+                       /* Delete the command from pending_cmds list */
+                       spin_lock_irqsave(&hwq->hsq_slock, lock_flags);
+                       list_del(&cmd->list);
+                       spin_unlock_irqrestore(&hwq->hsq_slock, lock_flags);
+
                        cxlflash_schedule_async_reset(cfg);
                        break;
                }