1b28be59aSCK Hu // SPDX-License-Identifier: GPL-2.0
2b28be59aSCK Hu /*
3b28be59aSCK Hu  * Copyright (c) 2018 MediaTek Inc.
4b28be59aSCK Hu  * Author: Chunhui Dai <chunhui.dai@mediatek.com>
5b28be59aSCK Hu  */
6b28be59aSCK Hu 
7b28be59aSCK Hu #include "phy-mtk-hdmi.h"
8*cff81a61SChunfeng Yun #include "phy-mtk-io.h"
9b28be59aSCK Hu 
10b28be59aSCK Hu #define HDMI_CON0	0x00
11b0870c01SChunfeng Yun #define RG_HDMITX_DRV_IBIAS_MASK	GENMASK(5, 0)
12b0870c01SChunfeng Yun #define RG_HDMITX_EN_SER_MASK		GENMASK(15, 12)
13b0870c01SChunfeng Yun #define RG_HDMITX_EN_SLDO_MASK		GENMASK(19, 16)
14b0870c01SChunfeng Yun #define RG_HDMITX_EN_PRED_MASK		GENMASK(23, 20)
15b0870c01SChunfeng Yun #define RG_HDMITX_EN_IMP_MASK		GENMASK(27, 24)
16b0870c01SChunfeng Yun #define RG_HDMITX_EN_DRV_MASK		GENMASK(31, 28)
17b28be59aSCK Hu 
18b28be59aSCK Hu #define HDMI_CON1	0x04
19b0870c01SChunfeng Yun #define RG_HDMITX_PRED_IBIAS_MASK	GENMASK(21, 18)
20b0870c01SChunfeng Yun #define RG_HDMITX_PRED_IMP		BIT(22)
21b0870c01SChunfeng Yun #define RG_HDMITX_DRV_IMP_MASK		GENMASK(31, 26)
22b28be59aSCK Hu 
23b28be59aSCK Hu #define HDMI_CON2	0x08
24b0870c01SChunfeng Yun #define RG_HDMITX_EN_TX_CKLDO		BIT(0)
25b0870c01SChunfeng Yun #define RG_HDMITX_EN_TX_POSDIV		BIT(1)
26b0870c01SChunfeng Yun #define RG_HDMITX_TX_POSDIV_MASK	GENMASK(4, 3)
27b0870c01SChunfeng Yun #define RG_HDMITX_EN_MBIAS		BIT(6)
28b0870c01SChunfeng Yun #define RG_HDMITX_MBIAS_LPF_EN		BIT(7)
29b28be59aSCK Hu 
30b28be59aSCK Hu #define HDMI_CON4	0x10
31b0870c01SChunfeng Yun #define RG_HDMITX_RESERVE_MASK		GENMASK(31, 0)
32b28be59aSCK Hu 
33b28be59aSCK Hu #define HDMI_CON6	0x18
34b0870c01SChunfeng Yun #define RG_HTPLL_BR_MASK		GENMASK(1, 0)
35b0870c01SChunfeng Yun #define RG_HTPLL_BC_MASK		GENMASK(3, 2)
36b0870c01SChunfeng Yun #define RG_HTPLL_BP_MASK		GENMASK(7, 4)
37b0870c01SChunfeng Yun #define RG_HTPLL_IR_MASK		GENMASK(11, 8)
38b0870c01SChunfeng Yun #define RG_HTPLL_IC_MASK		GENMASK(15, 12)
39b0870c01SChunfeng Yun #define RG_HTPLL_POSDIV_MASK		GENMASK(17, 16)
40b0870c01SChunfeng Yun #define RG_HTPLL_PREDIV_MASK		GENMASK(19, 18)
41b0870c01SChunfeng Yun #define RG_HTPLL_FBKSEL_MASK		GENMASK(21, 20)
42b0870c01SChunfeng Yun #define RG_HTPLL_RLH_EN			BIT(22)
43b0870c01SChunfeng Yun #define RG_HTPLL_FBKDIV_MASK		GENMASK(30, 24)
44b0870c01SChunfeng Yun #define RG_HTPLL_EN			BIT(31)
45b28be59aSCK Hu 
46b28be59aSCK Hu #define HDMI_CON7	0x1c
47b0870c01SChunfeng Yun #define RG_HTPLL_AUTOK_EN		BIT(23)
48b0870c01SChunfeng Yun #define RG_HTPLL_DIVEN_MASK		GENMASK(30, 28)
49b28be59aSCK Hu 
mtk_hdmi_pll_prepare(struct clk_hw * hw)50b28be59aSCK Hu static int mtk_hdmi_pll_prepare(struct clk_hw *hw)
51b28be59aSCK Hu {
52b28be59aSCK Hu 	struct mtk_hdmi_phy *hdmi_phy = to_mtk_hdmi_phy(hw);
53*cff81a61SChunfeng Yun 	void __iomem *base = hdmi_phy->regs;
54b28be59aSCK Hu 
55*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON7, RG_HTPLL_AUTOK_EN);
56*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON6, RG_HTPLL_RLH_EN);
57*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON6, RG_HTPLL_POSDIV_MASK);
58*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON2, RG_HDMITX_EN_MBIAS);
59b28be59aSCK Hu 	usleep_range(80, 100);
60*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON6, RG_HTPLL_EN);
61*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON2, RG_HDMITX_EN_TX_CKLDO);
62*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON0, RG_HDMITX_EN_SLDO_MASK);
63b28be59aSCK Hu 	usleep_range(80, 100);
64*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON2, RG_HDMITX_MBIAS_LPF_EN);
65*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON0, RG_HDMITX_EN_SER_MASK);
66*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON0, RG_HDMITX_EN_PRED_MASK);
67*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON0, RG_HDMITX_EN_DRV_MASK);
68b28be59aSCK Hu 	usleep_range(80, 100);
69b28be59aSCK Hu 	return 0;
70b28be59aSCK Hu }
71b28be59aSCK Hu 
mtk_hdmi_pll_unprepare(struct clk_hw * hw)72b28be59aSCK Hu static void mtk_hdmi_pll_unprepare(struct clk_hw *hw)
73b28be59aSCK Hu {
74b28be59aSCK Hu 	struct mtk_hdmi_phy *hdmi_phy = to_mtk_hdmi_phy(hw);
75*cff81a61SChunfeng Yun 	void __iomem *base = hdmi_phy->regs;
76b28be59aSCK Hu 
77*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON0, RG_HDMITX_EN_DRV_MASK);
78*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON0, RG_HDMITX_EN_PRED_MASK);
79*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON0, RG_HDMITX_EN_SER_MASK);
80*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON2, RG_HDMITX_MBIAS_LPF_EN);
81b28be59aSCK Hu 	usleep_range(80, 100);
82*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON0, RG_HDMITX_EN_SLDO_MASK);
83*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON2, RG_HDMITX_EN_TX_CKLDO);
84*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON6, RG_HTPLL_EN);
85b28be59aSCK Hu 	usleep_range(80, 100);
86*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON2, RG_HDMITX_EN_MBIAS);
87*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON6, RG_HTPLL_POSDIV_MASK);
88*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON6, RG_HTPLL_RLH_EN);
89*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON7, RG_HTPLL_AUTOK_EN);
90b28be59aSCK Hu 	usleep_range(80, 100);
91b28be59aSCK Hu }
92b28be59aSCK Hu 
mtk_hdmi_pll_round_rate(struct clk_hw * hw,unsigned long rate,unsigned long * parent_rate)93b28be59aSCK Hu static long mtk_hdmi_pll_round_rate(struct clk_hw *hw, unsigned long rate,
94b28be59aSCK Hu 				    unsigned long *parent_rate)
95b28be59aSCK Hu {
96b28be59aSCK Hu 	return rate;
97b28be59aSCK Hu }
98b28be59aSCK Hu 
mtk_hdmi_pll_set_rate(struct clk_hw * hw,unsigned long rate,unsigned long parent_rate)99b28be59aSCK Hu static int mtk_hdmi_pll_set_rate(struct clk_hw *hw, unsigned long rate,
100b28be59aSCK Hu 				 unsigned long parent_rate)
101b28be59aSCK Hu {
102b28be59aSCK Hu 	struct mtk_hdmi_phy *hdmi_phy = to_mtk_hdmi_phy(hw);
103*cff81a61SChunfeng Yun 	void __iomem *base = hdmi_phy->regs;
104b28be59aSCK Hu 	u32 pos_div;
105b28be59aSCK Hu 
106b28be59aSCK Hu 	if (rate <= 64000000)
107b28be59aSCK Hu 		pos_div = 3;
108b28be59aSCK Hu 	else if (rate <= 128000000)
109b28be59aSCK Hu 		pos_div = 2;
110b28be59aSCK Hu 	else
111b28be59aSCK Hu 		pos_div = 1;
112b28be59aSCK Hu 
113*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON6, RG_HTPLL_PREDIV_MASK);
114*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON6, RG_HTPLL_POSDIV_MASK);
115*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON2, RG_HDMITX_EN_TX_POSDIV);
116*cff81a61SChunfeng Yun 	mtk_phy_update_field(base + HDMI_CON6, RG_HTPLL_IC_MASK, 0x1);
117*cff81a61SChunfeng Yun 	mtk_phy_update_field(base + HDMI_CON6, RG_HTPLL_IR_MASK, 0x1);
118*cff81a61SChunfeng Yun 	mtk_phy_update_field(base + HDMI_CON2, RG_HDMITX_TX_POSDIV_MASK, pos_div);
119*cff81a61SChunfeng Yun 	mtk_phy_update_field(base + HDMI_CON6, RG_HTPLL_FBKSEL_MASK, 1);
120*cff81a61SChunfeng Yun 	mtk_phy_update_field(base + HDMI_CON6, RG_HTPLL_FBKDIV_MASK, 19);
121*cff81a61SChunfeng Yun 	mtk_phy_update_field(base + HDMI_CON7, RG_HTPLL_DIVEN_MASK, 0x2);
122*cff81a61SChunfeng Yun 	mtk_phy_update_field(base + HDMI_CON6, RG_HTPLL_BP_MASK, 0xc);
123*cff81a61SChunfeng Yun 	mtk_phy_update_field(base + HDMI_CON6, RG_HTPLL_BC_MASK, 0x2);
124*cff81a61SChunfeng Yun 	mtk_phy_update_field(base + HDMI_CON6, RG_HTPLL_BR_MASK, 0x1);
125b28be59aSCK Hu 
126*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON1, RG_HDMITX_PRED_IMP);
127*cff81a61SChunfeng Yun 	mtk_phy_update_field(base + HDMI_CON1, RG_HDMITX_PRED_IBIAS_MASK, 0x3);
128*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON0, RG_HDMITX_EN_IMP_MASK);
129*cff81a61SChunfeng Yun 	mtk_phy_update_field(base + HDMI_CON1, RG_HDMITX_DRV_IMP_MASK, 0x28);
130*cff81a61SChunfeng Yun 	mtk_phy_update_field(base + HDMI_CON4, RG_HDMITX_RESERVE_MASK, 0x28);
131*cff81a61SChunfeng Yun 	mtk_phy_update_field(base + HDMI_CON0, RG_HDMITX_DRV_IBIAS_MASK, 0xa);
132b28be59aSCK Hu 	return 0;
133b28be59aSCK Hu }
134b28be59aSCK Hu 
mtk_hdmi_pll_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)135b28be59aSCK Hu static unsigned long mtk_hdmi_pll_recalc_rate(struct clk_hw *hw,
136b28be59aSCK Hu 					      unsigned long parent_rate)
137b28be59aSCK Hu {
138b28be59aSCK Hu 	struct mtk_hdmi_phy *hdmi_phy = to_mtk_hdmi_phy(hw);
139b28be59aSCK Hu 	unsigned long out_rate, val;
140a98d935eSChunfeng Yun 	u32 tmp;
141b28be59aSCK Hu 
142a98d935eSChunfeng Yun 	tmp = readl(hdmi_phy->regs + HDMI_CON6);
143a98d935eSChunfeng Yun 	val = FIELD_GET(RG_HTPLL_PREDIV_MASK, tmp);
144b28be59aSCK Hu 	switch (val) {
145b28be59aSCK Hu 	case 0x00:
146b28be59aSCK Hu 		out_rate = parent_rate;
147b28be59aSCK Hu 		break;
148b28be59aSCK Hu 	case 0x01:
149b28be59aSCK Hu 		out_rate = parent_rate / 2;
150b28be59aSCK Hu 		break;
151b28be59aSCK Hu 	default:
152b28be59aSCK Hu 		out_rate = parent_rate / 4;
153b28be59aSCK Hu 		break;
154b28be59aSCK Hu 	}
155b28be59aSCK Hu 
156a98d935eSChunfeng Yun 	val = FIELD_GET(RG_HTPLL_FBKDIV_MASK, tmp);
157b28be59aSCK Hu 	out_rate *= (val + 1) * 2;
158b28be59aSCK Hu 
159a98d935eSChunfeng Yun 	tmp = readl(hdmi_phy->regs + HDMI_CON2);
160a98d935eSChunfeng Yun 	val = FIELD_GET(RG_HDMITX_TX_POSDIV_MASK, tmp);
161a98d935eSChunfeng Yun 	out_rate >>= val;
162a98d935eSChunfeng Yun 
163a98d935eSChunfeng Yun 	if (tmp & RG_HDMITX_EN_TX_POSDIV)
164b28be59aSCK Hu 		out_rate /= 5;
165b28be59aSCK Hu 
166b28be59aSCK Hu 	return out_rate;
167b28be59aSCK Hu }
168b28be59aSCK Hu 
169b28be59aSCK Hu static const struct clk_ops mtk_hdmi_phy_pll_ops = {
170b28be59aSCK Hu 	.prepare = mtk_hdmi_pll_prepare,
171b28be59aSCK Hu 	.unprepare = mtk_hdmi_pll_unprepare,
172b28be59aSCK Hu 	.set_rate = mtk_hdmi_pll_set_rate,
173b28be59aSCK Hu 	.round_rate = mtk_hdmi_pll_round_rate,
174b28be59aSCK Hu 	.recalc_rate = mtk_hdmi_pll_recalc_rate,
175b28be59aSCK Hu };
176b28be59aSCK Hu 
mtk_hdmi_phy_enable_tmds(struct mtk_hdmi_phy * hdmi_phy)177b28be59aSCK Hu static void mtk_hdmi_phy_enable_tmds(struct mtk_hdmi_phy *hdmi_phy)
178b28be59aSCK Hu {
179*cff81a61SChunfeng Yun 	void __iomem *base = hdmi_phy->regs;
180*cff81a61SChunfeng Yun 
181*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON7, RG_HTPLL_AUTOK_EN);
182*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON6, RG_HTPLL_RLH_EN);
183*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON6, RG_HTPLL_POSDIV_MASK);
184*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON2, RG_HDMITX_EN_MBIAS);
185b28be59aSCK Hu 	usleep_range(80, 100);
186*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON6, RG_HTPLL_EN);
187*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON2, RG_HDMITX_EN_TX_CKLDO);
188*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON0, RG_HDMITX_EN_SLDO_MASK);
189b28be59aSCK Hu 	usleep_range(80, 100);
190*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON2, RG_HDMITX_MBIAS_LPF_EN);
191*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON0, RG_HDMITX_EN_SER_MASK);
192*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON0, RG_HDMITX_EN_PRED_MASK);
193*cff81a61SChunfeng Yun 	mtk_phy_set_bits(base + HDMI_CON0, RG_HDMITX_EN_DRV_MASK);
194b28be59aSCK Hu 	usleep_range(80, 100);
195b28be59aSCK Hu }
196b28be59aSCK Hu 
mtk_hdmi_phy_disable_tmds(struct mtk_hdmi_phy * hdmi_phy)197b28be59aSCK Hu static void mtk_hdmi_phy_disable_tmds(struct mtk_hdmi_phy *hdmi_phy)
198b28be59aSCK Hu {
199*cff81a61SChunfeng Yun 	void __iomem *base = hdmi_phy->regs;
200*cff81a61SChunfeng Yun 
201*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON0, RG_HDMITX_EN_DRV_MASK);
202*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON0, RG_HDMITX_EN_PRED_MASK);
203*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON0, RG_HDMITX_EN_SER_MASK);
204*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON2, RG_HDMITX_MBIAS_LPF_EN);
205b28be59aSCK Hu 	usleep_range(80, 100);
206*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON0, RG_HDMITX_EN_SLDO_MASK);
207*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON2, RG_HDMITX_EN_TX_CKLDO);
208*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON6, RG_HTPLL_EN);
209b28be59aSCK Hu 	usleep_range(80, 100);
210*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON2, RG_HDMITX_EN_MBIAS);
211*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON6, RG_HTPLL_POSDIV_MASK);
212*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON6, RG_HTPLL_RLH_EN);
213*cff81a61SChunfeng Yun 	mtk_phy_clear_bits(base + HDMI_CON7, RG_HTPLL_AUTOK_EN);
214b28be59aSCK Hu 	usleep_range(80, 100);
215b28be59aSCK Hu }
216b28be59aSCK Hu 
217b28be59aSCK Hu struct mtk_hdmi_phy_conf mtk_hdmi_phy_2701_conf = {
218b28be59aSCK Hu 	.flags = CLK_SET_RATE_GATE,
21909e872d5Schunhui dai 	.pll_default_off = true,
220b28be59aSCK Hu 	.hdmi_phy_clk_ops = &mtk_hdmi_phy_pll_ops,
221b28be59aSCK Hu 	.hdmi_phy_enable_tmds = mtk_hdmi_phy_enable_tmds,
222b28be59aSCK Hu 	.hdmi_phy_disable_tmds = mtk_hdmi_phy_disable_tmds,
223b28be59aSCK Hu };
224b28be59aSCK Hu 
225b28be59aSCK Hu MODULE_AUTHOR("Chunhui Dai <chunhui.dai@mediatek.com>");
226b28be59aSCK Hu MODULE_DESCRIPTION("MediaTek HDMI PHY Driver");
227b28be59aSCK Hu MODULE_LICENSE("GPL v2");
228