2 * Copyright (c) 2016 Michal Meloun <mmel@FreeBSD.org>
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
14 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27 #include <sys/cdefs.h>
28 __FBSDID("$FreeBSD$");
30 #include <sys/param.h>
31 #include <sys/systm.h>
34 #include <sys/mutex.h>
37 #include <machine/bus.h>
39 #include <dev/extres/clk/clk.h>
41 #include <gnu/dts/include/dt-bindings/clock/tegra124-car.h>
42 #include "tegra124_car.h"
46 #define SMF_HAVE_DIVIDER_2 1
48 struct super_mux_def {
49 struct clknode_init_def clkdef;
56 #define PLIST(x) static const char *x[]
57 #define SM(_id, cn, pl, r, x, d, f) \
61 .clkdef.parent_names = pl, \
62 .clkdef.parent_cnt = nitems(pl), \
63 .clkdef.flags = CLK_NODE_STATIC_STRINGS, \
70 PLIST(cclk_g_parents) = {
71 "clk_m", "pllC_out0", "clk_s", "pllM_out0",
72 "pllP_out0", "pllP_out4", "pllC2_out0", "pllC3_out0",
73 "pllX_out", NULL, NULL, NULL,
74 NULL, NULL, NULL,NULL, // "dfllCPU_out0"
77 PLIST(cclk_lp_parents) = {
78 "clk_m", "pllC_out0", "clk_s", "pllM_out0",
79 "pllP_out0", "pllP_out4", "pllC2_out0", "pllC3_out0",
80 "pllX_out", NULL, NULL, NULL,
81 NULL, NULL, NULL, NULL,
85 PLIST(sclk_parents) = {
86 "clk_m", "pllC_out1", "pllP_out4", "pllP_out0",
87 "pllP_out2", "pllC_out0", "clk_s", "pllM_out1",
90 static struct super_mux_def super_mux_def[] = {
91 SM(TEGRA124_CLK_CCLK_G, "cclk_g", cclk_g_parents, CCLKG_BURST_POLICY, 0, 0, 0),
92 SM(TEGRA124_CLK_CCLK_LP, "cclk_lp", cclk_lp_parents, CCLKLP_BURST_POLICY, 8, 16, SMF_HAVE_DIVIDER_2),
93 SM(TEGRA124_CLK_SCLK, "sclk", sclk_parents, SCLK_BURST_POLICY, 0, 0, 0),
96 static int super_mux_init(struct clknode *clk, device_t dev);
97 static int super_mux_set_mux(struct clknode *clk, int idx);
109 static clknode_method_t super_mux_methods[] = {
110 /* Device interface */
111 CLKNODEMETHOD(clknode_init, super_mux_init),
112 CLKNODEMETHOD(clknode_set_mux, super_mux_set_mux),
115 DEFINE_CLASS_1(tegra124_super_mux, tegra124_super_mux_class, super_mux_methods,
116 sizeof(struct super_mux_sc), clknode_class);
119 #define SUPER_MUX_STATE_STDBY 0
120 #define SUPER_MUX_STATE_IDLE 1
121 #define SUPER_MUX_STATE_RUN 2
122 #define SUPER_MUX_STATE_IRQ 3
123 #define SUPER_MUX_STATE_FIQ 4
125 /* Mux register bits. */
126 #define SUPER_MUX_STATE_BIT_SHIFT 28
127 #define SUPER_MUX_STATE_BIT_MASK 0xF
128 /* State is Priority encoded */
129 #define SUPER_MUX_STATE_BIT_STDBY 0x00
130 #define SUPER_MUX_STATE_BIT_IDLE 0x01
131 #define SUPER_MUX_STATE_BIT_RUN 0x02
132 #define SUPER_MUX_STATE_BIT_IRQ 0x04
133 #define SUPER_MUX_STATE_BIT_FIQ 0x08
135 #define SUPER_MUX_MUX_WIDTH 4
136 #define SUPER_MUX_LP_DIV2_BYPASS (1 << 16)
139 super_mux_get_state(uint32_t reg)
141 reg = (reg >> SUPER_MUX_STATE_BIT_SHIFT) & SUPER_MUX_STATE_BIT_MASK;
142 if (reg & SUPER_MUX_STATE_BIT_FIQ)
143 return (SUPER_MUX_STATE_FIQ);
144 if (reg & SUPER_MUX_STATE_BIT_IRQ)
145 return (SUPER_MUX_STATE_IRQ);
146 if (reg & SUPER_MUX_STATE_BIT_RUN)
147 return (SUPER_MUX_STATE_RUN);
148 if (reg & SUPER_MUX_STATE_BIT_IDLE)
149 return (SUPER_MUX_STATE_IDLE);
150 return (SUPER_MUX_STATE_STDBY);
154 super_mux_init(struct clknode *clk, device_t dev)
156 struct super_mux_sc *sc;
160 sc = clknode_get_softc(clk);
163 RD4(sc, sc->base_reg, ®);
165 state = super_mux_get_state(reg);
167 if ((state != SUPER_MUX_STATE_RUN) &&
168 (state != SUPER_MUX_STATE_IDLE)) {
169 panic("Unexpected super mux state: %u", state);
172 shift = state * SUPER_MUX_MUX_WIDTH;
174 sc->mux = (reg >> shift) & ((1 << SUPER_MUX_MUX_WIDTH) - 1);
177 * CCLKLP uses PLLX/2 as source if LP_DIV2_BYPASS isn't set
178 * and source mux is set to PLLX.
180 if (sc->flags & SMF_HAVE_DIVIDER_2) {
181 if (((reg & SUPER_MUX_LP_DIV2_BYPASS) == 0) &&
182 (sc->mux == sc->src_pllx))
183 sc->mux = sc->src_div2;
185 clknode_init_parent_idx(clk, sc->mux);
191 super_mux_set_mux(struct clknode *clk, int idx)
194 struct super_mux_sc *sc;
198 sc = clknode_get_softc(clk);
201 RD4(sc, sc->base_reg, ®);
202 state = super_mux_get_state(reg);
204 if ((state != SUPER_MUX_STATE_RUN) &&
205 (state != SUPER_MUX_STATE_IDLE)) {
206 panic("Unexpected super mux state: %u", state);
208 shift = (state - 1) * SUPER_MUX_MUX_WIDTH;
210 if (sc->flags & SMF_HAVE_DIVIDER_2) {
211 if (idx == sc->src_div2) {
213 reg &= ~SUPER_MUX_LP_DIV2_BYPASS;
214 WR4(sc, sc->base_reg, reg);
215 RD4(sc, sc->base_reg, &dummy);
216 } else if (idx == sc->src_pllx) {
217 reg = SUPER_MUX_LP_DIV2_BYPASS;
218 WR4(sc, sc->base_reg, reg);
219 RD4(sc, sc->base_reg, &dummy);
222 reg &= ~(((1 << SUPER_MUX_MUX_WIDTH) - 1) << shift);
225 WR4(sc, sc->base_reg, reg);
226 RD4(sc, sc->base_reg, &dummy);
233 super_mux_register(struct clkdom *clkdom, struct super_mux_def *clkdef)
236 struct super_mux_sc *sc;
238 clk = clknode_create(clkdom, &tegra124_super_mux_class,
243 sc = clknode_get_softc(clk);
244 sc->clkdev = clknode_get_device(clk);
245 sc->base_reg = clkdef->base_reg;
246 sc->src_pllx = clkdef->src_pllx;
247 sc->src_div2 = clkdef->src_div2;
248 sc->flags = clkdef->flags;
250 clknode_register(clkdom, clk);
255 tegra124_super_mux_clock(struct tegra124_car_softc *sc)
259 for (i = 0; i < nitems(super_mux_def); i++) {
260 rv = super_mux_register(sc->clkdom, &super_mux_def[i]);
262 panic("super_mux_register failed");