1 /*
2 * Copyright (c) 2013 Samsung Electronics Co., Ltd.
3 * Copyright (c) 2013 Linaro Ltd.
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2 as
7 * published by the Free Software Foundation.
8 *
9 * This file contains the utility functions to register the pll clocks.
10 */
11
12 #include <linux/errno.h>
13 #include <linux/hrtimer.h>
14 #include <linux/delay.h>
15 #include <linux/slab.h>
16 #include <linux/clkdev.h>
17 #include "clk.h"
18 #include "clk-pll.h"
19
20 #define PLL_TIMEOUT_MS 10
21
22 struct samsung_clk_pll {
23 struct clk_hw hw;
24 void __iomem *lock_reg;
25 void __iomem *con_reg;
26 enum samsung_pll_type type;
27 unsigned int rate_count;
28 const struct samsung_pll_rate_table *rate_table;
29 };
30
31 #define to_clk_pll(_hw) container_of(_hw, struct samsung_clk_pll, hw)
32
samsung_get_pll_settings(struct samsung_clk_pll * pll,unsigned long rate)33 static const struct samsung_pll_rate_table *samsung_get_pll_settings(
34 struct samsung_clk_pll *pll, unsigned long rate)
35 {
36 const struct samsung_pll_rate_table *rate_table = pll->rate_table;
37 int i;
38
39 for (i = 0; i < pll->rate_count; i++) {
40 if (rate == rate_table[i].rate)
41 return &rate_table[i];
42 }
43
44 return NULL;
45 }
46
samsung_pll_round_rate(struct clk_hw * hw,unsigned long drate,unsigned long * prate)47 static long samsung_pll_round_rate(struct clk_hw *hw,
48 unsigned long drate, unsigned long *prate)
49 {
50 struct samsung_clk_pll *pll = to_clk_pll(hw);
51 const struct samsung_pll_rate_table *rate_table = pll->rate_table;
52 int i;
53
54 /* Assumming rate_table is in descending order */
55 for (i = 0; i < pll->rate_count; i++) {
56 if (drate >= rate_table[i].rate)
57 return rate_table[i].rate;
58 }
59
60 /* return minimum supported value */
61 return rate_table[i - 1].rate;
62 }
63
64 /*
65 * PLL2126 Clock Type
66 */
67
68 #define PLL2126_MDIV_MASK (0xff)
69 #define PLL2126_PDIV_MASK (0x3f)
70 #define PLL2126_SDIV_MASK (0x3)
71 #define PLL2126_MDIV_SHIFT (16)
72 #define PLL2126_PDIV_SHIFT (8)
73 #define PLL2126_SDIV_SHIFT (0)
74
samsung_pll2126_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)75 static unsigned long samsung_pll2126_recalc_rate(struct clk_hw *hw,
76 unsigned long parent_rate)
77 {
78 struct samsung_clk_pll *pll = to_clk_pll(hw);
79 u32 pll_con, mdiv, pdiv, sdiv;
80 u64 fvco = parent_rate;
81
82 pll_con = __raw_readl(pll->con_reg);
83 mdiv = (pll_con >> PLL2126_MDIV_SHIFT) & PLL2126_MDIV_MASK;
84 pdiv = (pll_con >> PLL2126_PDIV_SHIFT) & PLL2126_PDIV_MASK;
85 sdiv = (pll_con >> PLL2126_SDIV_SHIFT) & PLL2126_SDIV_MASK;
86
87 fvco *= (mdiv + 8);
88 do_div(fvco, (pdiv + 2) << sdiv);
89
90 return (unsigned long)fvco;
91 }
92
93 static const struct clk_ops samsung_pll2126_clk_ops = {
94 .recalc_rate = samsung_pll2126_recalc_rate,
95 };
96
97 /*
98 * PLL3000 Clock Type
99 */
100
101 #define PLL3000_MDIV_MASK (0xff)
102 #define PLL3000_PDIV_MASK (0x3)
103 #define PLL3000_SDIV_MASK (0x3)
104 #define PLL3000_MDIV_SHIFT (16)
105 #define PLL3000_PDIV_SHIFT (8)
106 #define PLL3000_SDIV_SHIFT (0)
107
samsung_pll3000_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)108 static unsigned long samsung_pll3000_recalc_rate(struct clk_hw *hw,
109 unsigned long parent_rate)
110 {
111 struct samsung_clk_pll *pll = to_clk_pll(hw);
112 u32 pll_con, mdiv, pdiv, sdiv;
113 u64 fvco = parent_rate;
114
115 pll_con = __raw_readl(pll->con_reg);
116 mdiv = (pll_con >> PLL3000_MDIV_SHIFT) & PLL3000_MDIV_MASK;
117 pdiv = (pll_con >> PLL3000_PDIV_SHIFT) & PLL3000_PDIV_MASK;
118 sdiv = (pll_con >> PLL3000_SDIV_SHIFT) & PLL3000_SDIV_MASK;
119
120 fvco *= (2 * (mdiv + 8));
121 do_div(fvco, pdiv << sdiv);
122
123 return (unsigned long)fvco;
124 }
125
126 static const struct clk_ops samsung_pll3000_clk_ops = {
127 .recalc_rate = samsung_pll3000_recalc_rate,
128 };
129
130 /*
131 * PLL35xx Clock Type
132 */
133 /* Maximum lock time can be 270 * PDIV cycles */
134 #define PLL35XX_LOCK_FACTOR (270)
135
136 #define PLL35XX_MDIV_MASK (0x3FF)
137 #define PLL35XX_PDIV_MASK (0x3F)
138 #define PLL35XX_SDIV_MASK (0x7)
139 #define PLL35XX_LOCK_STAT_MASK (0x1)
140 #define PLL35XX_MDIV_SHIFT (16)
141 #define PLL35XX_PDIV_SHIFT (8)
142 #define PLL35XX_SDIV_SHIFT (0)
143 #define PLL35XX_LOCK_STAT_SHIFT (29)
144
samsung_pll35xx_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)145 static unsigned long samsung_pll35xx_recalc_rate(struct clk_hw *hw,
146 unsigned long parent_rate)
147 {
148 struct samsung_clk_pll *pll = to_clk_pll(hw);
149 u32 mdiv, pdiv, sdiv, pll_con;
150 u64 fvco = parent_rate;
151
152 pll_con = __raw_readl(pll->con_reg);
153 mdiv = (pll_con >> PLL35XX_MDIV_SHIFT) & PLL35XX_MDIV_MASK;
154 pdiv = (pll_con >> PLL35XX_PDIV_SHIFT) & PLL35XX_PDIV_MASK;
155 sdiv = (pll_con >> PLL35XX_SDIV_SHIFT) & PLL35XX_SDIV_MASK;
156
157 fvco *= mdiv;
158 do_div(fvco, (pdiv << sdiv));
159
160 return (unsigned long)fvco;
161 }
162
samsung_pll35xx_mp_change(const struct samsung_pll_rate_table * rate,u32 pll_con)163 static inline bool samsung_pll35xx_mp_change(
164 const struct samsung_pll_rate_table *rate, u32 pll_con)
165 {
166 u32 old_mdiv, old_pdiv;
167
168 old_mdiv = (pll_con >> PLL35XX_MDIV_SHIFT) & PLL35XX_MDIV_MASK;
169 old_pdiv = (pll_con >> PLL35XX_PDIV_SHIFT) & PLL35XX_PDIV_MASK;
170
171 return (rate->mdiv != old_mdiv || rate->pdiv != old_pdiv);
172 }
173
samsung_pll35xx_set_rate(struct clk_hw * hw,unsigned long drate,unsigned long prate)174 static int samsung_pll35xx_set_rate(struct clk_hw *hw, unsigned long drate,
175 unsigned long prate)
176 {
177 struct samsung_clk_pll *pll = to_clk_pll(hw);
178 const struct samsung_pll_rate_table *rate;
179 u32 tmp;
180
181 /* Get required rate settings from table */
182 rate = samsung_get_pll_settings(pll, drate);
183 if (!rate) {
184 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
185 drate, clk_hw_get_name(hw));
186 return -EINVAL;
187 }
188
189 tmp = __raw_readl(pll->con_reg);
190
191 if (!(samsung_pll35xx_mp_change(rate, tmp))) {
192 /* If only s change, change just s value only*/
193 tmp &= ~(PLL35XX_SDIV_MASK << PLL35XX_SDIV_SHIFT);
194 tmp |= rate->sdiv << PLL35XX_SDIV_SHIFT;
195 __raw_writel(tmp, pll->con_reg);
196
197 return 0;
198 }
199
200 /* Set PLL lock time. */
201 __raw_writel(rate->pdiv * PLL35XX_LOCK_FACTOR,
202 pll->lock_reg);
203
204 /* Change PLL PMS values */
205 tmp &= ~((PLL35XX_MDIV_MASK << PLL35XX_MDIV_SHIFT) |
206 (PLL35XX_PDIV_MASK << PLL35XX_PDIV_SHIFT) |
207 (PLL35XX_SDIV_MASK << PLL35XX_SDIV_SHIFT));
208 tmp |= (rate->mdiv << PLL35XX_MDIV_SHIFT) |
209 (rate->pdiv << PLL35XX_PDIV_SHIFT) |
210 (rate->sdiv << PLL35XX_SDIV_SHIFT);
211 __raw_writel(tmp, pll->con_reg);
212
213 /* wait_lock_time */
214 do {
215 cpu_relax();
216 tmp = __raw_readl(pll->con_reg);
217 } while (!(tmp & (PLL35XX_LOCK_STAT_MASK
218 << PLL35XX_LOCK_STAT_SHIFT)));
219 return 0;
220 }
221
222 static const struct clk_ops samsung_pll35xx_clk_ops = {
223 .recalc_rate = samsung_pll35xx_recalc_rate,
224 .round_rate = samsung_pll_round_rate,
225 .set_rate = samsung_pll35xx_set_rate,
226 };
227
228 static const struct clk_ops samsung_pll35xx_clk_min_ops = {
229 .recalc_rate = samsung_pll35xx_recalc_rate,
230 };
231
232 /*
233 * PLL36xx Clock Type
234 */
235 /* Maximum lock time can be 3000 * PDIV cycles */
236 #define PLL36XX_LOCK_FACTOR (3000)
237
238 #define PLL36XX_KDIV_MASK (0xFFFF)
239 #define PLL36XX_MDIV_MASK (0x1FF)
240 #define PLL36XX_PDIV_MASK (0x3F)
241 #define PLL36XX_SDIV_MASK (0x7)
242 #define PLL36XX_MDIV_SHIFT (16)
243 #define PLL36XX_PDIV_SHIFT (8)
244 #define PLL36XX_SDIV_SHIFT (0)
245 #define PLL36XX_KDIV_SHIFT (0)
246 #define PLL36XX_LOCK_STAT_SHIFT (29)
247
samsung_pll36xx_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)248 static unsigned long samsung_pll36xx_recalc_rate(struct clk_hw *hw,
249 unsigned long parent_rate)
250 {
251 struct samsung_clk_pll *pll = to_clk_pll(hw);
252 u32 mdiv, pdiv, sdiv, pll_con0, pll_con1;
253 s16 kdiv;
254 u64 fvco = parent_rate;
255
256 pll_con0 = __raw_readl(pll->con_reg);
257 pll_con1 = __raw_readl(pll->con_reg + 4);
258 mdiv = (pll_con0 >> PLL36XX_MDIV_SHIFT) & PLL36XX_MDIV_MASK;
259 pdiv = (pll_con0 >> PLL36XX_PDIV_SHIFT) & PLL36XX_PDIV_MASK;
260 sdiv = (pll_con0 >> PLL36XX_SDIV_SHIFT) & PLL36XX_SDIV_MASK;
261 kdiv = (s16)(pll_con1 & PLL36XX_KDIV_MASK);
262
263 fvco *= (mdiv << 16) + kdiv;
264 do_div(fvco, (pdiv << sdiv));
265 fvco >>= 16;
266
267 return (unsigned long)fvco;
268 }
269
samsung_pll36xx_mpk_change(const struct samsung_pll_rate_table * rate,u32 pll_con0,u32 pll_con1)270 static inline bool samsung_pll36xx_mpk_change(
271 const struct samsung_pll_rate_table *rate, u32 pll_con0, u32 pll_con1)
272 {
273 u32 old_mdiv, old_pdiv, old_kdiv;
274
275 old_mdiv = (pll_con0 >> PLL36XX_MDIV_SHIFT) & PLL36XX_MDIV_MASK;
276 old_pdiv = (pll_con0 >> PLL36XX_PDIV_SHIFT) & PLL36XX_PDIV_MASK;
277 old_kdiv = (pll_con1 >> PLL36XX_KDIV_SHIFT) & PLL36XX_KDIV_MASK;
278
279 return (rate->mdiv != old_mdiv || rate->pdiv != old_pdiv ||
280 rate->kdiv != old_kdiv);
281 }
282
samsung_pll36xx_set_rate(struct clk_hw * hw,unsigned long drate,unsigned long parent_rate)283 static int samsung_pll36xx_set_rate(struct clk_hw *hw, unsigned long drate,
284 unsigned long parent_rate)
285 {
286 struct samsung_clk_pll *pll = to_clk_pll(hw);
287 u32 tmp, pll_con0, pll_con1;
288 const struct samsung_pll_rate_table *rate;
289
290 rate = samsung_get_pll_settings(pll, drate);
291 if (!rate) {
292 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
293 drate, clk_hw_get_name(hw));
294 return -EINVAL;
295 }
296
297 pll_con0 = __raw_readl(pll->con_reg);
298 pll_con1 = __raw_readl(pll->con_reg + 4);
299
300 if (!(samsung_pll36xx_mpk_change(rate, pll_con0, pll_con1))) {
301 /* If only s change, change just s value only*/
302 pll_con0 &= ~(PLL36XX_SDIV_MASK << PLL36XX_SDIV_SHIFT);
303 pll_con0 |= (rate->sdiv << PLL36XX_SDIV_SHIFT);
304 __raw_writel(pll_con0, pll->con_reg);
305
306 return 0;
307 }
308
309 /* Set PLL lock time. */
310 __raw_writel(rate->pdiv * PLL36XX_LOCK_FACTOR, pll->lock_reg);
311
312 /* Change PLL PMS values */
313 pll_con0 &= ~((PLL36XX_MDIV_MASK << PLL36XX_MDIV_SHIFT) |
314 (PLL36XX_PDIV_MASK << PLL36XX_PDIV_SHIFT) |
315 (PLL36XX_SDIV_MASK << PLL36XX_SDIV_SHIFT));
316 pll_con0 |= (rate->mdiv << PLL36XX_MDIV_SHIFT) |
317 (rate->pdiv << PLL36XX_PDIV_SHIFT) |
318 (rate->sdiv << PLL36XX_SDIV_SHIFT);
319 __raw_writel(pll_con0, pll->con_reg);
320
321 pll_con1 &= ~(PLL36XX_KDIV_MASK << PLL36XX_KDIV_SHIFT);
322 pll_con1 |= rate->kdiv << PLL36XX_KDIV_SHIFT;
323 __raw_writel(pll_con1, pll->con_reg + 4);
324
325 /* wait_lock_time */
326 do {
327 cpu_relax();
328 tmp = __raw_readl(pll->con_reg);
329 } while (!(tmp & (1 << PLL36XX_LOCK_STAT_SHIFT)));
330
331 return 0;
332 }
333
334 static const struct clk_ops samsung_pll36xx_clk_ops = {
335 .recalc_rate = samsung_pll36xx_recalc_rate,
336 .set_rate = samsung_pll36xx_set_rate,
337 .round_rate = samsung_pll_round_rate,
338 };
339
340 static const struct clk_ops samsung_pll36xx_clk_min_ops = {
341 .recalc_rate = samsung_pll36xx_recalc_rate,
342 };
343
344 /*
345 * PLL45xx Clock Type
346 */
347 #define PLL4502_LOCK_FACTOR 400
348 #define PLL4508_LOCK_FACTOR 240
349
350 #define PLL45XX_MDIV_MASK (0x3FF)
351 #define PLL45XX_PDIV_MASK (0x3F)
352 #define PLL45XX_SDIV_MASK (0x7)
353 #define PLL45XX_AFC_MASK (0x1F)
354 #define PLL45XX_MDIV_SHIFT (16)
355 #define PLL45XX_PDIV_SHIFT (8)
356 #define PLL45XX_SDIV_SHIFT (0)
357 #define PLL45XX_AFC_SHIFT (0)
358
359 #define PLL45XX_ENABLE BIT(31)
360 #define PLL45XX_LOCKED BIT(29)
361
samsung_pll45xx_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)362 static unsigned long samsung_pll45xx_recalc_rate(struct clk_hw *hw,
363 unsigned long parent_rate)
364 {
365 struct samsung_clk_pll *pll = to_clk_pll(hw);
366 u32 mdiv, pdiv, sdiv, pll_con;
367 u64 fvco = parent_rate;
368
369 pll_con = __raw_readl(pll->con_reg);
370 mdiv = (pll_con >> PLL45XX_MDIV_SHIFT) & PLL45XX_MDIV_MASK;
371 pdiv = (pll_con >> PLL45XX_PDIV_SHIFT) & PLL45XX_PDIV_MASK;
372 sdiv = (pll_con >> PLL45XX_SDIV_SHIFT) & PLL45XX_SDIV_MASK;
373
374 if (pll->type == pll_4508)
375 sdiv = sdiv - 1;
376
377 fvco *= mdiv;
378 do_div(fvco, (pdiv << sdiv));
379
380 return (unsigned long)fvco;
381 }
382
samsung_pll45xx_mp_change(u32 pll_con0,u32 pll_con1,const struct samsung_pll_rate_table * rate)383 static bool samsung_pll45xx_mp_change(u32 pll_con0, u32 pll_con1,
384 const struct samsung_pll_rate_table *rate)
385 {
386 u32 old_mdiv, old_pdiv, old_afc;
387
388 old_mdiv = (pll_con0 >> PLL45XX_MDIV_SHIFT) & PLL45XX_MDIV_MASK;
389 old_pdiv = (pll_con0 >> PLL45XX_PDIV_SHIFT) & PLL45XX_PDIV_MASK;
390 old_afc = (pll_con1 >> PLL45XX_AFC_SHIFT) & PLL45XX_AFC_MASK;
391
392 return (old_mdiv != rate->mdiv || old_pdiv != rate->pdiv
393 || old_afc != rate->afc);
394 }
395
samsung_pll45xx_set_rate(struct clk_hw * hw,unsigned long drate,unsigned long prate)396 static int samsung_pll45xx_set_rate(struct clk_hw *hw, unsigned long drate,
397 unsigned long prate)
398 {
399 struct samsung_clk_pll *pll = to_clk_pll(hw);
400 const struct samsung_pll_rate_table *rate;
401 u32 con0, con1;
402 ktime_t start;
403
404 /* Get required rate settings from table */
405 rate = samsung_get_pll_settings(pll, drate);
406 if (!rate) {
407 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
408 drate, clk_hw_get_name(hw));
409 return -EINVAL;
410 }
411
412 con0 = __raw_readl(pll->con_reg);
413 con1 = __raw_readl(pll->con_reg + 0x4);
414
415 if (!(samsung_pll45xx_mp_change(con0, con1, rate))) {
416 /* If only s change, change just s value only*/
417 con0 &= ~(PLL45XX_SDIV_MASK << PLL45XX_SDIV_SHIFT);
418 con0 |= rate->sdiv << PLL45XX_SDIV_SHIFT;
419 __raw_writel(con0, pll->con_reg);
420
421 return 0;
422 }
423
424 /* Set PLL PMS values. */
425 con0 &= ~((PLL45XX_MDIV_MASK << PLL45XX_MDIV_SHIFT) |
426 (PLL45XX_PDIV_MASK << PLL45XX_PDIV_SHIFT) |
427 (PLL45XX_SDIV_MASK << PLL45XX_SDIV_SHIFT));
428 con0 |= (rate->mdiv << PLL45XX_MDIV_SHIFT) |
429 (rate->pdiv << PLL45XX_PDIV_SHIFT) |
430 (rate->sdiv << PLL45XX_SDIV_SHIFT);
431
432 /* Set PLL AFC value. */
433 con1 = __raw_readl(pll->con_reg + 0x4);
434 con1 &= ~(PLL45XX_AFC_MASK << PLL45XX_AFC_SHIFT);
435 con1 |= (rate->afc << PLL45XX_AFC_SHIFT);
436
437 /* Set PLL lock time. */
438 switch (pll->type) {
439 case pll_4502:
440 __raw_writel(rate->pdiv * PLL4502_LOCK_FACTOR, pll->lock_reg);
441 break;
442 case pll_4508:
443 __raw_writel(rate->pdiv * PLL4508_LOCK_FACTOR, pll->lock_reg);
444 break;
445 default:
446 break;
447 }
448
449 /* Set new configuration. */
450 __raw_writel(con1, pll->con_reg + 0x4);
451 __raw_writel(con0, pll->con_reg);
452
453 /* Wait for locking. */
454 start = ktime_get();
455 while (!(__raw_readl(pll->con_reg) & PLL45XX_LOCKED)) {
456 ktime_t delta = ktime_sub(ktime_get(), start);
457
458 if (ktime_to_ms(delta) > PLL_TIMEOUT_MS) {
459 pr_err("%s: could not lock PLL %s\n",
460 __func__, clk_hw_get_name(hw));
461 return -EFAULT;
462 }
463
464 cpu_relax();
465 }
466
467 return 0;
468 }
469
470 static const struct clk_ops samsung_pll45xx_clk_ops = {
471 .recalc_rate = samsung_pll45xx_recalc_rate,
472 .round_rate = samsung_pll_round_rate,
473 .set_rate = samsung_pll45xx_set_rate,
474 };
475
476 static const struct clk_ops samsung_pll45xx_clk_min_ops = {
477 .recalc_rate = samsung_pll45xx_recalc_rate,
478 };
479
480 /*
481 * PLL46xx Clock Type
482 */
483 #define PLL46XX_LOCK_FACTOR 3000
484
485 #define PLL46XX_VSEL_MASK (1)
486 #define PLL46XX_MDIV_MASK (0x1FF)
487 #define PLL1460X_MDIV_MASK (0x3FF)
488
489 #define PLL46XX_PDIV_MASK (0x3F)
490 #define PLL46XX_SDIV_MASK (0x7)
491 #define PLL46XX_VSEL_SHIFT (27)
492 #define PLL46XX_MDIV_SHIFT (16)
493 #define PLL46XX_PDIV_SHIFT (8)
494 #define PLL46XX_SDIV_SHIFT (0)
495
496 #define PLL46XX_KDIV_MASK (0xFFFF)
497 #define PLL4650C_KDIV_MASK (0xFFF)
498 #define PLL46XX_KDIV_SHIFT (0)
499 #define PLL46XX_MFR_MASK (0x3F)
500 #define PLL46XX_MRR_MASK (0x1F)
501 #define PLL46XX_KDIV_SHIFT (0)
502 #define PLL46XX_MFR_SHIFT (16)
503 #define PLL46XX_MRR_SHIFT (24)
504
505 #define PLL46XX_ENABLE BIT(31)
506 #define PLL46XX_LOCKED BIT(29)
507 #define PLL46XX_VSEL BIT(27)
508
samsung_pll46xx_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)509 static unsigned long samsung_pll46xx_recalc_rate(struct clk_hw *hw,
510 unsigned long parent_rate)
511 {
512 struct samsung_clk_pll *pll = to_clk_pll(hw);
513 u32 mdiv, pdiv, sdiv, kdiv, pll_con0, pll_con1, shift;
514 u64 fvco = parent_rate;
515
516 pll_con0 = __raw_readl(pll->con_reg);
517 pll_con1 = __raw_readl(pll->con_reg + 4);
518 mdiv = (pll_con0 >> PLL46XX_MDIV_SHIFT) & ((pll->type == pll_1460x) ?
519 PLL1460X_MDIV_MASK : PLL46XX_MDIV_MASK);
520 pdiv = (pll_con0 >> PLL46XX_PDIV_SHIFT) & PLL46XX_PDIV_MASK;
521 sdiv = (pll_con0 >> PLL46XX_SDIV_SHIFT) & PLL46XX_SDIV_MASK;
522 kdiv = pll->type == pll_4650c ? pll_con1 & PLL4650C_KDIV_MASK :
523 pll_con1 & PLL46XX_KDIV_MASK;
524
525 shift = ((pll->type == pll_4600) || (pll->type == pll_1460x)) ? 16 : 10;
526
527 fvco *= (mdiv << shift) + kdiv;
528 do_div(fvco, (pdiv << sdiv));
529 fvco >>= shift;
530
531 return (unsigned long)fvco;
532 }
533
samsung_pll46xx_mpk_change(u32 pll_con0,u32 pll_con1,const struct samsung_pll_rate_table * rate)534 static bool samsung_pll46xx_mpk_change(u32 pll_con0, u32 pll_con1,
535 const struct samsung_pll_rate_table *rate)
536 {
537 u32 old_mdiv, old_pdiv, old_kdiv;
538
539 old_mdiv = (pll_con0 >> PLL46XX_MDIV_SHIFT) & PLL46XX_MDIV_MASK;
540 old_pdiv = (pll_con0 >> PLL46XX_PDIV_SHIFT) & PLL46XX_PDIV_MASK;
541 old_kdiv = (pll_con1 >> PLL46XX_KDIV_SHIFT) & PLL46XX_KDIV_MASK;
542
543 return (old_mdiv != rate->mdiv || old_pdiv != rate->pdiv
544 || old_kdiv != rate->kdiv);
545 }
546
samsung_pll46xx_set_rate(struct clk_hw * hw,unsigned long drate,unsigned long prate)547 static int samsung_pll46xx_set_rate(struct clk_hw *hw, unsigned long drate,
548 unsigned long prate)
549 {
550 struct samsung_clk_pll *pll = to_clk_pll(hw);
551 const struct samsung_pll_rate_table *rate;
552 u32 con0, con1, lock;
553 ktime_t start;
554
555 /* Get required rate settings from table */
556 rate = samsung_get_pll_settings(pll, drate);
557 if (!rate) {
558 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
559 drate, clk_hw_get_name(hw));
560 return -EINVAL;
561 }
562
563 con0 = __raw_readl(pll->con_reg);
564 con1 = __raw_readl(pll->con_reg + 0x4);
565
566 if (!(samsung_pll46xx_mpk_change(con0, con1, rate))) {
567 /* If only s change, change just s value only*/
568 con0 &= ~(PLL46XX_SDIV_MASK << PLL46XX_SDIV_SHIFT);
569 con0 |= rate->sdiv << PLL46XX_SDIV_SHIFT;
570 __raw_writel(con0, pll->con_reg);
571
572 return 0;
573 }
574
575 /* Set PLL lock time. */
576 lock = rate->pdiv * PLL46XX_LOCK_FACTOR;
577 if (lock > 0xffff)
578 /* Maximum lock time bitfield is 16-bit. */
579 lock = 0xffff;
580
581 /* Set PLL PMS and VSEL values. */
582 if (pll->type == pll_1460x) {
583 con0 &= ~((PLL1460X_MDIV_MASK << PLL46XX_MDIV_SHIFT) |
584 (PLL46XX_PDIV_MASK << PLL46XX_PDIV_SHIFT) |
585 (PLL46XX_SDIV_MASK << PLL46XX_SDIV_SHIFT));
586 } else {
587 con0 &= ~((PLL46XX_MDIV_MASK << PLL46XX_MDIV_SHIFT) |
588 (PLL46XX_PDIV_MASK << PLL46XX_PDIV_SHIFT) |
589 (PLL46XX_SDIV_MASK << PLL46XX_SDIV_SHIFT) |
590 (PLL46XX_VSEL_MASK << PLL46XX_VSEL_SHIFT));
591 con0 |= rate->vsel << PLL46XX_VSEL_SHIFT;
592 }
593
594 con0 |= (rate->mdiv << PLL46XX_MDIV_SHIFT) |
595 (rate->pdiv << PLL46XX_PDIV_SHIFT) |
596 (rate->sdiv << PLL46XX_SDIV_SHIFT);
597
598 /* Set PLL K, MFR and MRR values. */
599 con1 = __raw_readl(pll->con_reg + 0x4);
600 con1 &= ~((PLL46XX_KDIV_MASK << PLL46XX_KDIV_SHIFT) |
601 (PLL46XX_MFR_MASK << PLL46XX_MFR_SHIFT) |
602 (PLL46XX_MRR_MASK << PLL46XX_MRR_SHIFT));
603 con1 |= (rate->kdiv << PLL46XX_KDIV_SHIFT) |
604 (rate->mfr << PLL46XX_MFR_SHIFT) |
605 (rate->mrr << PLL46XX_MRR_SHIFT);
606
607 /* Write configuration to PLL */
608 __raw_writel(lock, pll->lock_reg);
609 __raw_writel(con0, pll->con_reg);
610 __raw_writel(con1, pll->con_reg + 0x4);
611
612 /* Wait for locking. */
613 start = ktime_get();
614 while (!(__raw_readl(pll->con_reg) & PLL46XX_LOCKED)) {
615 ktime_t delta = ktime_sub(ktime_get(), start);
616
617 if (ktime_to_ms(delta) > PLL_TIMEOUT_MS) {
618 pr_err("%s: could not lock PLL %s\n",
619 __func__, clk_hw_get_name(hw));
620 return -EFAULT;
621 }
622
623 cpu_relax();
624 }
625
626 return 0;
627 }
628
629 static const struct clk_ops samsung_pll46xx_clk_ops = {
630 .recalc_rate = samsung_pll46xx_recalc_rate,
631 .round_rate = samsung_pll_round_rate,
632 .set_rate = samsung_pll46xx_set_rate,
633 };
634
635 static const struct clk_ops samsung_pll46xx_clk_min_ops = {
636 .recalc_rate = samsung_pll46xx_recalc_rate,
637 };
638
639 /*
640 * PLL6552 Clock Type
641 */
642
643 #define PLL6552_MDIV_MASK 0x3ff
644 #define PLL6552_PDIV_MASK 0x3f
645 #define PLL6552_SDIV_MASK 0x7
646 #define PLL6552_MDIV_SHIFT 16
647 #define PLL6552_MDIV_SHIFT_2416 14
648 #define PLL6552_PDIV_SHIFT 8
649 #define PLL6552_PDIV_SHIFT_2416 5
650 #define PLL6552_SDIV_SHIFT 0
651
samsung_pll6552_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)652 static unsigned long samsung_pll6552_recalc_rate(struct clk_hw *hw,
653 unsigned long parent_rate)
654 {
655 struct samsung_clk_pll *pll = to_clk_pll(hw);
656 u32 mdiv, pdiv, sdiv, pll_con;
657 u64 fvco = parent_rate;
658
659 pll_con = __raw_readl(pll->con_reg);
660 if (pll->type == pll_6552_s3c2416) {
661 mdiv = (pll_con >> PLL6552_MDIV_SHIFT_2416) & PLL6552_MDIV_MASK;
662 pdiv = (pll_con >> PLL6552_PDIV_SHIFT_2416) & PLL6552_PDIV_MASK;
663 } else {
664 mdiv = (pll_con >> PLL6552_MDIV_SHIFT) & PLL6552_MDIV_MASK;
665 pdiv = (pll_con >> PLL6552_PDIV_SHIFT) & PLL6552_PDIV_MASK;
666 }
667 sdiv = (pll_con >> PLL6552_SDIV_SHIFT) & PLL6552_SDIV_MASK;
668
669 fvco *= mdiv;
670 do_div(fvco, (pdiv << sdiv));
671
672 return (unsigned long)fvco;
673 }
674
675 static const struct clk_ops samsung_pll6552_clk_ops = {
676 .recalc_rate = samsung_pll6552_recalc_rate,
677 };
678
679 /*
680 * PLL6553 Clock Type
681 */
682
683 #define PLL6553_MDIV_MASK 0xff
684 #define PLL6553_PDIV_MASK 0x3f
685 #define PLL6553_SDIV_MASK 0x7
686 #define PLL6553_KDIV_MASK 0xffff
687 #define PLL6553_MDIV_SHIFT 16
688 #define PLL6553_PDIV_SHIFT 8
689 #define PLL6553_SDIV_SHIFT 0
690 #define PLL6553_KDIV_SHIFT 0
691
samsung_pll6553_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)692 static unsigned long samsung_pll6553_recalc_rate(struct clk_hw *hw,
693 unsigned long parent_rate)
694 {
695 struct samsung_clk_pll *pll = to_clk_pll(hw);
696 u32 mdiv, pdiv, sdiv, kdiv, pll_con0, pll_con1;
697 u64 fvco = parent_rate;
698
699 pll_con0 = __raw_readl(pll->con_reg);
700 pll_con1 = __raw_readl(pll->con_reg + 0x4);
701 mdiv = (pll_con0 >> PLL6553_MDIV_SHIFT) & PLL6553_MDIV_MASK;
702 pdiv = (pll_con0 >> PLL6553_PDIV_SHIFT) & PLL6553_PDIV_MASK;
703 sdiv = (pll_con0 >> PLL6553_SDIV_SHIFT) & PLL6553_SDIV_MASK;
704 kdiv = (pll_con1 >> PLL6553_KDIV_SHIFT) & PLL6553_KDIV_MASK;
705
706 fvco *= (mdiv << 16) + kdiv;
707 do_div(fvco, (pdiv << sdiv));
708 fvco >>= 16;
709
710 return (unsigned long)fvco;
711 }
712
713 static const struct clk_ops samsung_pll6553_clk_ops = {
714 .recalc_rate = samsung_pll6553_recalc_rate,
715 };
716
717 /*
718 * PLL Clock Type of S3C24XX before S3C2443
719 */
720
721 #define PLLS3C2410_MDIV_MASK (0xff)
722 #define PLLS3C2410_PDIV_MASK (0x1f)
723 #define PLLS3C2410_SDIV_MASK (0x3)
724 #define PLLS3C2410_MDIV_SHIFT (12)
725 #define PLLS3C2410_PDIV_SHIFT (4)
726 #define PLLS3C2410_SDIV_SHIFT (0)
727
728 #define PLLS3C2410_ENABLE_REG_OFFSET 0x10
729
samsung_s3c2410_pll_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)730 static unsigned long samsung_s3c2410_pll_recalc_rate(struct clk_hw *hw,
731 unsigned long parent_rate)
732 {
733 struct samsung_clk_pll *pll = to_clk_pll(hw);
734 u32 pll_con, mdiv, pdiv, sdiv;
735 u64 fvco = parent_rate;
736
737 pll_con = __raw_readl(pll->con_reg);
738 mdiv = (pll_con >> PLLS3C2410_MDIV_SHIFT) & PLLS3C2410_MDIV_MASK;
739 pdiv = (pll_con >> PLLS3C2410_PDIV_SHIFT) & PLLS3C2410_PDIV_MASK;
740 sdiv = (pll_con >> PLLS3C2410_SDIV_SHIFT) & PLLS3C2410_SDIV_MASK;
741
742 fvco *= (mdiv + 8);
743 do_div(fvco, (pdiv + 2) << sdiv);
744
745 return (unsigned int)fvco;
746 }
747
samsung_s3c2440_mpll_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)748 static unsigned long samsung_s3c2440_mpll_recalc_rate(struct clk_hw *hw,
749 unsigned long parent_rate)
750 {
751 struct samsung_clk_pll *pll = to_clk_pll(hw);
752 u32 pll_con, mdiv, pdiv, sdiv;
753 u64 fvco = parent_rate;
754
755 pll_con = __raw_readl(pll->con_reg);
756 mdiv = (pll_con >> PLLS3C2410_MDIV_SHIFT) & PLLS3C2410_MDIV_MASK;
757 pdiv = (pll_con >> PLLS3C2410_PDIV_SHIFT) & PLLS3C2410_PDIV_MASK;
758 sdiv = (pll_con >> PLLS3C2410_SDIV_SHIFT) & PLLS3C2410_SDIV_MASK;
759
760 fvco *= (2 * (mdiv + 8));
761 do_div(fvco, (pdiv + 2) << sdiv);
762
763 return (unsigned int)fvco;
764 }
765
samsung_s3c2410_pll_set_rate(struct clk_hw * hw,unsigned long drate,unsigned long prate)766 static int samsung_s3c2410_pll_set_rate(struct clk_hw *hw, unsigned long drate,
767 unsigned long prate)
768 {
769 struct samsung_clk_pll *pll = to_clk_pll(hw);
770 const struct samsung_pll_rate_table *rate;
771 u32 tmp;
772
773 /* Get required rate settings from table */
774 rate = samsung_get_pll_settings(pll, drate);
775 if (!rate) {
776 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
777 drate, clk_hw_get_name(hw));
778 return -EINVAL;
779 }
780
781 tmp = __raw_readl(pll->con_reg);
782
783 /* Change PLL PMS values */
784 tmp &= ~((PLLS3C2410_MDIV_MASK << PLLS3C2410_MDIV_SHIFT) |
785 (PLLS3C2410_PDIV_MASK << PLLS3C2410_PDIV_SHIFT) |
786 (PLLS3C2410_SDIV_MASK << PLLS3C2410_SDIV_SHIFT));
787 tmp |= (rate->mdiv << PLLS3C2410_MDIV_SHIFT) |
788 (rate->pdiv << PLLS3C2410_PDIV_SHIFT) |
789 (rate->sdiv << PLLS3C2410_SDIV_SHIFT);
790 __raw_writel(tmp, pll->con_reg);
791
792 /* Time to settle according to the manual */
793 udelay(300);
794
795 return 0;
796 }
797
samsung_s3c2410_pll_enable(struct clk_hw * hw,int bit,bool enable)798 static int samsung_s3c2410_pll_enable(struct clk_hw *hw, int bit, bool enable)
799 {
800 struct samsung_clk_pll *pll = to_clk_pll(hw);
801 u32 pll_en = __raw_readl(pll->lock_reg + PLLS3C2410_ENABLE_REG_OFFSET);
802 u32 pll_en_orig = pll_en;
803
804 if (enable)
805 pll_en &= ~BIT(bit);
806 else
807 pll_en |= BIT(bit);
808
809 __raw_writel(pll_en, pll->lock_reg + PLLS3C2410_ENABLE_REG_OFFSET);
810
811 /* if we started the UPLL, then allow to settle */
812 if (enable && (pll_en_orig & BIT(bit)))
813 udelay(300);
814
815 return 0;
816 }
817
samsung_s3c2410_mpll_enable(struct clk_hw * hw)818 static int samsung_s3c2410_mpll_enable(struct clk_hw *hw)
819 {
820 return samsung_s3c2410_pll_enable(hw, 5, true);
821 }
822
samsung_s3c2410_mpll_disable(struct clk_hw * hw)823 static void samsung_s3c2410_mpll_disable(struct clk_hw *hw)
824 {
825 samsung_s3c2410_pll_enable(hw, 5, false);
826 }
827
samsung_s3c2410_upll_enable(struct clk_hw * hw)828 static int samsung_s3c2410_upll_enable(struct clk_hw *hw)
829 {
830 return samsung_s3c2410_pll_enable(hw, 7, true);
831 }
832
samsung_s3c2410_upll_disable(struct clk_hw * hw)833 static void samsung_s3c2410_upll_disable(struct clk_hw *hw)
834 {
835 samsung_s3c2410_pll_enable(hw, 7, false);
836 }
837
838 static const struct clk_ops samsung_s3c2410_mpll_clk_min_ops = {
839 .recalc_rate = samsung_s3c2410_pll_recalc_rate,
840 .enable = samsung_s3c2410_mpll_enable,
841 .disable = samsung_s3c2410_mpll_disable,
842 };
843
844 static const struct clk_ops samsung_s3c2410_upll_clk_min_ops = {
845 .recalc_rate = samsung_s3c2410_pll_recalc_rate,
846 .enable = samsung_s3c2410_upll_enable,
847 .disable = samsung_s3c2410_upll_disable,
848 };
849
850 static const struct clk_ops samsung_s3c2440_mpll_clk_min_ops = {
851 .recalc_rate = samsung_s3c2440_mpll_recalc_rate,
852 .enable = samsung_s3c2410_mpll_enable,
853 .disable = samsung_s3c2410_mpll_disable,
854 };
855
856 static const struct clk_ops samsung_s3c2410_mpll_clk_ops = {
857 .recalc_rate = samsung_s3c2410_pll_recalc_rate,
858 .enable = samsung_s3c2410_mpll_enable,
859 .disable = samsung_s3c2410_mpll_disable,
860 .round_rate = samsung_pll_round_rate,
861 .set_rate = samsung_s3c2410_pll_set_rate,
862 };
863
864 static const struct clk_ops samsung_s3c2410_upll_clk_ops = {
865 .recalc_rate = samsung_s3c2410_pll_recalc_rate,
866 .enable = samsung_s3c2410_upll_enable,
867 .disable = samsung_s3c2410_upll_disable,
868 .round_rate = samsung_pll_round_rate,
869 .set_rate = samsung_s3c2410_pll_set_rate,
870 };
871
872 static const struct clk_ops samsung_s3c2440_mpll_clk_ops = {
873 .recalc_rate = samsung_s3c2440_mpll_recalc_rate,
874 .enable = samsung_s3c2410_mpll_enable,
875 .disable = samsung_s3c2410_mpll_disable,
876 .round_rate = samsung_pll_round_rate,
877 .set_rate = samsung_s3c2410_pll_set_rate,
878 };
879
880 /*
881 * PLL2550x Clock Type
882 */
883
884 #define PLL2550X_R_MASK (0x1)
885 #define PLL2550X_P_MASK (0x3F)
886 #define PLL2550X_M_MASK (0x3FF)
887 #define PLL2550X_S_MASK (0x7)
888 #define PLL2550X_R_SHIFT (20)
889 #define PLL2550X_P_SHIFT (14)
890 #define PLL2550X_M_SHIFT (4)
891 #define PLL2550X_S_SHIFT (0)
892
893 struct samsung_clk_pll2550x {
894 struct clk_hw hw;
895 const void __iomem *reg_base;
896 unsigned long offset;
897 };
898
899 #define to_clk_pll2550x(_hw) container_of(_hw, struct samsung_clk_pll2550x, hw)
900
samsung_pll2550x_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)901 static unsigned long samsung_pll2550x_recalc_rate(struct clk_hw *hw,
902 unsigned long parent_rate)
903 {
904 struct samsung_clk_pll2550x *pll = to_clk_pll2550x(hw);
905 u32 r, p, m, s, pll_stat;
906 u64 fvco = parent_rate;
907
908 pll_stat = __raw_readl(pll->reg_base + pll->offset * 3);
909 r = (pll_stat >> PLL2550X_R_SHIFT) & PLL2550X_R_MASK;
910 if (!r)
911 return 0;
912 p = (pll_stat >> PLL2550X_P_SHIFT) & PLL2550X_P_MASK;
913 m = (pll_stat >> PLL2550X_M_SHIFT) & PLL2550X_M_MASK;
914 s = (pll_stat >> PLL2550X_S_SHIFT) & PLL2550X_S_MASK;
915
916 fvco *= m;
917 do_div(fvco, (p << s));
918
919 return (unsigned long)fvco;
920 }
921
922 static const struct clk_ops samsung_pll2550x_clk_ops = {
923 .recalc_rate = samsung_pll2550x_recalc_rate,
924 };
925
samsung_clk_register_pll2550x(const char * name,const char * pname,const void __iomem * reg_base,const unsigned long offset)926 struct clk * __init samsung_clk_register_pll2550x(const char *name,
927 const char *pname, const void __iomem *reg_base,
928 const unsigned long offset)
929 {
930 struct samsung_clk_pll2550x *pll;
931 struct clk *clk;
932 struct clk_init_data init;
933
934 pll = kzalloc(sizeof(*pll), GFP_KERNEL);
935 if (!pll) {
936 pr_err("%s: could not allocate pll clk %s\n", __func__, name);
937 return NULL;
938 }
939
940 init.name = name;
941 init.ops = &samsung_pll2550x_clk_ops;
942 init.flags = CLK_GET_RATE_NOCACHE;
943 init.parent_names = &pname;
944 init.num_parents = 1;
945
946 pll->hw.init = &init;
947 pll->reg_base = reg_base;
948 pll->offset = offset;
949
950 clk = clk_register(NULL, &pll->hw);
951 if (IS_ERR(clk)) {
952 pr_err("%s: failed to register pll clock %s\n", __func__,
953 name);
954 kfree(pll);
955 }
956
957 if (clk_register_clkdev(clk, name, NULL))
958 pr_err("%s: failed to register lookup for %s", __func__, name);
959
960 return clk;
961 }
962
963 /*
964 * PLL2550xx Clock Type
965 */
966
967 /* Maximum lock time can be 270 * PDIV cycles */
968 #define PLL2550XX_LOCK_FACTOR 270
969
970 #define PLL2550XX_M_MASK 0x3FF
971 #define PLL2550XX_P_MASK 0x3F
972 #define PLL2550XX_S_MASK 0x7
973 #define PLL2550XX_LOCK_STAT_MASK 0x1
974 #define PLL2550XX_M_SHIFT 9
975 #define PLL2550XX_P_SHIFT 3
976 #define PLL2550XX_S_SHIFT 0
977 #define PLL2550XX_LOCK_STAT_SHIFT 21
978
samsung_pll2550xx_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)979 static unsigned long samsung_pll2550xx_recalc_rate(struct clk_hw *hw,
980 unsigned long parent_rate)
981 {
982 struct samsung_clk_pll *pll = to_clk_pll(hw);
983 u32 mdiv, pdiv, sdiv, pll_con;
984 u64 fvco = parent_rate;
985
986 pll_con = __raw_readl(pll->con_reg);
987 mdiv = (pll_con >> PLL2550XX_M_SHIFT) & PLL2550XX_M_MASK;
988 pdiv = (pll_con >> PLL2550XX_P_SHIFT) & PLL2550XX_P_MASK;
989 sdiv = (pll_con >> PLL2550XX_S_SHIFT) & PLL2550XX_S_MASK;
990
991 fvco *= mdiv;
992 do_div(fvco, (pdiv << sdiv));
993
994 return (unsigned long)fvco;
995 }
996
samsung_pll2550xx_mp_change(u32 mdiv,u32 pdiv,u32 pll_con)997 static inline bool samsung_pll2550xx_mp_change(u32 mdiv, u32 pdiv, u32 pll_con)
998 {
999 u32 old_mdiv, old_pdiv;
1000
1001 old_mdiv = (pll_con >> PLL2550XX_M_SHIFT) & PLL2550XX_M_MASK;
1002 old_pdiv = (pll_con >> PLL2550XX_P_SHIFT) & PLL2550XX_P_MASK;
1003
1004 return mdiv != old_mdiv || pdiv != old_pdiv;
1005 }
1006
samsung_pll2550xx_set_rate(struct clk_hw * hw,unsigned long drate,unsigned long prate)1007 static int samsung_pll2550xx_set_rate(struct clk_hw *hw, unsigned long drate,
1008 unsigned long prate)
1009 {
1010 struct samsung_clk_pll *pll = to_clk_pll(hw);
1011 const struct samsung_pll_rate_table *rate;
1012 u32 tmp;
1013
1014 /* Get required rate settings from table */
1015 rate = samsung_get_pll_settings(pll, drate);
1016 if (!rate) {
1017 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
1018 drate, clk_hw_get_name(hw));
1019 return -EINVAL;
1020 }
1021
1022 tmp = __raw_readl(pll->con_reg);
1023
1024 if (!(samsung_pll2550xx_mp_change(rate->mdiv, rate->pdiv, tmp))) {
1025 /* If only s change, change just s value only*/
1026 tmp &= ~(PLL2550XX_S_MASK << PLL2550XX_S_SHIFT);
1027 tmp |= rate->sdiv << PLL2550XX_S_SHIFT;
1028 __raw_writel(tmp, pll->con_reg);
1029
1030 return 0;
1031 }
1032
1033 /* Set PLL lock time. */
1034 __raw_writel(rate->pdiv * PLL2550XX_LOCK_FACTOR, pll->lock_reg);
1035
1036 /* Change PLL PMS values */
1037 tmp &= ~((PLL2550XX_M_MASK << PLL2550XX_M_SHIFT) |
1038 (PLL2550XX_P_MASK << PLL2550XX_P_SHIFT) |
1039 (PLL2550XX_S_MASK << PLL2550XX_S_SHIFT));
1040 tmp |= (rate->mdiv << PLL2550XX_M_SHIFT) |
1041 (rate->pdiv << PLL2550XX_P_SHIFT) |
1042 (rate->sdiv << PLL2550XX_S_SHIFT);
1043 __raw_writel(tmp, pll->con_reg);
1044
1045 /* wait_lock_time */
1046 do {
1047 cpu_relax();
1048 tmp = __raw_readl(pll->con_reg);
1049 } while (!(tmp & (PLL2550XX_LOCK_STAT_MASK
1050 << PLL2550XX_LOCK_STAT_SHIFT)));
1051
1052 return 0;
1053 }
1054
1055 static const struct clk_ops samsung_pll2550xx_clk_ops = {
1056 .recalc_rate = samsung_pll2550xx_recalc_rate,
1057 .round_rate = samsung_pll_round_rate,
1058 .set_rate = samsung_pll2550xx_set_rate,
1059 };
1060
1061 static const struct clk_ops samsung_pll2550xx_clk_min_ops = {
1062 .recalc_rate = samsung_pll2550xx_recalc_rate,
1063 };
1064
1065 /*
1066 * PLL2650XX Clock Type
1067 */
1068
1069 /* Maximum lock time can be 3000 * PDIV cycles */
1070 #define PLL2650XX_LOCK_FACTOR 3000
1071
1072 #define PLL2650XX_MDIV_SHIFT 9
1073 #define PLL2650XX_PDIV_SHIFT 3
1074 #define PLL2650XX_SDIV_SHIFT 0
1075 #define PLL2650XX_KDIV_SHIFT 0
1076 #define PLL2650XX_MDIV_MASK 0x1ff
1077 #define PLL2650XX_PDIV_MASK 0x3f
1078 #define PLL2650XX_SDIV_MASK 0x7
1079 #define PLL2650XX_KDIV_MASK 0xffff
1080 #define PLL2650XX_PLL_ENABLE_SHIFT 23
1081 #define PLL2650XX_PLL_LOCKTIME_SHIFT 21
1082 #define PLL2650XX_PLL_FOUTMASK_SHIFT 31
1083
samsung_pll2650xx_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)1084 static unsigned long samsung_pll2650xx_recalc_rate(struct clk_hw *hw,
1085 unsigned long parent_rate)
1086 {
1087 struct samsung_clk_pll *pll = to_clk_pll(hw);
1088 u32 mdiv, pdiv, sdiv, pll_con0, pll_con2;
1089 s16 kdiv;
1090 u64 fvco = parent_rate;
1091
1092 pll_con0 = __raw_readl(pll->con_reg);
1093 pll_con2 = __raw_readl(pll->con_reg + 8);
1094 mdiv = (pll_con0 >> PLL2650XX_MDIV_SHIFT) & PLL2650XX_MDIV_MASK;
1095 pdiv = (pll_con0 >> PLL2650XX_PDIV_SHIFT) & PLL2650XX_PDIV_MASK;
1096 sdiv = (pll_con0 >> PLL2650XX_SDIV_SHIFT) & PLL2650XX_SDIV_MASK;
1097 kdiv = (s16)(pll_con2 & PLL2650XX_KDIV_MASK);
1098
1099 fvco *= (mdiv << 16) + kdiv;
1100 do_div(fvco, (pdiv << sdiv));
1101 fvco >>= 16;
1102
1103 return (unsigned long)fvco;
1104 }
1105
samsung_pll2650xx_set_rate(struct clk_hw * hw,unsigned long drate,unsigned long parent_rate)1106 static int samsung_pll2650xx_set_rate(struct clk_hw *hw, unsigned long drate,
1107 unsigned long parent_rate)
1108 {
1109 struct samsung_clk_pll *pll = to_clk_pll(hw);
1110 u32 tmp, pll_con0, pll_con2;
1111 const struct samsung_pll_rate_table *rate;
1112
1113 rate = samsung_get_pll_settings(pll, drate);
1114 if (!rate) {
1115 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
1116 drate, clk_hw_get_name(hw));
1117 return -EINVAL;
1118 }
1119
1120 pll_con0 = __raw_readl(pll->con_reg);
1121 pll_con2 = __raw_readl(pll->con_reg + 8);
1122
1123 /* Change PLL PMS values */
1124 pll_con0 &= ~(PLL2650XX_MDIV_MASK << PLL2650XX_MDIV_SHIFT |
1125 PLL2650XX_PDIV_MASK << PLL2650XX_PDIV_SHIFT |
1126 PLL2650XX_SDIV_MASK << PLL2650XX_SDIV_SHIFT);
1127 pll_con0 |= rate->mdiv << PLL2650XX_MDIV_SHIFT;
1128 pll_con0 |= rate->pdiv << PLL2650XX_PDIV_SHIFT;
1129 pll_con0 |= rate->sdiv << PLL2650XX_SDIV_SHIFT;
1130 pll_con0 |= 1 << PLL2650XX_PLL_ENABLE_SHIFT;
1131 pll_con0 |= 1 << PLL2650XX_PLL_FOUTMASK_SHIFT;
1132
1133 pll_con2 &= ~(PLL2650XX_KDIV_MASK << PLL2650XX_KDIV_SHIFT);
1134 pll_con2 |= ((~(rate->kdiv) + 1) & PLL2650XX_KDIV_MASK)
1135 << PLL2650XX_KDIV_SHIFT;
1136
1137 /* Set PLL lock time. */
1138 __raw_writel(PLL2650XX_LOCK_FACTOR * rate->pdiv, pll->lock_reg);
1139
1140 __raw_writel(pll_con0, pll->con_reg);
1141 __raw_writel(pll_con2, pll->con_reg + 8);
1142
1143 do {
1144 tmp = __raw_readl(pll->con_reg);
1145 } while (!(tmp & (0x1 << PLL2650XX_PLL_LOCKTIME_SHIFT)));
1146
1147 return 0;
1148 }
1149
1150 static const struct clk_ops samsung_pll2650xx_clk_ops = {
1151 .recalc_rate = samsung_pll2650xx_recalc_rate,
1152 .set_rate = samsung_pll2650xx_set_rate,
1153 .round_rate = samsung_pll_round_rate,
1154 };
1155
1156 static const struct clk_ops samsung_pll2650xx_clk_min_ops = {
1157 .recalc_rate = samsung_pll2650xx_recalc_rate,
1158 };
1159
_samsung_clk_register_pll(struct samsung_clk_provider * ctx,const struct samsung_pll_clock * pll_clk,void __iomem * base)1160 static void __init _samsung_clk_register_pll(struct samsung_clk_provider *ctx,
1161 const struct samsung_pll_clock *pll_clk,
1162 void __iomem *base)
1163 {
1164 struct samsung_clk_pll *pll;
1165 struct clk *clk;
1166 struct clk_init_data init;
1167 int ret, len;
1168
1169 pll = kzalloc(sizeof(*pll), GFP_KERNEL);
1170 if (!pll) {
1171 pr_err("%s: could not allocate pll clk %s\n",
1172 __func__, pll_clk->name);
1173 return;
1174 }
1175
1176 init.name = pll_clk->name;
1177 init.flags = pll_clk->flags;
1178 init.parent_names = &pll_clk->parent_name;
1179 init.num_parents = 1;
1180
1181 if (pll_clk->rate_table) {
1182 /* find count of rates in rate_table */
1183 for (len = 0; pll_clk->rate_table[len].rate != 0; )
1184 len++;
1185
1186 pll->rate_count = len;
1187 pll->rate_table = kmemdup(pll_clk->rate_table,
1188 pll->rate_count *
1189 sizeof(struct samsung_pll_rate_table),
1190 GFP_KERNEL);
1191 WARN(!pll->rate_table,
1192 "%s: could not allocate rate table for %s\n",
1193 __func__, pll_clk->name);
1194 }
1195
1196 switch (pll_clk->type) {
1197 case pll_2126:
1198 init.ops = &samsung_pll2126_clk_ops;
1199 break;
1200 case pll_3000:
1201 init.ops = &samsung_pll3000_clk_ops;
1202 break;
1203 /* clk_ops for 35xx and 2550 are similar */
1204 case pll_35xx:
1205 case pll_2550:
1206 case pll_1450x:
1207 case pll_1451x:
1208 case pll_1452x:
1209 if (!pll->rate_table)
1210 init.ops = &samsung_pll35xx_clk_min_ops;
1211 else
1212 init.ops = &samsung_pll35xx_clk_ops;
1213 break;
1214 case pll_4500:
1215 init.ops = &samsung_pll45xx_clk_min_ops;
1216 break;
1217 case pll_4502:
1218 case pll_4508:
1219 if (!pll->rate_table)
1220 init.ops = &samsung_pll45xx_clk_min_ops;
1221 else
1222 init.ops = &samsung_pll45xx_clk_ops;
1223 break;
1224 /* clk_ops for 36xx and 2650 are similar */
1225 case pll_36xx:
1226 case pll_2650:
1227 if (!pll->rate_table)
1228 init.ops = &samsung_pll36xx_clk_min_ops;
1229 else
1230 init.ops = &samsung_pll36xx_clk_ops;
1231 break;
1232 case pll_6552:
1233 case pll_6552_s3c2416:
1234 init.ops = &samsung_pll6552_clk_ops;
1235 break;
1236 case pll_6553:
1237 init.ops = &samsung_pll6553_clk_ops;
1238 break;
1239 case pll_4600:
1240 case pll_4650:
1241 case pll_4650c:
1242 case pll_1460x:
1243 if (!pll->rate_table)
1244 init.ops = &samsung_pll46xx_clk_min_ops;
1245 else
1246 init.ops = &samsung_pll46xx_clk_ops;
1247 break;
1248 case pll_s3c2410_mpll:
1249 if (!pll->rate_table)
1250 init.ops = &samsung_s3c2410_mpll_clk_min_ops;
1251 else
1252 init.ops = &samsung_s3c2410_mpll_clk_ops;
1253 break;
1254 case pll_s3c2410_upll:
1255 if (!pll->rate_table)
1256 init.ops = &samsung_s3c2410_upll_clk_min_ops;
1257 else
1258 init.ops = &samsung_s3c2410_upll_clk_ops;
1259 break;
1260 case pll_s3c2440_mpll:
1261 if (!pll->rate_table)
1262 init.ops = &samsung_s3c2440_mpll_clk_min_ops;
1263 else
1264 init.ops = &samsung_s3c2440_mpll_clk_ops;
1265 break;
1266 case pll_2550xx:
1267 if (!pll->rate_table)
1268 init.ops = &samsung_pll2550xx_clk_min_ops;
1269 else
1270 init.ops = &samsung_pll2550xx_clk_ops;
1271 break;
1272 case pll_2650xx:
1273 if (!pll->rate_table)
1274 init.ops = &samsung_pll2650xx_clk_min_ops;
1275 else
1276 init.ops = &samsung_pll2650xx_clk_ops;
1277 break;
1278 default:
1279 pr_warn("%s: Unknown pll type for pll clk %s\n",
1280 __func__, pll_clk->name);
1281 }
1282
1283 pll->hw.init = &init;
1284 pll->type = pll_clk->type;
1285 pll->lock_reg = base + pll_clk->lock_offset;
1286 pll->con_reg = base + pll_clk->con_offset;
1287
1288 clk = clk_register(NULL, &pll->hw);
1289 if (IS_ERR(clk)) {
1290 pr_err("%s: failed to register pll clock %s : %ld\n",
1291 __func__, pll_clk->name, PTR_ERR(clk));
1292 kfree(pll);
1293 return;
1294 }
1295
1296 samsung_clk_add_lookup(ctx, clk, pll_clk->id);
1297
1298 if (!pll_clk->alias)
1299 return;
1300
1301 ret = clk_register_clkdev(clk, pll_clk->alias, pll_clk->dev_name);
1302 if (ret)
1303 pr_err("%s: failed to register lookup for %s : %d",
1304 __func__, pll_clk->name, ret);
1305 }
1306
samsung_clk_register_pll(struct samsung_clk_provider * ctx,const struct samsung_pll_clock * pll_list,unsigned int nr_pll,void __iomem * base)1307 void __init samsung_clk_register_pll(struct samsung_clk_provider *ctx,
1308 const struct samsung_pll_clock *pll_list,
1309 unsigned int nr_pll, void __iomem *base)
1310 {
1311 int cnt;
1312
1313 for (cnt = 0; cnt < nr_pll; cnt++)
1314 _samsung_clk_register_pll(ctx, &pll_list[cnt], base);
1315 }
1316