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

Commit b0008fd8 authored by Padmanabhan Komanduru's avatar Padmanabhan Komanduru
Browse files

msm: mdss: update DSI ULPS entry/exit sequence



Update the DSI ULPS sequence as per h/w recommended sequence
to add support for ULPS feature for SDM439 target.

Change-Id: Id57263ac300057aab6f0549d0ece747b5f1f2098
Signed-off-by: default avatarPadmanabhan Komanduru <pkomandu@codeaurora.org>
parent f5c48058
Loading
Loading
Loading
Loading
+3 −3
Original line number Original line Diff line number Diff line
@@ -2989,9 +2989,9 @@ static int mdss_dsi_ctrl_clock_init(struct platform_device *ctrl_pdev,
	info.core_clks.mmss_misc_ahb_clk =
	info.core_clks.mmss_misc_ahb_clk =
		ctrl_pdata->shared_data->mmss_misc_ahb_clk;
		ctrl_pdata->shared_data->mmss_misc_ahb_clk;


	info.link_clks.esc_clk = ctrl_pdata->esc_clk;
	info.link_lp_clks.esc_clk = ctrl_pdata->esc_clk;
	info.link_clks.byte_clk = ctrl_pdata->byte_clk;
	info.link_hs_clks.byte_clk = ctrl_pdata->byte_clk;
	info.link_clks.pixel_clk = ctrl_pdata->pixel_clk;
	info.link_hs_clks.pixel_clk = ctrl_pdata->pixel_clk;


	info.pre_clkoff_cb = mdss_dsi_pre_clkoff_cb;
	info.pre_clkoff_cb = mdss_dsi_pre_clkoff_cb;
	info.post_clkon_cb = mdss_dsi_post_clkon_cb;
	info.post_clkon_cb = mdss_dsi_post_clkon_cb;
+4 −0
Original line number Original line Diff line number Diff line
@@ -616,15 +616,19 @@ void mdss_dsi_shadow_clk_deinit(struct device *dev,
			struct mdss_dsi_ctrl_pdata *ctrl_pdata);
			struct mdss_dsi_ctrl_pdata *ctrl_pdata);
int mdss_dsi_pre_clkoff_cb(void *priv,
int mdss_dsi_pre_clkoff_cb(void *priv,
			   enum mdss_dsi_clk_type clk_type,
			   enum mdss_dsi_clk_type clk_type,
			   enum mdss_dsi_lclk_type l_type,
			   enum mdss_dsi_clk_state new_state);
			   enum mdss_dsi_clk_state new_state);
int mdss_dsi_post_clkoff_cb(void *priv,
int mdss_dsi_post_clkoff_cb(void *priv,
			    enum mdss_dsi_clk_type clk_type,
			    enum mdss_dsi_clk_type clk_type,
			    enum mdss_dsi_lclk_type l_type,
			    enum mdss_dsi_clk_state curr_state);
			    enum mdss_dsi_clk_state curr_state);
int mdss_dsi_post_clkon_cb(void *priv,
int mdss_dsi_post_clkon_cb(void *priv,
			   enum mdss_dsi_clk_type clk_type,
			   enum mdss_dsi_clk_type clk_type,
			   enum mdss_dsi_lclk_type l_type,
			   enum mdss_dsi_clk_state curr_state);
			   enum mdss_dsi_clk_state curr_state);
int mdss_dsi_pre_clkon_cb(void *priv,
int mdss_dsi_pre_clkon_cb(void *priv,
			  enum mdss_dsi_clk_type clk_type,
			  enum mdss_dsi_clk_type clk_type,
			  enum mdss_dsi_lclk_type l_type,
			  enum mdss_dsi_clk_state new_state);
			  enum mdss_dsi_clk_state new_state);
int mdss_dsi_panel_reset(struct mdss_panel_data *pdata, int enable);
int mdss_dsi_panel_reset(struct mdss_panel_data *pdata, int enable);
void mdss_dsi_phy_disable(struct mdss_dsi_ctrl_pdata *ctrl);
void mdss_dsi_phy_disable(struct mdss_dsi_ctrl_pdata *ctrl);
+240 −106
Original line number Original line Diff line number Diff line
@@ -27,11 +27,9 @@ struct dsi_core_clks {
};
};


struct dsi_link_clks {
struct dsi_link_clks {
	struct mdss_dsi_link_clk_info clks;
	struct mdss_dsi_link_hs_clk_info hs_clks;
	struct mdss_dsi_link_lp_clk_info lp_clks;
	u32 current_clk_state;
	u32 current_clk_state;
	u32 byte_clk_rate;
	u32 pix_clk_rate;
	u32 esc_clk_rate;
};
};


struct mdss_dsi_clk_mngr {
struct mdss_dsi_clk_mngr {
@@ -73,28 +71,27 @@ static int dsi_core_clk_start(struct dsi_core_clks *c_clks)


	rc = clk_prepare_enable(c_clks->clks.mdp_core_clk);
	rc = clk_prepare_enable(c_clks->clks.mdp_core_clk);
	if (rc) {
	if (rc) {
		pr_err("%s: failed to enable mdp_core_clock. rc=%d\n",
		pr_err("failed to enable mdp_core_clock. rc=%d\n", rc);
							 __func__, rc);
		goto error;
		goto error;
	}
	}


