10050ea24SMichal Meloun /*-
24d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause
30050ea24SMichal Meloun *
40050ea24SMichal Meloun * Copyright (c) 2019 Emmanuel Vadot <manu@freebsd.org>
50050ea24SMichal Meloun *
60050ea24SMichal Meloun * Copyright (c) 2020 Oskar Holmlund <oskar.holmlund@ohdata.se>
70050ea24SMichal Meloun *
80050ea24SMichal Meloun * Redistribution and use in source and binary forms, with or without
90050ea24SMichal Meloun * modification, are permitted provided that the following conditions
100050ea24SMichal Meloun * are met:
110050ea24SMichal Meloun * 1. Redistributions of source code must retain the above copyright
120050ea24SMichal Meloun * notice, this list of conditions and the following disclaimer.
130050ea24SMichal Meloun * 2. Redistributions in binary form must reproduce the above copyright
140050ea24SMichal Meloun * notice, this list of conditions and the following disclaimer in the
150050ea24SMichal Meloun * documentation and/or other materials provided with the distribution.
160050ea24SMichal Meloun *
170050ea24SMichal Meloun * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
180050ea24SMichal Meloun * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
190050ea24SMichal Meloun * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
200050ea24SMichal Meloun * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
210050ea24SMichal Meloun * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
220050ea24SMichal Meloun * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
230050ea24SMichal Meloun * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
240050ea24SMichal Meloun * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
250050ea24SMichal Meloun * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
260050ea24SMichal Meloun * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
270050ea24SMichal Meloun * SUCH DAMAGE.
280050ea24SMichal Meloun *
290050ea24SMichal Meloun * based on sys/arm/allwinner/clkng/aw_clk_np.c
300050ea24SMichal Meloun */
310050ea24SMichal Meloun
320050ea24SMichal Meloun #include <sys/param.h>
330050ea24SMichal Meloun #include <sys/systm.h>
340050ea24SMichal Meloun #include <sys/bus.h>
350050ea24SMichal Meloun
36*be82b3a0SEmmanuel Vadot #include <dev/clk/clk.h>
370050ea24SMichal Meloun
380050ea24SMichal Meloun #include <arm/ti/clk/ti_clk_dpll.h>
390050ea24SMichal Meloun
400050ea24SMichal Meloun #include "clkdev_if.h"
410050ea24SMichal Meloun
420050ea24SMichal Meloun /*
430050ea24SMichal Meloun * clknode for clocks matching the formula :
440050ea24SMichal Meloun *
450050ea24SMichal Meloun * clk = clkin * n / p
460050ea24SMichal Meloun *
470050ea24SMichal Meloun */
480050ea24SMichal Meloun
490050ea24SMichal Meloun struct ti_dpll_clknode_sc {
500050ea24SMichal Meloun uint32_t ti_clkmode_offset; /* control */
510050ea24SMichal Meloun uint8_t ti_clkmode_flags;
520050ea24SMichal Meloun
530050ea24SMichal Meloun uint32_t ti_idlest_offset;
540050ea24SMichal Meloun
550050ea24SMichal Meloun uint32_t ti_clksel_offset; /* mult-div1 */
560050ea24SMichal Meloun struct ti_clk_factor n; /* ti_clksel_mult */
570050ea24SMichal Meloun struct ti_clk_factor p; /* ti_clksel_div */
580050ea24SMichal Meloun
590050ea24SMichal Meloun uint32_t ti_autoidle_offset;
600050ea24SMichal Meloun };
610050ea24SMichal Meloun
620050ea24SMichal Meloun #define WRITE4(_clk, off, val) \
630050ea24SMichal Meloun CLKDEV_WRITE_4(clknode_get_device(_clk), off, val)
640050ea24SMichal Meloun #define READ4(_clk, off, val) \
650050ea24SMichal Meloun CLKDEV_READ_4(clknode_get_device(_clk), off, val)
660050ea24SMichal Meloun #define DEVICE_LOCK(_clk) \
670050ea24SMichal Meloun CLKDEV_DEVICE_LOCK(clknode_get_device(_clk))
680050ea24SMichal Meloun #define DEVICE_UNLOCK(_clk) \
690050ea24SMichal Meloun CLKDEV_DEVICE_UNLOCK(clknode_get_device(_clk))
700050ea24SMichal Meloun
710050ea24SMichal Meloun static int
ti_dpll_clk_init(struct clknode * clk,device_t dev)720050ea24SMichal Meloun ti_dpll_clk_init(struct clknode *clk, device_t dev)
730050ea24SMichal Meloun {
740050ea24SMichal Meloun clknode_init_parent_idx(clk, 0);
750050ea24SMichal Meloun return (0);
760050ea24SMichal Meloun }
770050ea24SMichal Meloun
780050ea24SMichal Meloun /* helper to keep aw_clk_np_find_best "intact" */
790050ea24SMichal Meloun static inline uint32_t
ti_clk_factor_get_max(struct ti_clk_factor * factor)800050ea24SMichal Meloun ti_clk_factor_get_max(struct ti_clk_factor *factor)
810050ea24SMichal Meloun {
820050ea24SMichal Meloun uint32_t max;
830050ea24SMichal Meloun
840050ea24SMichal Meloun if (factor->flags & TI_CLK_FACTOR_FIXED)
850050ea24SMichal Meloun max = factor->value;
860050ea24SMichal Meloun else {
870050ea24SMichal Meloun max = (1 << factor->width);
880050ea24SMichal Meloun }
890050ea24SMichal Meloun
900050ea24SMichal Meloun return (max);
910050ea24SMichal Meloun }
920050ea24SMichal Meloun
930050ea24SMichal Meloun static inline uint32_t
ti_clk_factor_get_min(struct ti_clk_factor * factor)940050ea24SMichal Meloun ti_clk_factor_get_min(struct ti_clk_factor *factor)
950050ea24SMichal Meloun {
960050ea24SMichal Meloun uint32_t min;
970050ea24SMichal Meloun
980050ea24SMichal Meloun if (factor->flags & TI_CLK_FACTOR_FIXED)
990050ea24SMichal Meloun min = factor->value;
1000050ea24SMichal Meloun else if (factor->flags & TI_CLK_FACTOR_ZERO_BASED)
1010050ea24SMichal Meloun min = 0;
1020050ea24SMichal Meloun else if (factor->flags & TI_CLK_FACTOR_MIN_VALUE)
1030050ea24SMichal Meloun min = factor->min_value;
1040050ea24SMichal Meloun else
1050050ea24SMichal Meloun min = 1;
1060050ea24SMichal Meloun
1070050ea24SMichal Meloun return (min);
1080050ea24SMichal Meloun }
1090050ea24SMichal Meloun
1100050ea24SMichal Meloun static uint64_t
ti_dpll_clk_find_best(struct ti_dpll_clknode_sc * sc,uint64_t fparent,uint64_t * fout,uint32_t * factor_n,uint32_t * factor_p)1110050ea24SMichal Meloun ti_dpll_clk_find_best(struct ti_dpll_clknode_sc *sc, uint64_t fparent,
1120050ea24SMichal Meloun uint64_t *fout, uint32_t *factor_n, uint32_t *factor_p)
1130050ea24SMichal Meloun {
1140050ea24SMichal Meloun uint64_t cur, best;
1150050ea24SMichal Meloun uint32_t n, p, max_n, max_p, min_n, min_p;
1160050ea24SMichal Meloun
1170050ea24SMichal Meloun *factor_n = *factor_p = 0;
1180050ea24SMichal Meloun
1190050ea24SMichal Meloun max_n = ti_clk_factor_get_max(&sc->n);
1200050ea24SMichal Meloun max_p = ti_clk_factor_get_max(&sc->p);
1210050ea24SMichal Meloun min_n = ti_clk_factor_get_min(&sc->n);
1220050ea24SMichal Meloun min_p = ti_clk_factor_get_min(&sc->p);
1230050ea24SMichal Meloun
1240050ea24SMichal Meloun for (p = min_p; p <= max_p; ) {
1250050ea24SMichal Meloun for (n = min_n; n <= max_n; ) {
1260050ea24SMichal Meloun cur = fparent * n / p;
1270050ea24SMichal Meloun if (abs(*fout - cur) < abs(*fout - best)) {
1280050ea24SMichal Meloun best = cur;
1290050ea24SMichal Meloun *factor_n = n;
1300050ea24SMichal Meloun *factor_p = p;
1310050ea24SMichal Meloun }
1320050ea24SMichal Meloun
1330050ea24SMichal Meloun n++;
1340050ea24SMichal Meloun }
1350050ea24SMichal Meloun p++;
1360050ea24SMichal Meloun }
1370050ea24SMichal Meloun
1380050ea24SMichal Meloun return (best);
1390050ea24SMichal Meloun }
1400050ea24SMichal Meloun
1410050ea24SMichal Meloun static inline uint32_t
ti_clk_get_factor(uint32_t val,struct ti_clk_factor * factor)1420050ea24SMichal Meloun ti_clk_get_factor(uint32_t val, struct ti_clk_factor *factor)
1430050ea24SMichal Meloun {
1440050ea24SMichal Meloun uint32_t factor_val;
1450050ea24SMichal Meloun
1460050ea24SMichal Meloun if (factor->flags & TI_CLK_FACTOR_FIXED)
1470050ea24SMichal Meloun return (factor->value);
1480050ea24SMichal Meloun
1490050ea24SMichal Meloun factor_val = (val & factor->mask) >> factor->shift;
1500050ea24SMichal Meloun if (!(factor->flags & TI_CLK_FACTOR_ZERO_BASED))
1510050ea24SMichal Meloun factor_val += 1;
1520050ea24SMichal Meloun
1530050ea24SMichal Meloun return (factor_val);
1540050ea24SMichal Meloun }
1550050ea24SMichal Meloun
1560050ea24SMichal Meloun static inline uint32_t
ti_clk_factor_get_value(struct ti_clk_factor * factor,uint32_t raw)1570050ea24SMichal Meloun ti_clk_factor_get_value(struct ti_clk_factor *factor, uint32_t raw)
1580050ea24SMichal Meloun {
1590050ea24SMichal Meloun uint32_t val;
1600050ea24SMichal Meloun
1610050ea24SMichal Meloun if (factor->flags & TI_CLK_FACTOR_FIXED)
1620050ea24SMichal Meloun return (factor->value);
1630050ea24SMichal Meloun
1640050ea24SMichal Meloun if (factor->flags & TI_CLK_FACTOR_ZERO_BASED)
1650050ea24SMichal Meloun val = raw;
1660050ea24SMichal Meloun else if (factor->flags & TI_CLK_FACTOR_MAX_VALUE &&
1670050ea24SMichal Meloun raw > factor->max_value)
1680050ea24SMichal Meloun val = factor->max_value;
1690050ea24SMichal Meloun else
1700050ea24SMichal Meloun val = raw - 1;
1710050ea24SMichal Meloun
1720050ea24SMichal Meloun return (val);
1730050ea24SMichal Meloun }
1740050ea24SMichal Meloun
1750050ea24SMichal Meloun static int
ti_dpll_clk_set_freq(struct clknode * clk,uint64_t fparent,uint64_t * fout,int flags,int * stop)1760050ea24SMichal Meloun ti_dpll_clk_set_freq(struct clknode *clk, uint64_t fparent, uint64_t *fout,
1770050ea24SMichal Meloun int flags, int *stop)
1780050ea24SMichal Meloun {
1790050ea24SMichal Meloun struct ti_dpll_clknode_sc *sc;
1800050ea24SMichal Meloun uint64_t cur, best;
1810050ea24SMichal Meloun uint32_t val, n, p, best_n, best_p, timeout;
1820050ea24SMichal Meloun
1830050ea24SMichal Meloun sc = clknode_get_softc(clk);
1840050ea24SMichal Meloun
1850050ea24SMichal Meloun best = cur = 0;
1860050ea24SMichal Meloun
1870050ea24SMichal Meloun best = ti_dpll_clk_find_best(sc, fparent, fout,
1880050ea24SMichal Meloun &best_n, &best_p);
1890050ea24SMichal Meloun
1900050ea24SMichal Meloun if ((flags & CLK_SET_DRYRUN) != 0) {
1910050ea24SMichal Meloun *fout = best;
1920050ea24SMichal Meloun *stop = 1;
1930050ea24SMichal Meloun return (0);
1940050ea24SMichal Meloun }
1950050ea24SMichal Meloun
1960050ea24SMichal Meloun if ((best < *fout) &&
1970050ea24SMichal Meloun (flags == CLK_SET_ROUND_DOWN)) {
1980050ea24SMichal Meloun *stop = 1;
1990050ea24SMichal Meloun return (ERANGE);
2000050ea24SMichal Meloun }
2010050ea24SMichal Meloun if ((best > *fout) &&
2020050ea24SMichal Meloun (flags == CLK_SET_ROUND_UP)) {
2030050ea24SMichal Meloun *stop = 1;
2040050ea24SMichal Meloun return (ERANGE);
2050050ea24SMichal Meloun }
2060050ea24SMichal Meloun
2070050ea24SMichal Meloun DEVICE_LOCK(clk);
2080050ea24SMichal Meloun /* 1 switch PLL to bypass mode */
2090050ea24SMichal Meloun WRITE4(clk, sc->ti_clkmode_offset, DPLL_EN_MN_BYPASS_MODE);
2100050ea24SMichal Meloun
2110050ea24SMichal Meloun /* 2 Ensure PLL is in bypass */
2120050ea24SMichal Meloun timeout = 10000;
2130050ea24SMichal Meloun do {
2140050ea24SMichal Meloun DELAY(10);
2150050ea24SMichal Meloun READ4(clk, sc->ti_idlest_offset, &val);
2160050ea24SMichal Meloun } while (!(val & ST_MN_BYPASS_MASK) && timeout--);
2170050ea24SMichal Meloun
2180050ea24SMichal Meloun if (timeout == 0) {
2190050ea24SMichal Meloun DEVICE_UNLOCK(clk);
2200050ea24SMichal Meloun return (ERANGE); // FIXME: Better return value?
2210050ea24SMichal Meloun }
2220050ea24SMichal Meloun
2230050ea24SMichal Meloun /* 3 Set DPLL_MULT & DPLL_DIV bits */
2240050ea24SMichal Meloun READ4(clk, sc->ti_clksel_offset, &val);
2250050ea24SMichal Meloun
2260050ea24SMichal Meloun n = ti_clk_factor_get_value(&sc->n, best_n);
2270050ea24SMichal Meloun p = ti_clk_factor_get_value(&sc->p, best_p);
2280050ea24SMichal Meloun val &= ~sc->n.mask;
2290050ea24SMichal Meloun val &= ~sc->p.mask;
2300050ea24SMichal Meloun val |= n << sc->n.shift;
2310050ea24SMichal Meloun val |= p << sc->p.shift;
2320050ea24SMichal Meloun
2330050ea24SMichal Meloun WRITE4(clk, sc->ti_clksel_offset, val);
2340050ea24SMichal Meloun
2350050ea24SMichal Meloun /* 4. configure M2, M4, M5 and M6 */
2360050ea24SMichal Meloun /*
2370050ea24SMichal Meloun * FIXME: According to documentation M2/M4/M5/M6 can be set "later"
2380050ea24SMichal Meloun * See note in TRM 8.1.6.7.1
2390050ea24SMichal Meloun */
2400050ea24SMichal Meloun
2410050ea24SMichal Meloun /* 5 Switch over to lock mode */
2420050ea24SMichal Meloun WRITE4(clk, sc->ti_clkmode_offset, DPLL_EN_LOCK_MODE);
2430050ea24SMichal Meloun
2440050ea24SMichal Meloun /* 6 Ensure PLL is locked */
2450050ea24SMichal Meloun timeout = 10000;
2460050ea24SMichal Meloun do {
2470050ea24SMichal Meloun DELAY(10);
2480050ea24SMichal Meloun READ4(clk, sc->ti_idlest_offset, &val);
2490050ea24SMichal Meloun } while (!(val & ST_DPLL_CLK_MASK) && timeout--);
2500050ea24SMichal Meloun
2510050ea24SMichal Meloun DEVICE_UNLOCK(clk);
2520050ea24SMichal Meloun if (timeout == 0) {
2530050ea24SMichal Meloun return (ERANGE); // FIXME: Better return value?
2540050ea24SMichal Meloun }
2550050ea24SMichal Meloun
2560050ea24SMichal Meloun *fout = best;
2570050ea24SMichal Meloun *stop = 1;
2580050ea24SMichal Meloun
2590050ea24SMichal Meloun return (0);
2600050ea24SMichal Meloun }
2610050ea24SMichal Meloun
2620050ea24SMichal Meloun static int
ti_dpll_clk_recalc(struct clknode * clk,uint64_t * freq)2630050ea24SMichal Meloun ti_dpll_clk_recalc(struct clknode *clk, uint64_t *freq)
2640050ea24SMichal Meloun {
2650050ea24SMichal Meloun struct ti_dpll_clknode_sc *sc;
2660050ea24SMichal Meloun uint32_t val, n, p;
2670050ea24SMichal Meloun
2680050ea24SMichal Meloun sc = clknode_get_softc(clk);
2690050ea24SMichal Meloun
2700050ea24SMichal Meloun DEVICE_LOCK(clk);
2710050ea24SMichal Meloun READ4(clk, sc->ti_clksel_offset, &val);
2720050ea24SMichal Meloun DEVICE_UNLOCK(clk);
2730050ea24SMichal Meloun
2740050ea24SMichal Meloun n = ti_clk_get_factor(val, &sc->n);
2750050ea24SMichal Meloun p = ti_clk_get_factor(val, &sc->p);
2760050ea24SMichal Meloun
2770050ea24SMichal Meloun *freq = *freq * n / p;
2780050ea24SMichal Meloun
2790050ea24SMichal Meloun return (0);
2800050ea24SMichal Meloun }
2810050ea24SMichal Meloun
2820050ea24SMichal Meloun static clknode_method_t ti_dpll_clknode_methods[] = {
2830050ea24SMichal Meloun /* Device interface */
2840050ea24SMichal Meloun CLKNODEMETHOD(clknode_init, ti_dpll_clk_init),
2850050ea24SMichal Meloun CLKNODEMETHOD(clknode_recalc_freq, ti_dpll_clk_recalc),
2860050ea24SMichal Meloun CLKNODEMETHOD(clknode_set_freq, ti_dpll_clk_set_freq),
2870050ea24SMichal Meloun CLKNODEMETHOD_END
2880050ea24SMichal Meloun };
2890050ea24SMichal Meloun
2900050ea24SMichal Meloun DEFINE_CLASS_1(ti_dpll_clknode, ti_dpll_clknode_class, ti_dpll_clknode_methods,
2910050ea24SMichal Meloun sizeof(struct ti_dpll_clknode_sc), clknode_class);
2920050ea24SMichal Meloun
2930050ea24SMichal Meloun int
ti_clknode_dpll_register(struct clkdom * clkdom,struct ti_clk_dpll_def * clkdef)2940050ea24SMichal Meloun ti_clknode_dpll_register(struct clkdom *clkdom, struct ti_clk_dpll_def *clkdef)
2950050ea24SMichal Meloun {
2960050ea24SMichal Meloun struct clknode *clk;
2970050ea24SMichal Meloun struct ti_dpll_clknode_sc *sc;
2980050ea24SMichal Meloun
2990050ea24SMichal Meloun clk = clknode_create(clkdom, &ti_dpll_clknode_class, &clkdef->clkdef);
3000050ea24SMichal Meloun if (clk == NULL)
3010050ea24SMichal Meloun return (1);
3020050ea24SMichal Meloun
3030050ea24SMichal Meloun sc = clknode_get_softc(clk);
3040050ea24SMichal Meloun
3050050ea24SMichal Meloun sc->ti_clkmode_offset = clkdef->ti_clkmode_offset;
3060050ea24SMichal Meloun sc->ti_clkmode_flags = clkdef->ti_clkmode_flags;
3070050ea24SMichal Meloun sc->ti_idlest_offset = clkdef->ti_idlest_offset;
3080050ea24SMichal Meloun sc->ti_clksel_offset = clkdef->ti_clksel_offset;
3090050ea24SMichal Meloun
3100050ea24SMichal Meloun sc->n.shift = clkdef->ti_clksel_mult.shift;
3110050ea24SMichal Meloun sc->n.mask = clkdef->ti_clksel_mult.mask;
3120050ea24SMichal Meloun sc->n.width = clkdef->ti_clksel_mult.width;
3130050ea24SMichal Meloun sc->n.value = clkdef->ti_clksel_mult.value;
3140050ea24SMichal Meloun sc->n.min_value = clkdef->ti_clksel_mult.min_value;
3150050ea24SMichal Meloun sc->n.max_value = clkdef->ti_clksel_mult.max_value;
3160050ea24SMichal Meloun sc->n.flags = clkdef->ti_clksel_mult.flags;
3170050ea24SMichal Meloun
3180050ea24SMichal Meloun sc->p.shift = clkdef->ti_clksel_div.shift;
3190050ea24SMichal Meloun sc->p.mask = clkdef->ti_clksel_div.mask;
3200050ea24SMichal Meloun sc->p.width = clkdef->ti_clksel_div.width;
3210050ea24SMichal Meloun sc->p.value = clkdef->ti_clksel_div.value;
3220050ea24SMichal Meloun sc->p.min_value = clkdef->ti_clksel_div.min_value;
3230050ea24SMichal Meloun sc->p.max_value = clkdef->ti_clksel_div.max_value;
3240050ea24SMichal Meloun sc->p.flags = clkdef->ti_clksel_div.flags;
3250050ea24SMichal Meloun
3260050ea24SMichal Meloun sc->ti_autoidle_offset = clkdef->ti_autoidle_offset;
3270050ea24SMichal Meloun
3280050ea24SMichal Meloun clknode_register(clkdom, clk);
3290050ea24SMichal Meloun
3300050ea24SMichal Meloun return (0);
3310050ea24SMichal Meloun }
332