Lines Matching +full:vco +full:- +full:hz
1 // SPDX-License-Identifier: GPL-2.0
4 * Author: Andy Yan <andy.yan@rock-chips.com>
9 #include <clk-uclass.h>
11 #include <dt-structs.h>
21 #include <dt-bindings/clock/rk3368-cru.h>
43 #define PLL_DIVISORS(hz, _nr, _no) { \ argument
44 .nr = _nr, .nf = (u32)((u64)hz * _nr * _no / OSC_HZ), .no = _no}; \
45 _Static_assert(((u64)hz * _nr * _no / OSC_HZ) * OSC_HZ /\
46 (_nr * _no) == hz, #hz "Hz cannot be hit with PLL " \
66 struct rk3368_pll *pll = &cru->pll[pll_id]; in rkclk_pll_get_rate()
68 con = readl(&pll->con3); in rkclk_pll_get_rate()
74 con = readl(&pll->con0); in rkclk_pll_get_rate()
77 con = readl(&pll->con1); in rkclk_pll_get_rate()
91 struct rk3368_pll *pll = &cru->pll[pll_id]; in rkclk_set_pll()
92 /* All PLLs have same VCO and output frequency range restrictions*/ in rkclk_set_pll()
93 uint vco_hz = OSC_HZ / 1000 * div->nf / div->nr * 1000; in rkclk_set_pll()
94 uint output_hz = vco_hz / div->no; in rkclk_set_pll()
96 debug("PLL at %p: nf=%d, nr=%d, no=%d, vco=%u Hz, output=%u Hz\n", in rkclk_set_pll()
97 pll, div->nf, div->nr, div->no, vco_hz, output_hz); in rkclk_set_pll()
100 rk_clrsetreg(&pll->con3, PLL_MODE_MASK | PLL_RESET_MASK, in rkclk_set_pll()
103 rk_clrsetreg(&pll->con0, PLL_NR_MASK | PLL_OD_MASK, in rkclk_set_pll()
104 ((div->nr - 1) << PLL_NR_SHIFT) | in rkclk_set_pll()
105 ((div->no - 1) << PLL_OD_SHIFT)); in rkclk_set_pll()
106 writel((div->nf - 1) << PLL_NF_SHIFT, &pll->con1); in rkclk_set_pll()
111 clrsetbits_le32(&pll->con2, PLL_BWADJ_MASK, (div->nf >> 1) - 1); in rkclk_set_pll()
116 rk_clrreg(&pll->con3, PLL_RESET_MASK); in rkclk_set_pll()
119 while (!(readl(&pll->con1) & PLL_LOCK_STA)) in rkclk_set_pll()
122 rk_clrsetreg(&pll->con3, PLL_MODE_MASK, in rkclk_set_pll()
173 return -EINVAL; in rk3368_mmc_get_clk()
176 con = readl(&cru->clksel_con[con_id]); in rk3368_mmc_get_clk()
189 return -EINVAL; in rk3368_mmc_get_clk()
194 debug("%s: raw rate %d (post-divide by 2)\n", __func__, rate); in rk3368_mmc_get_clk()
218 * Find the largest rate no larger than the target-rate for in rk3368_mmc_find_best_rate_and_parent()
226 debug("%s: rate %ld, parent-mux %d, parent-rate %ld, div %d\n", in rk3368_mmc_find_best_rate_and_parent()
230 if ((div - 1) > MMC_CLK_DIV_MASK) in rk3368_mmc_find_best_rate_and_parent()
240 *best_div = div - 1; in rk3368_mmc_find_best_rate_and_parent()
251 struct rk3368_clk_priv *priv = dev_get_priv(clk->dev); in rk3368_mmc_set_clk()
252 struct rk3368_cru *cru = priv->cru; in rk3368_mmc_set_clk()
253 ulong clk_id = clk->id; in rk3368_mmc_set_clk()
270 return -EINVAL; in rk3368_mmc_set_clk()
273 rk_clrsetreg(&cru->clksel_con[con_id], in rk3368_mmc_set_clk()
320 if (readl(&cru->clksel_con[43]) & GMAC_MUX_SEL_EXTCLK) { in rk3368_gmac_set_clk()
324 u32 con = readl(&cru->clksel_con[43]); in rk3368_gmac_set_clk()
336 return -EPERM; in rk3368_gmac_set_clk()
338 div = DIV_ROUND_UP(pll_rate, set_rate) - 1; in rk3368_gmac_set_clk()
340 rk_clrsetreg(&cru->clksel_con[43], GMAC_DIV_CON_MASK, in rk3368_gmac_set_clk()
353 * RK3368 SPI clocks have a common divider-width (7 bits) and a single bit
354 * to select either CPLL or GPLL as the clock-parent. The location within
375 return (val >> shift) & ((1 << width) - 1); in extract_bits()
385 spiclk = &spi_clkregs[clk_id - SCLK_SPI0]; in rk3368_spi_get_clk()
389 pr_err("%s: SPI clk-id %ld not supported\n", __func__, clk_id); in rk3368_spi_get_clk()
390 return -EINVAL; in rk3368_spi_get_clk()
393 val = readl(&cru->clksel_con[spiclk->reg]); in rk3368_spi_get_clk()
394 div = extract_bits(val, 7, spiclk->div_shift); in rk3368_spi_get_clk()
400 static ulong rk3368_spi_set_clk(struct rk3368_cru *cru, ulong clk_id, uint hz) in rk3368_spi_set_clk() argument
405 src_clk_div = DIV_ROUND_UP(GPLL_HZ, hz); in rk3368_spi_set_clk()
410 spiclk = &spi_clkregs[clk_id - SCLK_SPI0]; in rk3368_spi_set_clk()
414 pr_err("%s: SPI clk-id %ld not supported\n", __func__, clk_id); in rk3368_spi_set_clk()
415 return -EINVAL; in rk3368_spi_set_clk()
418 rk_clrsetreg(&cru->clksel_con[spiclk->reg], in rk3368_spi_set_clk()
419 ((0x7f << spiclk->div_shift) | in rk3368_spi_set_clk()
420 (0x1 << spiclk->sel_shift)), in rk3368_spi_set_clk()
421 ((src_clk_div << spiclk->div_shift) | in rk3368_spi_set_clk()
422 (1 << spiclk->sel_shift))); in rk3368_spi_set_clk()
431 val = readl(&cru->clksel_con[25]); in rk3368_saradc_get_clk()
438 static ulong rk3368_saradc_set_clk(struct rk3368_cru *cru, uint hz) in rk3368_saradc_set_clk() argument
442 src_clk_div = DIV_ROUND_UP(OSC_HZ, hz) - 1; in rk3368_saradc_set_clk()
445 rk_clrsetreg(&cru->clksel_con[25], in rk3368_saradc_set_clk()
454 struct rk3368_clk_priv *priv = dev_get_priv(clk->dev); in rk3368_clk_get_rate()
457 debug("%s: id %ld\n", __func__, clk->id); in rk3368_clk_get_rate()
458 switch (clk->id) { in rk3368_clk_get_rate()
460 rate = rkclk_pll_get_rate(priv->cru, CPLL); in rk3368_clk_get_rate()
463 rate = rkclk_pll_get_rate(priv->cru, GPLL); in rk3368_clk_get_rate()
466 rate = rk3368_spi_get_clk(priv->cru, clk->id); in rk3368_clk_get_rate()
471 rate = rk3368_mmc_get_clk(priv->cru, clk->id); in rk3368_clk_get_rate()
475 rate = rk3368_saradc_get_clk(priv->cru); in rk3368_clk_get_rate()
478 return -ENOENT; in rk3368_clk_get_rate()
486 __maybe_unused struct rk3368_clk_priv *priv = dev_get_priv(clk->dev); in rk3368_clk_set_rate()
489 debug("%s id:%ld rate:%ld\n", __func__, clk->id, rate); in rk3368_clk_set_rate()
490 switch (clk->id) { in rk3368_clk_set_rate()
492 ret = rk3368_spi_set_clk(priv->cru, clk->id, rate); in rk3368_clk_set_rate()
496 ret = rk3368_ddr_set_clk(priv->cru, rate); in rk3368_clk_set_rate()
508 ret = rk3368_gmac_set_clk(priv->cru, rate); in rk3368_clk_set_rate()
512 ret = rk3368_saradc_set_clk(priv->cru, rate); in rk3368_clk_set_rate()
515 return -ENOENT; in rk3368_clk_set_rate()
523 struct rk3368_clk_priv *priv = dev_get_priv(clk->dev); in rk3368_gmac_set_parent()
524 struct rk3368_cru *cru = priv->cru; in rk3368_gmac_set_parent()
529 * If the requested parent is in the same clock-controller and in rk3368_gmac_set_parent()
533 if ((parent->dev == clk->dev) && (parent->id == SCLK_MAC)) { in rk3368_gmac_set_parent()
535 rk_clrreg(&cru->clksel_con[43], GMAC_MUX_SEL_EXTCLK); in rk3368_gmac_set_parent()
540 * Otherwise, we need to check the clock-output-names of the in rk3368_gmac_set_parent()
543 ret = dev_read_string_index(parent->dev, "clock-output-names", in rk3368_gmac_set_parent()
544 parent->id, &clock_output_name); in rk3368_gmac_set_parent()
546 return -ENODATA; in rk3368_gmac_set_parent()
551 rk_setreg(&cru->clksel_con[43], GMAC_MUX_SEL_EXTCLK); in rk3368_gmac_set_parent()
555 return -EINVAL; in rk3368_gmac_set_parent()
560 switch (clk->id) { in rk3368_clk_set_parent()
565 debug("%s: unsupported clk %ld\n", __func__, clk->id); in rk3368_clk_set_parent()
566 return -ENOENT; in rk3368_clk_set_parent()
571 switch (clk->id) { in rk3368_clk_enable()
583 debug("%s: unsupported clk %ld\n", __func__, clk->id); in rk3368_clk_enable()
584 return -ENOENT; in rk3368_clk_enable()
602 priv->cru = map_sysmem(plat->dtd.reg[0], plat->dtd.reg[1]); in rk3368_clk_probe()
605 rkclk_init(priv->cru); in rk3368_clk_probe()
616 priv->cru = dev_read_addr_ptr(dev); in rk3368_clk_ofdata_to_platdata()
635 priv->glb_srst_fst_value = offsetof(struct rk3368_cru, in rk3368_clk_bind()
637 priv->glb_srst_snd_value = offsetof(struct rk3368_cru, in rk3368_clk_bind()
639 sys_child->priv = priv; in rk3368_clk_bind()
653 { .compatible = "rockchip,rk3368-cru" },