drm/omap: dsi: Fix PM for display blank with paired dss_pll calls

Currently dsi_display_init_dsi() calls dss_pll_enable() but it is not
paired with dss_pll_disable() in dsi_display_uninit_dsi(). This leaves
the DSS clocks enabled when the display is blanked wasting about extra
5mW of power while idle.

The clock that is left on by not calling dss_pll_disable() is
DSS_CLKCTRL bit 10 OPTFCLKEN_SYS_CLK that is the source clock for
DSI PLL.

We can fix this issue by by making the current dsi_pll_uninit() into
dsi_pll_disable(). This way we can just call dss_pll_disable() from
dsi_display_uninit_dsi() and the code becomes a bit easier to follow.

However, we need to also consider that DSI PLL can be muxed for DVI too
as pointed out by Tomi Valkeinen <tomi.valkeinen@ti.com>. In the DVI
case, we want to unconditionally disable the clocks. To get around this
issue, we separate out the DSI lane handling from dsi_pll_enable() and
dsi_pll_disable() as suggested by Tomi in an earlier experimental patch.

So we must only toggle the DSI regulator based on the vdds_dsi_enabled
flag from dsi_display_init_dsi() and dsi_display_uninit_dsi().

We need to make these two changes together to avoid breaking things
for DVI when fixing the DSI clock handling. And this all causes a
slight renumbering of the error path for dsi_display_init_dsi().

Suggested-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
Signed-off-by: Tony Lindgren <tony@atomide.com>
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
This commit is contained in:
Tony Lindgren 2019-02-07 07:45:16 -08:00 committed by Tomi Valkeinen
parent a0970e87b5
commit fe4ed1b457
1 changed files with 33 additions and 31 deletions

View File

@ -1342,12 +1342,9 @@ static int dsi_pll_enable(struct dss_pll *pll)
*/ */
dsi_enable_scp_clk(dsi); dsi_enable_scp_clk(dsi);
if (!dsi->vdds_dsi_enabled) { r = regulator_enable(dsi->vdds_dsi_reg);
r = regulator_enable(dsi->vdds_dsi_reg); if (r)
if (r) goto err0;
goto err0;
dsi->vdds_dsi_enabled = true;
}
/* XXX PLL does not come out of reset without this... */ /* XXX PLL does not come out of reset without this... */
dispc_pck_free_enable(dsi->dss->dispc, 1); dispc_pck_free_enable(dsi->dss->dispc, 1);
@ -1372,36 +1369,25 @@ static int dsi_pll_enable(struct dss_pll *pll)
return 0; return 0;
err1: err1:
if (dsi->vdds_dsi_enabled) { regulator_disable(dsi->vdds_dsi_reg);
regulator_disable(dsi->vdds_dsi_reg);
dsi->vdds_dsi_enabled = false;
}
err0: err0:
dsi_disable_scp_clk(dsi); dsi_disable_scp_clk(dsi);
dsi_runtime_put(dsi); dsi_runtime_put(dsi);
return r; return r;
} }
static void dsi_pll_uninit(struct dsi_data *dsi, bool disconnect_lanes)
{
dsi_pll_power(dsi, DSI_PLL_POWER_OFF);
if (disconnect_lanes) {
WARN_ON(!dsi->vdds_dsi_enabled);
regulator_disable(dsi->vdds_dsi_reg);
dsi->vdds_dsi_enabled = false;
}
dsi_disable_scp_clk(dsi);
dsi_runtime_put(dsi);
DSSDBG("PLL uninit done\n");
}
static void dsi_pll_disable(struct dss_pll *pll) static void dsi_pll_disable(struct dss_pll *pll)
{ {
struct dsi_data *dsi = container_of(pll, struct dsi_data, pll); struct dsi_data *dsi = container_of(pll, struct dsi_data, pll);
dsi_pll_uninit(dsi, true); dsi_pll_power(dsi, DSI_PLL_POWER_OFF);
regulator_disable(dsi->vdds_dsi_reg);
dsi_disable_scp_clk(dsi);
dsi_runtime_put(dsi);
DSSDBG("PLL disable done\n");
} }
static int dsi_dump_dsi_clocks(struct seq_file *s, void *p) static int dsi_dump_dsi_clocks(struct seq_file *s, void *p)
@ -4089,11 +4075,11 @@ static int dsi_display_init_dsi(struct dsi_data *dsi)
r = dss_pll_enable(&dsi->pll); r = dss_pll_enable(&dsi->pll);
if (r) if (r)
goto err0; return r;
r = dsi_configure_dsi_clocks(dsi); r = dsi_configure_dsi_clocks(dsi);
if (r) if (r)
goto err1; goto err0;
dss_select_dsi_clk_source(dsi->dss, dsi->module_id, dss_select_dsi_clk_source(dsi->dss, dsi->module_id,
dsi->module_id == 0 ? dsi->module_id == 0 ?
@ -4101,6 +4087,14 @@ static int dsi_display_init_dsi(struct dsi_data *dsi)
DSSDBG("PLL OK\n"); DSSDBG("PLL OK\n");
if (!dsi->vdds_dsi_enabled) {
r = regulator_enable(dsi->vdds_dsi_reg);
if (r)
goto err1;
dsi->vdds_dsi_enabled = true;
}
r = dsi_cio_init(dsi); r = dsi_cio_init(dsi);
if (r) if (r)
goto err2; goto err2;
@ -4129,10 +4123,13 @@ static int dsi_display_init_dsi(struct dsi_data *dsi)
err3: err3:
dsi_cio_uninit(dsi); dsi_cio_uninit(dsi);
err2: err2:
dss_select_dsi_clk_source(dsi->dss, dsi->module_id, DSS_CLK_SRC_FCK); regulator_disable(dsi->vdds_dsi_reg);
dsi->vdds_dsi_enabled = false;
err1: err1:
dss_pll_disable(&dsi->pll); dss_select_dsi_clk_source(dsi->dss, dsi->module_id, DSS_CLK_SRC_FCK);
err0: err0:
dss_pll_disable(&dsi->pll);
return r; return r;
} }
@ -4151,7 +4148,12 @@ static void dsi_display_uninit_dsi(struct dsi_data *dsi, bool disconnect_lanes,
dss_select_dsi_clk_source(dsi->dss, dsi->module_id, DSS_CLK_SRC_FCK); dss_select_dsi_clk_source(dsi->dss, dsi->module_id, DSS_CLK_SRC_FCK);
dsi_cio_uninit(dsi); dsi_cio_uninit(dsi);
dsi_pll_uninit(dsi, disconnect_lanes); dss_pll_disable(&dsi->pll);
if (disconnect_lanes) {
regulator_disable(dsi->vdds_dsi_reg);
dsi->vdds_dsi_enabled = false;
}
} }
static void dsi_display_enable(struct omap_dss_device *dssdev) static void dsi_display_enable(struct omap_dss_device *dssdev)