Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit 22d661f8 authored by David Collins's avatar David Collins
Browse files

clk: qcom: gdsc-regulator: remove explicit parent supply enablement



Some GDSCs have a hardware requirement that their parent supply
must be enabled in order for the GDSC registers to be accessible.
To account for this, gdsc-regulator regulator_ops callback
functions call regulator_enable() on the parent supply to
guarantee that the GDSC registers are accessible.

Unfortunately, this usage of regulator_enable()
(and regulator_disable()) can result in a mutex deadlock when
commit f8702f9e ("regulator: core: Use ww_mutex for regulators
locking") is present.

Therefore, remove all regulator_enable() and regulator_disable()
calls to eliminate the possibility of deadlock.  Return an error
if the parent is not enabled when necessary.  This imposes a
requirement that consumers must enable the parent or the GDSC
itself (which indirectly enables the parent) before calling
regulator_set_mode() on a GDSC.

Change-Id: I49315a30f9f2034e9a6ad715bfbe5f8ed0896e46
Signed-off-by: default avatarDavid Collins <collinsd@codeaurora.org>
parent 6c366b93
Loading
Loading
Loading
Loading
+117 −70
Original line number Diff line number Diff line
@@ -20,6 +20,8 @@
#include <linux/reset.h>
#include <linux/mfd/syscon.h>

#include "../../regulator/internal.h"

/* GDSCR */
#define PWR_ON_MASK		BIT(31)
#define CLK_DIS_WAIT_MASK	(0xF << 12)
@@ -51,7 +53,6 @@ struct gdsc {
	struct regmap           *hw_ctrl;
	struct regmap           *sw_reset;
	struct clk		**clocks;
	struct regulator	*parent_regulator;
	struct reset_control	**reset_clocks;
	bool			toggle_logic;
	bool			retain_ff_enable;
@@ -60,6 +61,8 @@ struct gdsc {
	bool			force_root_en;
	bool			no_status_check_on_disable;
	bool			is_gdsc_enabled;
	bool			is_gdsc_hw_ctrl_mode;
	bool			is_root_clk_voted;
	bool			reset_aon;
	int			clock_count;
	int			reset_count;
@@ -118,47 +121,33 @@ static int poll_gdsc_status(struct gdsc *sc, enum gdscr_status status)
	return -ETIMEDOUT;
}

static int gdsc_is_enabled(struct regulator_dev *rdev)
static int gdsc_init_is_enabled(struct gdsc *sc)
{
	struct gdsc *sc = rdev_get_drvdata(rdev);
	uint32_t regval;
	int ret;

	if (!sc->toggle_logic)
		return !sc->resets_asserted;

	if (sc->parent_regulator) {
		/*
		 * The parent regulator for the GDSC is required to be on to
		 * make any register accesses to the GDSC base. Return false
		 * if the parent supply is disabled.
		 */
		if (regulator_is_enabled(sc->parent_regulator) <= 0)
			return false;

		if (regulator_enable(sc->parent_regulator))
			return false;
	if (!sc->toggle_logic) {
		sc->is_gdsc_enabled = !sc->resets_asserted;
		return 0;
	}

	regmap_read(sc->regmap, REG_OFFSET, &regval);
	ret = regmap_read(sc->regmap, REG_OFFSET, &regval);
	if (ret < 0)
		return ret;

	if (regval & PWR_ON_MASK) {
		/*
		 * The GDSC might be turned on due to TZ/HYP vote on the
		 * votable GDS registers. Check the SW_COLLAPSE_MASK to
		 * determine if HLOS has voted for it.
		 */
		if (!(regval & SW_COLLAPSE_MASK)) {
			if (sc->parent_regulator)
				regulator_disable(sc->parent_regulator);
	sc->is_gdsc_enabled = !(regval & SW_COLLAPSE_MASK);

			return true;
		}
	return 0;
}

	if (sc->parent_regulator)
		regulator_disable(sc->parent_regulator);
static int gdsc_is_enabled(struct regulator_dev *rdev)
{
	struct gdsc *sc = rdev_get_drvdata(rdev);

	return false;
	if (!sc->toggle_logic)
		return !sc->resets_asserted;

	return sc->is_gdsc_enabled;
}

static int gdsc_enable(struct regulator_dev *rdev)
@@ -167,8 +156,10 @@ static int gdsc_enable(struct regulator_dev *rdev)
	uint32_t regval, hw_ctrl_regval = 0x0;
	int i, ret = 0;

	if (sc->root_en || sc->force_root_en)
	if (sc->root_en || sc->force_root_en) {
		clk_prepare_enable(sc->clocks[sc->root_clk_idx]);
		sc->is_root_clk_voted = true;
	}

	regmap_read(sc->regmap, REG_OFFSET, &regval);
	if (regval & HW_CONTROL_MASK) {
@@ -294,8 +285,10 @@ static int gdsc_enable(struct regulator_dev *rdev)
	/* Delay to account for staggered memory powerup. */
	udelay(1);

	if (sc->force_root_en)
	if (sc->force_root_en) {
		clk_disable_unprepare(sc->clocks[sc->root_clk_idx]);
		sc->is_root_clk_voted = false;
	}

	sc->is_gdsc_enabled = true;

@@ -308,8 +301,27 @@ static int gdsc_disable(struct regulator_dev *rdev)
	uint32_t regval;
	int i, ret = 0;

	if (sc->force_root_en)
	if (rdev->supply) {
		regulator_lock(rdev->supply->rdev);
		ret = regulator_is_enabled(rdev->supply);
		if (ret < 0) {
			dev_err(&rdev->dev, "%s unable to check parent enable state, ret=%d\n",
				sc->rdesc.name, ret);
			goto done;
		}

		if (!ret) {
			dev_err(&rdev->dev, "%s cannot disable GDSC while parent is disabled\n",
				sc->rdesc.name);
			ret = -EIO;
			goto done;
		}
	}

	if (sc->force_root_en) {
		clk_prepare_enable(sc->clocks[sc->root_clk_idx]);
		sc->is_root_clk_voted = true;
	}

	/* Delay to account for staggered memory powerdown. */
	udelay(1);
@@ -353,35 +365,40 @@ static int gdsc_disable(struct regulator_dev *rdev)
	 * Check if gdsc_enable was called for this GDSC. If not, the root
	 * clock will not have been enabled prior to this.
	 */
	if ((sc->is_gdsc_enabled && sc->root_en) || sc->force_root_en)
	if ((sc->is_root_clk_voted && sc->root_en) || sc->force_root_en) {
		clk_disable_unprepare(sc->clocks[sc->root_clk_idx]);
		sc->is_root_clk_voted = false;
	}

	sc->is_gdsc_enabled = false;

done:
	if (rdev->supply)
		regulator_unlock(rdev->supply->rdev);

	return ret;
}

static unsigned int gdsc_get_mode(struct regulator_dev *rdev)
static int gdsc_init_hw_ctrl_mode(struct gdsc *sc)
{
	struct gdsc *sc = rdev_get_drvdata(rdev);
	uint32_t regval;
	int ret;

	if (sc->parent_regulator) {
		ret = regulator_enable(sc->parent_regulator);
		if (ret)
	ret = regmap_read(sc->regmap, REG_OFFSET, &regval);
	if (ret < 0)
		return ret;
	}

	regmap_read(sc->regmap, REG_OFFSET, &regval);
	sc->is_gdsc_hw_ctrl_mode = regval & HW_CONTROL_MASK;

	if (sc->parent_regulator)
		regulator_disable(sc->parent_regulator);
	return 0;
}

	if (regval & HW_CONTROL_MASK)
		return REGULATOR_MODE_FAST;
static unsigned int gdsc_get_mode(struct regulator_dev *rdev)
{
	struct gdsc *sc = rdev_get_drvdata(rdev);

	return REGULATOR_MODE_NORMAL;
	return sc->is_gdsc_hw_ctrl_mode ? REGULATOR_MODE_FAST
					: REGULATOR_MODE_NORMAL;
}

static int gdsc_set_mode(struct regulator_dev *rdev, unsigned int mode)
@@ -390,19 +407,41 @@ static int gdsc_set_mode(struct regulator_dev *rdev, unsigned int mode)
	uint32_t regval;
	int ret = 0;

	if (sc->parent_regulator) {
		ret = regulator_enable(sc->parent_regulator);
		if (ret)
			return ret;
	if (rdev->supply) {
		/*
		 * Ensure that the GDSC parent supply is enabled before
		 * continuing.  This is needed to avoid an unclocked access
		 * of the GDSC control register for GDSCs whose register access
		 * is gated by the parent supply enable state in hardware.
		 * Explicit parent supply locking ensures that the parent enable
		 * state cannot change after checking due to a race with another
		 * consumer.
		 */
		regulator_lock(rdev->supply->rdev);
		ret = regulator_is_enabled(rdev->supply);
		if (ret < 0) {
			dev_err(&rdev->dev, "%s unable to check parent enable state, ret=%d\n",
				sc->rdesc.name, ret);
			goto done;
		} else if (WARN(!ret,
				"%s cannot change GDSC HW/SW control mode while parent is disabled\n",
				sc->rdesc.name)) {
			ret = -EIO;
			goto done;
		}
	}

	regmap_read(sc->regmap, REG_OFFSET, &regval);
	ret = regmap_read(sc->regmap, REG_OFFSET, &regval);
	if (ret < 0)
		goto done;

	switch (mode) {
	case REGULATOR_MODE_FAST:
		/* Turn on HW trigger mode */
		regval |= HW_CONTROL_MASK;
		regmap_write(sc->regmap, REG_OFFSET, regval);
		ret = regmap_write(sc->regmap, REG_OFFSET, regval);
		if (ret < 0)
			goto done;
		/*
		 * There may be a race with internal HW trigger signal,
		 * that will result in GDSC going through a power down and
@@ -414,11 +453,14 @@ static int gdsc_set_mode(struct regulator_dev *rdev, unsigned int mode)
		 */
		gdsc_mb(sc);
		udelay(1);
		sc->is_gdsc_hw_ctrl_mode = true;
		break;
	case REGULATOR_MODE_NORMAL:
		/* Turn off HW trigger mode */
		regval &= ~HW_CONTROL_MASK;
		regmap_write(sc->regmap, REG_OFFSET, regval);
		ret = regmap_write(sc->regmap, REG_OFFSET, regval);
		if (ret < 0)
			goto done;
		/*
		 * There may be a race with internal HW trigger signal,
		 * that will result in GDSC going through a power down and
@@ -427,14 +469,16 @@ static int gdsc_set_mode(struct regulator_dev *rdev, unsigned int mode)
		 */
		gdsc_mb(sc);
		udelay(1);
		sc->is_gdsc_hw_ctrl_mode = false;
		break;
	default:
		ret = -EINVAL;
		break;
	}

	if (sc->parent_regulator)
		regulator_disable(sc->parent_regulator);
done:
	if (rdev->supply)
		regulator_unlock(rdev->supply->rdev);

	return ret;
}
@@ -562,17 +606,6 @@ static int gdsc_get_resources(struct gdsc *sc, struct platform_device *pdev)
		return -EINVAL;
	}

	if (of_find_property(dev->of_node, "vdd_parent-supply", NULL)) {
		sc->parent_regulator = devm_regulator_get(dev, "vdd_parent");
		if (IS_ERR(sc->parent_regulator)) {
			ret = PTR_ERR(sc->parent_regulator);
			if (ret != -EPROBE_DEFER)
				dev_err(dev, "Unable to get vdd_parent regulator, ret=%d\n",
					ret);
			return ret;
		}
	}

	sc->clocks = devm_kcalloc(dev, sc->clock_count, sizeof(*sc->clocks),
				  GFP_KERNEL);
	if (sc->clock_count && !sc->clocks)
@@ -682,6 +715,20 @@ static int gdsc_probe(struct platform_device *pdev)
		}
	}

	ret = gdsc_init_is_enabled(sc);
	if (ret) {
		dev_err(dev, "%s failed to get initial enable state, ret=%d\n",
			sc->rdesc.name, ret);
		return ret;
	}

	ret = gdsc_init_hw_ctrl_mode(sc);
	if (ret) {
		dev_err(dev, "%s failed to get initial hw_ctrl state, ret=%d\n",
			sc->rdesc.name, ret);
		return ret;
	}

	sc->rdesc.id = atomic_inc_return(&gdsc_count);
	sc->rdesc.ops = &gdsc_ops;
	sc->rdesc.type = REGULATOR_VOLTAGE;