提交 2f026833 编写于 作者: L Luo Jiaxing 提交者: Zheng Zengkai

scsi: hisi_sas: Reset controller for internal abort timeout

mainline inclusion
from mainline-master
commit 63ece9eb
category: bugfix
bugzilla: 175270
CVE: NA

Reference: https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/commit/?id=63ece9eb350312ee33327269480482dfac8555db

------------------------------------------------------------------------

If an internal task abort timeout occurs, the controller has developed a
fault, and needs to be reset to be recovered. However if a timeout occurs
during SCSI error handling, issuing a controller reset immediately may
conflict with the error handling.

To handle internal abort in these two scenarios, only queue the reset when
not in an error handling function. In the case of a timeout during error
handling, do nothing and rely on the inevitable ha nexus reset to reset the
controller.

Link: https://lore.kernel.org/r/1623058179-80434-5-git-send-email-john.garry@huawei.comSigned-off-by: NLuo Jiaxing <luojiaxing@huawei.com>
Signed-off-by: NJohn Garry <john.garry@huawei.com>
Signed-off-by: NMartin K. Petersen <martin.petersen@oracle.com>
Reviewed-by: NOuyangdelong <ouyangdelong@huawei.com>
Signed-off-by: NNifujia <nifujia1@hisilicon.com>
Signed-off-by: NZheng Zengkai <zhengzengkai@huawei.com>
上级 eeaae6e7
...@@ -15,7 +15,7 @@ static int hisi_sas_debug_issue_ssp_tmf(struct domain_device *device, ...@@ -15,7 +15,7 @@ static int hisi_sas_debug_issue_ssp_tmf(struct domain_device *device,
static int static int
hisi_sas_internal_task_abort(struct hisi_hba *hisi_hba, hisi_sas_internal_task_abort(struct hisi_hba *hisi_hba,
struct domain_device *device, struct domain_device *device,
int abort_flag, int tag); int abort_flag, int tag, bool rst_to_recover);
static int hisi_sas_softreset_ata_disk(struct domain_device *device); static int hisi_sas_softreset_ata_disk(struct domain_device *device);
static int hisi_sas_control_phy(struct asd_sas_phy *sas_phy, enum phy_func func, static int hisi_sas_control_phy(struct asd_sas_phy *sas_phy, enum phy_func func,
void *funcdata); void *funcdata);
...@@ -1074,7 +1074,7 @@ static void hisi_sas_dev_gone(struct domain_device *device) ...@@ -1074,7 +1074,7 @@ static void hisi_sas_dev_gone(struct domain_device *device)
down(&hisi_hba->sem); down(&hisi_hba->sem);
if (!test_bit(HISI_SAS_RESET_BIT, &hisi_hba->flags)) { if (!test_bit(HISI_SAS_RESET_BIT, &hisi_hba->flags)) {
hisi_sas_internal_task_abort(hisi_hba, device, hisi_sas_internal_task_abort(hisi_hba, device,
HISI_SAS_INT_ABT_DEV, 0); HISI_SAS_INT_ABT_DEV, 0, true);
hisi_sas_dereg_device(hisi_hba, device); hisi_sas_dereg_device(hisi_hba, device);
...@@ -1516,7 +1516,8 @@ static void hisi_sas_terminate_stp_reject(struct hisi_hba *hisi_hba) ...@@ -1516,7 +1516,8 @@ static void hisi_sas_terminate_stp_reject(struct hisi_hba *hisi_hba)
continue; continue;
rc = hisi_sas_internal_task_abort(hisi_hba, device, rc = hisi_sas_internal_task_abort(hisi_hba, device,
HISI_SAS_INT_ABT_DEV, 0); HISI_SAS_INT_ABT_DEV, 0,
false);
if (rc < 0) if (rc < 0)
dev_err(dev, "STP reject: abort dev failed %d\n", rc); dev_err(dev, "STP reject: abort dev failed %d\n", rc);
} }
...@@ -1671,7 +1672,8 @@ static int hisi_sas_abort_task(struct sas_task *task) ...@@ -1671,7 +1672,8 @@ static int hisi_sas_abort_task(struct sas_task *task)
&tmf_task); &tmf_task);
rc2 = hisi_sas_internal_task_abort(hisi_hba, device, rc2 = hisi_sas_internal_task_abort(hisi_hba, device,
HISI_SAS_INT_ABT_CMD, tag); HISI_SAS_INT_ABT_CMD, tag,
false);
if (rc2 < 0) { if (rc2 < 0) {
dev_err(dev, "abort task: internal abort (%d)\n", rc2); dev_err(dev, "abort task: internal abort (%d)\n", rc2);
return TMF_RESP_FUNC_FAILED; return TMF_RESP_FUNC_FAILED;
...@@ -1693,7 +1695,7 @@ static int hisi_sas_abort_task(struct sas_task *task) ...@@ -1693,7 +1695,7 @@ static int hisi_sas_abort_task(struct sas_task *task)
if (task->dev->dev_type == SAS_SATA_DEV) { if (task->dev->dev_type == SAS_SATA_DEV) {
rc = hisi_sas_internal_task_abort(hisi_hba, device, rc = hisi_sas_internal_task_abort(hisi_hba, device,
HISI_SAS_INT_ABT_DEV, HISI_SAS_INT_ABT_DEV,
0); 0, false);
if (rc < 0) { if (rc < 0) {
dev_err(dev, "abort task: internal abort failed\n"); dev_err(dev, "abort task: internal abort failed\n");
goto out; goto out;
...@@ -1708,7 +1710,8 @@ static int hisi_sas_abort_task(struct sas_task *task) ...@@ -1708,7 +1710,8 @@ static int hisi_sas_abort_task(struct sas_task *task)
struct hisi_sas_cq *cq = &hisi_hba->cq[slot->dlvry_queue]; struct hisi_sas_cq *cq = &hisi_hba->cq[slot->dlvry_queue];
rc = hisi_sas_internal_task_abort(hisi_hba, device, rc = hisi_sas_internal_task_abort(hisi_hba, device,
HISI_SAS_INT_ABT_CMD, tag); HISI_SAS_INT_ABT_CMD, tag,
false);
if (((rc < 0) || (rc == TMF_RESP_FUNC_FAILED)) && if (((rc < 0) || (rc == TMF_RESP_FUNC_FAILED)) &&
task->lldd_task) { task->lldd_task) {
/* /*
...@@ -1734,7 +1737,7 @@ static int hisi_sas_abort_task_set(struct domain_device *device, u8 *lun) ...@@ -1734,7 +1737,7 @@ static int hisi_sas_abort_task_set(struct domain_device *device, u8 *lun)
int rc; int rc;
rc = hisi_sas_internal_task_abort(hisi_hba, device, rc = hisi_sas_internal_task_abort(hisi_hba, device,
HISI_SAS_INT_ABT_DEV, 0); HISI_SAS_INT_ABT_DEV, 0, false);
if (rc < 0) { if (rc < 0) {
dev_err(dev, "abort task set: internal abort rc=%d\n", rc); dev_err(dev, "abort task set: internal abort rc=%d\n", rc);
return TMF_RESP_FUNC_FAILED; return TMF_RESP_FUNC_FAILED;
...@@ -1828,7 +1831,7 @@ static int hisi_sas_I_T_nexus_reset(struct domain_device *device) ...@@ -1828,7 +1831,7 @@ static int hisi_sas_I_T_nexus_reset(struct domain_device *device)
int rc; int rc;
rc = hisi_sas_internal_task_abort(hisi_hba, device, rc = hisi_sas_internal_task_abort(hisi_hba, device,
HISI_SAS_INT_ABT_DEV, 0); HISI_SAS_INT_ABT_DEV, 0, false);
if (rc < 0) { if (rc < 0) {
dev_err(dev, "I_T nexus reset: internal abort (%d)\n", rc); dev_err(dev, "I_T nexus reset: internal abort (%d)\n", rc);
return TMF_RESP_FUNC_FAILED; return TMF_RESP_FUNC_FAILED;
...@@ -1858,7 +1861,7 @@ static int hisi_sas_lu_reset(struct domain_device *device, u8 *lun) ...@@ -1858,7 +1861,7 @@ static int hisi_sas_lu_reset(struct domain_device *device, u8 *lun)
/* Clear internal IO and then lu reset */ /* Clear internal IO and then lu reset */
rc = hisi_sas_internal_task_abort(hisi_hba, device, rc = hisi_sas_internal_task_abort(hisi_hba, device,
HISI_SAS_INT_ABT_DEV, 0); HISI_SAS_INT_ABT_DEV, 0, false);
if (rc < 0) { if (rc < 0) {
dev_err(dev, "lu_reset: internal abort failed\n"); dev_err(dev, "lu_reset: internal abort failed\n");
goto out; goto out;
...@@ -2054,11 +2057,13 @@ hisi_sas_internal_abort_task_exec(struct hisi_hba *hisi_hba, int device_id, ...@@ -2054,11 +2057,13 @@ hisi_sas_internal_abort_task_exec(struct hisi_hba *hisi_hba, int device_id,
* @tag: tag of IO to be aborted (only relevant to single * @tag: tag of IO to be aborted (only relevant to single
* IO mode) * IO mode)
* @dq: delivery queue for this internal abort command * @dq: delivery queue for this internal abort command
* @rst_to_recover: If rst_to_recover set, queue a controller
* reset if an internal abort times out.
*/ */
static int static int
_hisi_sas_internal_task_abort(struct hisi_hba *hisi_hba, _hisi_sas_internal_task_abort(struct hisi_hba *hisi_hba,
struct domain_device *device, int abort_flag, struct domain_device *device, int abort_flag,
int tag, struct hisi_sas_dq *dq) int tag, struct hisi_sas_dq *dq, bool rst_to_recover)
{ {
struct sas_task *task; struct sas_task *task;
struct hisi_sas_device *sas_dev = device->lldd_dev; struct hisi_sas_device *sas_dev = device->lldd_dev;
...@@ -2114,7 +2119,13 @@ _hisi_sas_internal_task_abort(struct hisi_hba *hisi_hba, ...@@ -2114,7 +2119,13 @@ _hisi_sas_internal_task_abort(struct hisi_hba *hisi_hba,
synchronize_irq(cq->irq_no); synchronize_irq(cq->irq_no);
slot->task = NULL; slot->task = NULL;
} }
dev_err(dev, "internal task abort: timeout and not done.\n");
if (rst_to_recover) {
dev_err(dev, "internal task abort: timeout and not done. Queuing reset.\n");
queue_work(hisi_hba->wq, &hisi_hba->rst_work);
} else {
dev_err(dev, "internal task abort: timeout and not done.\n");
}
res = -EIO; res = -EIO;
goto exit; goto exit;
...@@ -2147,7 +2158,7 @@ _hisi_sas_internal_task_abort(struct hisi_hba *hisi_hba, ...@@ -2147,7 +2158,7 @@ _hisi_sas_internal_task_abort(struct hisi_hba *hisi_hba,
static int static int
hisi_sas_internal_task_abort(struct hisi_hba *hisi_hba, hisi_sas_internal_task_abort(struct hisi_hba *hisi_hba,
struct domain_device *device, struct domain_device *device,
int abort_flag, int tag) int abort_flag, int tag, bool rst_to_recover)
{ {
struct hisi_sas_slot *slot; struct hisi_sas_slot *slot;
struct device *dev = hisi_hba->dev; struct device *dev = hisi_hba->dev;
...@@ -2159,7 +2170,8 @@ hisi_sas_internal_task_abort(struct hisi_hba *hisi_hba, ...@@ -2159,7 +2170,8 @@ hisi_sas_internal_task_abort(struct hisi_hba *hisi_hba,
slot = &hisi_hba->slot_info[tag]; slot = &hisi_hba->slot_info[tag];
dq = &hisi_hba->dq[slot->dlvry_queue]; dq = &hisi_hba->dq[slot->dlvry_queue];
return _hisi_sas_internal_task_abort(hisi_hba, device, return _hisi_sas_internal_task_abort(hisi_hba, device,
abort_flag, tag, dq); abort_flag, tag, dq,
rst_to_recover);
case HISI_SAS_INT_ABT_DEV: case HISI_SAS_INT_ABT_DEV:
for (i = 0; i < hisi_hba->cq_nvecs; i++) { for (i = 0; i < hisi_hba->cq_nvecs; i++) {
struct hisi_sas_cq *cq = &hisi_hba->cq[i]; struct hisi_sas_cq *cq = &hisi_hba->cq[i];
...@@ -2170,7 +2182,7 @@ hisi_sas_internal_task_abort(struct hisi_hba *hisi_hba, ...@@ -2170,7 +2182,7 @@ hisi_sas_internal_task_abort(struct hisi_hba *hisi_hba,
dq = &hisi_hba->dq[i]; dq = &hisi_hba->dq[i];
rc = _hisi_sas_internal_task_abort(hisi_hba, device, rc = _hisi_sas_internal_task_abort(hisi_hba, device,
abort_flag, tag, abort_flag, tag,
dq); dq, rst_to_recover);
if (rc) if (rc)
return rc; return rc;
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册