1 // SPDX-License-Identifier: GPL-2.0-only
3 * Copyright (c) 2016, The Linux Foundation. All rights reserved.
4 * Copyright (C) 2013 Red Hat
5 * Author: Rob Clark <robdclark@gmail.com>
8 #include <linux/clk-provider.h>
9 #include <linux/delay.h>
13 struct hdmi_pll_8960
{
14 struct platform_device
*pdev
;
21 #define hw_clk_to_pll(x) container_of(x, struct hdmi_pll_8960, clk_hw)
26 * To get the parent clock setup properly, we need to plug in hdmi pll
27 * configuration into common-clock-framework.
39 /* NOTE: keep sorted highest freq to lowest: */
40 static const struct pll_rate freqtbl
[] = {
42 { 0x08, REG_HDMI_8960_PHY_PLL_REFCLK_CFG
},
43 { 0x20, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG0
},
44 { 0xf9, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG1
},
45 { 0x02, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG0
},
46 { 0x03, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG1
},
47 { 0x3b, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG2
},
48 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG3
},
49 { 0x86, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG4
},
50 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG5
},
51 { 0x0d, REG_HDMI_8960_PHY_PLL_SDM_CFG0
},
52 { 0x4d, REG_HDMI_8960_PHY_PLL_SDM_CFG1
},
53 { 0x5e, REG_HDMI_8960_PHY_PLL_SDM_CFG2
},
54 { 0x42, REG_HDMI_8960_PHY_PLL_SDM_CFG3
},
55 { 0x00, REG_HDMI_8960_PHY_PLL_SDM_CFG4
},
58 /* 1080p60/1080p50 case */
60 { 0x02, REG_HDMI_8960_PHY_PLL_REFCLK_CFG
},
61 { 0x02, REG_HDMI_8960_PHY_PLL_CHRG_PUMP_CFG
},
62 { 0x01, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG0
},
63 { 0x33, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG1
},
64 { 0x2c, REG_HDMI_8960_PHY_PLL_IDAC_ADJ_CFG
},
65 { 0x06, REG_HDMI_8960_PHY_PLL_I_VI_KVCO_CFG
},
66 { 0x0a, REG_HDMI_8960_PHY_PLL_PWRDN_B
},
67 { 0x76, REG_HDMI_8960_PHY_PLL_SDM_CFG0
},
68 { 0x01, REG_HDMI_8960_PHY_PLL_SDM_CFG1
},
69 { 0x4c, REG_HDMI_8960_PHY_PLL_SDM_CFG2
},
70 { 0xc0, REG_HDMI_8960_PHY_PLL_SDM_CFG3
},
71 { 0x00, REG_HDMI_8960_PHY_PLL_SDM_CFG4
},
72 { 0x9a, REG_HDMI_8960_PHY_PLL_SSC_CFG0
},
73 { 0x00, REG_HDMI_8960_PHY_PLL_SSC_CFG1
},
74 { 0x00, REG_HDMI_8960_PHY_PLL_SSC_CFG2
},
75 { 0x00, REG_HDMI_8960_PHY_PLL_SSC_CFG3
},
76 { 0x10, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG0
},
77 { 0x1a, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG1
},
78 { 0x0d, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG2
},
79 { 0xe6, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG0
},
80 { 0x02, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG1
},
81 { 0x3b, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG2
},
82 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG3
},
83 { 0x86, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG4
},
84 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG5
},
85 { 0x33, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG6
},
86 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG7
},
90 { 0x08, REG_HDMI_8960_PHY_PLL_REFCLK_CFG
},
91 { 0x21, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG0
},
92 { 0xf9, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG1
},
93 { 0x1c, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG0
},
94 { 0x02, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG1
},
95 { 0x3b, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG2
},
96 { 0x86, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG4
},
97 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG5
},
98 { 0x49, REG_HDMI_8960_PHY_PLL_SDM_CFG0
},
99 { 0x49, REG_HDMI_8960_PHY_PLL_SDM_CFG1
},
100 { 0x00, REG_HDMI_8960_PHY_PLL_SDM_CFG2
},
101 { 0x00, REG_HDMI_8960_PHY_PLL_SDM_CFG3
},
102 { 0x00, REG_HDMI_8960_PHY_PLL_SDM_CFG4
},
105 /* 720p60/720p50/1080i60/1080i50/1080p24/1080p30/1080p25 */
107 { 0x0a, REG_HDMI_8960_PHY_PLL_PWRDN_B
},
108 { 0x12, REG_HDMI_8960_PHY_PLL_REFCLK_CFG
},
109 { 0x01, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG0
},
110 { 0x33, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG1
},
111 { 0x76, REG_HDMI_8960_PHY_PLL_SDM_CFG0
},
112 { 0xe6, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG0
},
113 { 0x02, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG1
},
114 { 0x3b, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG2
},
118 { 0x18, REG_HDMI_8960_PHY_PLL_REFCLK_CFG
},
119 { 0x20, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG0
},
120 { 0xf9, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG1
},
121 { 0xe5, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG0
},
122 { 0x02, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG1
},
123 { 0x3b, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG2
},
124 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG3
},
125 { 0x86, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG4
},
126 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG5
},
127 { 0x0c, REG_HDMI_8960_PHY_PLL_SDM_CFG0
},
128 { 0x4c, REG_HDMI_8960_PHY_PLL_SDM_CFG1
},
129 { 0x7d, REG_HDMI_8960_PHY_PLL_SDM_CFG2
},
130 { 0xbc, REG_HDMI_8960_PHY_PLL_SDM_CFG3
},
131 { 0x00, REG_HDMI_8960_PHY_PLL_SDM_CFG4
},
135 { 0x18, REG_HDMI_8960_PHY_PLL_REFCLK_CFG
},
136 { 0x20, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG0
},
137 { 0xf9, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG1
},
138 { 0x8a, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG0
},
139 { 0x02, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG1
},
140 { 0x3b, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG2
},
141 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG3
},
142 { 0x86, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG4
},
143 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG5
},
144 { 0x0b, REG_HDMI_8960_PHY_PLL_SDM_CFG0
},
145 { 0x4b, REG_HDMI_8960_PHY_PLL_SDM_CFG1
},
146 { 0x7b, REG_HDMI_8960_PHY_PLL_SDM_CFG2
},
147 { 0x09, REG_HDMI_8960_PHY_PLL_SDM_CFG3
},
148 { 0x00, REG_HDMI_8960_PHY_PLL_SDM_CFG4
},
153 { 0x0a, REG_HDMI_8960_PHY_PLL_PWRDN_B
},
154 { 0x38, REG_HDMI_8960_PHY_PLL_REFCLK_CFG
},
155 { 0x02, REG_HDMI_8960_PHY_PLL_CHRG_PUMP_CFG
},
156 { 0x20, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG0
},
157 { 0xff, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG1
},
158 { 0x00, REG_HDMI_8960_PHY_PLL_SDM_CFG0
},
159 { 0x4e, REG_HDMI_8960_PHY_PLL_SDM_CFG1
},
160 { 0xd7, REG_HDMI_8960_PHY_PLL_SDM_CFG2
},
161 { 0x03, REG_HDMI_8960_PHY_PLL_SDM_CFG3
},
162 { 0x00, REG_HDMI_8960_PHY_PLL_SDM_CFG4
},
163 { 0x2a, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG0
},
164 { 0x03, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG1
},
165 { 0x3b, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG2
},
166 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG3
},
167 { 0x86, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG4
},
168 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG5
},
169 { 0x33, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG6
},
170 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG7
},
175 { 0x32, REG_HDMI_8960_PHY_PLL_REFCLK_CFG
},
176 { 0x02, REG_HDMI_8960_PHY_PLL_CHRG_PUMP_CFG
},
177 { 0x01, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG0
},
178 { 0x33, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG1
},
179 { 0x2c, REG_HDMI_8960_PHY_PLL_IDAC_ADJ_CFG
},
180 { 0x06, REG_HDMI_8960_PHY_PLL_I_VI_KVCO_CFG
},
181 { 0x0a, REG_HDMI_8960_PHY_PLL_PWRDN_B
},
182 { 0x7b, REG_HDMI_8960_PHY_PLL_SDM_CFG0
},
183 { 0x01, REG_HDMI_8960_PHY_PLL_SDM_CFG1
},
184 { 0x4c, REG_HDMI_8960_PHY_PLL_SDM_CFG2
},
185 { 0xc0, REG_HDMI_8960_PHY_PLL_SDM_CFG3
},
186 { 0x00, REG_HDMI_8960_PHY_PLL_SDM_CFG4
},
187 { 0x9a, REG_HDMI_8960_PHY_PLL_SSC_CFG0
},
188 { 0x00, REG_HDMI_8960_PHY_PLL_SSC_CFG1
},
189 { 0x00, REG_HDMI_8960_PHY_PLL_SSC_CFG2
},
190 { 0x00, REG_HDMI_8960_PHY_PLL_SSC_CFG3
},
191 { 0x10, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG0
},
192 { 0x1a, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG1
},
193 { 0x0d, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG2
},
194 { 0x2a, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG0
},
195 { 0x03, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG1
},
196 { 0x3b, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG2
},
197 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG3
},
198 { 0x86, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG4
},
199 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG5
},
200 { 0x33, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG6
},
201 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG7
},
206 { 0x32, REG_HDMI_8960_PHY_PLL_REFCLK_CFG
},
207 { 0x02, REG_HDMI_8960_PHY_PLL_CHRG_PUMP_CFG
},
208 { 0x01, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG0
},
209 { 0x33, REG_HDMI_8960_PHY_PLL_LOOP_FLT_CFG1
},
210 { 0x2c, REG_HDMI_8960_PHY_PLL_IDAC_ADJ_CFG
},
211 { 0x06, REG_HDMI_8960_PHY_PLL_I_VI_KVCO_CFG
},
212 { 0x0a, REG_HDMI_8960_PHY_PLL_PWRDN_B
},
213 { 0x77, REG_HDMI_8960_PHY_PLL_SDM_CFG0
},
214 { 0x4c, REG_HDMI_8960_PHY_PLL_SDM_CFG1
},
215 { 0x00, REG_HDMI_8960_PHY_PLL_SDM_CFG2
},
216 { 0xc0, REG_HDMI_8960_PHY_PLL_SDM_CFG3
},
217 { 0x00, REG_HDMI_8960_PHY_PLL_SDM_CFG4
},
218 { 0x9a, REG_HDMI_8960_PHY_PLL_SSC_CFG0
},
219 { 0x00, REG_HDMI_8960_PHY_PLL_SSC_CFG1
},
220 { 0x00, REG_HDMI_8960_PHY_PLL_SSC_CFG2
},
221 { 0x20, REG_HDMI_8960_PHY_PLL_SSC_CFG3
},
222 { 0x10, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG0
},
223 { 0x1a, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG1
},
224 { 0x0d, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG2
},
225 { 0xf4, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG0
},
226 { 0x02, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG1
},
227 { 0x3b, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG2
},
228 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG3
},
229 { 0x86, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG4
},
230 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG5
},
231 { 0x33, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG6
},
232 { 0x00, REG_HDMI_8960_PHY_PLL_VCOCAL_CFG7
},
237 static inline void pll_write(struct hdmi_pll_8960
*pll
, u32 reg
, u32 data
)
239 msm_writel(data
, pll
->mmio
+ reg
);
242 static inline u32
pll_read(struct hdmi_pll_8960
*pll
, u32 reg
)
244 return msm_readl(pll
->mmio
+ reg
);
247 static inline struct hdmi_phy
*pll_get_phy(struct hdmi_pll_8960
*pll
)
249 return platform_get_drvdata(pll
->pdev
);
252 static int hdmi_pll_enable(struct clk_hw
*hw
)
254 struct hdmi_pll_8960
*pll
= hw_clk_to_pll(hw
);
255 struct hdmi_phy
*phy
= pll_get_phy(pll
);
256 int timeout_count
, pll_lock_retry
= 10;
261 /* Assert PLL S/W reset */
262 pll_write(pll
, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG2
, 0x8d);
263 pll_write(pll
, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG0
, 0x10);
264 pll_write(pll
, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG1
, 0x1a);
266 /* Wait for a short time before de-asserting
267 * to allow the hardware to complete its job.
268 * This much of delay should be fine for hardware
269 * to assert and de-assert.
273 /* De-assert PLL S/W reset */
274 pll_write(pll
, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG2
, 0x0d);
276 val
= hdmi_phy_read(phy
, REG_HDMI_8960_PHY_REG12
);
277 val
|= HDMI_8960_PHY_REG12_SW_RESET
;
278 /* Assert PHY S/W reset */
279 hdmi_phy_write(phy
, REG_HDMI_8960_PHY_REG12
, val
);
280 val
&= ~HDMI_8960_PHY_REG12_SW_RESET
;
282 * Wait for a short time before de-asserting to allow the hardware to
283 * complete its job. This much of delay should be fine for hardware to
284 * assert and de-assert.
287 /* De-assert PHY S/W reset */
288 hdmi_phy_write(phy
, REG_HDMI_8960_PHY_REG12
, val
);
289 hdmi_phy_write(phy
, REG_HDMI_8960_PHY_REG2
, 0x3f);
291 val
= hdmi_phy_read(phy
, REG_HDMI_8960_PHY_REG12
);
292 val
|= HDMI_8960_PHY_REG12_PWRDN_B
;
293 hdmi_phy_write(phy
, REG_HDMI_8960_PHY_REG12
, val
);
294 /* Wait 10 us for enabling global power for PHY */
298 val
= pll_read(pll
, REG_HDMI_8960_PHY_PLL_PWRDN_B
);
299 val
|= HDMI_8960_PHY_PLL_PWRDN_B_PLL_PWRDN_B
;
300 val
&= ~HDMI_8960_PHY_PLL_PWRDN_B_PD_PLL
;
301 pll_write(pll
, REG_HDMI_8960_PHY_PLL_PWRDN_B
, val
);
302 hdmi_phy_write(phy
, REG_HDMI_8960_PHY_REG2
, 0x80);
304 timeout_count
= 1000;
305 while (--pll_lock_retry
> 0) {
306 /* are we there yet? */
307 val
= pll_read(pll
, REG_HDMI_8960_PHY_PLL_STATUS0
);
308 if (val
& HDMI_8960_PHY_PLL_STATUS0_PLL_LOCK
)
313 if (--timeout_count
> 0)
317 * PLL has still not locked.
318 * Do a software reset and try again
319 * Assert PLL S/W reset first
321 pll_write(pll
, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG2
, 0x8d);
323 pll_write(pll
, REG_HDMI_8960_PHY_PLL_LOCKDET_CFG2
, 0x0d);
326 * Wait for a short duration for the PLL calibration
327 * before checking if the PLL gets locked
331 timeout_count
= 1000;
337 static void hdmi_pll_disable(struct clk_hw
*hw
)
339 struct hdmi_pll_8960
*pll
= hw_clk_to_pll(hw
);
340 struct hdmi_phy
*phy
= pll_get_phy(pll
);
345 val
= hdmi_phy_read(phy
, REG_HDMI_8960_PHY_REG12
);
346 val
&= ~HDMI_8960_PHY_REG12_PWRDN_B
;
347 hdmi_phy_write(phy
, REG_HDMI_8960_PHY_REG12
, val
);
349 val
= pll_read(pll
, REG_HDMI_8960_PHY_PLL_PWRDN_B
);
350 val
|= HDMI_8960_PHY_REG12_SW_RESET
;
351 val
&= ~HDMI_8960_PHY_REG12_PWRDN_B
;
352 pll_write(pll
, REG_HDMI_8960_PHY_PLL_PWRDN_B
, val
);
353 /* Make sure HDMI PHY/PLL are powered down */
357 static const struct pll_rate
*find_rate(unsigned long rate
)
361 for (i
= 1; i
< ARRAY_SIZE(freqtbl
); i
++)
362 if (rate
> freqtbl
[i
].rate
)
363 return &freqtbl
[i
- 1];
365 return &freqtbl
[i
- 1];
368 static unsigned long hdmi_pll_recalc_rate(struct clk_hw
*hw
,
369 unsigned long parent_rate
)
371 struct hdmi_pll_8960
*pll
= hw_clk_to_pll(hw
);
376 static long hdmi_pll_round_rate(struct clk_hw
*hw
, unsigned long rate
,
377 unsigned long *parent_rate
)
379 const struct pll_rate
*pll_rate
= find_rate(rate
);
381 return pll_rate
->rate
;
384 static int hdmi_pll_set_rate(struct clk_hw
*hw
, unsigned long rate
,
385 unsigned long parent_rate
)
387 struct hdmi_pll_8960
*pll
= hw_clk_to_pll(hw
);
388 const struct pll_rate
*pll_rate
= find_rate(rate
);
391 DBG("rate=%lu", rate
);
393 for (i
= 0; i
< pll_rate
->num_reg
; i
++)
394 pll_write(pll
, pll_rate
->conf
[i
].reg
, pll_rate
->conf
[i
].val
);
401 static const struct clk_ops hdmi_pll_ops
= {
402 .enable
= hdmi_pll_enable
,
403 .disable
= hdmi_pll_disable
,
404 .recalc_rate
= hdmi_pll_recalc_rate
,
405 .round_rate
= hdmi_pll_round_rate
,
406 .set_rate
= hdmi_pll_set_rate
,
409 static const char * const hdmi_pll_parents
[] = {
413 static struct clk_init_data pll_init
= {
415 .ops
= &hdmi_pll_ops
,
416 .parent_names
= hdmi_pll_parents
,
417 .num_parents
= ARRAY_SIZE(hdmi_pll_parents
),
418 .flags
= CLK_IGNORE_UNUSED
,
421 int msm_hdmi_pll_8960_init(struct platform_device
*pdev
)
423 struct device
*dev
= &pdev
->dev
;
424 struct hdmi_pll_8960
*pll
;
429 for (i
= 0; i
< (ARRAY_SIZE(freqtbl
) - 1); i
++)
430 if (WARN_ON(freqtbl
[i
].rate
< freqtbl
[i
+ 1].rate
))
433 pll
= devm_kzalloc(dev
, sizeof(*pll
), GFP_KERNEL
);
437 pll
->mmio
= msm_ioremap(pdev
, "hdmi_pll", "HDMI_PLL");
438 if (IS_ERR(pll
->mmio
)) {
439 DRM_DEV_ERROR(dev
, "failed to map pll base\n");
444 pll
->clk_hw
.init
= &pll_init
;
446 clk
= devm_clk_register(dev
, &pll
->clk_hw
);
448 DRM_DEV_ERROR(dev
, "failed to register pll clock\n");