Lines Matching +full:pll +full:- +full:0
1 // SPDX-License-Identifier: GPL-2.0
11 * In the most basic form, a Meson PLL is composed as follows:
13 * PLL
14 * +--------------------------------+
16 * | +--+ |
17 * in >>-----[ /N ]--->| | +-----+ |
18 * | | |------| DCO |---->> out
19 * | +--------->| | +--v--+ |
20 * | | +--+ | |
22 * | +--[ *(M + (F/Fmax) ]<--+ |
24 * +--------------------------------+
29 #include <linux/clk-provider.h>
36 #include "clk-regmap.h"
37 #include "clk-pll.h"
42 return (struct meson_clk_pll_data *)clk->data; in meson_clk_pll_data()
45 static int __pll_round_closest_mult(struct meson_clk_pll_data *pll) in __pll_round_closest_mult() argument
47 if ((pll->flags & CLK_MESON_PLL_ROUND_CLOSEST) && in __pll_round_closest_mult()
48 !MESON_PARM_APPLICABLE(&pll->frac)) in __pll_round_closest_mult()
51 return 0; in __pll_round_closest_mult()
57 struct meson_clk_pll_data *pll) in __pll_params_to_rate() argument
61 if (frac && MESON_PARM_APPLICABLE(&pll->frac)) { in __pll_params_to_rate()
65 (1 << pll->frac.width)); in __pll_params_to_rate()
75 struct meson_clk_pll_data *pll = meson_clk_pll_data(clk); in meson_clk_pll_recalc_rate() local
78 n = meson_parm_read(clk->map, &pll->n); in meson_clk_pll_recalc_rate()
85 if (n == 0) in meson_clk_pll_recalc_rate()
86 return 0; in meson_clk_pll_recalc_rate()
88 m = meson_parm_read(clk->map, &pll->m); in meson_clk_pll_recalc_rate()
90 frac = MESON_PARM_APPLICABLE(&pll->frac) ? in meson_clk_pll_recalc_rate()
91 meson_parm_read(clk->map, &pll->frac) : in meson_clk_pll_recalc_rate()
92 0; in meson_clk_pll_recalc_rate()
94 return __pll_params_to_rate(parent_rate, m, n, frac, pll); in meson_clk_pll_recalc_rate()
101 struct meson_clk_pll_data *pll) in __pll_params_with_frac() argument
103 unsigned int frac_max = (1 << pll->frac.width); in __pll_params_with_frac()
108 return 0; in __pll_params_with_frac()
110 if (pll->flags & CLK_MESON_PLL_ROUND_CLOSEST) in __pll_params_with_frac()
115 val -= m * frac_max; in __pll_params_with_frac()
117 return min((unsigned int)val, (frac_max - 1)); in __pll_params_with_frac()
123 struct meson_clk_pll_data *pll) in meson_clk_pll_is_better() argument
125 if (__pll_round_closest_mult(pll)) { in meson_clk_pll_is_better()
127 if (abs(now - rate) < abs(best - rate)) in meson_clk_pll_is_better()
141 struct meson_clk_pll_data *pll) in meson_clk_get_pll_table_index() argument
143 if (!pll->table[index].n) in meson_clk_get_pll_table_index()
144 return -EINVAL; in meson_clk_get_pll_table_index()
146 *m = pll->table[index].m; in meson_clk_get_pll_table_index()
147 *n = pll->table[index].n; in meson_clk_get_pll_table_index()
149 return 0; in meson_clk_get_pll_table_index()
155 struct meson_clk_pll_data *pll) in meson_clk_get_pll_range_m() argument
159 if (__pll_round_closest_mult(pll)) in meson_clk_get_pll_range_m()
170 struct meson_clk_pll_data *pll) in meson_clk_get_pll_range_index() argument
175 if (*n >= (1 << pll->n.width)) in meson_clk_get_pll_range_index()
176 return -EINVAL; in meson_clk_get_pll_range_index()
180 if (rate <= pll->range->min * parent_rate) { in meson_clk_get_pll_range_index()
181 *m = pll->range->min; in meson_clk_get_pll_range_index()
182 return -ENODATA; in meson_clk_get_pll_range_index()
183 } else if (rate >= pll->range->max * parent_rate) { in meson_clk_get_pll_range_index()
184 *m = pll->range->max; in meson_clk_get_pll_range_index()
185 return -ENODATA; in meson_clk_get_pll_range_index()
189 *m = meson_clk_get_pll_range_m(rate, parent_rate, *n, pll); in meson_clk_get_pll_range_index()
191 /* the pre-divider gives a multiplier too big - stop */ in meson_clk_get_pll_range_index()
192 if (*m >= (1 << pll->m.width)) in meson_clk_get_pll_range_index()
193 return -EINVAL; in meson_clk_get_pll_range_index()
195 return 0; in meson_clk_get_pll_range_index()
203 struct meson_clk_pll_data *pll) in meson_clk_get_pll_get_index() argument
205 if (pll->range) in meson_clk_get_pll_get_index()
207 index, m, n, pll); in meson_clk_get_pll_get_index()
208 else if (pll->table) in meson_clk_get_pll_get_index()
209 return meson_clk_get_pll_table_index(index, m, n, pll); in meson_clk_get_pll_get_index()
211 return -EINVAL; in meson_clk_get_pll_get_index()
218 struct meson_clk_pll_data *pll) in meson_clk_get_pll_settings() argument
220 unsigned long best = 0, now = 0; in meson_clk_get_pll_settings()
224 for (i = 0, ret = 0; !ret; i++) { in meson_clk_get_pll_settings()
226 i, &m, &n, pll); in meson_clk_get_pll_settings()
227 if (ret == -EINVAL) in meson_clk_get_pll_settings()
230 now = __pll_params_to_rate(parent_rate, m, n, 0, pll); in meson_clk_get_pll_settings()
231 if (meson_clk_pll_is_better(rate, best, now, pll)) { in meson_clk_get_pll_settings()
241 return best ? 0 : -EINVAL; in meson_clk_get_pll_settings()
248 struct meson_clk_pll_data *pll = meson_clk_pll_data(clk); in meson_clk_pll_determine_rate() local
253 ret = meson_clk_get_pll_settings(req->rate, req->best_parent_rate, in meson_clk_pll_determine_rate()
254 &m, &n, pll); in meson_clk_pll_determine_rate()
258 round = __pll_params_to_rate(req->best_parent_rate, m, n, 0, pll); in meson_clk_pll_determine_rate()
260 if (!MESON_PARM_APPLICABLE(&pll->frac) || req->rate == round) { in meson_clk_pll_determine_rate()
261 req->rate = round; in meson_clk_pll_determine_rate()
262 return 0; in meson_clk_pll_determine_rate()
269 frac = __pll_params_with_frac(req->rate, req->best_parent_rate, m, n, pll); in meson_clk_pll_determine_rate()
270 req->rate = __pll_params_to_rate(req->best_parent_rate, m, n, frac, pll); in meson_clk_pll_determine_rate()
272 return 0; in meson_clk_pll_determine_rate()
278 struct meson_clk_pll_data *pll = meson_clk_pll_data(clk); in meson_clk_pll_wait_lock() local
283 if (meson_parm_read(clk->map, &pll->l)) in meson_clk_pll_wait_lock()
284 return 0; in meson_clk_pll_wait_lock()
287 } while (--delay); in meson_clk_pll_wait_lock()
289 return -ETIMEDOUT; in meson_clk_pll_wait_lock()
295 struct meson_clk_pll_data *pll = meson_clk_pll_data(clk); in meson_clk_pll_init() local
297 if (pll->init_count) { in meson_clk_pll_init()
298 if (MESON_PARM_APPLICABLE(&pll->rst)) in meson_clk_pll_init()
299 meson_parm_write(clk->map, &pll->rst, 1); in meson_clk_pll_init()
301 regmap_multi_reg_write(clk->map, pll->init_regs, in meson_clk_pll_init()
302 pll->init_count); in meson_clk_pll_init()
304 if (MESON_PARM_APPLICABLE(&pll->rst)) in meson_clk_pll_init()
305 meson_parm_write(clk->map, &pll->rst, 0); in meson_clk_pll_init()
308 return 0; in meson_clk_pll_init()
314 struct meson_clk_pll_data *pll = meson_clk_pll_data(clk); in meson_clk_pll_is_enabled() local
316 if (MESON_PARM_APPLICABLE(&pll->rst) && in meson_clk_pll_is_enabled()
317 meson_parm_read(clk->map, &pll->rst)) in meson_clk_pll_is_enabled()
318 return 0; in meson_clk_pll_is_enabled()
320 if (!meson_parm_read(clk->map, &pll->en) || in meson_clk_pll_is_enabled()
321 !meson_parm_read(clk->map, &pll->l)) in meson_clk_pll_is_enabled()
322 return 0; in meson_clk_pll_is_enabled()
334 return 0; in meson_clk_pcie_pll_enable()
335 pr_info("Retry enabling PCIe PLL clock\n"); in meson_clk_pcie_pll_enable()
336 } while (--retries); in meson_clk_pcie_pll_enable()
338 return -EIO; in meson_clk_pcie_pll_enable()
344 struct meson_clk_pll_data *pll = meson_clk_pll_data(clk); in meson_clk_pll_enable() local
346 /* do nothing if the PLL is already enabled */ in meson_clk_pll_enable()
348 return 0; in meson_clk_pll_enable()
350 /* Make sure the pll is in reset */ in meson_clk_pll_enable()
351 if (MESON_PARM_APPLICABLE(&pll->rst)) in meson_clk_pll_enable()
352 meson_parm_write(clk->map, &pll->rst, 1); in meson_clk_pll_enable()
354 /* Enable the pll */ in meson_clk_pll_enable()
355 meson_parm_write(clk->map, &pll->en, 1); in meson_clk_pll_enable()
357 /* Take the pll out reset */ in meson_clk_pll_enable()
358 if (MESON_PARM_APPLICABLE(&pll->rst)) in meson_clk_pll_enable()
359 meson_parm_write(clk->map, &pll->rst, 0); in meson_clk_pll_enable()
362 * Compared with the previous SoCs, self-adaption current module in meson_clk_pll_enable()
363 * is newly added for A1, keep the new power-on sequence to enable the in meson_clk_pll_enable()
364 * PLL. The sequence is: in meson_clk_pll_enable()
365 * 1. enable the pll, delay for 10us in meson_clk_pll_enable()
366 * 2. enable the pll self-adaption current module, delay for 40us in meson_clk_pll_enable()
369 if (MESON_PARM_APPLICABLE(&pll->current_en)) { in meson_clk_pll_enable()
371 meson_parm_write(clk->map, &pll->current_en, 1); in meson_clk_pll_enable()
375 if (MESON_PARM_APPLICABLE(&pll->l_detect)) { in meson_clk_pll_enable()
376 meson_parm_write(clk->map, &pll->l_detect, 1); in meson_clk_pll_enable()
377 meson_parm_write(clk->map, &pll->l_detect, 0); in meson_clk_pll_enable()
381 return -EIO; in meson_clk_pll_enable()
383 return 0; in meson_clk_pll_enable()
389 struct meson_clk_pll_data *pll = meson_clk_pll_data(clk); in meson_clk_pll_disable() local
391 /* Put the pll is in reset */ in meson_clk_pll_disable()
392 if (MESON_PARM_APPLICABLE(&pll->rst)) in meson_clk_pll_disable()
393 meson_parm_write(clk->map, &pll->rst, 1); in meson_clk_pll_disable()
395 /* Disable the pll */ in meson_clk_pll_disable()
396 meson_parm_write(clk->map, &pll->en, 0); in meson_clk_pll_disable()
398 /* Disable PLL internal self-adaption current module */ in meson_clk_pll_disable()
399 if (MESON_PARM_APPLICABLE(&pll->current_en)) in meson_clk_pll_disable()
400 meson_parm_write(clk->map, &pll->current_en, 0); in meson_clk_pll_disable()
407 struct meson_clk_pll_data *pll = meson_clk_pll_data(clk); in meson_clk_pll_set_rate() local
408 unsigned int enabled, m, n, frac = 0; in meson_clk_pll_set_rate()
412 if (parent_rate == 0 || rate == 0) in meson_clk_pll_set_rate()
413 return -EINVAL; in meson_clk_pll_set_rate()
417 ret = meson_clk_get_pll_settings(rate, parent_rate, &m, &n, pll); in meson_clk_pll_set_rate()
421 enabled = meson_parm_read(clk->map, &pll->en); in meson_clk_pll_set_rate()
425 meson_parm_write(clk->map, &pll->n, n); in meson_clk_pll_set_rate()
426 meson_parm_write(clk->map, &pll->m, m); in meson_clk_pll_set_rate()
428 if (MESON_PARM_APPLICABLE(&pll->frac)) { in meson_clk_pll_set_rate()
429 frac = __pll_params_with_frac(rate, parent_rate, m, n, pll); in meson_clk_pll_set_rate()
430 meson_parm_write(clk->map, &pll->frac, frac); in meson_clk_pll_set_rate()
433 /* If the pll is stopped, bail out now */ in meson_clk_pll_set_rate()
435 return 0; in meson_clk_pll_set_rate()
439 pr_warn("%s: pll did not lock, trying to restore old rate %lu\n", in meson_clk_pll_set_rate()
454 * The Meson G12A PCIE PLL is fined tuned to deliver a very precise
456 * a strict register sequence to enable the PLL.
457 * To simplify, re-use the _init() op to enable the PLL and keep
486 MODULE_DESCRIPTION("Amlogic PLL driver");