scsi: hisi_sas: check IPTT is valid before using it for v3 hw
authorXiaofei Tan <tanxiaofei@huawei.com>
Fri, 23 Mar 2018 16:05:13 +0000 (00:05 +0800)
committerMartin K. Petersen <martin.petersen@oracle.com>
Wed, 18 Apr 2018 23:32:51 +0000 (19:32 -0400)
There is a bug of v3 hw development version. When AXI error happen, hw
may return an abnormal CQ that IPTT value is 0xffff.  This will cause
IPTT out-of-bounds reference.

This patch adds a check of IPTT in cq_tasklet_v3_hw() and discards
invalid slot. This workaround scheme is just to enhance fault-tolerance
of the driver. So, we will apply this scheme for all version of v3 hw,
although release version has fixed this SoC bug.

Signed-off-by: Xiaofei Tan <tanxiaofei@huawei.com>
Signed-off-by: John Garry <john.garry@huawei.com>
Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
drivers/scsi/hisi_sas/hisi_sas_v3_hw.c

index efe64bcfa4f240c9fac849284d3ca3fc84a82a5c..aa52d5e424f7a09d5e5728a6d0b0fee0ebdd90b0 100644 (file)
@@ -1731,15 +1731,19 @@ static void cq_tasklet_v3_hw(unsigned long val)
 
        while (rd_point != wr_point) {
                struct hisi_sas_complete_v3_hdr *complete_hdr;
+               struct device *dev = hisi_hba->dev;
                int iptt;
 
                complete_hdr = &complete_queue[rd_point];
 
                iptt = (complete_hdr->dw1) & CMPLT_HDR_IPTT_MSK;
-               slot = &hisi_hba->slot_info[iptt];
-               slot->cmplt_queue_slot = rd_point;
-               slot->cmplt_queue = queue;
-               slot_complete_v3_hw(hisi_hba, slot);
+               if (likely(iptt < HISI_SAS_COMMAND_ENTRIES_V3_HW)) {
+                       slot = &hisi_hba->slot_info[iptt];
+                       slot->cmplt_queue_slot = rd_point;
+                       slot->cmplt_queue = queue;
+                       slot_complete_v3_hw(hisi_hba, slot);
+               } else
+                       dev_err(dev, "IPTT %d is invalid, discard it.\n", iptt);
 
                if (++rd_point >= HISI_SAS_QUEUE_SLOTS)
                        rd_point = 0;