xref: /freebsd/sys/arm/ti/clk/ti_clk_dpll.c (revision be82b3a0bf72ed3b5f01ac9fcd8dcd3802e3c742)
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