/*- * SPDX-License-Identifier: BSD-2-Clause-FreeBSD * * Copyright (c) 2020 Oleksandr Tymoshenko * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * * $FreeBSD$ */ #include __FBSDID("$FreeBSD$"); #include #include #include #include #include #include "clkdev_if.h" struct imx_clk_frac_pll_sc { uint32_t offset; }; #define WRITE4(_clk, off, val) \ CLKDEV_WRITE_4(clknode_get_device(_clk), off, val) #define READ4(_clk, off, val) \ CLKDEV_READ_4(clknode_get_device(_clk), off, val) #define DEVICE_LOCK(_clk) \ CLKDEV_DEVICE_LOCK(clknode_get_device(_clk)) #define DEVICE_UNLOCK(_clk) \ CLKDEV_DEVICE_UNLOCK(clknode_get_device(_clk)) #define CFG0 0 #define CFG0_PLL_LOCK (1 << 31) #define CFG0_PD (1 << 19) #define CFG0_BYPASS (1 << 14) #define CFG0_NEWDIV_VAL (1 << 12) #define CFG0_NEWDIV_ACK (1 << 11) #define CFG0_OUTPUT_DIV_MASK (0x1f << 0) #define CFG0_OUTPUT_DIV_SHIFT 0 #define CFG1 4 #define CFG1_FRAC_DIV_MASK (0xffffff << 7) #define CFG1_FRAC_DIV_SHIFT 7 #define CFG1_INT_DIV_MASK (0x7f << 0) #define CFG1_INT_DIV_SHIFT 0 #if 0 #define dprintf(format, arg...) \ printf("%s:(%s)" format, __func__, clknode_get_name(clk), arg) #else #define dprintf(format, arg...) #endif static int imx_clk_frac_pll_init(struct clknode *clk, device_t dev) { clknode_init_parent_idx(clk, 0); return (0); } static int imx_clk_frac_pll_set_gate(struct clknode *clk, bool enable) { struct imx_clk_frac_pll_sc *sc; uint32_t cfg0; int timeout; sc = clknode_get_softc(clk); DEVICE_LOCK(clk); READ4(clk, sc->offset + CFG0, &cfg0); if (enable) cfg0 &= ~(CFG0_PD); else cfg0 |= CFG0_PD; WRITE4(clk, sc->offset + CFG0, cfg0); /* Wait for PLL to lock */ if (enable && ((cfg0 & CFG0_BYPASS) == 0)) { for (timeout = 1000; timeout; timeout--) { READ4(clk, sc->offset + CFG0, &cfg0); if (cfg0 & CFG0_PLL_LOCK) break; DELAY(1); } } DEVICE_UNLOCK(clk); return (0); } static int imx_clk_frac_pll_recalc(struct clknode *clk, uint64_t *freq) { struct imx_clk_frac_pll_sc *sc; uint32_t cfg0, cfg1; uint64_t div, divfi, divff, divf_val; sc = clknode_get_softc(clk); DEVICE_LOCK(clk); READ4(clk, sc->offset + CFG0, &cfg0); READ4(clk, sc->offset + CFG1, &cfg1); DEVICE_UNLOCK(clk); div = (cfg0 & CFG0_OUTPUT_DIV_MASK) >> CFG0_OUTPUT_DIV_SHIFT; div = (div + 1) * 2; divff = (cfg1 & CFG1_FRAC_DIV_MASK) >> CFG1_FRAC_DIV_SHIFT; divfi = (cfg1 & CFG1_INT_DIV_MASK) >> CFG1_INT_DIV_SHIFT; /* PLL is bypassed */ if (cfg0 & CFG0_BYPASS) return (0); divf_val = 1 + divfi + (divff/0x1000000); *freq = *freq * 8 * divf_val / div; return (0); } static clknode_method_t imx_clk_frac_pll_clknode_methods[] = { /* Device interface */ CLKNODEMETHOD(clknode_init, imx_clk_frac_pll_init), CLKNODEMETHOD(clknode_set_gate, imx_clk_frac_pll_set_gate), CLKNODEMETHOD(clknode_recalc_freq, imx_clk_frac_pll_recalc), CLKNODEMETHOD_END }; DEFINE_CLASS_1(imx_clk_frac_pll_clknode, imx_clk_frac_pll_clknode_class, imx_clk_frac_pll_clknode_methods, sizeof(struct imx_clk_frac_pll_sc), clknode_class); int imx_clk_frac_pll_register(struct clkdom *clkdom, struct imx_clk_frac_pll_def *clkdef) { struct clknode *clk; struct imx_clk_frac_pll_sc *sc; clk = clknode_create(clkdom, &imx_clk_frac_pll_clknode_class, &clkdef->clkdef); if (clk == NULL) return (1); sc = clknode_get_softc(clk); sc->offset = clkdef->offset; clknode_register(clkdom, clk); return (0); }