[PATCH] drm/panthor: Fix a race between the reset and suspend path
Steven Price
steven.price at arm.com
Mon Dec 23 16:22:40 UTC 2024
On 17/12/2024 09:24, Boris Brezillon wrote:
> If a reset is scheduled when the suspend happens, we drop the
> reset-pending info on the floor assuming the resume will fix things,
> but the resume logic might try a fast reset. If we're lucky, the
> fast reset fails and we fallback to a slow reset, but if the FW was
> corrupted in a way that makes it partially functional (it boots but
> doesn't quite do what it's expected to do), we won't notice immediately
> that things are not working correctly, leading to a new reset further
> down the road.
>
> Fixes: 5fe909cae118 ("drm/panthor: Add the device logical block")
> Signed-off-by: Boris Brezillon <boris.brezillon at collabora.com>
Reviewed-by: Steven Price <steven.price at arm.com>
> ---
> drivers/gpu/drm/panthor/panthor_device.c | 22 ++++++++++++----------
> 1 file changed, 12 insertions(+), 10 deletions(-)
>
> diff --git a/drivers/gpu/drm/panthor/panthor_device.c b/drivers/gpu/drm/panthor/panthor_device.c
> index 2c817e65e6be..3285ac42d2cd 100644
> --- a/drivers/gpu/drm/panthor/panthor_device.c
> +++ b/drivers/gpu/drm/panthor/panthor_device.c
> @@ -128,14 +128,11 @@ static void panthor_device_reset_work(struct work_struct *work)
> struct panthor_device *ptdev = container_of(work, struct panthor_device, reset.work);
> int ret = 0, cookie;
>
> - if (atomic_read(&ptdev->pm.state) != PANTHOR_DEVICE_PM_STATE_ACTIVE) {
> - /*
> - * No need for a reset as the device has been (or will be)
> - * powered down
> - */
> - atomic_set(&ptdev->reset.pending, 0);
> + /* If the device is entering suspend, we don't reset. A slow reset will
> + * be forced at resume time instead.
> + */
> + if (atomic_read(&ptdev->pm.state) != PANTHOR_DEVICE_PM_STATE_ACTIVE)
> return;
> - }
>
> if (!drm_dev_enter(&ptdev->base, &cookie))
> return;
> @@ -473,6 +470,14 @@ int panthor_device_resume(struct device *dev)
>
> if (panthor_device_is_initialized(ptdev) &&
> drm_dev_enter(&ptdev->base, &cookie)) {
> + /* If there was a reset pending at the time we suspended the
> + * device, we force a slow reset.
> + */
> + if (atomic_read(&ptdev->reset.pending)) {
> + ptdev->reset.fast = false;
> + atomic_set(&ptdev->reset.pending, 0);
> + }
> +
> ret = panthor_device_resume_hw_components(ptdev);
> if (ret && ptdev->reset.fast) {
> drm_err(&ptdev->base, "Fast reset failed, trying a slow reset");
> @@ -489,9 +494,6 @@ int panthor_device_resume(struct device *dev)
> goto err_suspend_devfreq;
> }
>
> - if (atomic_read(&ptdev->reset.pending))
> - queue_work(ptdev->reset.wq, &ptdev->reset.work);
> -
> /* Clear all IOMEM mappings pointing to this device after we've
> * resumed. This way the fake mappings pointing to the dummy pages
> * are removed and the real iomem mapping will be restored on next
More information about the dri-devel
mailing list