]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
i2c: designware: amdisp: Fix resume-probe race condition issue
authorPratap Nirujogi <pratap.nirujogi@amd.com>
Fri, 20 Mar 2026 20:12:22 +0000 (16:12 -0400)
committerAndi Shyti <andi.shyti@kernel.org>
Fri, 27 Mar 2026 12:51:21 +0000 (13:51 +0100)
Identified resume-probe race condition in kernel v7.0 with the commit
38fa29b01a6a ("i2c: designware: Combine the init functions"),but this
issue existed from the beginning though not detected.

The amdisp i2c device requires ISP to be in power-on state for probe
to succeed. To meet this requirement, this device is added to genpd
to control ISP power using runtime PM. The pm_runtime_get_sync() called
before i2c_dw_probe() triggers PM resume, which powers on ISP and also
invokes the amdisp i2c runtime resume before the probe completes resulting
in this race condition and a NULL dereferencing issue in v7.0

Fix this race condition by using the genpd APIs directly during probe:
  - Call dev_pm_genpd_resume() to Power ON ISP before probe
  - Call dev_pm_genpd_suspend() to Power OFF ISP after probe
  - Set the device to suspended state with pm_runtime_set_suspended()
  - Enable runtime PM only after the device is fully initialized

Fixes: d6263c468a761 ("i2c: amd-isp: Add ISP i2c-designware driver")
Co-developed-by: Bin Du <bin.du@amd.com>
Signed-off-by: Bin Du <bin.du@amd.com>
Signed-off-by: Pratap Nirujogi <pratap.nirujogi@amd.com>
Cc: <stable@vger.kernel.org> # v6.16+
Acked-by: Mika Westerberg <mika.westerberg@linux.intel.com>
Reviewed-by: Mario Limonciello (AMD) <superm1@kernel.org>
Reviewed-by: Andy Shevchenko <andriy.shevchenko@linux.intel.com>
Signed-off-by: Andi Shyti <andi.shyti@kernel.org>
Link: https://lore.kernel.org/r/20260320201302.3490570-1-pratap.nirujogi@amd.com
drivers/i2c/busses/i2c-designware-amdisp.c

index c48728ad9f6f25174515defe55603d67b2491b72..9f0ec0fae6f23552e82e6a9ee450abf5ee164c33 100644 (file)
@@ -7,6 +7,7 @@
 
 #include <linux/module.h>
 #include <linux/platform_device.h>
+#include <linux/pm_domain.h>
 #include <linux/pm_runtime.h>
 #include <linux/soc/amd/isp4_misc.h>
 
@@ -76,22 +77,20 @@ static int amd_isp_dw_i2c_plat_probe(struct platform_device *pdev)
 
        device_enable_async_suspend(&pdev->dev);
 
-       pm_runtime_enable(&pdev->dev);
-       pm_runtime_get_sync(&pdev->dev);
-
+       dev_pm_genpd_resume(&pdev->dev);
        ret = i2c_dw_probe(isp_i2c_dev);
        if (ret) {
                dev_err_probe(&pdev->dev, ret, "i2c_dw_probe failed\n");
                goto error_release_rpm;
        }
-
-       pm_runtime_put_sync(&pdev->dev);
+       dev_pm_genpd_suspend(&pdev->dev);
+       pm_runtime_set_suspended(&pdev->dev);
+       pm_runtime_enable(&pdev->dev);
 
        return 0;
 
 error_release_rpm:
        amd_isp_dw_i2c_plat_pm_cleanup(isp_i2c_dev);
-       pm_runtime_put_sync(&pdev->dev);
        return ret;
 }