[Freedreno] [DPU PATCH v4 4/5] drm/msm/dp: add support for DP PLL driver
varar at codeaurora.org
varar at codeaurora.org
Fri Mar 27 20:33:01 UTC 2020
Hi Stephen Boyd,
I have updated our comments inline for the queries.
Could you please check and let know your views.
I have addressed most of the comments from Rob/Jordon and planning to
update new patchset in couple of days.
Thanks,
--Vara
On 2020-03-27 13:28, varar at codeaurora.org wrote:
> On 2020-03-19 15:17, Stephen Boyd wrote:
>> Quoting Vara Reddy (2020-03-04 16:10:27)
>>> From: Chandan Uddaraju <chandanu at codeaurora.org>
>>>
>>> Add the needed DP PLL specific files to support
>>> display port interface on msm targets.
>>>
>>> The DP driver calls the DP PLL driver registration.
>>> The DP driver sets the link and pixel clock sources.
>>>
>>> Changes in v2:
>>> -- Update copyright markings on all relevant files.
>>> -- Use DRM_DEBUG_DP for debug msgs.
>>>
>>> Changes in v4:
>>> -- Update the DP link clock provider names
>>>
>>> Signed-off-by: Chandan Uddaraju <chandanu at codeaurora.org>
>>> Signed-off-by: Vara Reddy <varar at codeaurora.org>
>>> ---
>>> drivers/gpu/drm/msm/Kconfig | 13 +
>>> drivers/gpu/drm/msm/Makefile | 4 +
>>> drivers/gpu/drm/msm/dp/dp_display.c | 51 +++
>>> drivers/gpu/drm/msm/dp/dp_display.h | 3 +
>>> drivers/gpu/drm/msm/dp/dp_parser.h | 4 +
>>> drivers/gpu/drm/msm/dp/dp_power.h | 1 -
>>> drivers/gpu/drm/msm/dp/pll/dp_pll.c | 136 +++++++
>>> drivers/gpu/drm/msm/dp/pll/dp_pll.h | 57 +++
>>> drivers/gpu/drm/msm/dp/pll/dp_pll_10nm.c | 406
>>> ++++++++++++++++++++
>>> drivers/gpu/drm/msm/dp/pll/dp_pll_10nm.h | 86 +++++
>>> drivers/gpu/drm/msm/dp/pll/dp_pll_10nm_util.c | 524
>>> ++++++++++++++++++++++++++
>>> 11 files changed, 1284 insertions(+), 1 deletion(-)
>>> create mode 100644 drivers/gpu/drm/msm/dp/pll/dp_pll.c
>>> create mode 100644 drivers/gpu/drm/msm/dp/pll/dp_pll.h
>>> create mode 100644 drivers/gpu/drm/msm/dp/pll/dp_pll_10nm.c
>>> create mode 100644 drivers/gpu/drm/msm/dp/pll/dp_pll_10nm.h
>>> create mode 100644 drivers/gpu/drm/msm/dp/pll/dp_pll_10nm_util.c
>>>
>>> diff --git a/drivers/gpu/drm/msm/Kconfig
>>> b/drivers/gpu/drm/msm/Kconfig
>>> index 7946cb1..e73ad23 100644
>>> --- a/drivers/gpu/drm/msm/Kconfig
>>> +++ b/drivers/gpu/drm/msm/Kconfig
>>> @@ -66,6 +66,19 @@ config DRM_MSM_DP
>>> display support is enabled through this config option. It
>>> can
>>> be primary or secondary display on device.
>>>
>>> +config DRM_MSM_DP_PLL
>>> + bool "Enable DP PLL driver in MSM DRM"
>>> + depends on DRM_MSM_DP && COMMON_CLK
>>> + help
>>> + Choose this option to enable DP PLL driver which provides
>>> DP
>>> + source clocks under common clock framework.
>>> +
>>> +config DRM_MSM_DP_10NM_PLL
>>> + bool "Enable DP 10nm PLL driver in MSM DRM (used by SDM845)"
>>> + depends on DRM_MSM_DP_PLL
>>> + help
>>> + Choose this option if DP PLL on SDM845 is used on the
>>> platform.
>>> +
>>> config DRM_MSM_DSI
>>> bool "Enable DSI support in MSM DRM driver"
>>> depends on DRM_MSM
>>> diff --git a/drivers/gpu/drm/msm/Makefile
>>> b/drivers/gpu/drm/msm/Makefile
>>> index c846599..1621f3f 100644
>>> --- a/drivers/gpu/drm/msm/Makefile
>>> +++ b/drivers/gpu/drm/msm/Makefile
>>> @@ -140,4 +140,8 @@ msm-$(CONFIG_DRM_MSM_DSI_14NM_PHY) +=
>>> dsi/pll/dsi_pll_14nm.o
>>> msm-$(CONFIG_DRM_MSM_DSI_10NM_PHY) += dsi/pll/dsi_pll_10nm.o
>>> endif
>>>
>>> +msm-$(CONFIG_DRM_MSM_DP_PLL)+= dp/pll/dp_pll.o
>>> +msm-$(CONFIG_DRM_MSM_DP_10NM_PLL)+= dp/pll/dp_pll_10nm.o \
>>> + dp/pll/dp_pll_10nm_util.o
>>> +
>>> obj-$(CONFIG_DRM_MSM) += msm.o
>>> diff --git a/drivers/gpu/drm/msm/dp/dp_display.c
>>> b/drivers/gpu/drm/msm/dp/dp_display.c
>>> index 5525405..ded8f5c 100644
>>> --- a/drivers/gpu/drm/msm/dp/dp_display.c
>>> +++ b/drivers/gpu/drm/msm/dp/dp_display.c
>>> @@ -61,6 +61,51 @@ struct dp_display_private {
>>> {}
>>> };
>>>
>>> +static int dp_get_pll(struct dp_display_private *dp_priv)
>>> +{
>>> + struct platform_device *pdev = NULL;
>>> + struct platform_device *pll_pdev;
>>> + struct device_node *pll_node;
>>> + struct dp_parser *dp_parser = NULL;
>>> +
>>> + if (!dp_priv) {
>>> + DRM_ERROR("Invalid Arguments\n");
>>
>> It's a static function. Is this even possible? The bad code would be
>> in
>> the same file.
>>
>>> + return -EINVAL;
>>> + }
>>> +
>>> + pdev = dp_priv->pdev;
>>> + dp_parser = dp_priv->parser;
>>> +
>>> + if (!dp_parser) {
>>> + DRM_DEV_ERROR(&pdev->dev, "%s: Parser not
>>> initialized\n",
>>> + __func__);
>>> + return -EINVAL;
>>> + }
>>> +
>>> + pll_node = of_parse_phandle(pdev->dev.of_node, "pll-node",
>>> 0);
>>> + if (!pll_node) {
>>> + DRM_DEV_ERROR(&pdev->dev, "%s: cannot find pll
>>> device\n",
>>> + __func__);
>>> + return -ENXIO;
>>> + }
>>> +
>>> + pll_pdev = of_find_device_by_node(pll_node);
>>> + if (pll_pdev)
>>> + dp_parser->pll = platform_get_drvdata(pll_pdev);
>>> +
>>> + of_node_put(pll_node);
>>> +
>>> + if (!pll_pdev || !dp_parser->pll) {
>>> + DRM_DEV_ERROR(&pdev->dev, "%s: pll driver is not
>>> ready\n",
>>> + __func__);
>>> + return -EPROBE_DEFER;
>>> + }
>>> +
>>> + dp_parser->pll_dev = get_device(&pll_pdev->dev);
>>> +
>>> + return 0;
>>> +}
>>> +
>>> static irqreturn_t dp_display_irq(int irq, void *dev_id)
>>> {
>>> struct dp_display_private *dp = dev_id;
>>> @@ -114,6 +159,10 @@ static int dp_display_bind(struct device *dev,
>>> struct device *master,
>>> goto end;
>>> }
>>>
>>> + rc = dp_get_pll(dp);
>>> + if (rc)
>>> + goto end;
>>> +
>>> rc = dp_aux_register(dp->aux);
>>> if (rc) {
>>> DRM_ERROR("DRM DP AUX register failed\n");
>>> @@ -837,6 +886,7 @@ int __init msm_dp_register(void)
>>> {
>>> int ret;
>>>
>>> + msm_dp_pll_driver_register();
>>> ret = platform_driver_register(&dp_display_driver);
>>> if (ret) {
>>> DRM_ERROR("driver register failed");
>>> @@ -848,6 +898,7 @@ int __init msm_dp_register(void)
>>>
>>> void __exit msm_dp_unregister(void)
>>> {
>>> + msm_dp_pll_driver_unregister();
>>> platform_driver_unregister(&dp_display_driver);
>>> }
>>>
>>> diff --git a/drivers/gpu/drm/msm/dp/dp_display.h
>>> b/drivers/gpu/drm/msm/dp/dp_display.h
>>> index dad8610..4c53ed5 100644
>>> --- a/drivers/gpu/drm/msm/dp/dp_display.h
>>> +++ b/drivers/gpu/drm/msm/dp/dp_display.h
>>> @@ -25,4 +25,7 @@ int dp_display_get_modes(struct msm_dp *dp_display,
>>> bool dp_display_check_video_test(struct msm_dp *dp_display);
>>> int dp_display_get_test_bpp(struct msm_dp *dp_display);
>>>
>>> +void __init msm_dp_pll_driver_register(void);
>>> +void __exit msm_dp_pll_driver_unregister(void);
>>> +
>>> #endif /* _DP_DISPLAY_H_ */
>>> diff --git a/drivers/gpu/drm/msm/dp/dp_parser.h
>>> b/drivers/gpu/drm/msm/dp/dp_parser.h
>>> index 1dc592f..e775cbc 100644
>>> --- a/drivers/gpu/drm/msm/dp/dp_parser.h
>>> +++ b/drivers/gpu/drm/msm/dp/dp_parser.h
>>> @@ -6,6 +6,8 @@
>>> #ifndef _DP_PARSER_H_
>>> #define _DP_PARSER_H_
>>>
>>> +#include "pll/dp_pll.h"
>>> +
>>> #define DP_LABEL "MDSS DP DISPLAY"
>>> #define AUX_CFG_LEN 10
>>> #define DP_MAX_PIXEL_CLK_KHZ 675000
>>> @@ -193,6 +195,8 @@ struct dp_parser {
>>> bool combo_phy_en;
>>> struct dp_io io;
>>> struct dp_display_data disp_data;
>>> + struct msm_dp_pll *pll;
>>> + struct device *pll_dev;
>>> const struct dp_regulator_cfg *regulator_cfg;
>>> u8 l_map[4];
>>> struct dp_aux_cfg aux_cfg[AUX_CFG_LEN];
>>> diff --git a/drivers/gpu/drm/msm/dp/dp_power.h
>>> b/drivers/gpu/drm/msm/dp/dp_power.h
>>> index 52b6194..b24b2c5 100644
>>> --- a/drivers/gpu/drm/msm/dp/dp_power.h
>>> +++ b/drivers/gpu/drm/msm/dp/dp_power.h
>>> @@ -14,7 +14,6 @@
>>> * @init: initializes the regulators/core clocks/GPIOs/pinctrl
>>> * @deinit: turns off the regulators/core clocks/GPIOs/pinctrl
>>> * @clk_enable: enable/disable the DP clocks
>>> - * @set_link_clk_parent: set the parent of DP link clock
>>> * @set_pixel_clk_parent: set the parent of DP pixel clock
>>> */
>>> struct dp_power {
>>
>> This can be another patch to update documentation?
>>
>>> diff --git a/drivers/gpu/drm/msm/dp/pll/dp_pll.c
>>> b/drivers/gpu/drm/msm/dp/pll/dp_pll.c
>>> new file mode 100644
>>> index 0000000..11e26cf
>>> --- /dev/null
>>> +++ b/drivers/gpu/drm/msm/dp/pll/dp_pll.c
>>> @@ -0,0 +1,136 @@
>>> +// SPDX-License-Identifier: GPL-2.0-only
>>> +/*
>>> + * Copyright (c) 2016-2020, The Linux Foundation. All rights
>>> reserved.
>>> + */
>>> +
>>> +#include "dp_pll.h"
>>> +
>>> +int msm_dp_pll_util_parse_dt_clock(struct platform_device *pdev,
>>> + struct msm_dp_pll *pll)
>>> +{
>>> + u32 i = 0, rc = 0;
>>> + struct dss_module_power *mp = &pll->mp;
>>> + const char *clock_name;
>>> + u32 clock_rate;
>>> +
>>> + mp->num_clk = of_property_count_strings(pdev->dev.of_node,
>>> +
>>> "clock-names");
>>> + if (mp->num_clk <= 0) {
>>> + DRM_DEV_ERROR(&pdev->dev, "%s:clocks are not
>>> defined\n",
>>> + __func__);
>>> + goto clk_err;
>>> + }
>>> +
>>> + mp->clk_config = devm_kzalloc(&pdev->dev,
>>
>> Use devm_kcalloc() for arrays.
>>
>>> + sizeof(struct dss_clk) * mp->num_clk,
>>> GFP_KERNEL);
>>> + if (!mp->clk_config) {
>>> + rc = -ENOMEM;
>>> + mp->num_clk = 0;
>>> + goto clk_err;
>>> + }
>>> +
>>> + for (i = 0; i < mp->num_clk; i++) {
>>> + of_property_read_string_index(pdev->dev.of_node,
>>> "clock-names",
>>> + i,
>>> &clock_name);
>>> + strlcpy(mp->clk_config[i].clk_name, clock_name,
>>> + sizeof(mp->clk_config[i].clk_name));
>>> +
>>> + of_property_read_u32_index(pdev->dev.of_node,
>>> "clock-rate",
>>> + i,
>>> &clock_rate);
>>> + mp->clk_config[i].rate = clock_rate;
>>> +
>>> + if (!clock_rate)
>>> + mp->clk_config[i].type = DSS_CLK_AHB;
>>> + else
>>> + mp->clk_config[i].type = DSS_CLK_PCLK;
>>> + }
>>> +
>>> +clk_err:
>>> + return rc;
>>> +}
>>> +
>>> +struct msm_dp_pll *msm_dp_pll_init(struct platform_device *pdev,
>>> + enum msm_dp_pll_type type, int id)
>>> +{
>>> + struct device *dev = &pdev->dev;
>>> + struct msm_dp_pll *pll;
>>> +
>>> + switch (type) {
>>> + case MSM_DP_PLL_10NM:
>>> + pll = msm_dp_pll_10nm_init(pdev, id);
>>> + break;
>>> + default:
>>> + pll = ERR_PTR(-ENXIO);
>>> + break;
>>> + }
>>> +
>>> + if (IS_ERR(pll)) {
>>> + DRM_DEV_ERROR(dev, "%s: failed to init DP PLL\n",
>>> __func__);
>>> + return pll;
>>> + }
>>> +
>>> + pll->type = type;
>>> +
>>> + DRM_DEBUG_DP("DP:%d PLL registered", id);
>>> +
>>> + return pll;
>>> +}
>>> +
>>> +static const struct of_device_id dp_pll_dt_match[] = {
>>> + { .compatible = "qcom,dp-pll-10nm" },
>>
>> Add a terminator here { }.
>>
>>> +};
>>> +
>>> +static int dp_pll_driver_probe(struct platform_device *pdev)
>>> +{
>>> + struct msm_dp_pll *pll;
>>> + struct device *dev = &pdev->dev;
>>> + const struct of_device_id *match;
>>> +
>>> + match = of_match_node(dp_pll_dt_match, dev->of_node);
>>> + if (!match)
>>> + return -ENODEV;
>>> +
>>> + /* Currently supporting only 10nm-DP PLL */
>>
>> Please remove this useless comment that can only require being updated
>> in the future.
>>
>>> + pll = msm_dp_pll_init(pdev, MSM_DP_PLL_10NM, 0);
>>> + if (IS_ERR_OR_NULL(pll)) {
>>> + DRM_DEV_ERROR(dev,
>>> + "%s: pll init failed: %ld, need separate pll
>>> clk driver\n",
>>
>> What does this error mean?
>>
>>> + __func__, PTR_ERR(pll));
>>> + return -ENODEV;
>>> + }
>>> +
>>> + platform_set_drvdata(pdev, pll);
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static int dp_pll_driver_remove(struct platform_device *pdev)
>>> +{
>>> + struct msm_dp_pll *pll = platform_get_drvdata(pdev);
>>> +
>>> + if (pll)
>>> + pll = NULL;
>>> +
>>> + platform_set_drvdata(pdev, NULL);
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static struct platform_driver dp_pll_platform_driver = {
>>> + .probe = dp_pll_driver_probe,
>>> + .remove = dp_pll_driver_remove,
>>> + .driver = {
>>> + .name = "msm_dp_pll",
>>> + .of_match_table = dp_pll_dt_match,
>>> + },
>>> +};
>>> +
>>> +void __init msm_dp_pll_driver_register(void)
>>> +{
>>> + platform_driver_register(&dp_pll_platform_driver);
>>> +}
>>> +
>>> +void __exit msm_dp_pll_driver_unregister(void)
>>> +{
>>> + platform_driver_unregister(&dp_pll_platform_driver);
>>> +}
>>> diff --git a/drivers/gpu/drm/msm/dp/pll/dp_pll.h
>>> b/drivers/gpu/drm/msm/dp/pll/dp_pll.h
>>> new file mode 100644
>>> index 0000000..185f1e9
>>> --- /dev/null
>>> +++ b/drivers/gpu/drm/msm/dp/pll/dp_pll.h
>>> @@ -0,0 +1,57 @@
>>> +/* SPDX-License-Identifier: GPL-2.0-only */
>>> +/*
>>> + * Copyright (c) 2016-2020, The Linux Foundation. All rights
>>> reserved.
>>> + */
>>> +
>>> +#ifndef __DP_PLL_H
>>> +#define __DP_PLL_H
>>> +
>>> +#include <linux/clk.h>
>>> +#include <linux/clk-provider.h>
>>
>> You're making a clk provider. Please Cc clk maintainers and
>> linux-clk at vger.kernel.org
>>
>>> +#include <linux/platform_device.h>
>>> +
>>> +#include "dpu_io_util.h"
>>> +#include "msm_drv.h"
>>> +
>>> +#define PLL_REG_W(base, offset, data) \
>>> + writel((data), (base) + (offset))
>>> +#define PLL_REG_R(base, offset) readl((base) + (offset))
>>> +
>>> +enum msm_dp_pll_type {
>>> + MSM_DP_PLL_10NM,
>>> + MSM_DP_PLL_MAX
>>> +};
>>> +
>>> +struct msm_dp_pll {
>>> + enum msm_dp_pll_type type;
>>> + struct clk_hw clk_hw;
>>> + unsigned long rate; /* current vco rate */
>>> + u64 min_rate; /* min vco rate */
>>> + u64 max_rate; /* max vco rate */
>>> + bool pll_on;
>>
>> Are these struct members here to save/restore the rate across
>> suspend/resume?
>
> We are trying to retain the current vco rate across suspend/resume.
> min_rate and max_rate are the minimum and maximum vco rates that are
> possible for the chipset.
>
>>
>>> + void *priv;
>>> + /* Pll specific resources like GPIO, power supply, clocks,
>>> etc*/
>>> + struct dss_module_power mp;
>>> + int (*get_provider)(struct msm_dp_pll *pll,
>>> + struct clk **link_clk_provider,
>>> + struct clk **pixel_clk_provider);
>>> +};
>>> +
>>> +#define hw_clk_to_pll(x) container_of(x, struct msm_dp_pll, clk_hw)
>>
>> Why not to_msm_dp_pll() for this macro.
>>
>>> +
>>> +struct msm_dp_pll *msm_dp_pll_init(struct platform_device *pdev,
>>> + enum msm_dp_pll_type type, int id);
>>> +
>>> +int msm_dp_pll_util_parse_dt_clock(struct platform_device *pdev,
>>> + struct msm_dp_pll *pll);
>>> +
>>> +#ifdef CONFIG_DRM_MSM_DP_10NM_PLL
>>> +struct msm_dp_pll *msm_dp_pll_10nm_init(struct platform_device
>>> *pdev, int id);
>>> +#else
>>> +static inline struct msm_dp_pll *msm_dp_pll_10nm_init
>>> + (struct platform_device *pdev, int
>>> id)
>>
>> I would write this as
>>
>> static inline struct msm_dp_pll *
>> msm_dp_pll_10nm_init(struct platform_device *pdev, int id)
>>
>> so that we don't have parameters all by themselves.
>>
>>> +{
>>> + return ERR_PTR(-ENODEV);
>>> +}
>>> +#endif
>>> +#endif /* __DP_PLL_H */
>>> diff --git a/drivers/gpu/drm/msm/dp/pll/dp_pll_10nm.c
>>> b/drivers/gpu/drm/msm/dp/pll/dp_pll_10nm.c
>>> new file mode 100644
>>> index 0000000..5991e09
>>> --- /dev/null
>>> +++ b/drivers/gpu/drm/msm/dp/pll/dp_pll_10nm.c
>>> @@ -0,0 +1,406 @@
>>> +// SPDX-License-Identifier: GPL-2.0-only
>>> +/*
>>> + * Copyright (c) 2016-2020, The Linux Foundation. All rights
>>> reserved.
>>> + */
>>> +
>>> +/*
>>> + * Display Port PLL driver block diagram for branch clocks
>>> + *
>>> + * +------------------------------+
>>> + * | DP_VCO_CLK |
>>> + * | |
>>> + * | +-------------------+ |
>>> + * | | (DP PLL/VCO) | |
>>> + * | +---------+---------+ |
>>> + * | v |
>>> + * | +----------+-----------+ |
>>> + * | | hsclk_divsel_clk_src | |
>>> + * | +----------+-----------+ |
>>> + * +------------------------------+
>>> + * |
>>> + * +---------<---------v------------>----------+
>>> + * | |
>>> + * +--------v---------+ |
>>> + * | dp_phy_pll | |
>>> + * | link_clk | |
>>> + * +--------+---------+ |
>>> + * | |
>>> + * | |
>>> + * v v
>>> + * Input to DISPCC block |
>>> + * for link clk, crypto clk |
>>> + * and interface clock |
>>> + * |
>>> + * |
>>> + * +--------<------------+-----------------+---<---+
>>> + * | | |
>>> + * +----v---------+ +--------v-----+ +--------v------+
>>> + * | vco_divided | | vco_divided | | vco_divided |
>>> + * | _clk_src | | _clk_src | | _clk_src |
>>> + * | | | | | |
>>> + * |divsel_six | | divsel_two | | divsel_four |
>>> + * +-------+------+ +-----+--------+ +--------+------+
>>> + * | | |
>>> + * v---->----------v-------------<------v
>>> + * |
>>> + * +----------+---------+
>>> + * | dp_phy_pll_vco |
>>> + * | div_clk |
>>> + * +---------+----------+
>>> + * |
>>> + * v
>>> + * Input to DISPCC block
>>> + * for DP pixel clock
>>> + *
>>
>> Awesome diagram!
>>
>>> + */
>>> +
>>> +#include <linux/clk.h>
>>
>> Please try to avoid making a clk provider a clk consumer at the same
>> time.
>>
>>> +#include <linux/delay.h>
>>> +#include <linux/err.h>
>>> +#include <linux/kernel.h>
>>> +#include <linux/regmap.h>
>>> +
>>> +#include "dp_pll_10nm.h"
>>> +
>>> +#define NUM_PROVIDED_CLKS 2
>>> +
>>> +#define DP_LINK_CLK_SRC 0
>>> +#define DP_PIXEL_CLK_SRC 1
>>> +
>>> +static struct dp_pll_10nm *dp_pdb;
>>> +
>>> +static const struct clk_ops dp_10nm_vco_clk_ops = {
>>> + .recalc_rate = dp_vco_recalc_rate_10nm,
>>> + .set_rate = dp_vco_set_rate_10nm,
>>> + .round_rate = dp_vco_round_rate_10nm,
>>> + .prepare = dp_vco_prepare_10nm,
>>> + .unprepare = dp_vco_unprepare_10nm,
>>> +};
>>> +
>>> +struct dp_pll_10nm_pclksel {
>>> + struct clk_hw hw;
>>> +
>>> + /* divider params */
>>> + u8 shift;
>>> + u8 width;
>>> + u8 flags; /* same flags as used by clk_divider struct */
>>> +
>>> + struct dp_pll_10nm *pll;
>>> +};
>>> +#define to_pll_10nm_pclksel(_hw) \
>>> + container_of(_hw, struct dp_pll_10nm_pclksel, hw)
>>> +
>>> +static int dp_mux_set_parent_10nm(struct clk_hw *hw, u8 val)
>>> +{
>>> + struct dp_pll_10nm_pclksel *pclksel =
>>> to_pll_10nm_pclksel(hw);
>>> + struct dp_pll_10nm *dp_res = pclksel->pll;
>>> + u32 auxclk_div;
>>> +
>>> + auxclk_div = PLL_REG_R(dp_res->phy_base, REG_DP_PHY_VCO_DIV);
>>> + auxclk_div &= ~0x03; /* bits 0 to 1 */
>>> +
>>> + if (val == 0) /* mux parent index = 0 */
>>> + auxclk_div |= 1;
>>> + else if (val == 1) /* mux parent index = 1 */
>>> + auxclk_div |= 2;
>>> + else if (val == 2) /* mux parent index = 2 */
>>> + auxclk_div |= 0;
>>
>> These comments are just saying what the code says. Can you remove
>> them?
>>
>>> +
>>> + PLL_REG_W(dp_res->phy_base,
>>> + REG_DP_PHY_VCO_DIV, auxclk_div);
>>> + DRM_DEBUG_DP("%s: mux=%d auxclk_div=%x\n", __func__, val,
>>> auxclk_div);
>>
>> Is it a mux and divider combined? The diagram seems to imply that the
>> mux picks between fixed dividers but I don't see why that isn't just a
>> divider clk because the parent is always the same for the fixed
>> dividers. Put another way, why is a parent API being used for this? It
>> would make more sense to just treat this like a divider and use rate
>> setting APIs instead.
>
> I am not sure if I completely understand the concern. I am thinking
> that you are suggesting that we remove the mux altogether
> and just use the fixed dividers, while programming the registers as
> part of the the set_rate for those dividers.
> If that is the case, then we would have to list 3 parents for the RCG
> clock in display clock controller (dispcc). However, as per the
> clock plan, the RCG clocks have a well defined set of parents (DP PLL,
> or the reference clock). It would be incorrect
> to list all these PLL specific clocks as parents of the RCG. We are
> required to list a a single provider clock for DP Pixel clock.
> Hence the choice of a mux clock.
>
>>
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static u8 dp_mux_get_parent_10nm(struct clk_hw *hw)
>>> +{
>>> + u32 auxclk_div = 0;
>>> + struct dp_pll_10nm_pclksel *pclksel =
>>> to_pll_10nm_pclksel(hw);
>>> + struct dp_pll_10nm *dp_res = pclksel->pll;
>>> + u8 val = 0;
>>> +
>>> + auxclk_div = PLL_REG_R(dp_res->phy_base, REG_DP_PHY_VCO_DIV);
>>> + auxclk_div &= 0x03;
>>> +
>>> + if (auxclk_div == 1) /* Default divider */
>>> + val = 0;
>>> + else if (auxclk_div == 2)
>>> + val = 1;
>>> + else if (auxclk_div == 0)
>>> + val = 2;
>>> +
>>> + DRM_DEBUG_DP("%s: auxclk_div=%d, val=%d\n", __func__,
>>> auxclk_div, val);
>>> +
>>> + return val;
>>> +}
>>> +
>>> +static int clk_mux_determine_rate(struct clk_hw *hw,
>>> + struct clk_rate_request *req)
>>> +{
>>> + unsigned long rate = 0;
>>> + int ret = 0;
>>> +
>>> + rate = clk_get_rate(hw->clk);
>>> +
>>> + if (rate <= 0) {
>>> + DRM_ERROR("Rate is not set properly\n");
>>> + return -EINVAL;
>>> + }
>>> +
>>> + req->rate = rate;
>>> +
>>> + DRM_DEBUG_DP("%s: rate=%ld\n", __func__, req->rate);
>>> + /* Set the new parent of mux if there is a new valid parent
>>> */
>>> + if (hw->clk && req->best_parent_hw->clk) {
>>> + ret = clk_set_parent(hw->clk,
>>> req->best_parent_hw->clk);
>>> + if (ret) {
>>> + DRM_ERROR("%s: clk_set_parent failed:
>>> ret=%d\n",
>>> + __func__, ret);
>>> + return ret;
>>> + }
>>> + }
>>> + return 0;
>>> +}
>>> +
>>> +static unsigned long mux_recalc_rate(struct clk_hw *hw,
>>
>> Please prefix all functions with dp_pll_
>>
>>> + unsigned long parent_rate)
>>> +{
>>> + struct clk *div_clk = NULL, *vco_clk = NULL;
>>> + struct msm_dp_pll *vco = NULL;
>>> +
>>> + div_clk = clk_get_parent(hw->clk);
>>
>> We have clk_hw_get_parent() for this stuff. Please use it.
>>
>>> + if (!div_clk)
>>> + return 0;
>>> +
>>> + vco_clk = clk_get_parent(div_clk);
>>> + if (!vco_clk)
>>> + return 0;
>>> +
>>> + vco = hw_clk_to_pll(__clk_get_hw(vco_clk));
>>
>> Please don't use __clk_get_hw().
> Could you please elaborate on this or what to use.
> Should we use the api implementation directly.
>>
>>> + if (!vco)
>>> + return 0;
>>> +
>>> + if (vco->rate == DP_VCO_HSCLK_RATE_8100MHZDIV1000)
>>> + return (vco->rate / 6);
>>> + else if (vco->rate == DP_VCO_HSCLK_RATE_5400MHZDIV1000)
>>> + return (vco->rate / 4);
>>> + else
>>> + return (vco->rate / 2);
>>
>> else return is shunned. Just have return instead.
>>
>>> +}
>>> +
>>> +static int dp_pll_10nm_get_provider(struct msm_dp_pll *pll,
>>> + struct clk **link_clk_provider,
>>> + struct clk **pixel_clk_provider)
>>> +{
>>> + struct dp_pll_10nm *pll_10nm = to_dp_pll_10nm(pll);
>>> + struct clk_hw_onecell_data *hw_data = pll_10nm->hw_data;
>>> +
>>> + if (link_clk_provider)
>>> + *link_clk_provider =
>>> hw_data->hws[DP_LINK_CLK_SRC]->clk;
>>> + if (pixel_clk_provider)
>>> + *pixel_clk_provider =
>>> hw_data->hws[DP_PIXEL_CLK_SRC]->clk;
>>
>> I don't know what this is but it looks weird. Is this a clk provider
>> on
>> top of another clk provider?
> Here we are setting the parent of RCG.
> RCG needs called by the consumer to set the parent.
> DP is a consumer setting a rate on provider.
>
>>
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static const struct clk_ops dp_10nm_pclksel_clk_ops = {
>>> + .get_parent = dp_mux_get_parent_10nm,
>>> + .set_parent = dp_mux_set_parent_10nm,
>>> + .recalc_rate = mux_recalc_rate,
>>> + .determine_rate = clk_mux_determine_rate,
>>> +};
>>> +
>>> +static struct clk_hw *dp_pll_10nm_pixel_clk_sel(struct dp_pll_10nm
>>> *pll_10nm)
>>> +{
>>> + struct device *dev = &pll_10nm->pdev->dev;
>>> + struct dp_pll_10nm_pclksel *pll_pclksel;
>>> + struct clk_init_data pclksel_init = {
>>> + .parent_names = (const char *[]){
>>> + "dp_vco_divsel_two_clk_src",
>>> + "dp_vco_divsel_four_clk_src",
>>> + "dp_vco_divsel_six_clk_src" },
>>
>> It looks like a clk divider still, not a mux.
>>
>>> + .num_parents = 3,
>>> + .name = "dp_phy_pll_vco_div_clk",
>>> + .flags = CLK_IGNORE_UNUSED,
>>
>> Why? Please comment why this is needed in the code. But more
>> importantly, please just don't use this flag.
> We are working to remove this flag CLK_IGNORE_UNUSED.
>>
>>> + .ops = &dp_10nm_pclksel_clk_ops,
>>> + };
>>> + int ret;
>>> +
>>> + pll_pclksel = devm_kzalloc(dev, sizeof(*pll_pclksel),
>>> GFP_KERNEL);
>>> + if (!pll_pclksel)
>>> + return ERR_PTR(-ENOMEM);
>>> +
>>> + pll_pclksel->pll = pll_10nm;
>>> + pll_pclksel->shift = 0;
>>> + pll_pclksel->width = 4;
>>> + pll_pclksel->flags = CLK_DIVIDER_ONE_BASED;
>>> + pll_pclksel->hw.init = &pclksel_init;
>>> +
>>> + ret = clk_hw_register(dev, &pll_pclksel->hw);
>>> + if (ret)
>>> + return ERR_PTR(ret);
>>> +
>>> + return &pll_pclksel->hw;
>>> +}
>>> +
>>> +static int dp_pll_10nm_register(struct dp_pll_10nm *pll_10nm)
>>> +{
>>> + char clk_name[32], parent[32], vco_name[32];
>>> + struct clk_init_data vco_init = {
>>> + .parent_names = (const char *[]){ "bi_tcxo" },
>>
>> Please use the new way of specifying clk parents through .fw_name and
>> DT
>> indices.
>>
>>> + .num_parents = 1,
>>> + .name = vco_name,
>>> + .flags = CLK_IGNORE_UNUSED,
>>
>> Ugh there's that flag again.
>>
>>> + .ops = &dp_10nm_vco_clk_ops,
>>> + };
>>> + struct device *dev = &pll_10nm->pdev->dev;
>>> + struct clk_hw **hws = pll_10nm->hws;
>>> + struct clk_hw_onecell_data *hw_data;
>>> + struct clk_hw *hw;
>>> + int num = 0;
>>> + int ret;
>>> +
>>> + DRM_DEBUG_DP("DP->id = %d", pll_10nm->id);
>>> +
>>> + hw_data = devm_kzalloc(dev, sizeof(*hw_data) +
>>
>> Use devm_kcalloc()
>>
>>> + NUM_PROVIDED_CLKS * sizeof(struct
>>> clk_hw *),
>>> + GFP_KERNEL);
>>> + if (!hw_data)
>>> + return -ENOMEM;
>>> +
>>> + snprintf(vco_name, 32, "dp_vco_clk");
>>> + pll_10nm->base.clk_hw.init = &vco_init;
>>> + ret = clk_hw_register(dev, &pll_10nm->base.clk_hw);
>>> + if (ret)
>>> + return ret;
>>> + hws[num++] = &pll_10nm->base.clk_hw;
>>> +
>>> + snprintf(clk_name, 32, "dp_phy_pll_link_clk");
>>> + snprintf(parent, 32, "dp_vco_clk");
>>
>> Is there a need to make these names with snprintf?
>>
>>> + hw = clk_hw_register_fixed_factor(dev, clk_name, parent,
>>> + CLK_SET_RATE_PARENT, 1,
>>> 10);
>>> + if (IS_ERR(hw))
>>> + return PTR_ERR(hw);
>>> + hws[num++] = hw;
>>> + hw_data->hws[DP_LINK_CLK_SRC] = hw;
>>> +
>>> + snprintf(clk_name, 32, "dp_vco_divsel_two_clk_src");
>>> + snprintf(parent, 32, "dp_vco_clk");
>>> + hw = clk_hw_register_fixed_factor(dev, clk_name, parent,
>>> + 0, 1, 2);
>>> + if (IS_ERR(hw))
>>> + return PTR_ERR(hw);
>>> + hws[num++] = hw;
>>> +
>>> + snprintf(clk_name, 32, "dp_vco_divsel_four_clk_src");
>>> + snprintf(parent, 32, "dp_vco_clk");
>>> + hw = clk_hw_register_fixed_factor(dev, clk_name, parent,
>>> + 0, 1, 4);
>>> + if (IS_ERR(hw))
>>> + return PTR_ERR(hw);
>>> + hws[num++] = hw;
>>> +
>>> + snprintf(clk_name, 32, "dp_vco_divsel_six_clk_src");
>>> + snprintf(parent, 32, "dp_vco_clk");
>>> + hw = clk_hw_register_fixed_factor(dev, clk_name, parent,
>>> + 0, 1, 6);
>>> + if (IS_ERR(hw))
>>> + return PTR_ERR(hw);
>>> + hws[num++] = hw;
>>> +
>>> + hw = dp_pll_10nm_pixel_clk_sel(pll_10nm);
>>> + if (IS_ERR(hw))
>>> + return PTR_ERR(hw);
>>> +
>>> + hws[num++] = hw;
>>> + hw_data->hws[DP_PIXEL_CLK_SRC] = hw;
>>> +
>>> + pll_10nm->num_hws = num;
>>> +
>>> + hw_data->num = NUM_PROVIDED_CLKS;
>>> + pll_10nm->hw_data = hw_data;
>>> +
>>> + ret = of_clk_add_hw_provider(dev->of_node,
>>> of_clk_hw_onecell_get,
>>> + pll_10nm->hw_data);
>>> + if (ret) {
>>> + DRM_DEV_ERROR(dev, "failed to register clk provider:
>>> %d\n",
>>> + ret);
>>> + return ret;
>>> + }
>>> +
>>> + return ret;
>>> +}
>>> +
>>> +struct msm_dp_pll *msm_dp_pll_10nm_init(struct platform_device
>>> *pdev, int id)
>>> +{
>>> + struct dp_pll_10nm *dp_10nm_pll;
>>> + struct msm_dp_pll *pll;
>>> + int ret;
>>> +
>>> + if (!pdev)
>>> + return ERR_PTR(-ENODEV);
>>> +
>>> + dp_10nm_pll = devm_kzalloc(&pdev->dev,
>>> + sizeof(*dp_10nm_pll),
>>> GFP_KERNEL);
>>> + if (!dp_10nm_pll)
>>> + return ERR_PTR(-ENOMEM);
>>> +
>>> + DRM_DEBUG_DP("DP PLL%d", id);
>>> +
>>> + dp_10nm_pll->pdev = pdev;
>>> + dp_10nm_pll->id = id;
>>> + dp_pdb = dp_10nm_pll;
>>> +
>>> + dp_10nm_pll->pll_base = msm_ioremap(pdev, "pll_base",
>>> "DP_PLL");
>>> + if (IS_ERR_OR_NULL(dp_10nm_pll->pll_base)) {
>>> + DRM_DEV_ERROR(&pdev->dev, "failed to map CMN PLL
>>> base\n");
>>> + return ERR_PTR(-ENOMEM);
>>> + }
>>> +
>>> + dp_10nm_pll->phy_base = msm_ioremap(pdev, "phy_base",
>>> "DP_PHY");
>>> + if (IS_ERR_OR_NULL(dp_10nm_pll->phy_base)) {
>>> + DRM_DEV_ERROR(&pdev->dev, "failed to map CMN PHY
>>> base\n");
>>> + return ERR_PTR(-ENOMEM);
>>> + }
>>> +
>>> + dp_10nm_pll->ln_tx0_base = msm_ioremap(pdev, "ln_tx0_base",
>>> + "DP_LN_TX0");
>>> + if (IS_ERR_OR_NULL(dp_10nm_pll->ln_tx0_base)) {
>>> + DRM_DEV_ERROR(&pdev->dev, "failed to map CMN LN_TX0
>>> base\n");
>>> + return ERR_PTR(-ENOMEM);
>>> + }
>>> +
>>> + dp_10nm_pll->ln_tx1_base = msm_ioremap(pdev, "ln_tx1_base",
>>> + "DP_LN_TX1");
>>> + if (IS_ERR_OR_NULL(dp_10nm_pll->ln_tx1_base)) {
>>> + DRM_DEV_ERROR(&pdev->dev, "failed to map CMN LN_TX1
>>> base\n");
>>> + return ERR_PTR(-ENOMEM);
>>> + }
>>> +
>>> + ret = of_property_read_u32(pdev->dev.of_node, "cell-index",
>>> + &dp_10nm_pll->index);
>>> + if (ret) {
>>> + DRM_ERROR("Unable to get the cell-index ret=%d\n",
>>> ret);
>>> + dp_10nm_pll->index = 0;
>>> + }
>>> +
>>> + ret = msm_dp_pll_util_parse_dt_clock(pdev,
>>> &dp_10nm_pll->base);
>>> + if (ret) {
>>> + DRM_ERROR("Unable to parse dt clocks ret=%d\n", ret);
>>> + return ERR_PTR(ret);
>>> + }
>>> +
>>> + ret = dp_pll_10nm_register(dp_10nm_pll);
>>> + if (ret) {
>>> + DRM_DEV_ERROR(&pdev->dev, "failed to register PLL:
>>> %d\n", ret);
>>> + return ERR_PTR(ret);
>>> + }
>>> +
>>> + pll = &dp_10nm_pll->base;
>>> + pll->min_rate = DP_VCO_HSCLK_RATE_1620MHZDIV1000;
>>> + pll->max_rate = DP_VCO_HSCLK_RATE_8100MHZDIV1000;
>>
>> Maybe min/max can be set with clk_hw_set_rate_range()?
> Here pll->min_rate and pll->max_rate stores the VCO range in local
> structure.
>
>>
>>> + pll->get_provider = dp_pll_10nm_get_provider;
>>> +
>>> + return pll;
>>> +}
>>> diff --git a/drivers/gpu/drm/msm/dp/pll/dp_pll_10nm.h
>>> b/drivers/gpu/drm/msm/dp/pll/dp_pll_10nm.h
>>> new file mode 100644
>>> index 0000000..8466463
>>> --- /dev/null
>>> +++ b/drivers/gpu/drm/msm/dp/pll/dp_pll_10nm.h
>>> @@ -0,0 +1,86 @@
>>> +/* SPDX-License-Identifier: GPL-2.0-only */
>>> +/*
>>> + * Copyright (c) 2016-2020, The Linux Foundation. All rights
>>> reserved.
>>> + */
>>> +
>>> +#ifndef __DP_PLL_10NM_H
>>> +#define __DP_PLL_10NM_H
>>> +
>>> +#include "dp_pll.h"
>>> +#include "dp_reg.h"
>>> +
>>> +#define DP_VCO_HSCLK_RATE_1620MHZDIV1000 1620000UL
>>> +#define DP_VCO_HSCLK_RATE_2700MHZDIV1000 2700000UL
>>> +#define DP_VCO_HSCLK_RATE_5400MHZDIV1000 5400000UL
>>> +#define DP_VCO_HSCLK_RATE_8100MHZDIV1000 8100000UL
>>> +
>>> +#define NUM_DP_CLOCKS_MAX 6
>>> +
>>> +#define DP_PHY_PLL_POLL_SLEEP_US 500
>>> +#define DP_PHY_PLL_POLL_TIMEOUT_US 10000
>>> +
>>> +#define DP_VCO_RATE_8100MHZDIV1000 8100000UL
>>> +#define DP_VCO_RATE_9720MHZDIV1000 9720000UL
>>> +#define DP_VCO_RATE_10800MHZDIV1000 10800000UL
>>> +
>>> +struct dp_pll_10nm {
>>> + struct msm_dp_pll base;
>>> +
>>> + int id;
>>> + struct platform_device *pdev;
>>> +
>>> + void __iomem *pll_base;
>>> + void __iomem *phy_base;
>>> + void __iomem *ln_tx0_base;
>>> + void __iomem *ln_tx1_base;
>>> +
>>> + /* private clocks: */
>>> + struct clk_hw *hws[NUM_DP_CLOCKS_MAX];
>>> + u32 num_hws;
>>> +
>>> + /* clock-provider: */
>>> + struct clk_hw_onecell_data *hw_data;
>>> +
>>> + /* lane and orientation settings */
>>> + u8 lane_cnt;
>>> + u8 orientation;
>>> +
>>> + /* COM PHY settings */
>>> + u32 hsclk_sel;
>>> + u32 dec_start_mode0;
>>> + u32 div_frac_start1_mode0;
>>> + u32 div_frac_start2_mode0;
>>> + u32 div_frac_start3_mode0;
>>> + u32 integloop_gain0_mode0;
>>> + u32 integloop_gain1_mode0;
>>> + u32 vco_tune_map;
>>> + u32 lock_cmp1_mode0;
>>> + u32 lock_cmp2_mode0;
>>> + u32 lock_cmp3_mode0;
>>> + u32 lock_cmp_en;
>>> +
>>> + /* PHY vco divider */
>>> + u32 phy_vco_div;
>>> + /*
>>> + * Certain pll's needs to update the same vco rate after
>>> resume in
>>> + * suspend/resume scenario. Cached the vco rate for such
>>> plls.
>>> + */
>>> + unsigned long vco_cached_rate;
>>> + u32 cached_cfg0;
>>> + u32 cached_cfg1;
>>> + u32 cached_outdiv;
>>> +
>>> + uint32_t index;
>>> +};
>>> +
>>> +#define to_dp_pll_10nm(x) container_of(x, struct dp_pll_10nm,
>>> base)
>>
>> Why is this struct in the header file? Does something outside of the
>> PLL
>> driver need to access this struct?
>>
>>> +
>>> +int dp_vco_set_rate_10nm(struct clk_hw *hw, unsigned long rate,
>>> + unsigned long parent_rate);
>>> +unsigned long dp_vco_recalc_rate_10nm(struct clk_hw *hw,
>>> + unsigned long parent_rate);
>>> +long dp_vco_round_rate_10nm(struct clk_hw *hw, unsigned long rate,
>>> + unsigned long *parent_rate);
>>> +int dp_vco_prepare_10nm(struct clk_hw *hw);
>>> +void dp_vco_unprepare_10nm(struct clk_hw *hw);
>>> +#endif /* __DP_PLL_10NM_H */
>>> diff --git a/drivers/gpu/drm/msm/dp/pll/dp_pll_10nm_util.c
>>> b/drivers/gpu/drm/msm/dp/pll/dp_pll_10nm_util.c
>>> new file mode 100644
>>> index 0000000..db3983d5
>>> --- /dev/null
>>> +++ b/drivers/gpu/drm/msm/dp/pll/dp_pll_10nm_util.c
>>> @@ -0,0 +1,524 @@
>>> +// SPDX-License-Identifier: GPL-2.0-only
>>> +/*
>>> + * Copyright (c) 2016-2020, The Linux Foundation. All rights
>>> reserved.
>>> + */
>>> +
>>> +#define pr_fmt(fmt) "%s: " fmt, __func__
>>> +
>>> +#include <linux/delay.h>
>>> +#include <linux/err.h>
>>> +#include <linux/iopoll.h>
>>> +#include <linux/kernel.h>
>>> +
>>> +#include "dp_hpd.h"
>>> +#include "dp_pll.h"
>>> +#include "dp_pll_10nm.h"
>>> +
>>> +static int dp_vco_pll_init_db_10nm(struct msm_dp_pll *pll,
>>> + unsigned long rate)
>>> +{
>>> + struct dp_pll_10nm *dp_res = to_dp_pll_10nm(pll);
>>> + u32 spare_value = 0;
>>> +
>>> + spare_value = PLL_REG_R(dp_res->phy_base, REG_DP_PHY_SPARE0);
>>> + dp_res->lane_cnt = spare_value & 0x0F;
>>> + dp_res->orientation = (spare_value & 0xF0) >> 4;
>>
>> Can this be read at any time from the spare register? Just curious why
>> we need to read this here in init vs. when it's actually used. We
>> could
>> avoid stashing away these values in the struct. It also looks like the
>> PLL part is intimately tied to the phy, but there isn't an actual phy
>> driver. Why? It looks like the phy driver is a clk driver, and the clk
>> framework is not prepared to handle things like lane count and
>> orientation. I'd expect that to come through a PHY framework and then
>> have the phy driver control the PLL by powering it back up and
>> changing
>> it's rates, etc. Not everything needs to be done through clk APIs.
>
> Presently we need this spare register information in many functions
> to program dp hardware. There are atleast 6 instances we use them, so
> we are trying
> to store in the structure once.
> Unfortunately msm DP pll and phy is one hardware tightly bound.
>
>>
>>> +
>>> + DRM_DEBUG_DP("%s: spare_value=0x%x, ln_cnt=0x%x,
>>> orientation=0x%x\n",
>>> + __func__, spare_value, dp_res->lane_cnt,
>>> + dp_res->orientation);
>>> +
>>> + switch (rate) {
>>> + case DP_VCO_HSCLK_RATE_1620MHZDIV1000:
>>> + DRM_DEBUG_DP("%s: VCO rate: %ld\n", __func__,
>>> + DP_VCO_RATE_9720MHZDIV1000);
>>> + dp_res->hsclk_sel = 0x0c;
>>> + dp_res->dec_start_mode0 = 0x69;
>>> + dp_res->div_frac_start1_mode0 = 0x00;
>>> + dp_res->div_frac_start2_mode0 = 0x80;
>>> + dp_res->div_frac_start3_mode0 = 0x07;
>>> + dp_res->integloop_gain0_mode0 = 0x3f;
>>> + dp_res->integloop_gain1_mode0 = 0x00;
>>> + dp_res->vco_tune_map = 0x00;
>>> + dp_res->lock_cmp1_mode0 = 0x6f;
>>> + dp_res->lock_cmp2_mode0 = 0x08;
>>> + dp_res->lock_cmp3_mode0 = 0x00;
>>> + dp_res->phy_vco_div = 0x1;
>>> + dp_res->lock_cmp_en = 0x00;
>>> + break;
>>> + case DP_VCO_HSCLK_RATE_2700MHZDIV1000:
>>> + DRM_DEBUG_DP("%s: VCO rate: %ld\n", __func__,
>>> + DP_VCO_RATE_10800MHZDIV1000);
>>> + dp_res->hsclk_sel = 0x04;
>>> + dp_res->dec_start_mode0 = 0x69;
>>> + dp_res->div_frac_start1_mode0 = 0x00;
>>> + dp_res->div_frac_start2_mode0 = 0x80;
>>> + dp_res->div_frac_start3_mode0 = 0x07;
>>> + dp_res->integloop_gain0_mode0 = 0x3f;
>>> + dp_res->integloop_gain1_mode0 = 0x00;
>>> + dp_res->vco_tune_map = 0x00;
>>> + dp_res->lock_cmp1_mode0 = 0x0f;
>>> + dp_res->lock_cmp2_mode0 = 0x0e;
>>> + dp_res->lock_cmp3_mode0 = 0x00;
>>> + dp_res->phy_vco_div = 0x1;
>>> + dp_res->lock_cmp_en = 0x00;
>>> + break;
>>> + case DP_VCO_HSCLK_RATE_5400MHZDIV1000:
>>> + DRM_DEBUG_DP("%s: VCO rate: %ld\n", __func__,
>>> + DP_VCO_RATE_10800MHZDIV1000);
>>> + dp_res->hsclk_sel = 0x00;
>>> + dp_res->dec_start_mode0 = 0x8c;
>>> + dp_res->div_frac_start1_mode0 = 0x00;
>>> + dp_res->div_frac_start2_mode0 = 0x00;
>>> + dp_res->div_frac_start3_mode0 = 0x0a;
>>> + dp_res->integloop_gain0_mode0 = 0x3f;
>>> + dp_res->integloop_gain1_mode0 = 0x00;
>>> + dp_res->vco_tune_map = 0x00;
>>> + dp_res->lock_cmp1_mode0 = 0x1f;
>>> + dp_res->lock_cmp2_mode0 = 0x1c;
>>> + dp_res->lock_cmp3_mode0 = 0x00;
>>> + dp_res->phy_vco_div = 0x2;
>>> + dp_res->lock_cmp_en = 0x00;
>>> + break;
>>> + case DP_VCO_HSCLK_RATE_8100MHZDIV1000:
>>> + DRM_DEBUG_DP("%s: VCO rate: %ld\n", __func__,
>>> + DP_VCO_RATE_8100MHZDIV1000);
>>> + dp_res->hsclk_sel = 0x03;
>>> + dp_res->dec_start_mode0 = 0x69;
>>> + dp_res->div_frac_start1_mode0 = 0x00;
>>> + dp_res->div_frac_start2_mode0 = 0x80;
>>> + dp_res->div_frac_start3_mode0 = 0x07;
>>> + dp_res->integloop_gain0_mode0 = 0x3f;
>>> + dp_res->integloop_gain1_mode0 = 0x00;
>>> + dp_res->vco_tune_map = 0x00;
>>> + dp_res->lock_cmp1_mode0 = 0x2f;
>>> + dp_res->lock_cmp2_mode0 = 0x2a;
>>> + dp_res->lock_cmp3_mode0 = 0x00;
>>> + dp_res->phy_vco_div = 0x0;
>>> + dp_res->lock_cmp_en = 0x08;
>>> + break;
>>> + default:
>>> + return -EINVAL;
>>> + }
>>> + return 0;
>>> +}
>>> +
>>> +static int dp_config_vco_rate_10nm(struct msm_dp_pll *pll,
>>> + unsigned long rate)
>>> +{
>>> + u32 res = 0;
>>> + struct dp_pll_10nm *dp_res = to_dp_pll_10nm(pll);
>>> +
>>> + res = dp_vco_pll_init_db_10nm(pll, rate);
>>> + if (res) {
>>> + DRM_ERROR("VCO Init DB failed\n");
>>> + return res;
>>> + }
>>> +
>>> + if (dp_res->lane_cnt != 4) {
>>> + if (dp_res->orientation == ORIENTATION_CC2)
>>> + PLL_REG_W(dp_res->phy_base,
>>> REG_DP_PHY_PD_CTL, 0x6d);
>>> + else
>>> + PLL_REG_W(dp_res->phy_base,
>>> REG_DP_PHY_PD_CTL, 0x75);
>>> + } else {
>>> + PLL_REG_W(dp_res->phy_base, REG_DP_PHY_PD_CTL, 0x7d);
>>> + }
>>> +
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_SVS_MODE_CLK_SEL,
>>> 0x01);
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_SYSCLK_EN_SEL, 0x37);
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_SYS_CLK_CTRL, 0x02);
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_CLK_ENABLE1, 0x0e);
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_SYSCLK_BUF_ENABLE,
>>> 0x06);
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_CLK_SEL, 0x30);
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_CMN_CONFIG, 0x02);
>>> +
>>> + /* Different for each clock rates */
>>> + PLL_REG_W(dp_res->pll_base,
>>> + QSERDES_COM_HSCLK_SEL, dp_res->hsclk_sel);
>>> + PLL_REG_W(dp_res->pll_base,
>>> + QSERDES_COM_DEC_START_MODE0,
>>> dp_res->dec_start_mode0);
>>> + PLL_REG_W(dp_res->pll_base,
>>> + QSERDES_COM_DIV_FRAC_START1_MODE0,
>>> + dp_res->div_frac_start1_mode0);
>>> + PLL_REG_W(dp_res->pll_base,
>>> + QSERDES_COM_DIV_FRAC_START2_MODE0,
>>> + dp_res->div_frac_start2_mode0);
>>> + PLL_REG_W(dp_res->pll_base,
>>> + QSERDES_COM_DIV_FRAC_START3_MODE0,
>>> + dp_res->div_frac_start3_mode0);
>>> + PLL_REG_W(dp_res->pll_base,
>>> + QSERDES_COM_INTEGLOOP_GAIN0_MODE0,
>>> + dp_res->integloop_gain0_mode0);
>>> + PLL_REG_W(dp_res->pll_base,
>>> + QSERDES_COM_INTEGLOOP_GAIN1_MODE0,
>>> + dp_res->integloop_gain1_mode0);
>>> + PLL_REG_W(dp_res->pll_base,
>>> + QSERDES_COM_VCO_TUNE_MAP, dp_res->vco_tune_map);
>>> + PLL_REG_W(dp_res->pll_base,
>>> + QSERDES_COM_LOCK_CMP1_MODE0,
>>> dp_res->lock_cmp1_mode0);
>>> + PLL_REG_W(dp_res->pll_base,
>>> + QSERDES_COM_LOCK_CMP2_MODE0,
>>> dp_res->lock_cmp2_mode0);
>>> + PLL_REG_W(dp_res->pll_base,
>>> + QSERDES_COM_LOCK_CMP3_MODE0,
>>> dp_res->lock_cmp3_mode0);
>>> +
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_BG_TIMER, 0x0a);
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_CORECLK_DIV_MODE0,
>>> 0x0a);
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_VCO_TUNE_CTRL, 0x00);
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_BIAS_EN_CLKBUFLR_EN,
>>> 0x3f);
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_CORE_CLK_EN, 0x1f);
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_PLL_IVCO, 0x07);
>>> + PLL_REG_W(dp_res->pll_base,
>>> + QSERDES_COM_LOCK_CMP_EN, dp_res->lock_cmp_en);
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_PLL_CCTRL_MODE0,
>>> 0x36);
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_PLL_RCTRL_MODE0,
>>> 0x16);
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_CP_CTRL_MODE0, 0x06);
>>> +
>>> + if (dp_res->orientation == ORIENTATION_CC2)
>>> + PLL_REG_W(dp_res->phy_base, REG_DP_PHY_MODE, 0x4c);
>>> + else
>>> + PLL_REG_W(dp_res->phy_base, REG_DP_PHY_MODE, 0x5c);
>>> +
>>> + /* TX Lane configuration */
>>> + PLL_REG_W(dp_res->phy_base,
>>> + REG_DP_PHY_TX0_TX1_LANE_CTL, 0x05);
>>> + PLL_REG_W(dp_res->phy_base,
>>> + REG_DP_PHY_TX2_TX3_LANE_CTL, 0x05);
>>> +
>>> + /* TX-0 register configuration */
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_TRANSCEIVER_BIAS_EN, 0x1a);
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_VMODE_CTRL1, 0x40);
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_PRE_STALL_LDO_BOOST_EN, 0x30);
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_INTERFACE_SELECT, 0x3d);
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_CLKBUF_ENABLE, 0x0f);
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_RESET_TSYNC_EN, 0x03);
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_TRAN_DRVR_EMP_EN, 0x03);
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_PARRATE_REC_DETECT_IDLE_EN,
>>> 0x00);
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_TX_INTERFACE_MODE, 0x00);
>>> + PLL_REG_W(dp_res->ln_tx0_base, REG_DP_PHY_TXn_TX_BAND, 0x4);
>>> +
>>> + /* TX-1 register configuration */
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_TRANSCEIVER_BIAS_EN, 0x1a);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_VMODE_CTRL1, 0x40);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_PRE_STALL_LDO_BOOST_EN, 0x30);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_INTERFACE_SELECT, 0x3d);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_CLKBUF_ENABLE, 0x0f);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_RESET_TSYNC_EN, 0x03);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_TRAN_DRVR_EMP_EN, 0x03);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_PARRATE_REC_DETECT_IDLE_EN,
>>> 0x00);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_TX_INTERFACE_MODE, 0x00);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_TX_BAND, 0x4);
>>> +
>>> + /* dependent on the vco frequency */
>>> + PLL_REG_W(dp_res->phy_base,
>>> + REG_DP_PHY_VCO_DIV, dp_res->phy_vco_div);
>>> +
>>> + return res;
>>> +}
>>> +
>>> +static bool dp_10nm_pll_lock_status(struct dp_pll_10nm *dp_res)
>>> +{
>>> + u32 status;
>>> + bool pll_locked;
>>> +
>>> + /* poll for PLL lock status */
>>> + if (readl_poll_timeout_atomic((dp_res->pll_base +
>>> + QSERDES_COM_C_READY_STATUS),
>>> + status,
>>> + ((status & BIT(0)) > 0),
>>> + DP_PHY_PLL_POLL_SLEEP_US,
>>> + DP_PHY_PLL_POLL_TIMEOUT_US)) {
>>> + DRM_ERROR("%s: C_READY status is not high.
>>> Status=%x\n",
>>> + __func__, status);
>>> + pll_locked = false;
>>> + } else {
>>> + pll_locked = true;
>>> + }
>>> +
>>> + return pll_locked;
>>> +}
>>> +
>>> +static bool dp_10nm_phy_rdy_status(struct dp_pll_10nm *dp_res)
>>> +{
>>> + u32 status;
>>> + bool phy_ready = true;
>>> +
>>> + /* poll for PHY ready status */
>>> + if (readl_poll_timeout_atomic((dp_res->phy_base +
>>> + REG_DP_PHY_STATUS),
>>> + status,
>>> + ((status & (BIT(1))) > 0),
>>> + DP_PHY_PLL_POLL_SLEEP_US,
>>> + DP_PHY_PLL_POLL_TIMEOUT_US)) {
>>> + DRM_ERROR("%s: Phy_ready is not high. Status=%x\n",
>>> + __func__, status);
>>> + phy_ready = false;
>>> + }
>>> +
>>> + return phy_ready;
>>> +}
>>> +
>>> +static int dp_pll_enable_10nm(struct clk_hw *hw)
>>> +{
>>> + int rc = 0;
>>> + struct msm_dp_pll *pll = hw_clk_to_pll(hw);
>>> + struct dp_pll_10nm *dp_res = to_dp_pll_10nm(pll);
>>> + u32 bias_en, drvr_en;
>>> +
>>> + PLL_REG_W(dp_res->phy_base, REG_DP_PHY_AUX_CFG2, 0x04);
>>> + PLL_REG_W(dp_res->phy_base, REG_DP_PHY_CFG, 0x01);
>>> + PLL_REG_W(dp_res->phy_base, REG_DP_PHY_CFG, 0x05);
>>> + PLL_REG_W(dp_res->phy_base, REG_DP_PHY_CFG, 0x01);
>>> + PLL_REG_W(dp_res->phy_base, REG_DP_PHY_CFG, 0x09);
>>> +
>>> + PLL_REG_W(dp_res->pll_base, QSERDES_COM_RESETSM_CNTRL, 0x20);
>>> +
>>> + if (!dp_10nm_pll_lock_status(dp_res)) {
>>> + rc = -EINVAL;
>>> + goto lock_err;
>>> + }
>>> +
>>> + PLL_REG_W(dp_res->phy_base, REG_DP_PHY_CFG, 0x19);
>>> + /* poll for PHY ready status */
>>> + if (!dp_10nm_phy_rdy_status(dp_res)) {
>>> + rc = -EINVAL;
>>> + goto lock_err;
>>> + }
>>> +
>>> + DRM_DEBUG_DP("%s: PLL is locked\n", __func__);
>>> +
>>> + if (dp_res->lane_cnt == 1) {
>>> + bias_en = 0x3e;
>>> + drvr_en = 0x13;
>>> + } else {
>>> + bias_en = 0x3f;
>>> + drvr_en = 0x10;
>>> + }
>>> +
>>> + if (dp_res->lane_cnt != 4) {
>>> + if (dp_res->orientation == ORIENTATION_CC1) {
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_HIGHZ_DRVR_EN,
>>> drvr_en);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_TRANSCEIVER_BIAS_EN,
>>> bias_en);
>>> + } else {
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_HIGHZ_DRVR_EN,
>>> drvr_en);
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_TRANSCEIVER_BIAS_EN,
>>> bias_en);
>>> + }
>>> + } else {
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_HIGHZ_DRVR_EN,
>>> drvr_en);
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_TRANSCEIVER_BIAS_EN,
>>> bias_en);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_HIGHZ_DRVR_EN,
>>> drvr_en);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_TRANSCEIVER_BIAS_EN,
>>> bias_en);
>>> + }
>>> +
>>> + PLL_REG_W(dp_res->ln_tx0_base, REG_DP_PHY_TXn_TX_POL_INV,
>>> 0x0a);
>>> + PLL_REG_W(dp_res->ln_tx1_base, REG_DP_PHY_TXn_TX_POL_INV,
>>> 0x0a);
>>> + PLL_REG_W(dp_res->phy_base, REG_DP_PHY_CFG, 0x18);
>>> + udelay(2000);
>>> +
>>> + PLL_REG_W(dp_res->phy_base, REG_DP_PHY_CFG, 0x19);
>>> +
>>> + /* poll for PHY ready status */
>>> + if (!dp_10nm_phy_rdy_status(dp_res)) {
>>> + rc = -EINVAL;
>>> + goto lock_err;
>>> + }
>>> +
>>> + PLL_REG_W(dp_res->ln_tx0_base, REG_DP_PHY_TXn_TX_DRV_LVL,
>>> 0x38);
>>> + PLL_REG_W(dp_res->ln_tx1_base, REG_DP_PHY_TXn_TX_DRV_LVL,
>>> 0x38);
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> REG_DP_PHY_TXn_TX_EMP_POST1_LVL, 0x20);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> REG_DP_PHY_TXn_TX_EMP_POST1_LVL, 0x20);
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_RES_CODE_LANE_OFFSET_TX,
>>> 0x06);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_RES_CODE_LANE_OFFSET_TX,
>>> 0x06);
>>> + PLL_REG_W(dp_res->ln_tx0_base,
>>> + REG_DP_PHY_TXn_RES_CODE_LANE_OFFSET_RX,
>>> 0x07);
>>> + PLL_REG_W(dp_res->ln_tx1_base,
>>> + REG_DP_PHY_TXn_RES_CODE_LANE_OFFSET_RX,
>>> 0x07);
>>> +
>>> +lock_err:
>>> + return rc;
>>> +}
>>> +
>>> +static int dp_pll_disable_10nm(struct clk_hw *hw)
>>> +{
>>> + int rc = 0;
>>> + struct msm_dp_pll *pll = hw_clk_to_pll(hw);
>>> + struct dp_pll_10nm *dp_res = to_dp_pll_10nm(pll);
>>> +
>>> + /* Assert DP PHY power down */
>>> + PLL_REG_W(dp_res->phy_base, REG_DP_PHY_PD_CTL, 0x2);
>>> +
>>> + return rc;
>>> +}
>>> +
>>> +
>>> +int dp_vco_prepare_10nm(struct clk_hw *hw)
>>> +{
>>> + int rc = 0;
>>> + struct msm_dp_pll *pll = hw_clk_to_pll(hw);
>>> + struct dp_pll_10nm *dp_res = to_dp_pll_10nm(pll);
>>> +
>>> + DRM_DEBUG_DP("%s: rate = %ld\n", __func__, pll->rate);
>>> + if ((dp_res->vco_cached_rate != 0)
>>> + && (dp_res->vco_cached_rate == pll->rate)) {
>>> + rc = dp_vco_set_rate_10nm(hw,
>>> + dp_res->vco_cached_rate,
>>> dp_res->vco_cached_rate);
>>> + if (rc) {
>>> + DRM_ERROR("index=%d vco_set_rate failed.
>>> rc=%d\n",
>>> + rc, dp_res->index);
>>> + goto error;
>>> + }
>>> + }
>>> +
>>> + rc = dp_pll_enable_10nm(hw);
>>> + if (rc) {
>>> + DRM_ERROR("ndx=%d failed to enable dp pll\n",
>>> + dp_res->index);
>>> + goto error;
>>> + }
>>> +
>>> + pll->pll_on = true;
>>> +error:
>>> + return rc;
>>> +}
>>> +
>>> +void dp_vco_unprepare_10nm(struct clk_hw *hw)
>>> +{
>>> + struct msm_dp_pll *pll = hw_clk_to_pll(hw);
>>> + struct dp_pll_10nm *dp_res = to_dp_pll_10nm(pll);
>>> +
>>> + if (!dp_res) {
>>> + DRM_ERROR("Invalid input parameter\n");
>>> + return;
>>> + }
>>> +
>>> + if (!pll->pll_on) {
>>> + DRM_ERROR("pll resource can't be enabled\n");
>>> + return;
>>> + }
>>> + dp_res->vco_cached_rate = pll->rate;
>>> + dp_pll_disable_10nm(hw);
>>> +
>>> + pll->pll_on = false;
>>> +}
>>> +
>>> +int dp_vco_set_rate_10nm(struct clk_hw *hw, unsigned long rate,
>>> + unsigned long parent_rate)
>>> +{
>>> + struct msm_dp_pll *pll = hw_clk_to_pll(hw);
>>> + int rc;
>>> +
>>> + DRM_DEBUG_DP("DP lane CLK rate=%ld\n", rate);
>>> +
>>> + rc = dp_config_vco_rate_10nm(pll, rate);
>>> + if (rc)
>>> + DRM_ERROR("%s: Failed to set clk rate\n", __func__);
>>> +
>>> + pll->rate = rate;
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +unsigned long dp_vco_recalc_rate_10nm(struct clk_hw *hw,
>>> + unsigned long parent_rate)
>>> +{
>>> + struct msm_dp_pll *pll = hw_clk_to_pll(hw);
>>> + struct dp_pll_10nm *dp_res = to_dp_pll_10nm(pll);
>>> + u32 div, hsclk_div, link_clk_div = 0;
>>> + u64 vco_rate;
>>> +
>>> + div = PLL_REG_R(dp_res->pll_base, QSERDES_COM_HSCLK_SEL);
>>> + div &= 0x0f;
>>> +
>>> + if (div == 12)
>>> + hsclk_div = 6; /* Default */
>>> + else if (div == 4)
>>> + hsclk_div = 4;
>>> + else if (div == 0)
>>> + hsclk_div = 2;
>>> + else if (div == 3)
>>> + hsclk_div = 1;
>>> + else {
>>> + DRM_DEBUG_DP("unknown divider. forcing to
>>> default\n");
>>> + hsclk_div = 5;
>>> + }
>>> +
>>> + div = PLL_REG_R(dp_res->phy_base, REG_DP_PHY_AUX_CFG2);
>>> + div >>= 2;
>>> +
>>> + if ((div & 0x3) == 0)
>>> + link_clk_div = 5;
>>> + else if ((div & 0x3) == 1)
>>> + link_clk_div = 10;
>>> + else if ((div & 0x3) == 2)
>>> + link_clk_div = 20;
>>> + else
>>> + DRM_ERROR("%s: unsupported div. Phy_mode: %d\n",
>>> __func__, div);
>>> +
>>> + if (link_clk_div == 20) {
>>> + vco_rate = DP_VCO_HSCLK_RATE_2700MHZDIV1000;
>>> + } else {
>>> + if (hsclk_div == 6)
>>> + vco_rate = DP_VCO_HSCLK_RATE_1620MHZDIV1000;
>>> + else if (hsclk_div == 4)
>>> + vco_rate = DP_VCO_HSCLK_RATE_2700MHZDIV1000;
>>> + else if (hsclk_div == 2)
>>> + vco_rate = DP_VCO_HSCLK_RATE_5400MHZDIV1000;
>>> + else
>>> + vco_rate = DP_VCO_HSCLK_RATE_8100MHZDIV1000;
>>> + }
>>> +
>>> + DRM_DEBUG_DP("returning vco rate = %lu\n", (unsigned
>>> long)vco_rate);
>>> +
>>> + dp_res->vco_cached_rate = pll->rate = vco_rate;
>>> + return (unsigned long)vco_rate;
>>> +}
>>> +
>>> +long dp_vco_round_rate_10nm(struct clk_hw *hw, unsigned long rate,
>>> + unsigned long *parent_rate)
>>> +{
>>> + unsigned long rrate = rate;
>>> + struct msm_dp_pll *pll = hw_clk_to_pll(hw);
>>> +
>>> + if (rate <= pll->min_rate)
>>> + rrate = pll->min_rate;
>>> + else if (rate <= DP_VCO_HSCLK_RATE_2700MHZDIV1000)
>>> + rrate = DP_VCO_HSCLK_RATE_2700MHZDIV1000;
>>> + else if (rate <= DP_VCO_HSCLK_RATE_5400MHZDIV1000)
>>> + rrate = DP_VCO_HSCLK_RATE_5400MHZDIV1000;
>>> + else
>>> + rrate = pll->max_rate;
>>
>> A switch statement would be more readable. Not sure why pll->min_rate
>> is
>> needed though.
>
> pll->min_rate is the minimum vco clock that is supported on the
> chipset.
> We are trying to make sure that we dont go below the min vco
> level for that chipset.
>
>>
>> switch (rate) {
>> case 0 .. pll->min_rate:
>> rrate =
>> break;
>> case pll->min_rate .. DP_VCO_HSCLK_RATE_2700MHZDIV1000:
>> ...
>> }
>>
>>> +
>>> + DRM_DEBUG_DP("%s: rrate=%ld\n", __func__, rrate);
>>> +
>>> + *parent_rate = rrate;
>>> + return rrate;
>>> +}
More information about the Freedreno
mailing list