	rc = clk_prepare_enable(c_clks->clks.ahb_clk);
	rc = clk_prepare_enable(c_clks->clks.ahb_clk);
	if (rc) {
	if (rc) {
		pr_err("%s: failed to enable ahb clock. rc=%d\n", __func__, rc);
		pr_err("failed to enable ahb clock. rc=%d\n", rc);
		goto disable_core_clk;
		goto disable_core_clk;
	}
	}


	rc = clk_prepare_enable(c_clks->clks.axi_clk);
	rc = clk_prepare_enable(c_clks->clks.axi_clk);
	if (rc) {
	if (rc) {
		pr_err("%s: failed to enable ahb clock. rc=%d\n", __func__, rc);
		pr_err("failed to enable ahb clock. rc=%d\n", rc);
		goto disable_ahb_clk;
		goto disable_ahb_clk;
	}
	}


	if (c_clks->clks.mmss_misc_ahb_clk) {
	if (c_clks->clks.mmss_misc_ahb_clk) {
		rc = clk_prepare_enable(c_clks->clks.mmss_misc_ahb_clk);
		rc = clk_prepare_enable(c_clks->clks.mmss_misc_ahb_clk);
		if (rc) {
		if (rc) {
			pr_err("%s: failed to enable mmss misc ahb clk.rc=%d\n",
			pr_err("failed to enable mmss misc ahb clk.rc=%d\n",
				__func__, rc);
				rc);
			goto disable_axi_clk;
			goto disable_axi_clk;
		}
		}
	}
	}
@@ -140,12 +137,15 @@ static int dsi_core_clk_stop(struct dsi_core_clks *c_clks)
	return rc;
	return rc;
}
}


