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