Commit 10ff4c52 authored by Javier Martinez Canillas's avatar Javier Martinez Canillas Committed by Wolfram Sang

i2c: exynos5: Fix possible ABBA deadlock by keeping I2C clock prepared

The exynos5 I2C controller driver always prepares and enables a clock
before using it and then disables unprepares it when the clock is not
used anymore.

But this can cause a possible ABBA deadlock in some scenarios since a
driver that uses regmap to access its I2C registers, will first grab
the regmap lock and then the I2C xfer function will grab the prepare
lock when preparing the I2C clock. But since the clock driver also
uses regmap for I2C accesses, preparing a clock will first grab the
prepare lock and then the regmap lock when using the regmap API.

An example of this happens on the Exynos5422 Odroid XU4 board where a
s2mps11 PMIC is used and both the s2mps11 regulators and clk drivers
share the same I2C regmap.

The possible deadlock is reported by the kernel lockdep:

  Possible unsafe locking scenario:

        CPU0                    CPU1
        ----                    ----
   lock(sec_core:428:(regmap)->lock);
                                lock(prepare_lock);
                                lock(sec_core:428:(regmap)->lock);
   lock(prepare_lock);

  *** DEADLOCK ***

Fix it by leaving the code prepared on probe and use {en,dis}able in
the I2C transfer function.

This patch is similar to commit 34e81ad5 ("i2c: s3c2410: fix ABBA
deadlock by keeping clock prepared") that fixes the same bug in other
driver for an I2C controller found in Samsung SoCs.
Reported-by: default avatarAnand Moon <linux.amoon@gmail.com>
Signed-off-by: default avatarJavier Martinez Canillas <javier@osg.samsung.com>
Reviewed-by: default avatarAnand Moon <linux.amoon@gmail.com>
Reviewed-by: default avatarKrzysztof Kozlowski <k.kozlowski@samsung.com>
Signed-off-by: default avatarWolfram Sang <wsa@the-dreams.de>
Cc: stable@kernel.org
parent 609d5a1b
...@@ -671,7 +671,9 @@ static int exynos5_i2c_xfer(struct i2c_adapter *adap, ...@@ -671,7 +671,9 @@ static int exynos5_i2c_xfer(struct i2c_adapter *adap,
return -EIO; return -EIO;
} }
clk_prepare_enable(i2c->clk); ret = clk_enable(i2c->clk);
if (ret)
return ret;
for (i = 0; i < num; i++, msgs++) { for (i = 0; i < num; i++, msgs++) {
stop = (i == num - 1); stop = (i == num - 1);
...@@ -695,7 +697,7 @@ static int exynos5_i2c_xfer(struct i2c_adapter *adap, ...@@ -695,7 +697,7 @@ static int exynos5_i2c_xfer(struct i2c_adapter *adap,
} }
out: out:
clk_disable_unprepare(i2c->clk); clk_disable(i2c->clk);
return ret; return ret;
} }
...@@ -747,7 +749,9 @@ static int exynos5_i2c_probe(struct platform_device *pdev) ...@@ -747,7 +749,9 @@ static int exynos5_i2c_probe(struct platform_device *pdev)
return -ENOENT; return -ENOENT;
} }
clk_prepare_enable(i2c->clk); ret = clk_prepare_enable(i2c->clk);
if (ret)
return ret;
mem = platform_get_resource(pdev, IORESOURCE_MEM, 0); mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
i2c->regs = devm_ioremap_resource(&pdev->dev, mem); i2c->regs = devm_ioremap_resource(&pdev->dev, mem);
...@@ -799,6 +803,10 @@ static int exynos5_i2c_probe(struct platform_device *pdev) ...@@ -799,6 +803,10 @@ static int exynos5_i2c_probe(struct platform_device *pdev)
platform_set_drvdata(pdev, i2c); platform_set_drvdata(pdev, i2c);
clk_disable(i2c->clk);
return 0;
err_clk: err_clk:
clk_disable_unprepare(i2c->clk); clk_disable_unprepare(i2c->clk);
return ret; return ret;
...@@ -810,6 +818,8 @@ static int exynos5_i2c_remove(struct platform_device *pdev) ...@@ -810,6 +818,8 @@ static int exynos5_i2c_remove(struct platform_device *pdev)
i2c_del_adapter(&i2c->adap); i2c_del_adapter(&i2c->adap);
clk_unprepare(i2c->clk);
return 0; return 0;
} }
...@@ -821,6 +831,8 @@ static int exynos5_i2c_suspend_noirq(struct device *dev) ...@@ -821,6 +831,8 @@ static int exynos5_i2c_suspend_noirq(struct device *dev)
i2c->suspended = 1; i2c->suspended = 1;
clk_unprepare(i2c->clk);
return 0; return 0;
} }
...@@ -830,7 +842,9 @@ static int exynos5_i2c_resume_noirq(struct device *dev) ...@@ -830,7 +842,9 @@ static int exynos5_i2c_resume_noirq(struct device *dev)
struct exynos5_i2c *i2c = platform_get_drvdata(pdev); struct exynos5_i2c *i2c = platform_get_drvdata(pdev);
int ret = 0; int ret = 0;
clk_prepare_enable(i2c->clk); ret = clk_prepare_enable(i2c->clk);
if (ret)
return ret;
ret = exynos5_hsi2c_clock_setup(i2c); ret = exynos5_hsi2c_clock_setup(i2c);
if (ret) { if (ret) {
...@@ -839,7 +853,7 @@ static int exynos5_i2c_resume_noirq(struct device *dev) ...@@ -839,7 +853,7 @@ static int exynos5_i2c_resume_noirq(struct device *dev)
} }
exynos5_i2c_init(i2c); exynos5_i2c_init(i2c);
clk_disable_unprepare(i2c->clk); clk_disable(i2c->clk);
i2c->suspended = 0; i2c->suspended = 0;
return 0; return 0;
......
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