static int dsi_link_clk_set_rate(struct dsi_link_clks *l_clks)
static int dsi_link_hs_clk_set_rate(
	struct mdss_dsi_link_hs_clk_info *link_hs_clks)
{
{
	int rc = 0;
	int rc = 0;
	struct mdss_dsi_clk_mngr *mngr;
	struct mdss_dsi_clk_mngr *mngr;
	struct dsi_link_clks *l_clks;
	struct mdss_dsi_ctrl_pdata *ctrl;
	struct mdss_dsi_ctrl_pdata *ctrl;


	l_clks = container_of(link_hs_clks, struct dsi_link_clks, hs_clks);
	mngr = container_of(l_clks, struct mdss_dsi_clk_mngr, link_clks);
	mngr = container_of(l_clks, struct mdss_dsi_clk_mngr, link_clks);


	/*
	/*
@@ -160,19 +160,13 @@ static int dsi_link_clk_set_rate(struct dsi_link_clks *l_clks)
	if (ctrl->panel_data.panel_info.cont_splash_enabled)
	if (ctrl->panel_data.panel_info.cont_splash_enabled)
		return 0;
		return 0;


	rc = clk_set_rate(l_clks->clks.esc_clk, l_clks->esc_clk_rate);
	rc = clk_set_rate(link_hs_clks->byte_clk, link_hs_clks->byte_clk_rate);
	if (rc) {
		pr_err("clk_set_rate failed for esc_clk rc = %d\n", rc);
		goto error;
	}

	rc = clk_set_rate(l_clks->clks.byte_clk, l_clks->byte_clk_rate);
	if (rc) {
	if (rc) {
		pr_err("clk_set_rate failed for byte_clk rc = %d\n", rc);
		pr_err("clk_set_rate failed for byte_clk rc = %d\n", rc);
		goto error;
		goto error;
	}
	}


	rc = clk_set_rate(l_clks->clks.pixel_clk, l_clks->pix_clk_rate);
	rc = clk_set_rate(link_hs_clks->pixel_clk, link_hs_clks->pix_clk_rate);
	if (rc) {
	if (rc) {
		pr_err("clk_set_rate failed for pixel_clk rc = %d\n", rc);
		pr_err("clk_set_rate failed for pixel_clk rc = %d\n", rc);
		goto error;
		goto error;
@@ -182,141 +176,203 @@ static int dsi_link_clk_set_rate(struct dsi_link_clks *l_clks)
	return rc;
	return rc;
}
}


static int dsi_link_clk_prepare(struct dsi_link_clks *l_clks)
static int dsi_link_hs_clk_prepare(
	struct mdss_dsi_link_hs_clk_info *link_hs_clks)
{
{
	int rc = 0;
	int rc = 0;


	rc = clk_prepare(l_clks->clks.esc_clk);
	rc = clk_prepare(link_hs_clks->byte_clk);
	if (rc) {
		pr_err("%s: Failed to prepare dsi esc clk\n", __func__);
		goto esc_clk_err;
	}

	rc = clk_prepare(l_clks->clks.byte_clk);
	if (rc) {
	if (rc) {
		pr_err("%s: Failed to prepare dsi byte clk\n", __func__);
		pr_err("Failed to prepare dsi byte clk\n");
		goto byte_clk_err;
		goto byte_clk_err;
	}
	}


	rc = clk_prepare(l_clks->clks.pixel_clk);
	rc = clk_prepare(link_hs_clks->pixel_clk);
	if (rc) {
	if (rc) {
		pr_err("%s: Failed to prepare dsi pixel clk\n", __func__);
		pr_err("Failed to prepare dsi pixel_clk\n");
		goto pixel_clk_err;
		goto pixel_clk_err;
	}
	}


	return rc;
	return rc;


pixel_clk_err:
pixel_clk_err:
	clk_unprepare(l_clks->clks.byte_clk);
	clk_unprepare(link_hs_clks->byte_clk);
byte_clk_err:
byte_clk_err:
	clk_unprepare(l_clks->clks.esc_clk);
esc_clk_err:
	return rc;
	return rc;
}
}


static int dsi_link_clk_unprepare(struct dsi_link_clks *l_clks)
static int dsi_link_hs_clk_unprepare(
	struct mdss_dsi_link_hs_clk_info *link_hs_clks)
{
{
	int rc = 0;
	int rc = 0;


	clk_unprepare(l_clks->clks.pixel_clk);
	clk_unprepare(link_hs_clks->pixel_clk);
	clk_unprepare(l_clks->clks.byte_clk);
	clk_unprepare(link_hs_clks->byte_clk);
	clk_unprepare(l_clks->clks.esc_clk);


	return rc;
	return rc;
}
}


static int dsi_link_clk_enable(struct dsi_link_clks *l_clks)
static int dsi_link_hs_clk_enable(
	struct mdss_dsi_link_hs_clk_info *link_hs_clks)
{
{
	int rc = 0;
	int rc = 0;


	rc = clk_enable(l_clks->clks.esc_clk);
	rc = clk_enable(link_hs_clks->byte_clk);
	if (rc) {
	if (rc) {
		pr_err("%s: Failed to enable dsi esc clk\n", __func__);
		pr_err("Failed to enable dsi byte clk\n");
		goto esc_clk_err;
	}

	rc = clk_enable(l_clks->clks.byte_clk);
	if (rc) {
		pr_err("%s: Failed to enable dsi byte clk\n", __func__);
		goto byte_clk_err;
		goto byte_clk_err;
	}
	}


	rc = clk_enable(l_clks->clks.pixel_clk);
	rc = clk_enable(link_hs_clks->pixel_clk);
	if (rc) {
	if (rc) {
		pr_err("%s: Failed to enable dsi pixel clk\n", __func__);
		pr_err("Failed to enable dsi pixel_clk\n");
		goto pixel_clk_err;
		goto pixel_clk_err;
	}
	}


	return rc;
	return rc;


pixel_clk_err:
pixel_clk_err:
	clk_disable(l_clks->clks.byte_clk);
	clk_disable(link_hs_clks->byte_clk);
byte_clk_err:
byte_clk_err:
	clk_disable(l_clks->clks.esc_clk);
esc_clk_err:
	return rc;
	return rc;
}
}


static int dsi_link_clk_disable(struct dsi_link_clks *l_clks)
static int dsi_link_hs_clk_disable(
	struct mdss_dsi_link_hs_clk_info *link_hs_clks)
{
{
	int rc = 0;
	int rc = 0;


	clk_disable(l_clks->clks.esc_clk);
	clk_disable(link_hs_clks->pixel_clk);
	clk_disable(l_clks->clks.pixel_clk);
	clk_disable(link_hs_clks->byte_clk);
	clk_disable(l_clks->clks.byte_clk);


	return rc;
	return rc;
}
}




static int dsi_link_clk_start(struct dsi_link_clks *l_clks)
static int dsi_link_hs_clk_start(
	struct mdss_dsi_link_hs_clk_info *link_hs_clks,
	enum mdss_dsi_link_clk_op_type op_type)
{
{
	int rc = 0;
	int rc = 0;
	struct dsi_link_clks *l_clks;
	struct mdss_dsi_clk_mngr *mngr;
	struct mdss_dsi_clk_mngr *mngr;


	l_clks = container_of(link_hs_clks, struct dsi_link_clks, hs_clks);
	mngr = container_of(l_clks, struct mdss_dsi_clk_mngr, link_clks);
	mngr = container_of(l_clks, struct mdss_dsi_clk_mngr, link_clks);


	rc = dsi_link_clk_set_rate(l_clks);
	if (op_type & MDSS_DSI_LINK_CLK_SET_RATE) {
		rc = dsi_link_hs_clk_set_rate(link_hs_clks);
		if (rc) {
		if (rc) {
		pr_err("failed to set clk rates, rc = %d\n", rc);
			pr_err("failed to set HS clk rates, rc = %d\n", rc);
			goto error;
			goto error;
		}
		}
	}


	rc = dsi_link_clk_prepare(l_clks);
	if (op_type & MDSS_DSI_LINK_CLK_PREPARE) {
		rc = dsi_link_hs_clk_prepare(link_hs_clks);
		if (rc) {
		if (rc) {
		pr_err("failed to prepare link clks, rc = %d\n", rc);
			pr_err("failed to prepare link HS clks, rc = %d\n", rc);
			goto error;
			goto error;
		}
		}
	}


	rc = dsi_link_clk_enable(l_clks);
	if (op_type & MDSS_DSI_LINK_CLK_ENABLE) {
		rc = dsi_link_hs_clk_enable(link_hs_clks);
		if (rc) {
		if (rc) {
		pr_err("failed to enable link clks, rc = %d\n", rc);
			pr_err("failed to enable link HS clks, rc = %d\n", rc);
			goto error_unprepare;
			goto error_unprepare;
		}
		}
	}


	pr_debug("%s: LINK CLOCK IS ON\n", mngr->name);
	pr_debug("%s: LINK HS CLOCK IS ON\n", mngr->name);
	return rc;
	return rc;
error_unprepare:
error_unprepare:
	dsi_link_clk_unprepare(l_clks);
	dsi_link_hs_clk_unprepare(link_hs_clks);
error:
error:
	return rc;
	return rc;
}
}


static int dsi_link_clk_stop(struct dsi_link_clks *l_clks)
static int dsi_link_lp_clk_start(
	struct mdss_dsi_link_lp_clk_info *link_lp_clks)
{
{
	int rc = 0;
	int rc = 0;
	struct mdss_dsi_clk_mngr *mngr;
	struct mdss_dsi_clk_mngr *mngr;
	struct dsi_link_clks *l_clks;
	struct mdss_dsi_ctrl_pdata *ctrl;


	l_clks = container_of(link_lp_clks, struct dsi_link_clks, lp_clks);
	mngr = container_of(l_clks, struct mdss_dsi_clk_mngr, link_clks);
	mngr = container_of(l_clks, struct mdss_dsi_clk_mngr, link_clks);
	/*
	 * In an ideal world, cont_splash_enabled should not be required inside
	 * the clock manager. But, in the current driver cont_splash_enabled
	 * flag is set inside mdp driver and there is no interface event
	 * associated with this flag setting. Also, set rate for clock need not
	 * be called for every enable call. It should be done only once when
	 * coming out of suspend.
	 */
	ctrl = mngr->priv_data;
	if (ctrl->panel_data.panel_info.cont_splash_enabled)
		goto prepare;


	(void)dsi_link_clk_disable(l_clks);
	rc = clk_set_rate(link_lp_clks->esc_clk, link_lp_clks->esc_clk_rate);
	if (rc) {
		pr_err("clk_set_rate failed for esc_clk rc = %d\n", rc);
		goto error;
	}

prepare:
	rc = clk_prepare(link_lp_clks->esc_clk);
	if (rc) {
		pr_err("Failed to prepare dsi esc clk\n");
		goto error;
	}

	rc = clk_enable(link_lp_clks->esc_clk);
	if (rc) {
		pr_err("Failed to enable dsi esc clk\n");
		clk_unprepare(l_clks->lp_clks.esc_clk);
		goto error;
	}
error:
	pr_debug("%s: LINK LP CLOCK IS ON\n", mngr->name);
	return rc;
}

static int dsi_link_hs_clk_stop(
	struct mdss_dsi_link_hs_clk_info *link_hs_clks)
{
	int rc = 0;
	struct dsi_link_clks *l_clks;
	struct mdss_dsi_clk_mngr *mngr;

	l_clks = container_of(link_hs_clks, struct dsi_link_clks, hs_clks);
	mngr = container_of(l_clks, struct mdss_dsi_clk_mngr, link_clks);


	(void)dsi_link_clk_unprepare(l_clks);
	(void)dsi_link_hs_clk_disable(link_hs_clks);
	pr_debug("%s: LINK CLOCK IS OFF\n", mngr->name);

	(void)dsi_link_hs_clk_unprepare(link_hs_clks);
	pr_debug("%s: LINK HS CLOCK IS OFF\n", mngr->name);


	return rc;
	return rc;
}
}


static int dsi_link_lp_clk_stop(
	struct mdss_dsi_link_lp_clk_info *link_lp_clks)
{
	struct dsi_link_clks *l_clks;
	struct mdss_dsi_clk_mngr *mngr;

	l_clks = container_of(link_lp_clks, struct dsi_link_clks, lp_clks);
	mngr = container_of(l_clks, struct mdss_dsi_clk_mngr, link_clks);

	clk_disable(l_clks->lp_clks.esc_clk);
	clk_unprepare(l_clks->lp_clks.esc_clk);

	pr_debug("%s: LINK LP CLOCK IS OFF\n", mngr->name);
	return 0;
}


static int dsi_update_clk_state(struct dsi_core_clks *c_clks, u32 c_state,
static int dsi_update_clk_state(struct dsi_core_clks *c_clks, u32 c_state,
				struct dsi_link_clks *l_clks, u32 l_state)
				struct dsi_link_clks *l_clks, u32 l_state)
{
{
@@ -347,7 +403,7 @@ static int dsi_update_clk_state(struct dsi_core_clks *c_clks, u32 c_state,
	if (c_clks && (c_state == MDSS_DSI_CLK_ON)) {
	if (c_clks && (c_state == MDSS_DSI_CLK_ON)) {
		if (c_clks->current_clk_state == MDSS_DSI_CLK_OFF) {
		if (c_clks->current_clk_state == MDSS_DSI_CLK_OFF) {
			rc = mngr->pre_clkon_cb(mngr->priv_data,
			rc = mngr->pre_clkon_cb(mngr->priv_data,
						MDSS_DSI_CORE_CLK,
				MDSS_DSI_CORE_CLK, MDSS_DSI_LINK_NONE,
				MDSS_DSI_CLK_ON);
				MDSS_DSI_CLK_ON);
			if (rc) {
			if (rc) {
				pr_err("failed to turn on MDP FS rc= %d\n", rc);
				pr_err("failed to turn on MDP FS rc= %d\n", rc);
@@ -362,7 +418,7 @@ static int dsi_update_clk_state(struct dsi_core_clks *c_clks, u32 c_state,


		if (mngr->post_clkon_cb) {
		if (mngr->post_clkon_cb) {
			rc = mngr->post_clkon_cb(mngr->priv_data,
			rc = mngr->post_clkon_cb(mngr->priv_data,
						 MDSS_DSI_CORE_CLK,
				 MDSS_DSI_CORE_CLK, MDSS_DSI_LINK_NONE,
				 MDSS_DSI_CLK_ON);
				 MDSS_DSI_CLK_ON);
			if (rc)
			if (rc)
				pr_err("post clk on cb failed, rc = %d\n", rc);
				pr_err("post clk on cb failed, rc = %d\n", rc);
@@ -375,21 +431,50 @@ static int dsi_update_clk_state(struct dsi_core_clks *c_clks, u32 c_state,
		if (l_state == MDSS_DSI_CLK_ON) {
		if (l_state == MDSS_DSI_CLK_ON) {
			if (mngr->pre_clkon_cb) {
			if (mngr->pre_clkon_cb) {
				rc = mngr->pre_clkon_cb(mngr->priv_data,
				rc = mngr->pre_clkon_cb(mngr->priv_data,
					MDSS_DSI_LINK_CLK, l_state);
					MDSS_DSI_LINK_CLK, MDSS_DSI_LINK_LP_CLK,
					l_state);
				if (rc)
				if (rc)
					pr_err("pre link clk on cb failed\n");
					pr_err("pre link LP clk on cb failed\n");
			}
			}
			rc = dsi_link_clk_start(l_clks);
			rc = dsi_link_lp_clk_start(&l_clks->lp_clks);
			if (rc) {
			if (rc) {
				pr_err("failed to start link clk rc= %d\n", rc);
				pr_err("failed to start LP link clk clk\n");
				goto error;
				goto error;
			}
			}
			if (mngr->post_clkon_cb) {
			if (mngr->post_clkon_cb) {
				rc = mngr->post_clkon_cb(mngr->priv_data,
				rc = mngr->post_clkon_cb(mngr->priv_data,
							MDSS_DSI_LINK_CLK,
					MDSS_DSI_LINK_CLK, MDSS_DSI_LINK_LP_CLK,
					l_state);
				if (rc)
					pr_err("post LP clk on cb failed\n");
			}

			if (mngr->pre_clkon_cb) {
				rc = mngr->pre_clkon_cb(mngr->priv_data,
					MDSS_DSI_LINK_CLK, MDSS_DSI_LINK_HS_CLK,
					l_state);
					l_state);
				if (rc)
				if (rc)
					pr_err("post link clk on cb failed\n");
					pr_err("pre HS clk on cb failed\n");
			}
			rc = dsi_link_hs_clk_start(&l_clks->hs_clks,
				(MDSS_DSI_LINK_CLK_SET_RATE |
				MDSS_DSI_LINK_CLK_PREPARE));
			if (rc) {
				pr_err("failed to prepare HS clk rc= %d\n", rc);
				goto error;
			}
			if (mngr->post_clkon_cb) {
				rc = mngr->post_clkon_cb(mngr->priv_data,
					MDSS_DSI_LINK_CLK, MDSS_DSI_LINK_HS_CLK,
					l_state);
				if (rc)
					pr_err("post HS clk on cb failed\n");
			}
			rc = dsi_link_hs_clk_start(&l_clks->hs_clks,
				MDSS_DSI_LINK_CLK_ENABLE);
			if (rc) {
				pr_err("failed to enable HS clk rc= %d\n", rc);
				goto error;
			}
			}
		} else {
		} else {
			/*
			/*
@@ -418,9 +503,16 @@ static int dsi_update_clk_state(struct dsi_core_clks *c_clks, u32 c_state,
					goto error;
					goto error;
				}
				}


				rc = dsi_link_clk_start(l_clks);
				rc = dsi_link_lp_clk_start(&l_clks->lp_clks);
				if (rc) {
					pr_err("LP Link clks did not start\n");
					goto error;
				}

				rc = dsi_link_hs_clk_start(&l_clks->hs_clks,
						MDSS_DSI_LINK_CLK_START);
				if (rc) {
				if (rc) {
					pr_err("Link clks did not start\n");
					pr_err("HS Link clks did not start\n");
					goto error;
					goto error;
				}
				}
				l_c_on = true;
				l_c_on = true;
@@ -429,24 +521,50 @@ static int dsi_update_clk_state(struct dsi_core_clks *c_clks, u32 c_state,


			if (mngr->pre_clkoff_cb) {
			if (mngr->pre_clkoff_cb) {
				rc = mngr->pre_clkoff_cb(mngr->priv_data,
				rc = mngr->pre_clkoff_cb(mngr->priv_data,
					MDSS_DSI_LINK_CLK, l_state);
					MDSS_DSI_LINK_CLK, MDSS_DSI_LINK_HS_CLK,
					l_state);
				if (rc)
					pr_err("pre HS clk off cb failed\n");
			}

			rc = dsi_link_hs_clk_stop(&l_clks->hs_clks);
			if (rc) {
				pr_err("failed to stop HS clk, rc = %d\n",
				       rc);
				goto error;
			}

			if (mngr->post_clkoff_cb) {
				rc = mngr->post_clkoff_cb(mngr->priv_data,
					MDSS_DSI_LINK_CLK, MDSS_DSI_LINK_HS_CLK,
					l_state);
				if (rc)
					pr_err("post HS clk off cb failed\n");
			}

			if (mngr->pre_clkoff_cb) {
				rc = mngr->pre_clkoff_cb(mngr->priv_data,
					MDSS_DSI_LINK_CLK, MDSS_DSI_LINK_LP_CLK,
					l_state);
				if (rc)
				if (rc)
					pr_err("pre link clk off cb failed\n");
					pr_err("pre LP clk off cb failed\n");
			}
			}


			rc = dsi_link_clk_stop(l_clks);
			rc = dsi_link_lp_clk_stop(&l_clks->lp_clks);
			if (rc) {
			if (rc) {
				pr_err("failed to stop link clk, rc = %d\n",
				pr_err("failed to stop LP link clk, rc = %d\n",
				       rc);
				       rc);
				goto error;
				goto error;
			}
			}


			if (mngr->post_clkoff_cb) {
			if (mngr->post_clkoff_cb) {
				rc = mngr->post_clkoff_cb(mngr->priv_data,
				rc = mngr->post_clkoff_cb(mngr->priv_data,
					MDSS_DSI_LINK_CLK, l_state);
					MDSS_DSI_LINK_CLK, MDSS_DSI_LINK_LP_CLK,
					l_state);
				if (rc)
				if (rc)
					pr_err("post link clk off cb failed\n");
					pr_err("post LP clk off cb failed\n");
			}
			}

			/*
			/*
			 * This check is to save unnecessary clock state
			 * This check is to save unnecessary clock state
			 * change when going from EARLY_GATE to OFF. In the
			 * change when going from EARLY_GATE to OFF. In the
@@ -502,7 +620,7 @@ static int dsi_update_clk_state(struct dsi_core_clks *c_clks, u32 c_state,


		if (mngr->pre_clkoff_cb) {
		if (mngr->pre_clkoff_cb) {
			rc = mngr->pre_clkoff_cb(mngr->priv_data,
			rc = mngr->pre_clkoff_cb(mngr->priv_data,
						 MDSS_DSI_CORE_CLK,
				 MDSS_DSI_CORE_CLK, MDSS_DSI_LINK_NONE,
				 c_state);
				 c_state);
			if (rc)
			if (rc)
				pr_err("pre core clk off cb failed\n");
				pr_err("pre core clk off cb failed\n");
@@ -517,7 +635,7 @@ static int dsi_update_clk_state(struct dsi_core_clks *c_clks, u32 c_state,
		if (c_state == MDSS_DSI_CLK_OFF) {
		if (c_state == MDSS_DSI_CLK_OFF) {
			if (mngr->post_clkoff_cb) {
			if (mngr->post_clkoff_cb) {
				rc = mngr->post_clkoff_cb(mngr->priv_data,
				rc = mngr->post_clkoff_cb(mngr->priv_data,
						MDSS_DSI_CORE_CLK,
					MDSS_DSI_CORE_CLK, MDSS_DSI_LINK_NONE,
						MDSS_DSI_CLK_OFF);
						MDSS_DSI_CLK_OFF);
				if (rc)
				if (rc)
					pr_err("post clkoff cb fail, rc = %d\n",
					pr_err("post clkoff cb fail, rc = %d\n",
@@ -610,27 +728,30 @@ static int dsi_set_clk_rate(struct mdss_dsi_clk_mngr *mngr, int clk, u32 rate,
	MDSS_XLOG(clk, rate, flags);
	MDSS_XLOG(clk, rate, flags);
	switch (clk) {
	switch (clk) {
	case MDSS_DSI_LINK_ESC_CLK:
	case MDSS_DSI_LINK_ESC_CLK:
		mngr->link_clks.esc_clk_rate = rate;
		mngr->link_clks.lp_clks.esc_clk_rate = rate;
		if (!flags) {
		if (!flags) {
			rc = clk_set_rate(mngr->link_clks.clks.esc_clk, rate);
			rc = clk_set_rate(mngr->link_clks.lp_clks.esc_clk,
				rate);
			if (rc)
			if (rc)
				pr_err("set rate failed for esc clk rc=%d\n",
				pr_err("set rate failed for esc clk rc=%d\n",
				       rc);
				       rc);
		}
		}
		break;
		break;
	case MDSS_DSI_LINK_BYTE_CLK:
	case MDSS_DSI_LINK_BYTE_CLK:
		mngr->link_clks.byte_clk_rate = rate;
		mngr->link_clks.hs_clks.byte_clk_rate = rate;
		if (!flags) {
		if (!flags) {
			rc = clk_set_rate(mngr->link_clks.clks.byte_clk, rate);
			rc = clk_set_rate(mngr->link_clks.hs_clks.byte_clk,
				rate);
			if (rc)
			if (rc)
				pr_err("set rate failed for byte clk rc=%d\n",
				pr_err("set rate failed for byte clk rc=%d\n",
				       rc);
				       rc);
		}
		}
		break;
		break;
	case MDSS_DSI_LINK_PIX_CLK:
	case MDSS_DSI_LINK_PIX_CLK:
		mngr->link_clks.pix_clk_rate = rate;
		mngr->link_clks.hs_clks.pix_clk_rate = rate;
		if (!flags) {
		if (!flags) {
			rc = clk_set_rate(mngr->link_clks.clks.pixel_clk, rate);
			rc = clk_set_rate(mngr->link_clks.hs_clks.pixel_clk,
				rate);
			if (rc)
			if (rc)
				pr_err("failed to set rate for pix clk rc=%d\n",
				pr_err("failed to set rate for pix clk rc=%d\n",
				       rc);
				       rc);
@@ -888,8 +1009,10 @@ void *mdss_dsi_clk_init(struct mdss_dsi_clk_info *info)
	mutex_init(&mngr->clk_mutex);
	mutex_init(&mngr->clk_mutex);
	memcpy(&mngr->core_clks.clks, &info->core_clks, sizeof(struct
	memcpy(&mngr->core_clks.clks, &info->core_clks, sizeof(struct
						 mdss_dsi_core_clk_info));
						 mdss_dsi_core_clk_info));
	memcpy(&mngr->link_clks.clks, &info->link_clks, sizeof(struct
	memcpy(&mngr->link_clks.hs_clks, &info->link_hs_clks, sizeof(struct
						 mdss_dsi_link_clk_info));
						 mdss_dsi_link_hs_clk_info));
	memcpy(&mngr->link_clks.lp_clks, &info->link_lp_clks, sizeof(struct
						 mdss_dsi_link_lp_clk_info));


	INIT_LIST_HEAD(&mngr->client_list);
	INIT_LIST_HEAD(&mngr->client_list);
	mngr->pre_clkon_cb = info->pre_clkon_cb;
	mngr->pre_clkon_cb = info->pre_clkon_cb;
@@ -981,15 +1104,26 @@ int mdss_dsi_clk_force_toggle(void *client, u32 clk)
	if ((clk & MDSS_DSI_LINK_CLK) &&
	if ((clk & MDSS_DSI_LINK_CLK) &&
	    (mngr->link_clks.current_clk_state == MDSS_DSI_CLK_ON)) {
	    (mngr->link_clks.current_clk_state == MDSS_DSI_CLK_ON)) {


		rc = dsi_link_clk_stop(&mngr->link_clks);
		rc = dsi_link_hs_clk_stop(&mngr->link_clks.hs_clks);
		if (rc) {
		if (rc) {
			pr_err("failed to stop link clks\n");
			pr_err("failed to stop HS link clks\n");
			goto error;
			goto error;
		}
		}


		rc = dsi_link_clk_start(&mngr->link_clks);
		rc = dsi_link_lp_clk_stop(&mngr->link_clks.lp_clks);
		if (rc) {
			pr_err("failed to stop LP link clks\n");
			goto error;
		}

		rc = dsi_link_lp_clk_start(&mngr->link_clks.lp_clks);
		if (rc)
			pr_err("failed to start LP link clks\n");

		rc = dsi_link_hs_clk_start(&mngr->link_clks.hs_clks,
				MDSS_DSI_LINK_CLK_START);
		if (rc)
		if (rc)
			pr_err("failed to start link clks\n");
			pr_err("failed to start HS link clks\n");


	} else if (clk & MDSS_DSI_LINK_CLK) {
	} else if (clk & MDSS_DSI_LINK_CLK) {
		pr_err("cannot reset, link clock is off\n");
		pr_err("cannot reset, link clock is off\n");
+37 −9
Original line number Original line Diff line number Diff line
@@ -39,6 +39,13 @@ enum mdss_dsi_link_clk_type {
	MDSS_DSI_LINK_CLK_MAX,
	MDSS_DSI_LINK_CLK_MAX,
};
};


enum mdss_dsi_link_clk_op_type {
	MDSS_DSI_LINK_CLK_SET_RATE = BIT(0),
	MDSS_DSI_LINK_CLK_PREPARE = BIT(1),
	MDSS_DSI_LINK_CLK_ENABLE = BIT(2),
	MDSS_DSI_LINK_CLK_START = BIT(0) | BIT(1) | BIT(2),
};

enum mdss_dsi_clk_type {
enum mdss_dsi_clk_type {
	MDSS_DSI_CORE_CLK = BIT(0),
	MDSS_DSI_CORE_CLK = BIT(0),
	MDSS_DSI_LINK_CLK = BIT(1),
	MDSS_DSI_LINK_CLK = BIT(1),
@@ -46,52 +53,66 @@ enum mdss_dsi_clk_type {
	MDSS_DSI_CLKS_MAX = BIT(2),
	MDSS_DSI_CLKS_MAX = BIT(2),
};
};


enum mdss_dsi_lclk_type {
	MDSS_DSI_LINK_NONE = 0,
	MDSS_DSI_LINK_LP_CLK = BIT(0),
	MDSS_DSI_LINK_HS_CLK = BIT(1),
};

/**
/**
 * typedef *pre_clockoff_cb() - Callback before clock is turned off
 * typedef *pre_clockoff_cb() - Callback before clock is turned off
 * @priv: private data pointer.
 * @priv: private data pointer.
 * @clk_type: clock which is being turned off.
 * @clk_type: clock which is being turned off.
 * @l_type: specifies if the clock is HS or LP type. Valid only for link clocks.
 * @new_state: next state for the clock.
 * @new_state: next state for the clock.
 *
 *
 * @return: error code.
 * @return: error code.
 */
 */
typedef int (*pre_clockoff_cb)(void *priv,
typedef int (*pre_clockoff_cb)(void *priv,
				enum mdss_dsi_clk_type clk_type,
				enum mdss_dsi_clk_type clk_type,
				enum mdss_dsi_lclk_type l_type,
				enum mdss_dsi_clk_state new_state);
				enum mdss_dsi_clk_state new_state);


/**
/**
 * typedef *post_clockoff_cb() - Callback after clock is turned off
 * typedef *post_clockoff_cb() - Callback after clock is turned off
 * @priv: private data pointer.
 * @priv: private data pointer.
 * @clk_type: clock which was turned off.
 * @clk_type: clock which was turned off.
 * @l_type: specifies if the clock is HS or LP type. Valid only for link clocks.
 * @curr_state: current state for the clock.
 * @curr_state: current state for the clock.
 *
 *
 * @return: error code.
 * @return: error code.
 */
 */
typedef int (*post_clockoff_cb)(void *priv,
typedef int (*post_clockoff_cb)(void *priv,
				enum mdss_dsi_clk_type clk_type,
				enum mdss_dsi_clk_type clk_type,
				enum mdss_dsi_lclk_type l_type,
				enum mdss_dsi_clk_state curr_state);
				enum mdss_dsi_clk_state curr_state);


/**
/**
 * typedef *post_clockon_cb() - Callback after clock is turned on
 * typedef *post_clockon_cb() - Callback after clock is turned on
 * @priv: private data pointer.
 * @priv: private data pointer.
 * @clk_type: clock which was turned on.
 * @clk_type: clock which was turned on.
 * @l_type: specifies if the clock is HS or LP type. Valid only for link clocks.
 * @curr_state: current state for the clock.
 * @curr_state: current state for the clock.
 *
 *
 * @return: error code.
 * @return: error code.
 */
 */
typedef int (*post_clockon_cb)(void *priv,
typedef int (*post_clockon_cb)(void *priv,
				enum mdss_dsi_clk_type clk_type,
				enum mdss_dsi_clk_type clk_type,
				enum mdss_dsi_lclk_type l_type,
				enum mdss_dsi_clk_state curr_state);
				enum mdss_dsi_clk_state curr_state);


/**
/**
 * typedef *pre_clockon_cb() - Callback before clock is turned on
 * typedef *pre_clockon_cb() - Callback before clock is turned on
 * @priv: private data pointer.
 * @priv: private data pointer.
 * @clk_type: clock which is being turned on.
 * @clk_type: clock which is being turned on.
 * @l_type: specifies if the clock is HS or LP type. Valid only for link clocks.
 * @new_state: next state for the clock.
 * @new_state: next state for the clock.
 *
 *
 * @return: error code.
 * @return: error code.
 */
 */
typedef int (*pre_clockon_cb)(void *priv,
typedef int (*pre_clockon_cb)(void *priv,
				enum mdss_dsi_clk_type clk_type,
				enum mdss_dsi_clk_type clk_type,
				enum mdss_dsi_lclk_type l_type,
				enum mdss_dsi_clk_state new_state);
				enum mdss_dsi_clk_state new_state);


struct mdss_dsi_core_clk_info {
struct mdss_dsi_core_clk_info {
@@ -101,10 +122,16 @@ struct mdss_dsi_core_clk_info {
	struct clk *mmss_misc_ahb_clk;
	struct clk *mmss_misc_ahb_clk;
};
};


struct mdss_dsi_link_clk_info {
struct mdss_dsi_link_hs_clk_info {
	struct clk *esc_clk;
	struct clk *byte_clk;
	struct clk *byte_clk;
	struct clk *pixel_clk;
	struct clk *pixel_clk;
	u32 byte_clk_rate;
	u32 pix_clk_rate;
};

struct mdss_dsi_link_lp_clk_info {
	struct clk *esc_clk;
	u32 esc_clk_rate;
};
};


struct dsi_panel_clk_ctrl {
struct dsi_panel_clk_ctrl {
@@ -126,7 +153,8 @@ struct dsi_panel_clk_ctrl {
struct mdss_dsi_clk_info {
struct mdss_dsi_clk_info {
	char name[DSI_CLK_NAME_LEN];
	char name[DSI_CLK_NAME_LEN];
	struct mdss_dsi_core_clk_info core_clks;
	struct mdss_dsi_core_clk_info core_clks;
	struct mdss_dsi_link_clk_info link_clks;
	struct mdss_dsi_link_hs_clk_info link_hs_clks;
	struct mdss_dsi_link_lp_clk_info link_lp_clks;
	pre_clockoff_cb pre_clkoff_cb;
	pre_clockoff_cb pre_clkoff_cb;
	post_clockoff_cb post_clkoff_cb;
	post_clockoff_cb post_clkoff_cb;
	post_clockon_cb post_clkon_cb;
	post_clockon_cb post_clkon_cb;
+36 −32

File changed.

Preview size limit exceeded, changes collapsed.