blob: 9617825daabb8310788dde2e401c1d969889acac [file] [log] [blame]
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +09001/*
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>
Tomasz Figab4054ac2013-08-26 19:09:05 +020013#include <linux/hrtimer.h>
Heiko Stuebnerea5d6a82014-02-25 09:50:43 +090014#include <linux/delay.h>
Stephen Boyd6f1ed072015-06-19 15:00:46 -070015#include <linux/slab.h>
16#include <linux/clkdev.h>
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +090017#include "clk.h"
18#include "clk-pll.h"
19
Tomasz Figab4054ac2013-08-26 19:09:05 +020020#define PLL_TIMEOUT_MS 10
21
Yadwinder Singh Brar079dbea2013-06-11 15:01:06 +053022struct samsung_clk_pll {
23 struct clk_hw hw;
24 void __iomem *lock_reg;
25 void __iomem *con_reg;
Yadwinder Singh Brar07dc76f2013-06-11 15:01:07 +053026 enum samsung_pll_type type;
Yadwinder Singh Brar3ff6e0d2013-06-11 15:01:12 +053027 unsigned int rate_count;
28 const struct samsung_pll_rate_table *rate_table;
Yadwinder Singh Brar079dbea2013-06-11 15:01:06 +053029};
30
31#define to_clk_pll(_hw) container_of(_hw, struct samsung_clk_pll, hw)
32
Yadwinder Singh Brardfa89312013-06-11 15:01:13 +053033static 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
47static 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
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +090064/*
Heiko Stuebnera951b1d2014-02-19 09:25:41 +090065 * 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
75static 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
Matthew Leach4de10332016-06-08 19:30:56 +010082 pll_con = readl_relaxed(pll->con_reg);
Heiko Stuebnera951b1d2014-02-19 09:25:41 +090083 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
93static 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
108static 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
Matthew Leach4de10332016-06-08 19:30:56 +0100115 pll_con = readl_relaxed(pll->con_reg);
Heiko Stuebnera951b1d2014-02-19 09:25:41 +0900116 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
126static const struct clk_ops samsung_pll3000_clk_ops = {
127 .recalc_rate = samsung_pll3000_recalc_rate,
128};
129
130/*
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900131 * PLL35xx Clock Type
132 */
Yadwinder Singh Brardfa89312013-06-11 15:01:13 +0530133/* Maximum lock time can be 270 * PDIV cycles */
134#define PLL35XX_LOCK_FACTOR (270)
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900135
136#define PLL35XX_MDIV_MASK (0x3FF)
137#define PLL35XX_PDIV_MASK (0x3F)
138#define PLL35XX_SDIV_MASK (0x7)
Yadwinder Singh Brardfa89312013-06-11 15:01:13 +0530139#define PLL35XX_LOCK_STAT_MASK (0x1)
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900140#define PLL35XX_MDIV_SHIFT (16)
141#define PLL35XX_PDIV_SHIFT (8)
142#define PLL35XX_SDIV_SHIFT (0)
Yadwinder Singh Brardfa89312013-06-11 15:01:13 +0530143#define PLL35XX_LOCK_STAT_SHIFT (29)
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900144
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900145static unsigned long samsung_pll35xx_recalc_rate(struct clk_hw *hw,
146 unsigned long parent_rate)
147{
Yadwinder Singh Brar079dbea2013-06-11 15:01:06 +0530148 struct samsung_clk_pll *pll = to_clk_pll(hw);
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900149 u32 mdiv, pdiv, sdiv, pll_con;
150 u64 fvco = parent_rate;
151
Matthew Leach4de10332016-06-08 19:30:56 +0100152 pll_con = readl_relaxed(pll->con_reg);
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900153 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
Yadwinder Singh Brardfa89312013-06-11 15:01:13 +0530163static 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
174static 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__,
Stephen Boyd836ee0f2015-08-12 11:42:23 -0700185 drate, clk_hw_get_name(hw));
Yadwinder Singh Brardfa89312013-06-11 15:01:13 +0530186 return -EINVAL;
187 }
188
Matthew Leach4de10332016-06-08 19:30:56 +0100189 tmp = readl_relaxed(pll->con_reg);
Yadwinder Singh Brardfa89312013-06-11 15:01:13 +0530190
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;
Matthew Leach4de10332016-06-08 19:30:56 +0100195 writel_relaxed(tmp, pll->con_reg);
Yadwinder Singh Brardfa89312013-06-11 15:01:13 +0530196
197 return 0;
198 }
199
200 /* Set PLL lock time. */
Matthew Leach4de10332016-06-08 19:30:56 +0100201 writel_relaxed(rate->pdiv * PLL35XX_LOCK_FACTOR,
Yadwinder Singh Brardfa89312013-06-11 15:01:13 +0530202 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);
Matthew Leach4de10332016-06-08 19:30:56 +0100211 writel_relaxed(tmp, pll->con_reg);
Yadwinder Singh Brardfa89312013-06-11 15:01:13 +0530212
213 /* wait_lock_time */
214 do {
215 cpu_relax();
Matthew Leach4de10332016-06-08 19:30:56 +0100216 tmp = readl_relaxed(pll->con_reg);
Yadwinder Singh Brardfa89312013-06-11 15:01:13 +0530217 } while (!(tmp & (PLL35XX_LOCK_STAT_MASK
218 << PLL35XX_LOCK_STAT_SHIFT)));
219 return 0;
220}
221
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900222static const struct clk_ops samsung_pll35xx_clk_ops = {
223 .recalc_rate = samsung_pll35xx_recalc_rate,
Yadwinder Singh Brardfa89312013-06-11 15:01:13 +0530224 .round_rate = samsung_pll_round_rate,
225 .set_rate = samsung_pll35xx_set_rate,
226};
227
228static const struct clk_ops samsung_pll35xx_clk_min_ops = {
229 .recalc_rate = samsung_pll35xx_recalc_rate,
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900230};
231
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900232/*
233 * PLL36xx Clock Type
234 */
Vikas Sajjanbb4278d2013-06-11 15:01:14 +0530235/* Maximum lock time can be 3000 * PDIV cycles */
236#define PLL36XX_LOCK_FACTOR (3000)
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900237
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)
Vikas Sajjanbb4278d2013-06-11 15:01:14 +0530245#define PLL36XX_KDIV_SHIFT (0)
246#define PLL36XX_LOCK_STAT_SHIFT (29)
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900247
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900248static unsigned long samsung_pll36xx_recalc_rate(struct clk_hw *hw,
249 unsigned long parent_rate)
250{
Yadwinder Singh Brar079dbea2013-06-11 15:01:06 +0530251 struct samsung_clk_pll *pll = to_clk_pll(hw);
Doug Anderson071ff9a2013-06-11 08:24:05 -0700252 u32 mdiv, pdiv, sdiv, pll_con0, pll_con1;
253 s16 kdiv;
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900254 u64 fvco = parent_rate;
255
Matthew Leach4de10332016-06-08 19:30:56 +0100256 pll_con0 = readl_relaxed(pll->con_reg);
257 pll_con1 = readl_relaxed(pll->con_reg + 4);
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900258 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;
Doug Anderson071ff9a2013-06-11 08:24:05 -0700261 kdiv = (s16)(pll_con1 & PLL36XX_KDIV_MASK);
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900262
263 fvco *= (mdiv << 16) + kdiv;
264 do_div(fvco, (pdiv << sdiv));
265 fvco >>= 16;
266
267 return (unsigned long)fvco;
268}
269
Vikas Sajjanbb4278d2013-06-11 15:01:14 +0530270static 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
283static 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__,
Stephen Boyd836ee0f2015-08-12 11:42:23 -0700293 drate, clk_hw_get_name(hw));
Vikas Sajjanbb4278d2013-06-11 15:01:14 +0530294 return -EINVAL;
295 }
296
Matthew Leach4de10332016-06-08 19:30:56 +0100297 pll_con0 = readl_relaxed(pll->con_reg);
298 pll_con1 = readl_relaxed(pll->con_reg + 4);
Vikas Sajjanbb4278d2013-06-11 15:01:14 +0530299
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);
Matthew Leach4de10332016-06-08 19:30:56 +0100304 writel_relaxed(pll_con0, pll->con_reg);
Vikas Sajjanbb4278d2013-06-11 15:01:14 +0530305
306 return 0;
307 }
308
309 /* Set PLL lock time. */
Matthew Leach4de10332016-06-08 19:30:56 +0100310 writel_relaxed(rate->pdiv * PLL36XX_LOCK_FACTOR, pll->lock_reg);
Vikas Sajjanbb4278d2013-06-11 15:01:14 +0530311
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);
Matthew Leach4de10332016-06-08 19:30:56 +0100319 writel_relaxed(pll_con0, pll->con_reg);
Vikas Sajjanbb4278d2013-06-11 15:01:14 +0530320
321 pll_con1 &= ~(PLL36XX_KDIV_MASK << PLL36XX_KDIV_SHIFT);
322 pll_con1 |= rate->kdiv << PLL36XX_KDIV_SHIFT;
Matthew Leach4de10332016-06-08 19:30:56 +0100323 writel_relaxed(pll_con1, pll->con_reg + 4);
Vikas Sajjanbb4278d2013-06-11 15:01:14 +0530324
325 /* wait_lock_time */
326 do {
327 cpu_relax();
Matthew Leach4de10332016-06-08 19:30:56 +0100328 tmp = readl_relaxed(pll->con_reg);
Vikas Sajjanbb4278d2013-06-11 15:01:14 +0530329 } while (!(tmp & (1 << PLL36XX_LOCK_STAT_SHIFT)));
330
331 return 0;
332}
333
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900334static const struct clk_ops samsung_pll36xx_clk_ops = {
335 .recalc_rate = samsung_pll36xx_recalc_rate,
Vikas Sajjanbb4278d2013-06-11 15:01:14 +0530336 .set_rate = samsung_pll36xx_set_rate,
337 .round_rate = samsung_pll_round_rate,
338};
339
340static const struct clk_ops samsung_pll36xx_clk_min_ops = {
341 .recalc_rate = samsung_pll36xx_recalc_rate,
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900342};
343
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900344/*
345 * PLL45xx Clock Type
346 */
Tomasz Figab4054ac2013-08-26 19:09:05 +0200347#define PLL4502_LOCK_FACTOR 400
348#define PLL4508_LOCK_FACTOR 240
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900349
350#define PLL45XX_MDIV_MASK (0x3FF)
351#define PLL45XX_PDIV_MASK (0x3F)
352#define PLL45XX_SDIV_MASK (0x7)
Tomasz Figab4054ac2013-08-26 19:09:05 +0200353#define PLL45XX_AFC_MASK (0x1F)
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900354#define PLL45XX_MDIV_SHIFT (16)
355#define PLL45XX_PDIV_SHIFT (8)
356#define PLL45XX_SDIV_SHIFT (0)
Tomasz Figab4054ac2013-08-26 19:09:05 +0200357#define PLL45XX_AFC_SHIFT (0)
358
359#define PLL45XX_ENABLE BIT(31)
360#define PLL45XX_LOCKED BIT(29)
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900361
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900362static unsigned long samsung_pll45xx_recalc_rate(struct clk_hw *hw,
363 unsigned long parent_rate)
364{
Tomasz Figa52b06012013-08-26 19:09:04 +0200365 struct samsung_clk_pll *pll = to_clk_pll(hw);
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900366 u32 mdiv, pdiv, sdiv, pll_con;
367 u64 fvco = parent_rate;
368
Matthew Leach4de10332016-06-08 19:30:56 +0100369 pll_con = readl_relaxed(pll->con_reg);
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900370 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
Tomasz Figab4054ac2013-08-26 19:09:05 +0200383static 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
396static 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__,
Stephen Boyd836ee0f2015-08-12 11:42:23 -0700408 drate, clk_hw_get_name(hw));
Tomasz Figab4054ac2013-08-26 19:09:05 +0200409 return -EINVAL;
410 }
411
Matthew Leach4de10332016-06-08 19:30:56 +0100412 con0 = readl_relaxed(pll->con_reg);
413 con1 = readl_relaxed(pll->con_reg + 0x4);
Tomasz Figab4054ac2013-08-26 19:09:05 +0200414
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;
Matthew Leach4de10332016-06-08 19:30:56 +0100419 writel_relaxed(con0, pll->con_reg);
Tomasz Figab4054ac2013-08-26 19:09:05 +0200420
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. */
Matthew Leach4de10332016-06-08 19:30:56 +0100433 con1 = readl_relaxed(pll->con_reg + 0x4);
Tomasz Figab4054ac2013-08-26 19:09:05 +0200434 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:
Matthew Leach4de10332016-06-08 19:30:56 +0100440 writel_relaxed(rate->pdiv * PLL4502_LOCK_FACTOR, pll->lock_reg);
Tomasz Figab4054ac2013-08-26 19:09:05 +0200441 break;
442 case pll_4508:
Matthew Leach4de10332016-06-08 19:30:56 +0100443 writel_relaxed(rate->pdiv * PLL4508_LOCK_FACTOR, pll->lock_reg);
Tomasz Figab4054ac2013-08-26 19:09:05 +0200444 break;
445 default:
446 break;
Sachin Kamat8e31d192014-01-17 17:05:52 +0530447 }
Tomasz Figab4054ac2013-08-26 19:09:05 +0200448
449 /* Set new configuration. */
Matthew Leach4de10332016-06-08 19:30:56 +0100450 writel_relaxed(con1, pll->con_reg + 0x4);
451 writel_relaxed(con0, pll->con_reg);
Tomasz Figab4054ac2013-08-26 19:09:05 +0200452
453 /* Wait for locking. */
454 start = ktime_get();
Matthew Leach4de10332016-06-08 19:30:56 +0100455 while (!(readl_relaxed(pll->con_reg) & PLL45XX_LOCKED)) {
Tomasz Figab4054ac2013-08-26 19:09:05 +0200456 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",
Stephen Boyd836ee0f2015-08-12 11:42:23 -0700460 __func__, clk_hw_get_name(hw));
Tomasz Figab4054ac2013-08-26 19:09:05 +0200461 return -EFAULT;
462 }
463
464 cpu_relax();
465 }
466
467 return 0;
468}
469
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900470static const struct clk_ops samsung_pll45xx_clk_ops = {
471 .recalc_rate = samsung_pll45xx_recalc_rate,
Tomasz Figab4054ac2013-08-26 19:09:05 +0200472 .round_rate = samsung_pll_round_rate,
473 .set_rate = samsung_pll45xx_set_rate,
474};
475
476static const struct clk_ops samsung_pll45xx_clk_min_ops = {
477 .recalc_rate = samsung_pll45xx_recalc_rate,
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900478};
479
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900480/*
481 * PLL46xx Clock Type
482 */
Tomasz Figa5c896582013-08-26 19:09:07 +0200483#define PLL46XX_LOCK_FACTOR 3000
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900484
Tomasz Figa5c896582013-08-26 19:09:07 +0200485#define PLL46XX_VSEL_MASK (1)
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900486#define PLL46XX_MDIV_MASK (0x1FF)
Naveen Krishna Ch0c23e2a2014-09-22 10:17:01 +0530487#define PLL1460X_MDIV_MASK (0x3FF)
488
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900489#define PLL46XX_PDIV_MASK (0x3F)
490#define PLL46XX_SDIV_MASK (0x7)
Tomasz Figa5c896582013-08-26 19:09:07 +0200491#define PLL46XX_VSEL_SHIFT (27)
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900492#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)
Tomasz Figa5c896582013-08-26 19:09:07 +0200499#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)
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900508
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900509static unsigned long samsung_pll46xx_recalc_rate(struct clk_hw *hw,
510 unsigned long parent_rate)
511{
Tomasz Figac50d11f2013-08-26 19:09:06 +0200512 struct samsung_clk_pll *pll = to_clk_pll(hw);
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900513 u32 mdiv, pdiv, sdiv, kdiv, pll_con0, pll_con1, shift;
514 u64 fvco = parent_rate;
515
Matthew Leach4de10332016-06-08 19:30:56 +0100516 pll_con0 = readl_relaxed(pll->con_reg);
517 pll_con1 = readl_relaxed(pll->con_reg + 4);
Naveen Krishna Ch0c23e2a2014-09-22 10:17:01 +0530518 mdiv = (pll_con0 >> PLL46XX_MDIV_SHIFT) & ((pll->type == pll_1460x) ?
519 PLL1460X_MDIV_MASK : PLL46XX_MDIV_MASK);
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900520 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
Naveen Krishna Ch0c23e2a2014-09-22 10:17:01 +0530525 shift = ((pll->type == pll_4600) || (pll->type == pll_1460x)) ? 16 : 10;
526
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900527 fvco *= (mdiv << shift) + kdiv;
528 do_div(fvco, (pdiv << sdiv));
529 fvco >>= shift;
530
531 return (unsigned long)fvco;
532}
533
Tomasz Figa5c896582013-08-26 19:09:07 +0200534static 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
547static 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__,
Stephen Boyd836ee0f2015-08-12 11:42:23 -0700559 drate, clk_hw_get_name(hw));
Tomasz Figa5c896582013-08-26 19:09:07 +0200560 return -EINVAL;
561 }
562
Matthew Leach4de10332016-06-08 19:30:56 +0100563 con0 = readl_relaxed(pll->con_reg);
564 con1 = readl_relaxed(pll->con_reg + 0x4);
Tomasz Figa5c896582013-08-26 19:09:07 +0200565
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;
Matthew Leach4de10332016-06-08 19:30:56 +0100570 writel_relaxed(con0, pll->con_reg);
Tomasz Figa5c896582013-08-26 19:09:07 +0200571
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. */
Naveen Krishna Ch0c23e2a2014-09-22 10:17:01 +0530582 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) |
Tomasz Figa5c896582013-08-26 19:09:07 +0200588 (PLL46XX_PDIV_MASK << PLL46XX_PDIV_SHIFT) |
589 (PLL46XX_SDIV_MASK << PLL46XX_SDIV_SHIFT) |
590 (PLL46XX_VSEL_MASK << PLL46XX_VSEL_SHIFT));
Naveen Krishna Ch0c23e2a2014-09-22 10:17:01 +0530591 con0 |= rate->vsel << PLL46XX_VSEL_SHIFT;
592 }
593
Tomasz Figa5c896582013-08-26 19:09:07 +0200594 con0 |= (rate->mdiv << PLL46XX_MDIV_SHIFT) |
595 (rate->pdiv << PLL46XX_PDIV_SHIFT) |
Naveen Krishna Ch0c23e2a2014-09-22 10:17:01 +0530596 (rate->sdiv << PLL46XX_SDIV_SHIFT);
Tomasz Figa5c896582013-08-26 19:09:07 +0200597
598 /* Set PLL K, MFR and MRR values. */
Matthew Leach4de10332016-06-08 19:30:56 +0100599 con1 = readl_relaxed(pll->con_reg + 0x4);
Tomasz Figa5c896582013-08-26 19:09:07 +0200600 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 */
Matthew Leach4de10332016-06-08 19:30:56 +0100608 writel_relaxed(lock, pll->lock_reg);
609 writel_relaxed(con0, pll->con_reg);
610 writel_relaxed(con1, pll->con_reg + 0x4);
Tomasz Figa5c896582013-08-26 19:09:07 +0200611
612 /* Wait for locking. */
613 start = ktime_get();
Matthew Leach4de10332016-06-08 19:30:56 +0100614 while (!(readl_relaxed(pll->con_reg) & PLL46XX_LOCKED)) {
Tomasz Figa5c896582013-08-26 19:09:07 +0200615 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",
Stephen Boyd836ee0f2015-08-12 11:42:23 -0700619 __func__, clk_hw_get_name(hw));
Tomasz Figa5c896582013-08-26 19:09:07 +0200620 return -EFAULT;
621 }
622
623 cpu_relax();
624 }
625
626 return 0;
627}
628
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900629static const struct clk_ops samsung_pll46xx_clk_ops = {
630 .recalc_rate = samsung_pll46xx_recalc_rate,
Tomasz Figa5c896582013-08-26 19:09:07 +0200631 .round_rate = samsung_pll_round_rate,
632 .set_rate = samsung_pll46xx_set_rate,
633};
634
635static const struct clk_ops samsung_pll46xx_clk_min_ops = {
636 .recalc_rate = samsung_pll46xx_recalc_rate,
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900637};
638
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900639/*
Tomasz Figaeb527122013-07-23 01:49:19 +0200640 * PLL6552 Clock Type
641 */
642
Tomasz Figaeb527122013-07-23 01:49:19 +0200643#define PLL6552_MDIV_MASK 0x3ff
644#define PLL6552_PDIV_MASK 0x3f
645#define PLL6552_SDIV_MASK 0x7
646#define PLL6552_MDIV_SHIFT 16
Heiko Stuebner06654ac2014-02-19 09:25:36 +0900647#define PLL6552_MDIV_SHIFT_2416 14
Tomasz Figaeb527122013-07-23 01:49:19 +0200648#define PLL6552_PDIV_SHIFT 8
Heiko Stuebner06654ac2014-02-19 09:25:36 +0900649#define PLL6552_PDIV_SHIFT_2416 5
Tomasz Figaeb527122013-07-23 01:49:19 +0200650#define PLL6552_SDIV_SHIFT 0
651
Tomasz Figaeb527122013-07-23 01:49:19 +0200652static unsigned long samsung_pll6552_recalc_rate(struct clk_hw *hw,
653 unsigned long parent_rate)
654{
Tomasz Figa40ef7232013-08-21 02:33:21 +0200655 struct samsung_clk_pll *pll = to_clk_pll(hw);
Tomasz Figaeb527122013-07-23 01:49:19 +0200656 u32 mdiv, pdiv, sdiv, pll_con;
657 u64 fvco = parent_rate;
658
Matthew Leach4de10332016-06-08 19:30:56 +0100659 pll_con = readl_relaxed(pll->con_reg);
Heiko Stuebner06654ac2014-02-19 09:25:36 +0900660 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 }
Tomasz Figaeb527122013-07-23 01:49:19 +0200667 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
675static const struct clk_ops samsung_pll6552_clk_ops = {
676 .recalc_rate = samsung_pll6552_recalc_rate,
677};
678
Tomasz Figaeb527122013-07-23 01:49:19 +0200679/*
680 * PLL6553 Clock Type
681 */
682
Tomasz Figaeb527122013-07-23 01:49:19 +0200683#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
Tomasz Figaeb527122013-07-23 01:49:19 +0200692static unsigned long samsung_pll6553_recalc_rate(struct clk_hw *hw,
693 unsigned long parent_rate)
694{
Tomasz Figa40ef7232013-08-21 02:33:21 +0200695 struct samsung_clk_pll *pll = to_clk_pll(hw);
Tomasz Figaeb527122013-07-23 01:49:19 +0200696 u32 mdiv, pdiv, sdiv, kdiv, pll_con0, pll_con1;
697 u64 fvco = parent_rate;
698
Matthew Leach4de10332016-06-08 19:30:56 +0100699 pll_con0 = readl_relaxed(pll->con_reg);
700 pll_con1 = readl_relaxed(pll->con_reg + 0x4);
Tomasz Figaeb527122013-07-23 01:49:19 +0200701 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
713static const struct clk_ops samsung_pll6553_clk_ops = {
714 .recalc_rate = samsung_pll6553_recalc_rate,
715};
716
Tomasz Figaeb527122013-07-23 01:49:19 +0200717/*
Heiko Stuebnerea5d6a82014-02-25 09:50:43 +0900718 * 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
730static 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
Matthew Leach4de10332016-06-08 19:30:56 +0100737 pll_con = readl_relaxed(pll->con_reg);
Heiko Stuebnerea5d6a82014-02-25 09:50:43 +0900738 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
748static 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
Matthew Leach4de10332016-06-08 19:30:56 +0100755 pll_con = readl_relaxed(pll->con_reg);
Heiko Stuebnerea5d6a82014-02-25 09:50:43 +0900756 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
766static 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__,
Stephen Boyd836ee0f2015-08-12 11:42:23 -0700777 drate, clk_hw_get_name(hw));
Heiko Stuebnerea5d6a82014-02-25 09:50:43 +0900778 return -EINVAL;
779 }
780
Matthew Leach4de10332016-06-08 19:30:56 +0100781 tmp = readl_relaxed(pll->con_reg);
Heiko Stuebnerea5d6a82014-02-25 09:50:43 +0900782
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);
Matthew Leach4de10332016-06-08 19:30:56 +0100790 writel_relaxed(tmp, pll->con_reg);
Heiko Stuebnerea5d6a82014-02-25 09:50:43 +0900791
792 /* Time to settle according to the manual */
793 udelay(300);
794
795 return 0;
796}
797
798static int samsung_s3c2410_pll_enable(struct clk_hw *hw, int bit, bool enable)
799{
800 struct samsung_clk_pll *pll = to_clk_pll(hw);
Matthew Leach4de10332016-06-08 19:30:56 +0100801 u32 pll_en = readl_relaxed(pll->lock_reg + PLLS3C2410_ENABLE_REG_OFFSET);
Heiko Stuebnerea5d6a82014-02-25 09:50:43 +0900802 u32 pll_en_orig = pll_en;
803
804 if (enable)
805 pll_en &= ~BIT(bit);
806 else
807 pll_en |= BIT(bit);
808
Matthew Leach4de10332016-06-08 19:30:56 +0100809 writel_relaxed(pll_en, pll->lock_reg + PLLS3C2410_ENABLE_REG_OFFSET);
Heiko Stuebnerea5d6a82014-02-25 09:50:43 +0900810
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
818static int samsung_s3c2410_mpll_enable(struct clk_hw *hw)
819{
820 return samsung_s3c2410_pll_enable(hw, 5, true);
821}
822
823static void samsung_s3c2410_mpll_disable(struct clk_hw *hw)
824{
825 samsung_s3c2410_pll_enable(hw, 5, false);
826}
827
828static int samsung_s3c2410_upll_enable(struct clk_hw *hw)
829{
830 return samsung_s3c2410_pll_enable(hw, 7, true);
831}
832
833static void samsung_s3c2410_upll_disable(struct clk_hw *hw)
834{
835 samsung_s3c2410_pll_enable(hw, 7, false);
836}
837
838static 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
844static 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
850static 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
856static 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
864static 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
872static 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/*
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900881 * 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
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900893static unsigned long samsung_pll2550x_recalc_rate(struct clk_hw *hw,
894 unsigned long parent_rate)
895{
Sylwester Nawrocki1d9aa642016-08-18 17:01:20 +0200896 struct samsung_clk_pll *pll = to_clk_pll(hw);
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900897 u32 r, p, m, s, pll_stat;
898 u64 fvco = parent_rate;
899
Sylwester Nawrocki1d9aa642016-08-18 17:01:20 +0200900 pll_stat = readl_relaxed(pll->con_reg);
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900901 r = (pll_stat >> PLL2550X_R_SHIFT) & PLL2550X_R_MASK;
902 if (!r)
903 return 0;
904 p = (pll_stat >> PLL2550X_P_SHIFT) & PLL2550X_P_MASK;
905 m = (pll_stat >> PLL2550X_M_SHIFT) & PLL2550X_M_MASK;
906 s = (pll_stat >> PLL2550X_S_SHIFT) & PLL2550X_S_MASK;
907
908 fvco *= m;
909 do_div(fvco, (p << s));
910
911 return (unsigned long)fvco;
912}
913
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900914static const struct clk_ops samsung_pll2550x_clk_ops = {
915 .recalc_rate = samsung_pll2550x_recalc_rate,
Thomas Abraham1c4c5fe2013-03-09 17:02:48 +0900916};
917
Pankaj Dubey84329842014-03-12 20:26:45 +0530918/*
919 * PLL2550xx Clock Type
920 */
921
922/* Maximum lock time can be 270 * PDIV cycles */
923#define PLL2550XX_LOCK_FACTOR 270
924
925#define PLL2550XX_M_MASK 0x3FF
926#define PLL2550XX_P_MASK 0x3F
927#define PLL2550XX_S_MASK 0x7
928#define PLL2550XX_LOCK_STAT_MASK 0x1
929#define PLL2550XX_M_SHIFT 9
930#define PLL2550XX_P_SHIFT 3
931#define PLL2550XX_S_SHIFT 0
932#define PLL2550XX_LOCK_STAT_SHIFT 21
933
934static unsigned long samsung_pll2550xx_recalc_rate(struct clk_hw *hw,
935 unsigned long parent_rate)
936{
937 struct samsung_clk_pll *pll = to_clk_pll(hw);
938 u32 mdiv, pdiv, sdiv, pll_con;
939 u64 fvco = parent_rate;
940
Matthew Leach4de10332016-06-08 19:30:56 +0100941 pll_con = readl_relaxed(pll->con_reg);
Pankaj Dubey84329842014-03-12 20:26:45 +0530942 mdiv = (pll_con >> PLL2550XX_M_SHIFT) & PLL2550XX_M_MASK;
943 pdiv = (pll_con >> PLL2550XX_P_SHIFT) & PLL2550XX_P_MASK;
944 sdiv = (pll_con >> PLL2550XX_S_SHIFT) & PLL2550XX_S_MASK;
945
946 fvco *= mdiv;
947 do_div(fvco, (pdiv << sdiv));
948
949 return (unsigned long)fvco;
950}
951
952static inline bool samsung_pll2550xx_mp_change(u32 mdiv, u32 pdiv, u32 pll_con)
953{
954 u32 old_mdiv, old_pdiv;
955
956 old_mdiv = (pll_con >> PLL2550XX_M_SHIFT) & PLL2550XX_M_MASK;
957 old_pdiv = (pll_con >> PLL2550XX_P_SHIFT) & PLL2550XX_P_MASK;
958
959 return mdiv != old_mdiv || pdiv != old_pdiv;
960}
961
962static int samsung_pll2550xx_set_rate(struct clk_hw *hw, unsigned long drate,
963 unsigned long prate)
964{
965 struct samsung_clk_pll *pll = to_clk_pll(hw);
966 const struct samsung_pll_rate_table *rate;
967 u32 tmp;
968
969 /* Get required rate settings from table */
970 rate = samsung_get_pll_settings(pll, drate);
971 if (!rate) {
972 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
Stephen Boyd836ee0f2015-08-12 11:42:23 -0700973 drate, clk_hw_get_name(hw));
Pankaj Dubey84329842014-03-12 20:26:45 +0530974 return -EINVAL;
975 }
976
Matthew Leach4de10332016-06-08 19:30:56 +0100977 tmp = readl_relaxed(pll->con_reg);
Pankaj Dubey84329842014-03-12 20:26:45 +0530978
979 if (!(samsung_pll2550xx_mp_change(rate->mdiv, rate->pdiv, tmp))) {
980 /* If only s change, change just s value only*/
981 tmp &= ~(PLL2550XX_S_MASK << PLL2550XX_S_SHIFT);
982 tmp |= rate->sdiv << PLL2550XX_S_SHIFT;
Matthew Leach4de10332016-06-08 19:30:56 +0100983 writel_relaxed(tmp, pll->con_reg);
Pankaj Dubey84329842014-03-12 20:26:45 +0530984
985 return 0;
986 }
987
988 /* Set PLL lock time. */
Matthew Leach4de10332016-06-08 19:30:56 +0100989 writel_relaxed(rate->pdiv * PLL2550XX_LOCK_FACTOR, pll->lock_reg);
Pankaj Dubey84329842014-03-12 20:26:45 +0530990
991 /* Change PLL PMS values */
992 tmp &= ~((PLL2550XX_M_MASK << PLL2550XX_M_SHIFT) |
993 (PLL2550XX_P_MASK << PLL2550XX_P_SHIFT) |
994 (PLL2550XX_S_MASK << PLL2550XX_S_SHIFT));
995 tmp |= (rate->mdiv << PLL2550XX_M_SHIFT) |
996 (rate->pdiv << PLL2550XX_P_SHIFT) |
997 (rate->sdiv << PLL2550XX_S_SHIFT);
Matthew Leach4de10332016-06-08 19:30:56 +0100998 writel_relaxed(tmp, pll->con_reg);
Pankaj Dubey84329842014-03-12 20:26:45 +0530999
1000 /* wait_lock_time */
1001 do {
1002 cpu_relax();
Matthew Leach4de10332016-06-08 19:30:56 +01001003 tmp = readl_relaxed(pll->con_reg);
Pankaj Dubey84329842014-03-12 20:26:45 +05301004 } while (!(tmp & (PLL2550XX_LOCK_STAT_MASK
1005 << PLL2550XX_LOCK_STAT_SHIFT)));
1006
1007 return 0;
1008}
1009
1010static const struct clk_ops samsung_pll2550xx_clk_ops = {
1011 .recalc_rate = samsung_pll2550xx_recalc_rate,
1012 .round_rate = samsung_pll_round_rate,
1013 .set_rate = samsung_pll2550xx_set_rate,
1014};
1015
1016static const struct clk_ops samsung_pll2550xx_clk_min_ops = {
1017 .recalc_rate = samsung_pll2550xx_recalc_rate,
1018};
1019
Rahul Sharmaeefe1192014-03-12 20:26:46 +05301020/*
Sylwester Nawrockibe95d2c2016-09-09 10:09:05 +02001021 * PLL2650x Clock Type
1022 */
1023
1024/* Maximum lock time can be 3000 * PDIV cycles */
1025#define PLL2650X_LOCK_FACTOR 3000
1026
1027#define PLL2650X_M_MASK 0x1ff
1028#define PLL2650X_P_MASK 0x3f
1029#define PLL2650X_S_MASK 0x7
1030#define PLL2650X_K_MASK 0xffff
1031#define PLL2650X_LOCK_STAT_MASK 0x1
1032#define PLL2650X_M_SHIFT 16
1033#define PLL2650X_P_SHIFT 8
1034#define PLL2650X_S_SHIFT 0
1035#define PLL2650X_K_SHIFT 0
1036#define PLL2650X_LOCK_STAT_SHIFT 29
1037#define PLL2650X_PLL_ENABLE_SHIFT 31
1038
1039static unsigned long samsung_pll2650x_recalc_rate(struct clk_hw *hw,
1040 unsigned long parent_rate)
1041{
1042 struct samsung_clk_pll *pll = to_clk_pll(hw);
1043 u64 fout = parent_rate;
1044 u32 mdiv, pdiv, sdiv, pll_con0, pll_con1;
1045 s16 kdiv;
1046
1047 pll_con0 = readl_relaxed(pll->con_reg);
1048 mdiv = (pll_con0 >> PLL2650X_M_SHIFT) & PLL2650X_M_MASK;
1049 pdiv = (pll_con0 >> PLL2650X_P_SHIFT) & PLL2650X_P_MASK;
1050 sdiv = (pll_con0 >> PLL2650X_S_SHIFT) & PLL2650X_S_MASK;
1051
1052 pll_con1 = readl_relaxed(pll->con_reg + 4);
1053 kdiv = (s16)((pll_con1 >> PLL2650X_K_SHIFT) & PLL2650X_K_MASK);
1054
1055 fout *= (mdiv << 16) + kdiv;
1056 do_div(fout, (pdiv << sdiv));
1057 fout >>= 16;
1058
1059 return (unsigned long)fout;
1060}
1061
1062static int samsung_pll2650x_set_rate(struct clk_hw *hw, unsigned long drate,
1063 unsigned long prate)
1064{
1065 struct samsung_clk_pll *pll = to_clk_pll(hw);
1066 const struct samsung_pll_rate_table *rate;
1067 u32 con0, con1;
1068
1069 /* Get required rate settings from table */
1070 rate = samsung_get_pll_settings(pll, drate);
1071 if (!rate) {
1072 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
1073 drate, clk_hw_get_name(hw));
1074 return -EINVAL;
1075 }
1076
1077 con0 = readl_relaxed(pll->con_reg);
1078 con1 = readl_relaxed(pll->con_reg + 4);
1079
1080 /* Set PLL lock time. */
1081 writel_relaxed(rate->pdiv * PLL2650X_LOCK_FACTOR, pll->lock_reg);
1082
1083 /* Change PLL PMS values */
1084 con0 &= ~((PLL2650X_M_MASK << PLL2650X_M_SHIFT) |
1085 (PLL2650X_P_MASK << PLL2650X_P_SHIFT) |
1086 (PLL2650X_S_MASK << PLL2650X_S_SHIFT));
1087 con0 |= (rate->mdiv << PLL2650X_M_SHIFT) |
1088 (rate->pdiv << PLL2650X_P_SHIFT) |
1089 (rate->sdiv << PLL2650X_S_SHIFT);
1090 con0 |= (1 << PLL2650X_PLL_ENABLE_SHIFT);
1091 writel_relaxed(con0, pll->con_reg);
1092
1093 con1 &= ~(PLL2650X_K_MASK << PLL2650X_K_SHIFT);
1094 con1 |= ((rate->kdiv & PLL2650X_K_MASK) << PLL2650X_K_SHIFT);
1095 writel_relaxed(con1, pll->con_reg + 4);
1096
1097 do {
1098 cpu_relax();
1099 con0 = readl_relaxed(pll->con_reg);
1100 } while (!(con0 & (PLL2650X_LOCK_STAT_MASK
1101 << PLL2650X_LOCK_STAT_SHIFT)));
1102
1103 return 0;
1104}
1105
1106static const struct clk_ops samsung_pll2650x_clk_ops = {
1107 .recalc_rate = samsung_pll2650x_recalc_rate,
1108 .round_rate = samsung_pll_round_rate,
1109 .set_rate = samsung_pll2650x_set_rate,
1110};
1111
1112static const struct clk_ops samsung_pll2650x_clk_min_ops = {
1113 .recalc_rate = samsung_pll2650x_recalc_rate,
1114};
1115
1116/*
Rahul Sharmaeefe1192014-03-12 20:26:46 +05301117 * PLL2650XX Clock Type
1118 */
1119
1120/* Maximum lock time can be 3000 * PDIV cycles */
1121#define PLL2650XX_LOCK_FACTOR 3000
1122
1123#define PLL2650XX_MDIV_SHIFT 9
1124#define PLL2650XX_PDIV_SHIFT 3
1125#define PLL2650XX_SDIV_SHIFT 0
1126#define PLL2650XX_KDIV_SHIFT 0
1127#define PLL2650XX_MDIV_MASK 0x1ff
1128#define PLL2650XX_PDIV_MASK 0x3f
1129#define PLL2650XX_SDIV_MASK 0x7
1130#define PLL2650XX_KDIV_MASK 0xffff
1131#define PLL2650XX_PLL_ENABLE_SHIFT 23
1132#define PLL2650XX_PLL_LOCKTIME_SHIFT 21
1133#define PLL2650XX_PLL_FOUTMASK_SHIFT 31
1134
1135static unsigned long samsung_pll2650xx_recalc_rate(struct clk_hw *hw,
1136 unsigned long parent_rate)
1137{
1138 struct samsung_clk_pll *pll = to_clk_pll(hw);
1139 u32 mdiv, pdiv, sdiv, pll_con0, pll_con2;
1140 s16 kdiv;
1141 u64 fvco = parent_rate;
1142
Matthew Leach4de10332016-06-08 19:30:56 +01001143 pll_con0 = readl_relaxed(pll->con_reg);
1144 pll_con2 = readl_relaxed(pll->con_reg + 8);
Rahul Sharmaeefe1192014-03-12 20:26:46 +05301145 mdiv = (pll_con0 >> PLL2650XX_MDIV_SHIFT) & PLL2650XX_MDIV_MASK;
1146 pdiv = (pll_con0 >> PLL2650XX_PDIV_SHIFT) & PLL2650XX_PDIV_MASK;
1147 sdiv = (pll_con0 >> PLL2650XX_SDIV_SHIFT) & PLL2650XX_SDIV_MASK;
1148 kdiv = (s16)(pll_con2 & PLL2650XX_KDIV_MASK);
1149
1150 fvco *= (mdiv << 16) + kdiv;
1151 do_div(fvco, (pdiv << sdiv));
1152 fvco >>= 16;
1153
1154 return (unsigned long)fvco;
1155}
1156
1157static int samsung_pll2650xx_set_rate(struct clk_hw *hw, unsigned long drate,
1158 unsigned long parent_rate)
1159{
1160 struct samsung_clk_pll *pll = to_clk_pll(hw);
1161 u32 tmp, pll_con0, pll_con2;
1162 const struct samsung_pll_rate_table *rate;
1163
1164 rate = samsung_get_pll_settings(pll, drate);
1165 if (!rate) {
1166 pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
Stephen Boyd836ee0f2015-08-12 11:42:23 -07001167 drate, clk_hw_get_name(hw));
Rahul Sharmaeefe1192014-03-12 20:26:46 +05301168 return -EINVAL;
1169 }
1170
Matthew Leach4de10332016-06-08 19:30:56 +01001171 pll_con0 = readl_relaxed(pll->con_reg);
1172 pll_con2 = readl_relaxed(pll->con_reg + 8);
Rahul Sharmaeefe1192014-03-12 20:26:46 +05301173
1174 /* Change PLL PMS values */
1175 pll_con0 &= ~(PLL2650XX_MDIV_MASK << PLL2650XX_MDIV_SHIFT |
1176 PLL2650XX_PDIV_MASK << PLL2650XX_PDIV_SHIFT |
1177 PLL2650XX_SDIV_MASK << PLL2650XX_SDIV_SHIFT);
1178 pll_con0 |= rate->mdiv << PLL2650XX_MDIV_SHIFT;
1179 pll_con0 |= rate->pdiv << PLL2650XX_PDIV_SHIFT;
1180 pll_con0 |= rate->sdiv << PLL2650XX_SDIV_SHIFT;
1181 pll_con0 |= 1 << PLL2650XX_PLL_ENABLE_SHIFT;
1182 pll_con0 |= 1 << PLL2650XX_PLL_FOUTMASK_SHIFT;
1183
1184 pll_con2 &= ~(PLL2650XX_KDIV_MASK << PLL2650XX_KDIV_SHIFT);
1185 pll_con2 |= ((~(rate->kdiv) + 1) & PLL2650XX_KDIV_MASK)
1186 << PLL2650XX_KDIV_SHIFT;
1187
1188 /* Set PLL lock time. */
Matthew Leach4de10332016-06-08 19:30:56 +01001189 writel_relaxed(PLL2650XX_LOCK_FACTOR * rate->pdiv, pll->lock_reg);
Rahul Sharmaeefe1192014-03-12 20:26:46 +05301190
Matthew Leach4de10332016-06-08 19:30:56 +01001191 writel_relaxed(pll_con0, pll->con_reg);
1192 writel_relaxed(pll_con2, pll->con_reg + 8);
Rahul Sharmaeefe1192014-03-12 20:26:46 +05301193
1194 do {
Matthew Leach4de10332016-06-08 19:30:56 +01001195 tmp = readl_relaxed(pll->con_reg);
Rahul Sharmaeefe1192014-03-12 20:26:46 +05301196 } while (!(tmp & (0x1 << PLL2650XX_PLL_LOCKTIME_SHIFT)));
1197
1198 return 0;
1199}
1200
1201static const struct clk_ops samsung_pll2650xx_clk_ops = {
1202 .recalc_rate = samsung_pll2650xx_recalc_rate,
1203 .set_rate = samsung_pll2650xx_set_rate,
1204 .round_rate = samsung_pll_round_rate,
1205};
1206
1207static const struct clk_ops samsung_pll2650xx_clk_min_ops = {
1208 .recalc_rate = samsung_pll2650xx_recalc_rate,
1209};
1210
Rahul Sharma976face2014-03-12 20:26:44 +05301211static void __init _samsung_clk_register_pll(struct samsung_clk_provider *ctx,
Uwe Kleine-König4a1caed2015-05-28 10:45:51 +02001212 const struct samsung_pll_clock *pll_clk,
Rahul Sharma976face2014-03-12 20:26:44 +05301213 void __iomem *base)
Yadwinder Singh Brar07dc76f2013-06-11 15:01:07 +05301214{
1215 struct samsung_clk_pll *pll;
1216 struct clk *clk;
1217 struct clk_init_data init;
Yadwinder Singh Brar3ff6e0d2013-06-11 15:01:12 +05301218 int ret, len;
Yadwinder Singh Brar07dc76f2013-06-11 15:01:07 +05301219
1220 pll = kzalloc(sizeof(*pll), GFP_KERNEL);
1221 if (!pll) {
1222 pr_err("%s: could not allocate pll clk %s\n",
1223 __func__, pll_clk->name);
1224 return;
1225 }
1226
1227 init.name = pll_clk->name;
1228 init.flags = pll_clk->flags;
1229 init.parent_names = &pll_clk->parent_name;
1230 init.num_parents = 1;
1231
Yadwinder Singh Brar3ff6e0d2013-06-11 15:01:12 +05301232 if (pll_clk->rate_table) {
1233 /* find count of rates in rate_table */
1234 for (len = 0; pll_clk->rate_table[len].rate != 0; )
1235 len++;
1236
1237 pll->rate_count = len;
1238 pll->rate_table = kmemdup(pll_clk->rate_table,
1239 pll->rate_count *
1240 sizeof(struct samsung_pll_rate_table),
1241 GFP_KERNEL);
1242 WARN(!pll->rate_table,
1243 "%s: could not allocate rate table for %s\n",
1244 __func__, pll_clk->name);
1245 }
1246
Yadwinder Singh Brar07dc76f2013-06-11 15:01:07 +05301247 switch (pll_clk->type) {
Heiko Stuebnera951b1d2014-02-19 09:25:41 +09001248 case pll_2126:
1249 init.ops = &samsung_pll2126_clk_ops;
1250 break;
1251 case pll_3000:
1252 init.ops = &samsung_pll3000_clk_ops;
1253 break;
Yadwinder Singh Brar07dc76f2013-06-11 15:01:07 +05301254 /* clk_ops for 35xx and 2550 are similar */
1255 case pll_35xx:
1256 case pll_2550:
Naveen Krishna Ch0c23e2a2014-09-22 10:17:01 +05301257 case pll_1450x:
1258 case pll_1451x:
1259 case pll_1452x:
Yadwinder Singh Brardfa89312013-06-11 15:01:13 +05301260 if (!pll->rate_table)
1261 init.ops = &samsung_pll35xx_clk_min_ops;
1262 else
1263 init.ops = &samsung_pll35xx_clk_ops;
Yadwinder Singh Brar07dc76f2013-06-11 15:01:07 +05301264 break;
Tomasz Figa52b06012013-08-26 19:09:04 +02001265 case pll_4500:
Tomasz Figab4054ac2013-08-26 19:09:05 +02001266 init.ops = &samsung_pll45xx_clk_min_ops;
1267 break;
Tomasz Figa52b06012013-08-26 19:09:04 +02001268 case pll_4502:
1269 case pll_4508:
Tomasz Figab4054ac2013-08-26 19:09:05 +02001270 if (!pll->rate_table)
1271 init.ops = &samsung_pll45xx_clk_min_ops;
1272 else
1273 init.ops = &samsung_pll45xx_clk_ops;
Tomasz Figa52b06012013-08-26 19:09:04 +02001274 break;
Yadwinder Singh Brar07dc76f2013-06-11 15:01:07 +05301275 /* clk_ops for 36xx and 2650 are similar */
1276 case pll_36xx:
1277 case pll_2650:
Vikas Sajjanbb4278d2013-06-11 15:01:14 +05301278 if (!pll->rate_table)
1279 init.ops = &samsung_pll36xx_clk_min_ops;
1280 else
1281 init.ops = &samsung_pll36xx_clk_ops;
Yadwinder Singh Brar07dc76f2013-06-11 15:01:07 +05301282 break;
Tomasz Figa40ef7232013-08-21 02:33:21 +02001283 case pll_6552:
Heiko Stuebner06654ac2014-02-19 09:25:36 +09001284 case pll_6552_s3c2416:
Tomasz Figa40ef7232013-08-21 02:33:21 +02001285 init.ops = &samsung_pll6552_clk_ops;
1286 break;
1287 case pll_6553:
1288 init.ops = &samsung_pll6553_clk_ops;
1289 break;
Tomasz Figac50d11f2013-08-26 19:09:06 +02001290 case pll_4600:
1291 case pll_4650:
1292 case pll_4650c:
Naveen Krishna Ch0c23e2a2014-09-22 10:17:01 +05301293 case pll_1460x:
Tomasz Figa5c896582013-08-26 19:09:07 +02001294 if (!pll->rate_table)
1295 init.ops = &samsung_pll46xx_clk_min_ops;
1296 else
1297 init.ops = &samsung_pll46xx_clk_ops;
Tomasz Figac50d11f2013-08-26 19:09:06 +02001298 break;
Heiko Stuebnerea5d6a82014-02-25 09:50:43 +09001299 case pll_s3c2410_mpll:
1300 if (!pll->rate_table)
1301 init.ops = &samsung_s3c2410_mpll_clk_min_ops;
1302 else
1303 init.ops = &samsung_s3c2410_mpll_clk_ops;
1304 break;
1305 case pll_s3c2410_upll:
1306 if (!pll->rate_table)
1307 init.ops = &samsung_s3c2410_upll_clk_min_ops;
1308 else
1309 init.ops = &samsung_s3c2410_upll_clk_ops;
1310 break;
1311 case pll_s3c2440_mpll:
1312 if (!pll->rate_table)
1313 init.ops = &samsung_s3c2440_mpll_clk_min_ops;
1314 else
1315 init.ops = &samsung_s3c2440_mpll_clk_ops;
1316 break;
Sylwester Nawrocki1d9aa642016-08-18 17:01:20 +02001317 case pll_2550x:
1318 init.ops = &samsung_pll2550x_clk_ops;
1319 break;
Pankaj Dubey84329842014-03-12 20:26:45 +05301320 case pll_2550xx:
1321 if (!pll->rate_table)
1322 init.ops = &samsung_pll2550xx_clk_min_ops;
1323 else
1324 init.ops = &samsung_pll2550xx_clk_ops;
1325 break;
Sylwester Nawrockibe95d2c2016-09-09 10:09:05 +02001326 case pll_2650x:
1327 if (!pll->rate_table)
1328 init.ops = &samsung_pll2650x_clk_min_ops;
1329 else
1330 init.ops = &samsung_pll2650x_clk_ops;
1331 break;
Rahul Sharmaeefe1192014-03-12 20:26:46 +05301332 case pll_2650xx:
1333 if (!pll->rate_table)
1334 init.ops = &samsung_pll2650xx_clk_min_ops;
1335 else
1336 init.ops = &samsung_pll2650xx_clk_ops;
1337 break;
Yadwinder Singh Brar07dc76f2013-06-11 15:01:07 +05301338 default:
1339 pr_warn("%s: Unknown pll type for pll clk %s\n",
1340 __func__, pll_clk->name);
1341 }
1342
1343 pll->hw.init = &init;
1344 pll->type = pll_clk->type;
1345 pll->lock_reg = base + pll_clk->lock_offset;
1346 pll->con_reg = base + pll_clk->con_offset;
1347
1348 clk = clk_register(NULL, &pll->hw);
1349 if (IS_ERR(clk)) {
1350 pr_err("%s: failed to register pll clock %s : %ld\n",
1351 __func__, pll_clk->name, PTR_ERR(clk));
1352 kfree(pll);
1353 return;
1354 }
1355
Rahul Sharma976face2014-03-12 20:26:44 +05301356 samsung_clk_add_lookup(ctx, clk, pll_clk->id);
Yadwinder Singh Brar07dc76f2013-06-11 15:01:07 +05301357
1358 if (!pll_clk->alias)
1359 return;
1360
1361 ret = clk_register_clkdev(clk, pll_clk->alias, pll_clk->dev_name);
1362 if (ret)
1363 pr_err("%s: failed to register lookup for %s : %d",
1364 __func__, pll_clk->name, ret);
1365}
1366
Rahul Sharma976face2014-03-12 20:26:44 +05301367void __init samsung_clk_register_pll(struct samsung_clk_provider *ctx,
Uwe Kleine-König4a1caed2015-05-28 10:45:51 +02001368 const struct samsung_pll_clock *pll_list,
Rahul Sharma976face2014-03-12 20:26:44 +05301369 unsigned int nr_pll, void __iomem *base)
Yadwinder Singh Brar07dc76f2013-06-11 15:01:07 +05301370{
1371 int cnt;
1372
1373 for (cnt = 0; cnt < nr_pll; cnt++)
Rahul Sharma976face2014-03-12 20:26:44 +05301374 _samsung_clk_register_pll(ctx, &pll_list[cnt], base);
Yadwinder Singh Brar07dc76f2013-06-11 15:01:07 +05301375}