Commit 2824ec9f authored by Sayali Lokhande's avatar Sayali Lokhande Committed by Martin K. Petersen

scsi: ufs: Flush exception event before suspend

Exception event can be raised by the device when system suspend is in
progress. This will result in unclocked register access in exception event
handler as clocks will be turned off during suspend. This change makes sure
to flush exception event handler work in suspend before disabling clocks to
avoid unclocked register access issue.

Link: https://lore.kernel.org/r/1581392451-28743-2-git-send-email-cang@codeaurora.orgReviewed-by: default avatarBean Huo <beanhuo@micron.com>
Signed-off-by: default avatarSayali Lokhande <sayalil@codeaurora.org>
Signed-off-by: default avatarAsutosh Das <asutoshd@codeaurora.org>
Signed-off-by: default avatarCan Guo <cang@codeaurora.org>
Signed-off-by: default avatarMartin K. Petersen <martin.petersen@oracle.com>
parent 518b32f1
...@@ -4730,8 +4730,15 @@ ufshcd_transfer_rsp_status(struct ufs_hba *hba, struct ufshcd_lrb *lrbp) ...@@ -4730,8 +4730,15 @@ ufshcd_transfer_rsp_status(struct ufs_hba *hba, struct ufshcd_lrb *lrbp)
* UFS device needs urgent BKOPs. * UFS device needs urgent BKOPs.
*/ */
if (!hba->pm_op_in_progress && if (!hba->pm_op_in_progress &&
ufshcd_is_exception_event(lrbp->ucd_rsp_ptr)) ufshcd_is_exception_event(lrbp->ucd_rsp_ptr) &&
schedule_work(&hba->eeh_work); schedule_work(&hba->eeh_work)) {
/*
* Prevent suspend once eeh_work is scheduled
* to avoid deadlock between ufshcd_suspend
* and exception event handler.
*/
pm_runtime_get_noresume(hba->dev);
}
break; break;
case UPIU_TRANSACTION_REJECT_UPIU: case UPIU_TRANSACTION_REJECT_UPIU:
/* TODO: handle Reject UPIU Response */ /* TODO: handle Reject UPIU Response */
...@@ -5184,7 +5191,14 @@ static void ufshcd_exception_event_handler(struct work_struct *work) ...@@ -5184,7 +5191,14 @@ static void ufshcd_exception_event_handler(struct work_struct *work)
out: out:
ufshcd_scsi_unblock_requests(hba); ufshcd_scsi_unblock_requests(hba);
pm_runtime_put_sync(hba->dev); /*
* pm_runtime_get_noresume is called while scheduling
* eeh_work to avoid suspend racing with exception work.
* Hence decrement usage counter using pm_runtime_put_noidle
* to allow suspend on completion of exception event handler.
*/
pm_runtime_put_noidle(hba->dev);
pm_runtime_put(hba->dev);
return; return;
} }
...@@ -7925,6 +7939,7 @@ static int ufshcd_suspend(struct ufs_hba *hba, enum ufs_pm_op pm_op) ...@@ -7925,6 +7939,7 @@ static int ufshcd_suspend(struct ufs_hba *hba, enum ufs_pm_op pm_op)
goto enable_gating; goto enable_gating;
} }
flush_work(&hba->eeh_work);
ret = ufshcd_link_state_transition(hba, req_link_state, 1); ret = ufshcd_link_state_transition(hba, req_link_state, 1);
if (ret) if (ret)
goto set_dev_active; goto set_dev_active;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment