Set link rate by using OPP set rate api so that CX level will be set
accordingly base on the link rate.

Signed-off-by: Kuogee Hsieh <khs...@codeaurora.org>
---
 drivers/gpu/drm/msm/dp/dp_ctrl.c    | 33 ++++++++++++++++++++++-
 drivers/gpu/drm/msm/dp/dp_ctrl.h    |  2 +-
 drivers/gpu/drm/msm/dp/dp_display.c |  8 +++---
 drivers/gpu/drm/msm/dp/dp_power.c   | 42 ++++++++++++++++++++++++++---
 drivers/gpu/drm/msm/dp/dp_power.h   |  2 +-
 5 files changed, 77 insertions(+), 10 deletions(-)

diff --git a/drivers/gpu/drm/msm/dp/dp_ctrl.c b/drivers/gpu/drm/msm/dp/dp_ctrl.c
index 2e3e1917351f..e1595d829e04 100644
--- a/drivers/gpu/drm/msm/dp/dp_ctrl.c
+++ b/drivers/gpu/drm/msm/dp/dp_ctrl.c
@@ -10,6 +10,7 @@
 #include <linux/delay.h>
 #include <linux/phy/phy.h>
 #include <linux/phy/phy-dp.h>
+#include <linux/pm_opp.h>
 #include <drm/drm_fixed.h>
 #include <drm/drm_dp_helper.h>
 #include <drm/drm_print.h>
@@ -76,6 +77,8 @@ struct dp_ctrl_private {
        struct dp_parser *parser;
        struct dp_catalog *catalog;
 
+       struct opp_table *opp_table;
+
        struct completion idle_comp;
        struct completion video_comp;
 };
@@ -1836,6 +1839,7 @@ struct dp_ctrl *dp_ctrl_get(struct device *dev, struct 
dp_link *link,
                        struct dp_parser *parser)
 {
        struct dp_ctrl_private *ctrl;
+       int ret;
 
        if (!dev || !panel || !aux ||
            !link || !catalog) {
@@ -1849,6 +1853,21 @@ struct dp_ctrl *dp_ctrl_get(struct device *dev, struct 
dp_link *link,
                return ERR_PTR(-ENOMEM);
        }
 
+       ctrl->opp_table = dev_pm_opp_set_clkname(dev, "ctrl_link");
+
+       if (IS_ERR(ctrl->opp_table)) {
+               dev_err(dev, "invalid DP OPP table in device tree\n");
+               ctrl->opp_table = NULL;
+       } else {
+               /* OPP table is optional */
+               ret = dev_pm_opp_of_add_table(dev);
+               if (ret && ret != -ENODEV) {
+                       dev_err(dev, "add DP OPP table\n");
+                       dev_pm_opp_put_clkname(ctrl->opp_table);
+                       ctrl->opp_table = NULL;
+               }
+       }
+
        init_completion(&ctrl->idle_comp);
        init_completion(&ctrl->video_comp);
 
@@ -1864,6 +1883,18 @@ struct dp_ctrl *dp_ctrl_get(struct device *dev, struct 
dp_link *link,
        return &ctrl->dp_ctrl;
 }
 
-void dp_ctrl_put(struct dp_ctrl *dp_ctrl)
+void dp_ctrl_put(struct device *dev, struct dp_ctrl *dp_ctrl)
 {
+       struct dp_ctrl_private *ctrl;
+
+       if (!dp_ctrl)
+               return;
+
+       ctrl = container_of(dp_ctrl, struct dp_ctrl_private, dp_ctrl);
+
+       if (ctrl->opp_table != NULL) {
+               dev_pm_opp_of_remove_table(dev);
+               dev_pm_opp_put_clkname(ctrl->opp_table);
+               ctrl->opp_table = NULL;
+       }
 }
diff --git a/drivers/gpu/drm/msm/dp/dp_ctrl.h b/drivers/gpu/drm/msm/dp/dp_ctrl.h
index f60ba93c8678..19b412a93e02 100644
--- a/drivers/gpu/drm/msm/dp/dp_ctrl.h
+++ b/drivers/gpu/drm/msm/dp/dp_ctrl.h
@@ -31,6 +31,6 @@ struct dp_ctrl *dp_ctrl_get(struct device *dev, struct 
dp_link *link,
                        struct dp_panel *panel, struct drm_dp_aux *aux,
                        struct dp_power *power, struct dp_catalog *catalog,
                        struct dp_parser *parser);
-void dp_ctrl_put(struct dp_ctrl *dp_ctrl);
+void dp_ctrl_put(struct device *dev, struct dp_ctrl *dp_ctrl);
 
 #endif /* _DP_CTRL_H_ */
diff --git a/drivers/gpu/drm/msm/dp/dp_display.c 
b/drivers/gpu/drm/msm/dp/dp_display.c
index 431dff9de797..be941eedf4c6 100644
--- a/drivers/gpu/drm/msm/dp/dp_display.c
+++ b/drivers/gpu/drm/msm/dp/dp_display.c
@@ -648,8 +648,10 @@ static int dp_irq_hpd_handle(struct dp_display_private 
*dp, u32 data)
 
 static void dp_display_deinit_sub_modules(struct dp_display_private *dp)
 {
+       struct device *dev = &dp->pdev->dev;
+
        dp_debug_put(dp->debug);
-       dp_ctrl_put(dp->ctrl);
+       dp_ctrl_put(dev, dp->ctrl);
        dp_panel_put(dp->panel);
        dp_aux_put(dp->aux);
        dp_audio_put(dp->audio);
@@ -693,7 +695,7 @@ static int dp_init_sub_modules(struct dp_display_private 
*dp)
                goto error;
        }
 
-       dp->power = dp_power_get(dp->parser);
+       dp->power = dp_power_get(dev, dp->parser);
        if (IS_ERR(dp->power)) {
                rc = PTR_ERR(dp->power);
                DRM_ERROR("failed to initialize power, rc = %d\n", rc);
@@ -749,7 +751,7 @@ static int dp_init_sub_modules(struct dp_display_private 
*dp)
        return rc;
 
 error_audio:
-       dp_ctrl_put(dp->ctrl);
+       dp_ctrl_put(dev, dp->ctrl);
 error_ctrl:
        dp_panel_put(dp->panel);
 error_link:
diff --git a/drivers/gpu/drm/msm/dp/dp_power.c 
b/drivers/gpu/drm/msm/dp/dp_power.c
index 17c1fc6a2d44..3d75bf09e38f 100644
--- a/drivers/gpu/drm/msm/dp/dp_power.c
+++ b/drivers/gpu/drm/msm/dp/dp_power.c
@@ -8,12 +8,14 @@
 #include <linux/clk.h>
 #include <linux/clk-provider.h>
 #include <linux/regulator/consumer.h>
+#include <linux/pm_opp.h>
 #include "dp_power.h"
 #include "msm_drv.h"
 
 struct dp_power_private {
        struct dp_parser *parser;
        struct platform_device *pdev;
+       struct device *dev;
        struct clk *link_clk_src;
        struct clk *pixel_provider;
        struct clk *link_provider;
@@ -148,18 +150,49 @@ static int dp_power_clk_deinit(struct dp_power_private 
*power)
        return 0;
 }
 
+static int dp_power_clk_set_link_rate(struct dp_power_private *power,
+                       struct dss_clk *clk_arry, int num_clk, int enable)
+{
+       u32 rate;
+       int i, rc = 0;
+
+       for (i = 0; i < num_clk; i++) {
+               if (clk_arry[i].clk) {
+                       if (clk_arry[i].type == DSS_CLK_PCLK) {
+                               if (enable)
+                                       rate = clk_arry[i].rate;
+                               else
+                                       rate = 0;
+
+                               rc = dev_pm_opp_set_rate(power->dev, rate);
+                       }
+
+               }
+       }
+       return rc;
+}
+
 static int dp_power_clk_set_rate(struct dp_power_private *power,
                enum dp_pm_type module, bool enable)
 {
        int rc = 0;
        struct dss_module_power *mp = &power->parser->mp[module];
 
-       if (enable) {
-               rc = msm_dss_clk_set_rate(mp->clk_config, mp->num_clk);
+       if (module == DP_CTRL_PM) {
+               rc = dp_power_clk_set_link_rate(power, mp->clk_config, 
mp->num_clk, enable);
                if (rc) {
-                       DRM_ERROR("failed to set clks rate.\n");
+                       DRM_ERROR("failed to set link clks rate.\n");
                        return rc;
                }
+       } else {
+
+               if (enable) {
+                       rc = msm_dss_clk_set_rate(mp->clk_config, mp->num_clk);
+                       if (rc) {
+                               DRM_ERROR("failed to set clks rate.\n");
+                               return rc;
+                       }
+               }
        }
 
        rc = msm_dss_enable_clk(mp->clk_config, mp->num_clk, enable);
@@ -349,7 +382,7 @@ int dp_power_deinit(struct dp_power *dp_power)
        return 0;
 }
 
-struct dp_power *dp_power_get(struct dp_parser *parser)
+struct dp_power *dp_power_get(struct device *dev, struct dp_parser *parser)
 {
        struct dp_power_private *power;
        struct dp_power *dp_power;
@@ -365,6 +398,7 @@ struct dp_power *dp_power_get(struct dp_parser *parser)
 
        power->parser = parser;
        power->pdev = parser->pdev;
+       power->dev = dev;
 
        dp_power = &power->dp_power;
 
diff --git a/drivers/gpu/drm/msm/dp/dp_power.h 
b/drivers/gpu/drm/msm/dp/dp_power.h
index 76743d755833..7d0327bbc0d5 100644
--- a/drivers/gpu/drm/msm/dp/dp_power.h
+++ b/drivers/gpu/drm/msm/dp/dp_power.h
@@ -102,6 +102,6 @@ void dp_power_client_deinit(struct dp_power *power);
  * methods to be called by the client to configure the power related
  * modueles.
  */
-struct dp_power *dp_power_get(struct dp_parser *parser);
+struct dp_power *dp_power_get(struct device *dev, struct dp_parser *parser);
 
 #endif /* _DP_POWER_H_ */

base-commit: 3c0f462da069af12211901ddf26f7e16e6951d9b
prerequisite-patch-id: a109eaf08147f50149ad661a58122b6745a52445
-- 
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum,
a Linux Foundation Collaborative Project

_______________________________________________
dri-devel mailing list
dri-devel@lists.freedesktop.org
https://lists.freedesktop.org/mailman/listinfo/dri-devel

Reply via email to