Commit e0ea2d11 authored by Lucas Stach's avatar Lucas Stach Committed by Shawn Guo

soc: imx: gpc: fix power up sequencing

Currently we wait only until the PGC inverts the isolation setting
before disabling the peripheral clocks. This doesn't ensure that the
reset is properly propagated through the peripheral devices in the
power domain.

Wait until the PGC signals that the power up request is done and
wait a bit for resets to propagate before disabling the clocks.
Signed-off-by: default avatarLucas Stach <l.stach@pengutronix.de>
Signed-off-by: default avatarShawn Guo <shawnguo@kernel.org>
parent fc40200e
...@@ -87,8 +87,8 @@ static int imx6_pm_domain_power_off(struct generic_pm_domain *genpd) ...@@ -87,8 +87,8 @@ static int imx6_pm_domain_power_off(struct generic_pm_domain *genpd)
static int imx6_pm_domain_power_on(struct generic_pm_domain *genpd) static int imx6_pm_domain_power_on(struct generic_pm_domain *genpd)
{ {
struct imx_pm_domain *pd = to_imx_pm_domain(genpd); struct imx_pm_domain *pd = to_imx_pm_domain(genpd);
int i, ret, sw, sw2iso; int i, ret;
u32 val; u32 val, req;
if (pd->supply) { if (pd->supply) {
ret = regulator_enable(pd->supply); ret = regulator_enable(pd->supply);
...@@ -107,17 +107,18 @@ static int imx6_pm_domain_power_on(struct generic_pm_domain *genpd) ...@@ -107,17 +107,18 @@ static int imx6_pm_domain_power_on(struct generic_pm_domain *genpd)
regmap_update_bits(pd->regmap, pd->reg_offs + GPC_PGC_CTRL_OFFS, regmap_update_bits(pd->regmap, pd->reg_offs + GPC_PGC_CTRL_OFFS,
0x1, 0x1); 0x1, 0x1);
/* Read ISO and ISO2SW power up delays */
regmap_read(pd->regmap, pd->reg_offs + GPC_PGC_PUPSCR_OFFS, &val);
sw = val & 0x3f;
sw2iso = (val >> 8) & 0x3f;
/* Request GPC to power up domain */ /* Request GPC to power up domain */
val = BIT(pd->cntr_pdn_bit + 1); req = BIT(pd->cntr_pdn_bit + 1);
regmap_update_bits(pd->regmap, GPC_CNTR, val, val); regmap_update_bits(pd->regmap, GPC_CNTR, req, req);
/* Wait ISO + ISO2SW IPG clock cycles */ /* Wait for the PGC to handle the request */
udelay(DIV_ROUND_UP(sw + sw2iso, pd->ipg_rate_mhz)); ret = regmap_read_poll_timeout(pd->regmap, GPC_CNTR, val, !(val & req),
1, 50);
if (ret)
pr_err("powerup request on domain %s timed out\n", genpd->name);
/* Wait for reset to propagate through peripherals */
usleep_range(5, 10);
/* Disable reset clocks for all devices in the domain */ /* Disable reset clocks for all devices in the domain */
for (i = 0; i < pd->num_clks; i++) for (i = 0; i < pd->num_clks; i++)
...@@ -343,6 +344,7 @@ static const struct regmap_config imx_gpc_regmap_config = { ...@@ -343,6 +344,7 @@ static const struct regmap_config imx_gpc_regmap_config = {
.rd_table = &access_table, .rd_table = &access_table,
.wr_table = &access_table, .wr_table = &access_table,
.max_register = 0x2ac, .max_register = 0x2ac,
.fast_io = true,
}; };
static struct generic_pm_domain *imx_gpc_onecell_domains[] = { static struct generic_pm_domain *imx_gpc_onecell_domains[] = {
......
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