223 lines
6.1 KiB
Diff
223 lines
6.1 KiB
Diff
|
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
||
|
From: Cristian Ciocaltea <cristian.ciocaltea@collabora.com>
|
||
|
Date: Tue, 16 Jan 2024 19:27:40 +0200
|
||
|
Subject: phy: phy-rockchip-samsung-hdptx: Add clock provider
|
||
|
|
||
|
The HDMI PHY PLL can be used as an alternative dclk source to SoC CRU.
|
||
|
It provides more accurate clock rates required to properly support
|
||
|
various display modes, e.g. those relying on non-integer refresh rates.
|
||
|
|
||
|
Also note this only works for HDMI 2.0 or bellow, e.g. cannot be used to
|
||
|
support HDMI 2.1 4K@120Hz mode.
|
||
|
|
||
|
Signed-off-by: Cristian Ciocaltea <cristian.ciocaltea@collabora.com>
|
||
|
---
|
||
|
drivers/phy/rockchip/phy-rockchip-samsung-hdptx.c | 148 +++++++++-
|
||
|
1 file changed, 143 insertions(+), 5 deletions(-)
|
||
|
|
||
|
diff --git a/drivers/phy/rockchip/phy-rockchip-samsung-hdptx.c b/drivers/phy/rockchip/phy-rockchip-samsung-hdptx.c
|
||
|
index 111111111111..222222222222 100644
|
||
|
--- a/drivers/phy/rockchip/phy-rockchip-samsung-hdptx.c
|
||
|
+++ b/drivers/phy/rockchip/phy-rockchip-samsung-hdptx.c
|
||
|
@@ -8,6 +8,7 @@
|
||
|
*/
|
||
|
#include <linux/bitfield.h>
|
||
|
#include <linux/clk.h>
|
||
|
+#include <linux/clk-provider.h>
|
||
|
#include <linux/delay.h>
|
||
|
#include <linux/mfd/syscon.h>
|
||
|
#include <linux/module.h>
|
||
|
@@ -279,6 +280,12 @@ struct rk_hdptx_phy {
|
||
|
int nr_clks;
|
||
|
struct reset_control_bulk_data rsts[RST_MAX];
|
||
|
bool earc_en;
|
||
|
+
|
||
|
+ /* clk provider */
|
||
|
+ struct clk_hw hw;
|
||
|
+ unsigned long rate;
|
||
|
+ int id;
|
||
|
+ int count;
|
||
|
};
|
||
|
|
||
|
static const struct lcpll_config lcpll_cfg[] = {
|
||
|
@@ -1031,6 +1038,8 @@ static int rk_hdptx_ropll_tmds_cmn_config(struct rk_hdptx_phy *hdptx,
|
||
|
const struct ropll_config *cfg = NULL;
|
||
|
struct ropll_config rc = {0};
|
||
|
|
||
|
+ hdptx->rate = rate * 100;
|
||
|
+
|
||
|
if (color_depth)
|
||
|
rate = rate * 10 / 8;
|
||
|
|
||
|
@@ -1315,11 +1324,13 @@ static int rk_hdptx_phy_power_off(struct phy *phy)
|
||
|
{
|
||
|
struct rk_hdptx_phy *hdptx = phy_get_drvdata(phy);
|
||
|
u32 val;
|
||
|
- int ret;
|
||
|
+ int ret = 0;
|
||
|
|
||
|
- ret = regmap_read(hdptx->grf, GRF_HDPTX_STATUS, &val);
|
||
|
- if (ret == 0 && (val & HDPTX_O_PLL_LOCK_DONE))
|
||
|
- rk_hdptx_phy_disable(hdptx);
|
||
|
+ if (hdptx->count == 0) {
|
||
|
+ ret = regmap_read(hdptx->grf, GRF_HDPTX_STATUS, &val);
|
||
|
+ if (ret == 0 && (val & HDPTX_O_PLL_LOCK_DONE))
|
||
|
+ rk_hdptx_phy_disable(hdptx);
|
||
|
+ }
|
||
|
|
||
|
pm_runtime_put(hdptx->dev);
|
||
|
|
||
|
@@ -1332,6 +1343,129 @@ static const struct phy_ops rk_hdptx_phy_ops = {
|
||
|
.owner = THIS_MODULE,
|
||
|
};
|
||
|
|
||
|
+static struct rk_hdptx_phy *to_rk_hdptx_phy(struct clk_hw *hw)
|
||
|
+{
|
||
|
+ return container_of(hw, struct rk_hdptx_phy, hw);
|
||
|
+}
|
||
|
+
|
||
|
+static int rk_hdptx_phy_clk_prepare(struct clk_hw *hw)
|
||
|
+{
|
||
|
+ struct rk_hdptx_phy *hdptx = to_rk_hdptx_phy(hw);
|
||
|
+ int ret;
|
||
|
+
|
||
|
+ ret = pm_runtime_resume_and_get(hdptx->dev);
|
||
|
+ if (ret) {
|
||
|
+ dev_err(hdptx->dev, "Failed to resume phy clk: %d\n", ret);
|
||
|
+ return ret;
|
||
|
+ }
|
||
|
+
|
||
|
+ if (!hdptx->count && hdptx->rate) {
|
||
|
+ ret = rk_hdptx_ropll_tmds_cmn_config(hdptx, hdptx->rate / 100);
|
||
|
+ if (ret < 0) {
|
||
|
+ dev_err(hdptx->dev, "Failed to init PHY PLL: %d\n", ret);
|
||
|
+ pm_runtime_put(hdptx->dev);
|
||
|
+ return ret;
|
||
|
+ }
|
||
|
+ }
|
||
|
+
|
||
|
+ hdptx->count++;
|
||
|
+
|
||
|
+ return 0;
|
||
|
+}
|
||
|
+
|
||
|
+static void rk_hdptx_phy_clk_unprepare(struct clk_hw *hw)
|
||
|
+{
|
||
|
+ struct rk_hdptx_phy *hdptx = to_rk_hdptx_phy(hw);
|
||
|
+
|
||
|
+ if (hdptx->count == 1) {
|
||
|
+ u32 val;
|
||
|
+ int ret = regmap_read(hdptx->grf, GRF_HDPTX_STATUS, &val);
|
||
|
+ if (ret == 0 && (val & HDPTX_O_PLL_LOCK_DONE))
|
||
|
+ rk_hdptx_phy_disable(hdptx);
|
||
|
+ }
|
||
|
+
|
||
|
+ hdptx->count--;
|
||
|
+ pm_runtime_put(hdptx->dev);
|
||
|
+}
|
||
|
+
|
||
|
+static unsigned long rk_hdptx_phy_clk_recalc_rate(struct clk_hw *hw,
|
||
|
+ unsigned long parent_rate)
|
||
|
+{
|
||
|
+ struct rk_hdptx_phy *hdptx = to_rk_hdptx_phy(hw);
|
||
|
+
|
||
|
+ return hdptx->rate;
|
||
|
+}
|
||
|
+
|
||
|
+static long rk_hdptx_phy_clk_round_rate(struct clk_hw *hw, unsigned long rate,
|
||
|
+ unsigned long *parent_rate)
|
||
|
+{
|
||
|
+ const struct ropll_config *cfg = NULL;
|
||
|
+ u32 bit_rate = rate / 100;
|
||
|
+ int i;
|
||
|
+
|
||
|
+ if (rate > HDMI20_MAX_RATE)
|
||
|
+ return rate;
|
||
|
+
|
||
|
+ for (i = 0; i < ARRAY_SIZE(ropll_tmds_cfg); i++)
|
||
|
+ if (bit_rate == ropll_tmds_cfg[i].bit_rate) {
|
||
|
+ cfg = &ropll_tmds_cfg[i];
|
||
|
+ break;
|
||
|
+ }
|
||
|
+
|
||
|
+ if (!cfg && !rk_hdptx_phy_clk_pll_calc(bit_rate, NULL))
|
||
|
+ return -EINVAL;
|
||
|
+
|
||
|
+ return rate;
|
||
|
+}
|
||
|
+
|
||
|
+static int rk_hdptx_phy_clk_set_rate(struct clk_hw *hw, unsigned long rate,
|
||
|
+ unsigned long parent_rate)
|
||
|
+{
|
||
|
+ struct rk_hdptx_phy *hdptx = to_rk_hdptx_phy(hw);
|
||
|
+ u32 val;
|
||
|
+ int ret = regmap_read(hdptx->grf, GRF_HDPTX_STATUS, &val);
|
||
|
+ if (ret == 0 && (val & HDPTX_O_PLL_LOCK_DONE))
|
||
|
+ rk_hdptx_phy_disable(hdptx);
|
||
|
+
|
||
|
+ return rk_hdptx_ropll_tmds_cmn_config(hdptx, rate / 100);
|
||
|
+}
|
||
|
+
|
||
|
+static const struct clk_ops hdptx_phy_clk_ops = {
|
||
|
+ .prepare = rk_hdptx_phy_clk_prepare,
|
||
|
+ .unprepare = rk_hdptx_phy_clk_unprepare,
|
||
|
+ .recalc_rate = rk_hdptx_phy_clk_recalc_rate,
|
||
|
+ .round_rate = rk_hdptx_phy_clk_round_rate,
|
||
|
+ .set_rate = rk_hdptx_phy_clk_set_rate,
|
||
|
+};
|
||
|
+
|
||
|
+static int rk_hdptx_phy_clk_register(struct rk_hdptx_phy *hdptx)
|
||
|
+{
|
||
|
+ struct device *dev = hdptx->dev;
|
||
|
+ const char *name, *pname;
|
||
|
+ struct clk *refclk;
|
||
|
+ int ret;
|
||
|
+
|
||
|
+ refclk = devm_clk_get(dev, "ref");
|
||
|
+ if (IS_ERR(refclk))
|
||
|
+ return dev_err_probe(dev, PTR_ERR(refclk),
|
||
|
+ "Failed to get ref clock\n");
|
||
|
+
|
||
|
+ pname = __clk_get_name(refclk);
|
||
|
+ name = hdptx->id ? "clk_hdmiphy_pixel1" : "clk_hdmiphy_pixel0";
|
||
|
+ hdptx->hw.init = CLK_HW_INIT(name, pname, &hdptx_phy_clk_ops,
|
||
|
+ CLK_GET_RATE_NOCACHE);
|
||
|
+
|
||
|
+ ret = devm_clk_hw_register(dev, &hdptx->hw);
|
||
|
+ if (ret)
|
||
|
+ return dev_err_probe(dev, ret, "Failed to register clock\n");
|
||
|
+
|
||
|
+ ret = devm_of_clk_add_hw_provider(dev, of_clk_hw_simple_get, &hdptx->hw);
|
||
|
+ if (ret)
|
||
|
+ return dev_err_probe(dev, ret,
|
||
|
+ "Failed to register clk provider\n");
|
||
|
+ return 0;
|
||
|
+}
|
||
|
+
|
||
|
static int rk_hdptx_phy_runtime_suspend(struct device *dev)
|
||
|
{
|
||
|
struct rk_hdptx_phy *hdptx = dev_get_drvdata(dev);
|
||
|
@@ -1367,6 +1501,10 @@ static int rk_hdptx_phy_probe(struct platform_device *pdev)
|
||
|
|
||
|
hdptx->dev = dev;
|
||
|
|
||
|
+ hdptx->id = of_alias_get_id(dev->of_node, "hdptxphy");
|
||
|
+ if (hdptx->id < 0)
|
||
|
+ hdptx->id = 0;
|
||
|
+
|
||
|
regs = devm_platform_ioremap_resource(pdev, 0);
|
||
|
if (IS_ERR(regs))
|
||
|
return dev_err_probe(dev, PTR_ERR(regs),
|
||
|
@@ -1426,7 +1564,7 @@ static int rk_hdptx_phy_probe(struct platform_device *pdev)
|
||
|
reset_control_deassert(hdptx->rsts[RST_CMN].rstc);
|
||
|
reset_control_deassert(hdptx->rsts[RST_INIT].rstc);
|
||
|
|
||
|
- return 0;
|
||
|
+ return rk_hdptx_phy_clk_register(hdptx);
|
||
|
}
|
||
|
|
||
|
static const struct dev_pm_ops rk_hdptx_phy_pm_ops = {
|
||
|
--
|
||
|
Armbian
|
||
|
|