[<prev] [next>] [<thread-prev] [day] [month] [year] [list]
Message-Id: <20251014060406.1420475-3-palash.kambar@oss.qualcomm.com>
Date: Tue, 14 Oct 2025 11:34:06 +0530
From: palash.kambar@....qualcomm.com
To: mani@...nel.org, alim.akhtar@...sung.com, avri.altman@....com,
bvanassche@....org, peter.griffin@...aro.org, krzk@...nel.org,
peter.wang@...iatek.com, beanhuo@...ron.com, quic_nguyenb@...cinc.com,
adrian.hunter@...el.com, ebiggers@...nel.org,
neil.armstrong@...aro.org, James.Bottomley@...senPartnership.com,
martin.petersen@...cle.com
Cc: linux-arm-msm@...r.kernel.org, linux-scsi@...r.kernel.org,
linux-kernel@...r.kernel.org, quic_cang@...cinc.com,
quic_nitirawa@...cinc.com,
Palash Kambar <palash.kambar@....qualcomm.com>
Subject: [PATCH V1 2/2] ufs: ufs-qcom: Disable AHIT before SQ tail update to prevent race in MCQ mode
From: Can Guo <quic_cang@...cinc.com>
In MCQ mode, a race condition can occur on QCOM UFSHC V6 when the
Auto-Hibernate Idle Timer (AHIT) is close to expiring. If a data
command and a hibernate command are issued simultaneously to the
UniPro layer, the data command may be dropped.
To prevent this, AHIT is disabled by reprogramming it to 0 before
updating the SQ tail pointer. Once there are no active commands in
the UFS host controller, the timer is re-enabled.
This change ensures reliable command delivery in MCQ mode by
avoiding timing conflicts between data and hibernate operations.
Signed-off-by: Can Guo <quic_cang@...cinc.com>
Co-developed-by: Palash Kambar <palash.kambar@....qualcomm.com>
Signed-off-by: Palash Kambar <palash.kambar@....qualcomm.com>
---
drivers/ufs/host/ufs-qcom.c | 35 +++++++++++++++++++++++++++++++++++
drivers/ufs/host/ufs-qcom.h | 1 +
2 files changed, 36 insertions(+)
diff --git a/drivers/ufs/host/ufs-qcom.c b/drivers/ufs/host/ufs-qcom.c
index 89a3328a7a75..f31239f4fc50 100644
--- a/drivers/ufs/host/ufs-qcom.c
+++ b/drivers/ufs/host/ufs-qcom.c
@@ -1286,6 +1286,39 @@ static int ufs_qcom_icc_init(struct ufs_qcom_host *host)
return 0;
}
+static void ufs_qcom_send_command(struct ufs_hba *hba,
+ struct ufshcd_lrb *lrbp)
+{
+ struct ufs_qcom_host *host = ufshcd_get_variant(hba);
+ unsigned long flags;
+
+ if ((host->hw_ver.major == 6 && host->hw_ver.minor == 0 &&
+ host->hw_ver.step == 0) && hba->mcq_enabled) {
+ spin_lock_irqsave(hba->host->host_lock, flags);
+ if ((++host->active_cmds) == 1)
+ /* Stop the auto-hiberate idle timer */
+ ufshcd_writel(hba, 0, REG_AUTO_HIBERNATE_IDLE_TIMER);
+ spin_unlock_irqrestore(hba->host->host_lock, flags);
+ }
+}
+
+static void ufs_qcom_compl_command(struct ufs_hba *hba,
+ struct ufshcd_lrb *lrbp)
+{
+ struct ufs_qcom_host *host = ufshcd_get_variant(hba);
+ unsigned long flags;
+
+ if ((host->hw_ver.major == 6 && host->hw_ver.minor == 0 &&
+ host->hw_ver.step == 0) && hba->mcq_enabled) {
+ spin_lock_irqsave(hba->host->host_lock, flags);
+ if ((--host->active_cmds) == 0)
+ /* Activate the auto-hiberate idle timer */
+ ufshcd_writel(hba, hba->ahit,
+ REG_AUTO_HIBERNATE_IDLE_TIMER);
+ spin_unlock_irqrestore(hba->host->host_lock, flags);
+ }
+}
+
/**
* ufs_qcom_init - bind phy with controller
* @hba: host controller instance
@@ -2194,6 +2227,8 @@ static const struct ufs_hba_variant_ops ufs_hba_qcom_vops = {
.get_ufs_hci_version = ufs_qcom_get_ufs_hci_version,
.clk_scale_notify = ufs_qcom_clk_scale_notify,
.setup_clocks = ufs_qcom_setup_clocks,
+ .setup_xfer_req = ufs_qcom_send_command,
+ .compl_command = ufs_qcom_compl_command,
.hce_enable_notify = ufs_qcom_hce_enable_notify,
.link_startup_notify = ufs_qcom_link_startup_notify,
.pwr_change_notify = ufs_qcom_pwr_change_notify,
diff --git a/drivers/ufs/host/ufs-qcom.h b/drivers/ufs/host/ufs-qcom.h
index 380d02333d38..a97da99361a8 100644
--- a/drivers/ufs/host/ufs-qcom.h
+++ b/drivers/ufs/host/ufs-qcom.h
@@ -308,6 +308,7 @@ struct ufs_qcom_host {
u32 phy_gear;
bool esi_enabled;
+ unsigned long active_cmds;
};
struct ufs_qcom_drvdata {
--
2.34.1
Powered by blists - more mailing lists