xref: /openbmc/linux/drivers/clk/samsung/clk-pll.c (revision be95d2c7)
1 /*
2  * Copyright (c) 2013 Samsung Electronics Co., Ltd.
3  * Copyright (c) 2013 Linaro Ltd.
4  *
5  * This program is free software; you can redistribute it and/or modify
6  * it under the terms of the GNU General Public License version 2 as
7  * published by the Free Software Foundation.
8  *
9  * This file contains the utility functions to register the pll clocks.
10 */
11 
12 #include <linux/errno.h>
13 #include <linux/hrtimer.h>
14 #include <linux/delay.h>
15 #include <linux/slab.h>
16 #include <linux/clkdev.h>
17 #include "clk.h"
18 #include "clk-pll.h"
19 
20 #define PLL_TIMEOUT_MS		10
21 
22 struct samsung_clk_pll {
23 	struct clk_hw		hw;
24 	void __iomem		*lock_reg;
25 	void __iomem		*con_reg;
26 	enum samsung_pll_type	type;
27 	unsigned int		rate_count;
28 	const struct samsung_pll_rate_table *rate_table;
29 };
30 
31 #define to_clk_pll(_hw) container_of(_hw, struct samsung_clk_pll, hw)
32 
33 static const struct samsung_pll_rate_table *samsung_get_pll_settings(
34 				struct samsung_clk_pll *pll, unsigned long rate)
35 {
36 	const struct samsung_pll_rate_table  *rate_table = pll->rate_table;
37 	int i;
38 
39 	for (i = 0; i < pll->rate_count; i++) {
40 		if (rate == rate_table[i].rate)
41 			return &rate_table[i];
42 	}
43 
44 	return NULL;
45 }
46 
47 static long samsung_pll_round_rate(struct clk_hw *hw,
48 			unsigned long drate, unsigned long *prate)
49 {
50 	struct samsung_clk_pll *pll = to_clk_pll(hw);
51 	const struct samsung_pll_rate_table *rate_table = pll->rate_table;
52 	int i;
53 
54 	/* Assumming rate_table is in descending order */
55 	for (i = 0; i < pll->rate_count; i++) {
56 		if (drate >= rate_table[i].rate)
57 			return rate_table[i].rate;
58 	}
59 
60 	/* return minimum supported value */
61 	return rate_table[i - 1].rate;
62 }
63 
64 /*
65  * PLL2126 Clock Type
66  */
67 
68 #define PLL2126_MDIV_MASK	(0xff)
69 #define PLL2126_PDIV_MASK	(0x3f)
70 #define PLL2126_SDIV_MASK	(0x3)
71 #define PLL2126_MDIV_SHIFT	(16)
72 #define PLL2126_PDIV_SHIFT	(8)
73 #define PLL2126_SDIV_SHIFT	(0)
74 
75 static unsigned long samsung_pll2126_recalc_rate(struct clk_hw *hw,
76 				unsigned long parent_rate)
77 {
78 	struct samsung_clk_pll *pll = to_clk_pll(hw);
79 	u32 pll_con, mdiv, pdiv, sdiv;
80 	u64 fvco = parent_rate;
81 
82 	pll_con = readl_relaxed(pll->con_reg);
83 	mdiv = (pll_con >> PLL2126_MDIV_SHIFT) & PLL2126_MDIV_MASK;
84 	pdiv = (pll_con >> PLL2126_PDIV_SHIFT) & PLL2126_PDIV_MASK;
85 	sdiv = (pll_con >> PLL2126_SDIV_SHIFT) & PLL2126_SDIV_MASK;
86 
87 	fvco *= (mdiv + 8);
88 	do_div(fvco, (pdiv + 2) << sdiv);
89 
90 	return (unsigned long)fvco;
91 }
92 
93 static const struct clk_ops samsung_pll2126_clk_ops = {
94 	.recalc_rate = samsung_pll2126_recalc_rate,
95 };
96 
97 /*
98  * PLL3000 Clock Type
99  */
100 
101 #define PLL3000_MDIV_MASK	(0xff)
102 #define PLL3000_PDIV_MASK	(0x3)
103 #define PLL3000_SDIV_MASK	(0x3)
104 #define PLL3000_MDIV_SHIFT	(16)
105 #define PLL3000_PDIV_SHIFT	(8)
106 #define PLL3000_SDIV_SHIFT	(0)
107 
108 static unsigned long samsung_pll3000_recalc_rate(struct clk_hw *hw,
109 				unsigned long parent_rate)
110 {
111 	struct samsung_clk_pll *pll = to_clk_pll(hw);
112 	u32 pll_con, mdiv, pdiv, sdiv;
113 	u64 fvco = parent_rate;
114 
115 	pll_con = readl_relaxed(pll->con_reg);
116 	mdiv = (pll_con >> PLL3000_MDIV_SHIFT) & PLL3000_MDIV_MASK;
117 	pdiv = (pll_con >> PLL3000_PDIV_SHIFT) & PLL3000_PDIV_MASK;
118 	sdiv = (pll_con >> PLL3000_SDIV_SHIFT) & PLL3000_SDIV_MASK;
119 
120 	fvco *= (2 * (mdiv + 8));
121 	do_div(fvco, pdiv << sdiv);
122 
123 	return (unsigned long)fvco;
124 }
125 
126 static const struct clk_ops samsung_pll3000_clk_ops = {
127 	.recalc_rate = samsung_pll3000_recalc_rate,
128 };
129 
130 /*
131  * PLL35xx Clock Type
132  */
133 /* Maximum lock time can be 270 * PDIV cycles */
134 #define PLL35XX_LOCK_FACTOR	(270)
135 
136 #define PLL35XX_MDIV_MASK       (0x3FF)
137 #define PLL35XX_PDIV_MASK       (0x3F)
138 #define PLL35XX_SDIV_MASK       (0x7)
139 #define PLL35XX_LOCK_STAT_MASK	(0x1)
140 #define PLL35XX_MDIV_SHIFT      (16)
141 #define PLL35XX_PDIV_SHIFT      (8)
142 #define PLL35XX_SDIV_SHIFT      (0)
143 #define PLL35XX_LOCK_STAT_SHIFT	(29)
144 
145 static unsigned long samsung_pll35xx_recalc_rate(struct clk_hw *hw,
146 				unsigned long parent_rate)
147 {
148 	struct samsung_clk_pll *pll = to_clk_pll(hw);
149 	u32 mdiv, pdiv, sdiv, pll_con;
150 	u64 fvco = parent_rate;
151 
152 	pll_con = readl_relaxed(pll->con_reg);
153 	mdiv = (pll_con >> PLL35XX_MDIV_SHIFT) & PLL35XX_MDIV_MASK;
154 	pdiv = (pll_con >> PLL35XX_PDIV_SHIFT) & PLL35XX_PDIV_MASK;
155 	sdiv = (pll_con >> PLL35XX_SDIV_SHIFT) & PLL35XX_SDIV_MASK;
156 
157 	fvco *= mdiv;
158 	do_div(fvco, (pdiv << sdiv));
159 
160 	return (unsigned long)fvco;
161 }
162 
163 static inline bool samsung_pll35xx_mp_change(
164 		const struct samsung_pll_rate_table *rate, u32 pll_con)
165 {
166 	u32 old_mdiv, old_pdiv;
167 
168 	old_mdiv = (pll_con >> PLL35XX_MDIV_SHIFT) & PLL35XX_MDIV_MASK;
169 	old_pdiv = (pll_con >> PLL35XX_PDIV_SHIFT) & PLL35XX_PDIV_MASK;
170 
171 	return (rate->mdiv != old_mdiv || rate->pdiv != old_pdiv);
172 }
173 
174 static int samsung_pll35xx_set_rate(struct clk_hw *hw, unsigned long drate,
175 					unsigned long prate)
176 {
177 	struct samsung_clk_pll *pll = to_clk_pll(hw);
178 	const struct samsung_pll_rate_table *rate;
179 	u32 tmp;
180 
181 	/* Get required rate settings from table */
182 	rate = samsung_get_pll_settings(pll, drate);
183 	if (!rate) {
184 		pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
185 			drate, clk_hw_get_name(hw));
186 		return -EINVAL;
187 	}
188 
189 	tmp = readl_relaxed(pll->con_reg);
190 
191 	if (!(samsung_pll35xx_mp_change(rate, tmp))) {
192 		/* If only s change, change just s value only*/
193 		tmp &= ~(PLL35XX_SDIV_MASK << PLL35XX_SDIV_SHIFT);
194 		tmp |= rate->sdiv << PLL35XX_SDIV_SHIFT;
195 		writel_relaxed(tmp, pll->con_reg);
196 
197 		return 0;
198 	}
199 
200 	/* Set PLL lock time. */
201 	writel_relaxed(rate->pdiv * PLL35XX_LOCK_FACTOR,
202 			pll->lock_reg);
203 
204 	/* Change PLL PMS values */
205 	tmp &= ~((PLL35XX_MDIV_MASK << PLL35XX_MDIV_SHIFT) |
206 			(PLL35XX_PDIV_MASK << PLL35XX_PDIV_SHIFT) |
207 			(PLL35XX_SDIV_MASK << PLL35XX_SDIV_SHIFT));
208 	tmp |= (rate->mdiv << PLL35XX_MDIV_SHIFT) |
209 			(rate->pdiv << PLL35XX_PDIV_SHIFT) |
210 			(rate->sdiv << PLL35XX_SDIV_SHIFT);
211 	writel_relaxed(tmp, pll->con_reg);
212 
213 	/* wait_lock_time */
214 	do {
215 		cpu_relax();
216 		tmp = readl_relaxed(pll->con_reg);
217 	} while (!(tmp & (PLL35XX_LOCK_STAT_MASK
218 				<< PLL35XX_LOCK_STAT_SHIFT)));
219 	return 0;
220 }
221 
222 static const struct clk_ops samsung_pll35xx_clk_ops = {
223 	.recalc_rate = samsung_pll35xx_recalc_rate,
224 	.round_rate = samsung_pll_round_rate,
225 	.set_rate = samsung_pll35xx_set_rate,
226 };
227 
228 static const struct clk_ops samsung_pll35xx_clk_min_ops = {
229 	.recalc_rate = samsung_pll35xx_recalc_rate,
230 };
231 
232 /*
233  * PLL36xx Clock Type
234  */
235 /* Maximum lock time can be 3000 * PDIV cycles */
236 #define PLL36XX_LOCK_FACTOR    (3000)
237 
238 #define PLL36XX_KDIV_MASK	(0xFFFF)
239 #define PLL36XX_MDIV_MASK	(0x1FF)
240 #define PLL36XX_PDIV_MASK	(0x3F)
241 #define PLL36XX_SDIV_MASK	(0x7)
242 #define PLL36XX_MDIV_SHIFT	(16)
243 #define PLL36XX_PDIV_SHIFT	(8)
244 #define PLL36XX_SDIV_SHIFT	(0)
245 #define PLL36XX_KDIV_SHIFT	(0)
246 #define PLL36XX_LOCK_STAT_SHIFT	(29)
247 
248 static unsigned long samsung_pll36xx_recalc_rate(struct clk_hw *hw,
249 				unsigned long parent_rate)
250 {
251 	struct samsung_clk_pll *pll = to_clk_pll(hw);
252 	u32 mdiv, pdiv, sdiv, pll_con0, pll_con1;
253 	s16 kdiv;
254 	u64 fvco = parent_rate;
255 
256 	pll_con0 = readl_relaxed(pll->con_reg);
257 	pll_con1 = readl_relaxed(pll->con_reg + 4);
258 	mdiv = (pll_con0 >> PLL36XX_MDIV_SHIFT) & PLL36XX_MDIV_MASK;
259 	pdiv = (pll_con0 >> PLL36XX_PDIV_SHIFT) & PLL36XX_PDIV_MASK;
260 	sdiv = (pll_con0 >> PLL36XX_SDIV_SHIFT) & PLL36XX_SDIV_MASK;
261 	kdiv = (s16)(pll_con1 & PLL36XX_KDIV_MASK);
262 
263 	fvco *= (mdiv << 16) + kdiv;
264 	do_div(fvco, (pdiv << sdiv));
265 	fvco >>= 16;
266 
267 	return (unsigned long)fvco;
268 }
269 
270 static inline bool samsung_pll36xx_mpk_change(
271 	const struct samsung_pll_rate_table *rate, u32 pll_con0, u32 pll_con1)
272 {
273 	u32 old_mdiv, old_pdiv, old_kdiv;
274 
275 	old_mdiv = (pll_con0 >> PLL36XX_MDIV_SHIFT) & PLL36XX_MDIV_MASK;
276 	old_pdiv = (pll_con0 >> PLL36XX_PDIV_SHIFT) & PLL36XX_PDIV_MASK;
277 	old_kdiv = (pll_con1 >> PLL36XX_KDIV_SHIFT) & PLL36XX_KDIV_MASK;
278 
279 	return (rate->mdiv != old_mdiv || rate->pdiv != old_pdiv ||
280 		rate->kdiv != old_kdiv);
281 }
282 
283 static int samsung_pll36xx_set_rate(struct clk_hw *hw, unsigned long drate,
284 					unsigned long parent_rate)
285 {
286 	struct samsung_clk_pll *pll = to_clk_pll(hw);
287 	u32 tmp, pll_con0, pll_con1;
288 	const struct samsung_pll_rate_table *rate;
289 
290 	rate = samsung_get_pll_settings(pll, drate);
291 	if (!rate) {
292 		pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
293 			drate, clk_hw_get_name(hw));
294 		return -EINVAL;
295 	}
296 
297 	pll_con0 = readl_relaxed(pll->con_reg);
298 	pll_con1 = readl_relaxed(pll->con_reg + 4);
299 
300 	if (!(samsung_pll36xx_mpk_change(rate, pll_con0, pll_con1))) {
301 		/* If only s change, change just s value only*/
302 		pll_con0 &= ~(PLL36XX_SDIV_MASK << PLL36XX_SDIV_SHIFT);
303 		pll_con0 |= (rate->sdiv << PLL36XX_SDIV_SHIFT);
304 		writel_relaxed(pll_con0, pll->con_reg);
305 
306 		return 0;
307 	}
308 
309 	/* Set PLL lock time. */
310 	writel_relaxed(rate->pdiv * PLL36XX_LOCK_FACTOR, pll->lock_reg);
311 
312 	 /* Change PLL PMS values */
313 	pll_con0 &= ~((PLL36XX_MDIV_MASK << PLL36XX_MDIV_SHIFT) |
314 			(PLL36XX_PDIV_MASK << PLL36XX_PDIV_SHIFT) |
315 			(PLL36XX_SDIV_MASK << PLL36XX_SDIV_SHIFT));
316 	pll_con0 |= (rate->mdiv << PLL36XX_MDIV_SHIFT) |
317 			(rate->pdiv << PLL36XX_PDIV_SHIFT) |
318 			(rate->sdiv << PLL36XX_SDIV_SHIFT);
319 	writel_relaxed(pll_con0, pll->con_reg);
320 
321 	pll_con1 &= ~(PLL36XX_KDIV_MASK << PLL36XX_KDIV_SHIFT);
322 	pll_con1 |= rate->kdiv << PLL36XX_KDIV_SHIFT;
323 	writel_relaxed(pll_con1, pll->con_reg + 4);
324 
325 	/* wait_lock_time */
326 	do {
327 		cpu_relax();
328 		tmp = readl_relaxed(pll->con_reg);
329 	} while (!(tmp & (1 << PLL36XX_LOCK_STAT_SHIFT)));
330 
331 	return 0;
332 }
333 
334 static const struct clk_ops samsung_pll36xx_clk_ops = {
335 	.recalc_rate = samsung_pll36xx_recalc_rate,
336 	.set_rate = samsung_pll36xx_set_rate,
337 	.round_rate = samsung_pll_round_rate,
338 };
339 
340 static const struct clk_ops samsung_pll36xx_clk_min_ops = {
341 	.recalc_rate = samsung_pll36xx_recalc_rate,
342 };
343 
344 /*
345  * PLL45xx Clock Type
346  */
347 #define PLL4502_LOCK_FACTOR	400
348 #define PLL4508_LOCK_FACTOR	240
349 
350 #define PLL45XX_MDIV_MASK	(0x3FF)
351 #define PLL45XX_PDIV_MASK	(0x3F)
352 #define PLL45XX_SDIV_MASK	(0x7)
353 #define PLL45XX_AFC_MASK	(0x1F)
354 #define PLL45XX_MDIV_SHIFT	(16)
355 #define PLL45XX_PDIV_SHIFT	(8)
356 #define PLL45XX_SDIV_SHIFT	(0)
357 #define PLL45XX_AFC_SHIFT	(0)
358 
359 #define PLL45XX_ENABLE		BIT(31)
360 #define PLL45XX_LOCKED		BIT(29)
361 
362 static unsigned long samsung_pll45xx_recalc_rate(struct clk_hw *hw,
363 				unsigned long parent_rate)
364 {
365 	struct samsung_clk_pll *pll = to_clk_pll(hw);
366 	u32 mdiv, pdiv, sdiv, pll_con;
367 	u64 fvco = parent_rate;
368 
369 	pll_con = readl_relaxed(pll->con_reg);
370 	mdiv = (pll_con >> PLL45XX_MDIV_SHIFT) & PLL45XX_MDIV_MASK;
371 	pdiv = (pll_con >> PLL45XX_PDIV_SHIFT) & PLL45XX_PDIV_MASK;
372 	sdiv = (pll_con >> PLL45XX_SDIV_SHIFT) & PLL45XX_SDIV_MASK;
373 
374 	if (pll->type == pll_4508)
375 		sdiv = sdiv - 1;
376 
377 	fvco *= mdiv;
378 	do_div(fvco, (pdiv << sdiv));
379 
380 	return (unsigned long)fvco;
381 }
382 
383 static bool samsung_pll45xx_mp_change(u32 pll_con0, u32 pll_con1,
384 				const struct samsung_pll_rate_table *rate)
385 {
386 	u32 old_mdiv, old_pdiv, old_afc;
387 
388 	old_mdiv = (pll_con0 >> PLL45XX_MDIV_SHIFT) & PLL45XX_MDIV_MASK;
389 	old_pdiv = (pll_con0 >> PLL45XX_PDIV_SHIFT) & PLL45XX_PDIV_MASK;
390 	old_afc = (pll_con1 >> PLL45XX_AFC_SHIFT) & PLL45XX_AFC_MASK;
391 
392 	return (old_mdiv != rate->mdiv || old_pdiv != rate->pdiv
393 		|| old_afc != rate->afc);
394 }
395 
396 static int samsung_pll45xx_set_rate(struct clk_hw *hw, unsigned long drate,
397 					unsigned long prate)
398 {
399 	struct samsung_clk_pll *pll = to_clk_pll(hw);
400 	const struct samsung_pll_rate_table *rate;
401 	u32 con0, con1;
402 	ktime_t start;
403 
404 	/* Get required rate settings from table */
405 	rate = samsung_get_pll_settings(pll, drate);
406 	if (!rate) {
407 		pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
408 			drate, clk_hw_get_name(hw));
409 		return -EINVAL;
410 	}
411 
412 	con0 = readl_relaxed(pll->con_reg);
413 	con1 = readl_relaxed(pll->con_reg + 0x4);
414 
415 	if (!(samsung_pll45xx_mp_change(con0, con1, rate))) {
416 		/* If only s change, change just s value only*/
417 		con0 &= ~(PLL45XX_SDIV_MASK << PLL45XX_SDIV_SHIFT);
418 		con0 |= rate->sdiv << PLL45XX_SDIV_SHIFT;
419 		writel_relaxed(con0, pll->con_reg);
420 
421 		return 0;
422 	}
423 
424 	/* Set PLL PMS values. */
425 	con0 &= ~((PLL45XX_MDIV_MASK << PLL45XX_MDIV_SHIFT) |
426 			(PLL45XX_PDIV_MASK << PLL45XX_PDIV_SHIFT) |
427 			(PLL45XX_SDIV_MASK << PLL45XX_SDIV_SHIFT));
428 	con0 |= (rate->mdiv << PLL45XX_MDIV_SHIFT) |
429 			(rate->pdiv << PLL45XX_PDIV_SHIFT) |
430 			(rate->sdiv << PLL45XX_SDIV_SHIFT);
431 
432 	/* Set PLL AFC value. */
433 	con1 = readl_relaxed(pll->con_reg + 0x4);
434 	con1 &= ~(PLL45XX_AFC_MASK << PLL45XX_AFC_SHIFT);
435 	con1 |= (rate->afc << PLL45XX_AFC_SHIFT);
436 
437 	/* Set PLL lock time. */
438 	switch (pll->type) {
439 	case pll_4502:
440 		writel_relaxed(rate->pdiv * PLL4502_LOCK_FACTOR, pll->lock_reg);
441 		break;
442 	case pll_4508:
443 		writel_relaxed(rate->pdiv * PLL4508_LOCK_FACTOR, pll->lock_reg);
444 		break;
445 	default:
446 		break;
447 	}
448 
449 	/* Set new configuration. */
450 	writel_relaxed(con1, pll->con_reg + 0x4);
451 	writel_relaxed(con0, pll->con_reg);
452 
453 	/* Wait for locking. */
454 	start = ktime_get();
455 	while (!(readl_relaxed(pll->con_reg) & PLL45XX_LOCKED)) {
456 		ktime_t delta = ktime_sub(ktime_get(), start);
457 
458 		if (ktime_to_ms(delta) > PLL_TIMEOUT_MS) {
459 			pr_err("%s: could not lock PLL %s\n",
460 					__func__, clk_hw_get_name(hw));
461 			return -EFAULT;
462 		}
463 
464 		cpu_relax();
465 	}
466 
467 	return 0;
468 }
469 
470 static const struct clk_ops samsung_pll45xx_clk_ops = {
471 	.recalc_rate = samsung_pll45xx_recalc_rate,
472 	.round_rate = samsung_pll_round_rate,
473 	.set_rate = samsung_pll45xx_set_rate,
474 };
475 
476 static const struct clk_ops samsung_pll45xx_clk_min_ops = {
477 	.recalc_rate = samsung_pll45xx_recalc_rate,
478 };
479 
480 /*
481  * PLL46xx Clock Type
482  */
483 #define PLL46XX_LOCK_FACTOR	3000
484 
485 #define PLL46XX_VSEL_MASK	(1)
486 #define PLL46XX_MDIV_MASK	(0x1FF)
487 #define PLL1460X_MDIV_MASK	(0x3FF)
488 
489 #define PLL46XX_PDIV_MASK	(0x3F)
490 #define PLL46XX_SDIV_MASK	(0x7)
491 #define PLL46XX_VSEL_SHIFT	(27)
492 #define PLL46XX_MDIV_SHIFT	(16)
493 #define PLL46XX_PDIV_SHIFT	(8)
494 #define PLL46XX_SDIV_SHIFT	(0)
495 
496 #define PLL46XX_KDIV_MASK	(0xFFFF)
497 #define PLL4650C_KDIV_MASK	(0xFFF)
498 #define PLL46XX_KDIV_SHIFT	(0)
499 #define PLL46XX_MFR_MASK	(0x3F)
500 #define PLL46XX_MRR_MASK	(0x1F)
501 #define PLL46XX_KDIV_SHIFT	(0)
502 #define PLL46XX_MFR_SHIFT	(16)
503 #define PLL46XX_MRR_SHIFT	(24)
504 
505 #define PLL46XX_ENABLE		BIT(31)
506 #define PLL46XX_LOCKED		BIT(29)
507 #define PLL46XX_VSEL		BIT(27)
508 
509 static unsigned long samsung_pll46xx_recalc_rate(struct clk_hw *hw,
510 				unsigned long parent_rate)
511 {
512 	struct samsung_clk_pll *pll = to_clk_pll(hw);
513 	u32 mdiv, pdiv, sdiv, kdiv, pll_con0, pll_con1, shift;
514 	u64 fvco = parent_rate;
515 
516 	pll_con0 = readl_relaxed(pll->con_reg);
517 	pll_con1 = readl_relaxed(pll->con_reg + 4);
518 	mdiv = (pll_con0 >> PLL46XX_MDIV_SHIFT) & ((pll->type == pll_1460x) ?
519 				PLL1460X_MDIV_MASK : PLL46XX_MDIV_MASK);
520 	pdiv = (pll_con0 >> PLL46XX_PDIV_SHIFT) & PLL46XX_PDIV_MASK;
521 	sdiv = (pll_con0 >> PLL46XX_SDIV_SHIFT) & PLL46XX_SDIV_MASK;
522 	kdiv = pll->type == pll_4650c ? pll_con1 & PLL4650C_KDIV_MASK :
523 					pll_con1 & PLL46XX_KDIV_MASK;
524 
525 	shift = ((pll->type == pll_4600) || (pll->type == pll_1460x)) ? 16 : 10;
526 
527 	fvco *= (mdiv << shift) + kdiv;
528 	do_div(fvco, (pdiv << sdiv));
529 	fvco >>= shift;
530 
531 	return (unsigned long)fvco;
532 }
533 
534 static bool samsung_pll46xx_mpk_change(u32 pll_con0, u32 pll_con1,
535 				const struct samsung_pll_rate_table *rate)
536 {
537 	u32 old_mdiv, old_pdiv, old_kdiv;
538 
539 	old_mdiv = (pll_con0 >> PLL46XX_MDIV_SHIFT) & PLL46XX_MDIV_MASK;
540 	old_pdiv = (pll_con0 >> PLL46XX_PDIV_SHIFT) & PLL46XX_PDIV_MASK;
541 	old_kdiv = (pll_con1 >> PLL46XX_KDIV_SHIFT) & PLL46XX_KDIV_MASK;
542 
543 	return (old_mdiv != rate->mdiv || old_pdiv != rate->pdiv
544 		|| old_kdiv != rate->kdiv);
545 }
546 
547 static int samsung_pll46xx_set_rate(struct clk_hw *hw, unsigned long drate,
548 					unsigned long prate)
549 {
550 	struct samsung_clk_pll *pll = to_clk_pll(hw);
551 	const struct samsung_pll_rate_table *rate;
552 	u32 con0, con1, lock;
553 	ktime_t start;
554 
555 	/* Get required rate settings from table */
556 	rate = samsung_get_pll_settings(pll, drate);
557 	if (!rate) {
558 		pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
559 			drate, clk_hw_get_name(hw));
560 		return -EINVAL;
561 	}
562 
563 	con0 = readl_relaxed(pll->con_reg);
564 	con1 = readl_relaxed(pll->con_reg + 0x4);
565 
566 	if (!(samsung_pll46xx_mpk_change(con0, con1, rate))) {
567 		/* If only s change, change just s value only*/
568 		con0 &= ~(PLL46XX_SDIV_MASK << PLL46XX_SDIV_SHIFT);
569 		con0 |= rate->sdiv << PLL46XX_SDIV_SHIFT;
570 		writel_relaxed(con0, pll->con_reg);
571 
572 		return 0;
573 	}
574 
575 	/* Set PLL lock time. */
576 	lock = rate->pdiv * PLL46XX_LOCK_FACTOR;
577 	if (lock > 0xffff)
578 		/* Maximum lock time bitfield is 16-bit. */
579 		lock = 0xffff;
580 
581 	/* Set PLL PMS and VSEL values. */
582 	if (pll->type == pll_1460x) {
583 		con0 &= ~((PLL1460X_MDIV_MASK << PLL46XX_MDIV_SHIFT) |
584 			(PLL46XX_PDIV_MASK << PLL46XX_PDIV_SHIFT) |
585 			(PLL46XX_SDIV_MASK << PLL46XX_SDIV_SHIFT));
586 	} else {
587 		con0 &= ~((PLL46XX_MDIV_MASK << PLL46XX_MDIV_SHIFT) |
588 			(PLL46XX_PDIV_MASK << PLL46XX_PDIV_SHIFT) |
589 			(PLL46XX_SDIV_MASK << PLL46XX_SDIV_SHIFT) |
590 			(PLL46XX_VSEL_MASK << PLL46XX_VSEL_SHIFT));
591 		con0 |=	rate->vsel << PLL46XX_VSEL_SHIFT;
592 	}
593 
594 	con0 |= (rate->mdiv << PLL46XX_MDIV_SHIFT) |
595 			(rate->pdiv << PLL46XX_PDIV_SHIFT) |
596 			(rate->sdiv << PLL46XX_SDIV_SHIFT);
597 
598 	/* Set PLL K, MFR and MRR values. */
599 	con1 = readl_relaxed(pll->con_reg + 0x4);
600 	con1 &= ~((PLL46XX_KDIV_MASK << PLL46XX_KDIV_SHIFT) |
601 			(PLL46XX_MFR_MASK << PLL46XX_MFR_SHIFT) |
602 			(PLL46XX_MRR_MASK << PLL46XX_MRR_SHIFT));
603 	con1 |= (rate->kdiv << PLL46XX_KDIV_SHIFT) |
604 			(rate->mfr << PLL46XX_MFR_SHIFT) |
605 			(rate->mrr << PLL46XX_MRR_SHIFT);
606 
607 	/* Write configuration to PLL */
608 	writel_relaxed(lock, pll->lock_reg);
609 	writel_relaxed(con0, pll->con_reg);
610 	writel_relaxed(con1, pll->con_reg + 0x4);
611 
612 	/* Wait for locking. */
613 	start = ktime_get();
614 	while (!(readl_relaxed(pll->con_reg) & PLL46XX_LOCKED)) {
615 		ktime_t delta = ktime_sub(ktime_get(), start);
616 
617 		if (ktime_to_ms(delta) > PLL_TIMEOUT_MS) {
618 			pr_err("%s: could not lock PLL %s\n",
619 					__func__, clk_hw_get_name(hw));
620 			return -EFAULT;
621 		}
622 
623 		cpu_relax();
624 	}
625 
626 	return 0;
627 }
628 
629 static const struct clk_ops samsung_pll46xx_clk_ops = {
630 	.recalc_rate = samsung_pll46xx_recalc_rate,
631 	.round_rate = samsung_pll_round_rate,
632 	.set_rate = samsung_pll46xx_set_rate,
633 };
634 
635 static const struct clk_ops samsung_pll46xx_clk_min_ops = {
636 	.recalc_rate = samsung_pll46xx_recalc_rate,
637 };
638 
639 /*
640  * PLL6552 Clock Type
641  */
642 
643 #define PLL6552_MDIV_MASK	0x3ff
644 #define PLL6552_PDIV_MASK	0x3f
645 #define PLL6552_SDIV_MASK	0x7
646 #define PLL6552_MDIV_SHIFT	16
647 #define PLL6552_MDIV_SHIFT_2416	14
648 #define PLL6552_PDIV_SHIFT	8
649 #define PLL6552_PDIV_SHIFT_2416	5
650 #define PLL6552_SDIV_SHIFT	0
651 
652 static unsigned long samsung_pll6552_recalc_rate(struct clk_hw *hw,
653 						unsigned long parent_rate)
654 {
655 	struct samsung_clk_pll *pll = to_clk_pll(hw);
656 	u32 mdiv, pdiv, sdiv, pll_con;
657 	u64 fvco = parent_rate;
658 
659 	pll_con = readl_relaxed(pll->con_reg);
660 	if (pll->type == pll_6552_s3c2416) {
661 		mdiv = (pll_con >> PLL6552_MDIV_SHIFT_2416) & PLL6552_MDIV_MASK;
662 		pdiv = (pll_con >> PLL6552_PDIV_SHIFT_2416) & PLL6552_PDIV_MASK;
663 	} else {
664 		mdiv = (pll_con >> PLL6552_MDIV_SHIFT) & PLL6552_MDIV_MASK;
665 		pdiv = (pll_con >> PLL6552_PDIV_SHIFT) & PLL6552_PDIV_MASK;
666 	}
667 	sdiv = (pll_con >> PLL6552_SDIV_SHIFT) & PLL6552_SDIV_MASK;
668 
669 	fvco *= mdiv;
670 	do_div(fvco, (pdiv << sdiv));
671 
672 	return (unsigned long)fvco;
673 }
674 
675 static const struct clk_ops samsung_pll6552_clk_ops = {
676 	.recalc_rate = samsung_pll6552_recalc_rate,
677 };
678 
679 /*
680  * PLL6553 Clock Type
681  */
682 
683 #define PLL6553_MDIV_MASK	0xff
684 #define PLL6553_PDIV_MASK	0x3f
685 #define PLL6553_SDIV_MASK	0x7
686 #define PLL6553_KDIV_MASK	0xffff
687 #define PLL6553_MDIV_SHIFT	16
688 #define PLL6553_PDIV_SHIFT	8
689 #define PLL6553_SDIV_SHIFT	0
690 #define PLL6553_KDIV_SHIFT	0
691 
692 static unsigned long samsung_pll6553_recalc_rate(struct clk_hw *hw,
693 						unsigned long parent_rate)
694 {
695 	struct samsung_clk_pll *pll = to_clk_pll(hw);
696 	u32 mdiv, pdiv, sdiv, kdiv, pll_con0, pll_con1;
697 	u64 fvco = parent_rate;
698 
699 	pll_con0 = readl_relaxed(pll->con_reg);
700 	pll_con1 = readl_relaxed(pll->con_reg + 0x4);
701 	mdiv = (pll_con0 >> PLL6553_MDIV_SHIFT) & PLL6553_MDIV_MASK;
702 	pdiv = (pll_con0 >> PLL6553_PDIV_SHIFT) & PLL6553_PDIV_MASK;
703 	sdiv = (pll_con0 >> PLL6553_SDIV_SHIFT) & PLL6553_SDIV_MASK;
704 	kdiv = (pll_con1 >> PLL6553_KDIV_SHIFT) & PLL6553_KDIV_MASK;
705 
706 	fvco *= (mdiv << 16) + kdiv;
707 	do_div(fvco, (pdiv << sdiv));
708 	fvco >>= 16;
709 
710 	return (unsigned long)fvco;
711 }
712 
713 static const struct clk_ops samsung_pll6553_clk_ops = {
714 	.recalc_rate = samsung_pll6553_recalc_rate,
715 };
716 
717 /*
718  * PLL Clock Type of S3C24XX before S3C2443
719  */
720 
721 #define PLLS3C2410_MDIV_MASK		(0xff)
722 #define PLLS3C2410_PDIV_MASK		(0x1f)
723 #define PLLS3C2410_SDIV_MASK		(0x3)
724 #define PLLS3C2410_MDIV_SHIFT		(12)
725 #define PLLS3C2410_PDIV_SHIFT		(4)
726 #define PLLS3C2410_SDIV_SHIFT		(0)
727 
728 #define PLLS3C2410_ENABLE_REG_OFFSET	0x10
729 
730 static unsigned long samsung_s3c2410_pll_recalc_rate(struct clk_hw *hw,
731 					unsigned long parent_rate)
732 {
733 	struct samsung_clk_pll *pll = to_clk_pll(hw);
734 	u32 pll_con, mdiv, pdiv, sdiv;
735 	u64 fvco = parent_rate;
736 
737 	pll_con = readl_relaxed(pll->con_reg);
738 	mdiv = (pll_con >> PLLS3C2410_MDIV_SHIFT) & PLLS3C2410_MDIV_MASK;
739 	pdiv = (pll_con >> PLLS3C2410_PDIV_SHIFT) & PLLS3C2410_PDIV_MASK;
740 	sdiv = (pll_con >> PLLS3C2410_SDIV_SHIFT) & PLLS3C2410_SDIV_MASK;
741 
742 	fvco *= (mdiv + 8);
743 	do_div(fvco, (pdiv + 2) << sdiv);
744 
745 	return (unsigned int)fvco;
746 }
747 
748 static unsigned long samsung_s3c2440_mpll_recalc_rate(struct clk_hw *hw,
749 					unsigned long parent_rate)
750 {
751 	struct samsung_clk_pll *pll = to_clk_pll(hw);
752 	u32 pll_con, mdiv, pdiv, sdiv;
753 	u64 fvco = parent_rate;
754 
755 	pll_con = readl_relaxed(pll->con_reg);
756 	mdiv = (pll_con >> PLLS3C2410_MDIV_SHIFT) & PLLS3C2410_MDIV_MASK;
757 	pdiv = (pll_con >> PLLS3C2410_PDIV_SHIFT) & PLLS3C2410_PDIV_MASK;
758 	sdiv = (pll_con >> PLLS3C2410_SDIV_SHIFT) & PLLS3C2410_SDIV_MASK;
759 
760 	fvco *= (2 * (mdiv + 8));
761 	do_div(fvco, (pdiv + 2) << sdiv);
762 
763 	return (unsigned int)fvco;
764 }
765 
766 static int samsung_s3c2410_pll_set_rate(struct clk_hw *hw, unsigned long drate,
767 					unsigned long prate)
768 {
769 	struct samsung_clk_pll *pll = to_clk_pll(hw);
770 	const struct samsung_pll_rate_table *rate;
771 	u32 tmp;
772 
773 	/* Get required rate settings from table */
774 	rate = samsung_get_pll_settings(pll, drate);
775 	if (!rate) {
776 		pr_err("%s: Invalid rate : %lu for pll clk %s\n", __func__,
777 			drate, clk_hw_get_name(hw));
778 		return -EINVAL;
779 	}
780 
781 	tmp = readl_relaxed(pll->con_reg);
782 
783 	/* Change PLL PMS values */
784 	tmp &= ~((PLLS3C2410_MDIV_MASK << PLLS3C2410_MDIV_SHIFT) |
785 			(PLLS3C2410_PDIV_MASK << PLLS3C2410_PDIV_SHIFT) |
786 			(PLLS3C2410_SDIV_MASK << PLLS3C2410_SDIV_SHIFT));
787 	tmp |= (rate->mdiv << PLLS3C2410_MDIV_SHIFT) |
788 			(rate->pdiv << PLLS3C2410_PDIV_SHIFT) |
789 			(rate->sdiv << PLLS3C2410_SDIV_SHIFT);
790 	writel_relaxed(tmp, pll->con_reg);
791 
792 	/* Time to settle according to the manual */
793 	udelay(300);
794 
795 	return 0;
796 }
797 
798 static int samsung_s3c2410_pll_enable(struct clk_hw *hw, int bit, bool enable)
799 {
800 	struct samsung_clk_pll *pll = to_clk_pll(hw);
801 	u32 pll_en = readl_relaxed(pll->lock_reg + PLLS3C2410_ENABLE_REG_OFFSET);
802 	u32 pll_en_orig = pll_en;
803 
804 	if (enable)
805 		pll_en &= ~BIT(bit);
806 	else
807 		pll_en |= BIT(bit);
808 
809 	writel_relaxed(pll_en, pll->lock_reg + PLLS3C2410_ENABLE_REG_OFFSET);
810 
811 	/* if we started the UPLL, then allow to settle */
812 	if (enable && (pll_en_orig & BIT(bit)))
813 		udelay(300);
814 
815 	return 0;
816 }
817 
818 static int samsung_s3c2410_mpll_enable(struct clk_hw *hw)
819 {
820 	return samsung_s3c2410_pll_enable(hw, 5, true);
821 }
822 
823 static void samsung_s3c2410_mpll_disable(struct clk_hw *hw)
824 {
825 	samsung_s3c2410_pll_enable(hw, 5, false);
826 }
827 
828 static int samsung_s3c2410_upll_enable(struct clk_hw *hw)
829 {
830 	return samsung_s3c2410_pll_enable(hw, 7, true);
831 }
832 
833 static void samsung_s3c2410_upll_disable(struct clk_hw *hw)
834 {
835 	samsung_s3c2410_pll_enable(hw, 7, false);
836 }
837 
838 static const struct clk_ops samsung_s3c2410_mpll_clk_min_ops = {
839 	.recalc_rate = samsung_s3c2410_pll_recalc_rate,
840 	.enable = samsung_s3c2410_mpll_enable,
841 	.disable = samsung_s3c2410_mpll_disable,
842 };
843 
844 static const struct clk_ops samsung_s3c2410_upll_clk_min_ops = {
845 	.recalc_rate = samsung_s3c2410_pll_recalc_rate,
846 	.enable = samsung_s3c2410_upll_enable,
847 	.disable = samsung_s3c2410_upll_disable,
848 };
849 
850 static const struct clk_ops samsung_s3c2440_mpll_clk_min_ops = {
851 	.recalc_rate = samsung_s3c2440_mpll_recalc_rate,
852 	.enable = samsung_s3c2410_mpll_enable,
853 	.disable = samsung_s3c2410_mpll_disable,
854 };
855 
856 static const struct clk_ops samsung_s3c2410_mpll_clk_ops = {
857 	.recalc_rate = samsung_s3c2410_pll_recalc_rate,
858 	.enable = samsung_s3c2410_mpll_enable,
859 	.disable = samsung_s3c2410_mpll_disable,
860 	.round_rate = samsung_pll_round_rate,
861 	.set_rate = samsung_s3c2410_pll_set_rate,
862 };
863 
864 static const struct clk_ops samsung_s3c2410_upll_clk_ops = {
865 	.recalc_rate = samsung_s3c2410_pll_recalc_rate,
866 	.enable = samsung_s3c2410_upll_enable,
867 	.disable = samsung_s3c2410_upll_disable,
868 	.round_rate = samsung_pll_round_rate,
869 	.set_rate = samsung_s3c2410_pll_set_rate,
870 };
871 
872 static const struct clk_ops samsung_s3c2440_mpll_clk_ops = {
873 	.recalc_rate = samsung_s3c2440_mpll_recalc_rate,
874 	.enable = samsung_s3c2410_mpll_enable,
875 	.disable = samsung_s3c2410_mpll_disable,
876 	.round_rate = samsung_pll_round_rate,
877 	.set_rate = samsung_s3c2410_pll_set_rate,
878 };
879 
880 /*
881  * PLL2550x Clock Type
882  */
883 
884 #define PLL2550X_R_MASK       (0x1)
885 #define PLL2550X_P_MASK       (0x3F)
886 #define PLL2550X_M_MASK       (0x3FF)
887 #define PLL2550X_S_MASK       (0x7)
888 #define PLL2550X_R_SHIFT      (20)
889 #define PLL2550X_P_SHIFT      (14)
890 #define PLL2550X_M_SHIFT      (4)
891 #define PLL2550X_S_SHIFT      (0)
892 
893 static unsigned long samsung_pll2550x_recalc_rate(struct clk_hw *hw,
894 				unsigned long parent_rate)
895 {
896 	struct samsung_clk_pll *pll = to_clk_pll(hw);
897 	u32 r, p, m, s, pll_stat;
898 	u64 fvco = parent_rate;
899 
900 	pll_stat = readl_relaxed(pll->con_reg);
901 	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 
914 static const struct clk_ops samsung_pll2550x_clk_ops = {
915 	.recalc_rate = samsung_pll2550x_recalc_rate,
916 };
917 
918 /*
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 
934 static 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 
941 	pll_con = readl_relaxed(pll->con_reg);
942 	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 
952 static 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 
962 static 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__,
973 			drate, clk_hw_get_name(hw));
974 		return -EINVAL;
975 	}
976 
977 	tmp = readl_relaxed(pll->con_reg);
978 
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;
983 		writel_relaxed(tmp, pll->con_reg);
984 
985 		return 0;
986 	}
987 
988 	/* Set PLL lock time. */
989 	writel_relaxed(rate->pdiv * PLL2550XX_LOCK_FACTOR, pll->lock_reg);
990 
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);
998 	writel_relaxed(tmp, pll->con_reg);
999 
1000 	/* wait_lock_time */
1001 	do {
1002 		cpu_relax();
1003 		tmp = readl_relaxed(pll->con_reg);
1004 	} while (!(tmp & (PLL2550XX_LOCK_STAT_MASK
1005 			<< PLL2550XX_LOCK_STAT_SHIFT)));
1006 
1007 	return 0;
1008 }
1009 
1010 static 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 
1016 static const struct clk_ops samsung_pll2550xx_clk_min_ops = {
1017 	.recalc_rate = samsung_pll2550xx_recalc_rate,
1018 };
1019 
1020 /*
1021  * 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 
1039 static 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 
1062 static 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 
1106 static 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 
1112 static const struct clk_ops samsung_pll2650x_clk_min_ops = {
1113 	.recalc_rate = samsung_pll2650x_recalc_rate,
1114 };
1115 
1116 /*
1117  * 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 
1135 static 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 
1143 	pll_con0 = readl_relaxed(pll->con_reg);
1144 	pll_con2 = readl_relaxed(pll->con_reg + 8);
1145 	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 
1157 static 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__,
1167 			drate, clk_hw_get_name(hw));
1168 		return -EINVAL;
1169 	}
1170 
1171 	pll_con0 = readl_relaxed(pll->con_reg);
1172 	pll_con2 = readl_relaxed(pll->con_reg + 8);
1173 
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. */
1189 	writel_relaxed(PLL2650XX_LOCK_FACTOR * rate->pdiv, pll->lock_reg);
1190 
1191 	writel_relaxed(pll_con0, pll->con_reg);
1192 	writel_relaxed(pll_con2, pll->con_reg + 8);
1193 
1194 	do {
1195 		tmp = readl_relaxed(pll->con_reg);
1196 	} while (!(tmp & (0x1 << PLL2650XX_PLL_LOCKTIME_SHIFT)));
1197 
1198 	return 0;
1199 }
1200 
1201 static 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 
1207 static const struct clk_ops samsung_pll2650xx_clk_min_ops = {
1208 	.recalc_rate = samsung_pll2650xx_recalc_rate,
1209 };
1210 
1211 static void __init _samsung_clk_register_pll(struct samsung_clk_provider *ctx,
1212 				const struct samsung_pll_clock *pll_clk,
1213 				void __iomem *base)
1214 {
1215 	struct samsung_clk_pll *pll;
1216 	struct clk *clk;
1217 	struct clk_init_data init;
1218 	int ret, len;
1219 
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 
1232 	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 
1247 	switch (pll_clk->type) {
1248 	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;
1254 	/* clk_ops for 35xx and 2550 are similar */
1255 	case pll_35xx:
1256 	case pll_2550:
1257 	case pll_1450x:
1258 	case pll_1451x:
1259 	case pll_1452x:
1260 		if (!pll->rate_table)
1261 			init.ops = &samsung_pll35xx_clk_min_ops;
1262 		else
1263 			init.ops = &samsung_pll35xx_clk_ops;
1264 		break;
1265 	case pll_4500:
1266 		init.ops = &samsung_pll45xx_clk_min_ops;
1267 		break;
1268 	case pll_4502:
1269 	case pll_4508:
1270 		if (!pll->rate_table)
1271 			init.ops = &samsung_pll45xx_clk_min_ops;
1272 		else
1273 			init.ops = &samsung_pll45xx_clk_ops;
1274 		break;
1275 	/* clk_ops for 36xx and 2650 are similar */
1276 	case pll_36xx:
1277 	case pll_2650:
1278 		if (!pll->rate_table)
1279 			init.ops = &samsung_pll36xx_clk_min_ops;
1280 		else
1281 			init.ops = &samsung_pll36xx_clk_ops;
1282 		break;
1283 	case pll_6552:
1284 	case pll_6552_s3c2416:
1285 		init.ops = &samsung_pll6552_clk_ops;
1286 		break;
1287 	case pll_6553:
1288 		init.ops = &samsung_pll6553_clk_ops;
1289 		break;
1290 	case pll_4600:
1291 	case pll_4650:
1292 	case pll_4650c:
1293 	case pll_1460x:
1294 		if (!pll->rate_table)
1295 			init.ops = &samsung_pll46xx_clk_min_ops;
1296 		else
1297 			init.ops = &samsung_pll46xx_clk_ops;
1298 		break;
1299 	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;
1317 	case pll_2550x:
1318 		init.ops = &samsung_pll2550x_clk_ops;
1319 		break;
1320 	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;
1326 	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;
1332 	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;
1338 	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 
1356 	samsung_clk_add_lookup(ctx, clk, pll_clk->id);
1357 
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 
1367 void __init samsung_clk_register_pll(struct samsung_clk_provider *ctx,
1368 			const struct samsung_pll_clock *pll_list,
1369 			unsigned int nr_pll, void __iomem *base)
1370 {
1371 	int cnt;
1372 
1373 	for (cnt = 0; cnt < nr_pll; cnt++)
1374 		_samsung_clk_register_pll(ctx, &pll_list[cnt], base);
1375 }
1376