drivers/ufs/core/ufshcd.c | 1 + 1 file changed, 1 insertion(+)
From: Qilin Tan <qilin.tan@mediatek.com>
When kernel_power_off is called directly without freezing userspace,
it may cause UFS crashes:
Callback:
...... 0xBFFFFFC080C6156C()
vmlinux readl() + 52
vmlinux ufshcd_add_command_trace() + 552
vmlinux ufshcd_send_command() + 84
When kernel_power_off is executed, ufshcd_wl_shutdown is also called
to turn off the UFS reference clock, VCC, and VCCQ. If I/O requests
are still being sent to the UFS host and accessing the interrupt
status register at this time, AP read timeouts may occur, causing bus
hang crashes.
The root cause is that scsi_device_quiesce and blk_mq_freeze_queue
only drain the requests in the request queue but don't guarantee that
all requests have been dispatched to the UFS host and completed.
Requests may remain pending in the hardware dispatch queue and be
rescheduled later. If the UFS reference clock has already been turned
off at this point, a bus hang crash will occur.
Example of the race condition:
Thread 1 Thread 2
kernel_power_off
-> ufshcd_wl_shutdown
-> scsi_device_quiesce(sdev)
-> blk_mq_freeze_queue(q)
-> blk_mq_run_hw_queue(htx, false)
-> blk_mq_delay_run_hw_queue(hctx, 0) blk_mq_run_work_fn
-> ufshcd_suspend(hba) // disable ref clk -> blk_mq_dispatch_rq_list
-> blk_mq_run_hw_queue()
-> ufshcd_send_command()
-> ufshcd_add_command_trace()
-> ufshcd_readl(hba, REG_INTERRUPT_STATUS)
When Thread-2's dispatch request is delayed due to heavy CPU load,
the interrupt status register may be read after the reference clock
is disabled, resulting in a bus hang crash.
To avoid this issue, call ufshcd_wait_for_doorbell_clr to wait until
all requests are processed before disabling the reference clock.
Signed-off-by: Qilin Tan <qilin.tan@mediatek.com>
Signed-off-by: Bosser Ye <bo.ye@mediatek.com>
---
drivers/ufs/core/ufshcd.c | 1 +
1 file changed, 1 insertion(+)
diff --git a/drivers/ufs/core/ufshcd.c b/drivers/ufs/core/ufshcd.c
index 7735421e3991..a1013aea8e90 100644
--- a/drivers/ufs/core/ufshcd.c
+++ b/drivers/ufs/core/ufshcd.c
@@ -10262,6 +10262,7 @@ static void ufshcd_wl_shutdown(struct device *dev)
scsi_device_set_state(sdev, SDEV_OFFLINE);
mutex_unlock(&sdev->state_mutex);
}
+ ufshcd_wait_for_doorbell_clr(hba, 5 * USEC_PER_SEC);
__ufshcd_wl_suspend(hba, UFS_SHUTDOWN_PM);
/*
--
2.17.0
On 5/19/25 12:38 AM, Bo Ye wrote: > The root cause is that scsi_device_quiesce and blk_mq_freeze_queue > only drain the requests in the request queue but don't guarantee that > all requests have been dispatched to the UFS host and completed. > Requests may remain pending in the hardware dispatch queue and be > rescheduled later. The above is confusing. scsi_device_quiesce() drains both the request queue and the hardware queues for all commands associated with a single logical unit. The problem that you are encountering is probably that scsi_device_quiesce() is only called for the WLUN but not for the other logical units and hence that there still may be commands being processed for other logical units after scsi_device_quiesce() has returned. > diff --git a/drivers/ufs/core/ufshcd.c b/drivers/ufs/core/ufshcd.c > index 7735421e3991..a1013aea8e90 100644 > --- a/drivers/ufs/core/ufshcd.c > +++ b/drivers/ufs/core/ufshcd.c > @@ -10262,6 +10262,7 @@ static void ufshcd_wl_shutdown(struct device *dev) > scsi_device_set_state(sdev, SDEV_OFFLINE); > mutex_unlock(&sdev->state_mutex); > } > + ufshcd_wait_for_doorbell_clr(hba, 5 * USEC_PER_SEC); > __ufshcd_wl_suspend(hba, UFS_SHUTDOWN_PM); > > /* The name of the ufshcd_wait_for_doorbell_clr() function is confusing since "doorbell" refers to a legacy single doorbell mode concept while the function supports both legacy mode and MCQ. After this patch has been queued I plan to submit a patch that renames this function. If the patch description of this patch is improved I will add my Reviewed-by to this patch. Thanks, Bart.
On 5/19/25 12:38 AM, Bo Ye wrote: > diff --git a/drivers/ufs/core/ufshcd.c b/drivers/ufs/core/ufshcd.c > index 7735421e3991..a1013aea8e90 100644 > --- a/drivers/ufs/core/ufshcd.c > +++ b/drivers/ufs/core/ufshcd.c > @@ -10262,6 +10262,7 @@ static void ufshcd_wl_shutdown(struct device *dev) > scsi_device_set_state(sdev, SDEV_OFFLINE); > mutex_unlock(&sdev->state_mutex); > } > + ufshcd_wait_for_doorbell_clr(hba, 5 * USEC_PER_SEC); > __ufshcd_wl_suspend(hba, UFS_SHUTDOWN_PM); > > /* This code path is not only triggered when using a UFSHCI 3.0 controller but also when using a UFSHCI 4.0 controller. ufshcd_wait_for_doorbell_clr() only supports the legacy single doorbell mode. Please make sure that the fix supports both the legacy single doorbell mode and MCQ. Thanks, Bart.
On 5/19/25 12:38 AM, Bo Ye wrote:
> diff --git a/drivers/ufs/core/ufshcd.c b/drivers/ufs/core/ufshcd.c
> index 7735421e3991..a1013aea8e90 100644
> --- a/drivers/ufs/core/ufshcd.c
> +++ b/drivers/ufs/core/ufshcd.c
> @@ -10262,6 +10262,7 @@ static void ufshcd_wl_shutdown(struct device *dev)
> scsi_device_set_state(sdev, SDEV_OFFLINE);
> mutex_unlock(&sdev->state_mutex);
> }
> + ufshcd_wait_for_doorbell_clr(hba, 5 * USEC_PER_SEC);
> __ufshcd_wl_suspend(hba, UFS_SHUTDOWN_PM);
>
> /*
UFS driver kernel patches should be prepared against Martin's for-next
branch. Commit a3a951064f7e ("scsi: ufs: Rework
ufshcd_clock_scaling_prepare()") removed ufshcd_wait_for_doorbell_clr().
Please fix the build failure introduced by this patch.
Bart.
© 2016 - 2025 Red Hat, Inc.