xref: /freebsd/sys/arm64/nvidia/tegra210/tegra210_pmc.c (revision b9cd72b06d795a8c7b39df1f520e866ad7f11aa8)
1e9034789SMichal Meloun /*-
24d846d26SWarner Losh  * SPDX-License-Identifier: BSD-2-Clause
3e9034789SMichal Meloun  *
4e9034789SMichal Meloun  * Copyright 2020 Michal Meloun <mmel@FreeBSD.org>
5e9034789SMichal Meloun  *
6e9034789SMichal Meloun  * Redistribution and use in source and binary forms, with or without
7e9034789SMichal Meloun  * modification, are permitted provided that the following conditions
8e9034789SMichal Meloun  * are met:
9e9034789SMichal Meloun  * 1. Redistributions of source code must retain the above copyright
10e9034789SMichal Meloun  *    notice, this list of conditions and the following disclaimer.
11e9034789SMichal Meloun  * 2. Redistributions in binary form must reproduce the above copyright
12e9034789SMichal Meloun  *    notice, this list of conditions and the following disclaimer in the
13e9034789SMichal Meloun  *    documentation and/or other materials provided with the distribution.
14e9034789SMichal Meloun  *
15e9034789SMichal Meloun  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
16e9034789SMichal Meloun  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17e9034789SMichal Meloun  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18e9034789SMichal Meloun  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
19e9034789SMichal Meloun  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20e9034789SMichal Meloun  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
21e9034789SMichal Meloun  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22e9034789SMichal Meloun  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23e9034789SMichal Meloun  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24e9034789SMichal Meloun  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25e9034789SMichal Meloun  * SUCH DAMAGE.
26e9034789SMichal Meloun  */
27e9034789SMichal Meloun 
28e9034789SMichal Meloun #include <sys/param.h>
29e9034789SMichal Meloun #include <sys/systm.h>
30e9034789SMichal Meloun #include <sys/bus.h>
31e9034789SMichal Meloun #include <sys/kernel.h>
32e9034789SMichal Meloun #include <sys/module.h>
33e9034789SMichal Meloun #include <sys/malloc.h>
34e9034789SMichal Meloun #include <sys/mutex.h>
35e9034789SMichal Meloun #include <sys/rman.h>
36e9034789SMichal Meloun 
37e9034789SMichal Meloun #include <machine/bus.h>
38e9034789SMichal Meloun 
39be82b3a0SEmmanuel Vadot #include <dev/clk/clk.h>
401f469a9fSEmmanuel Vadot #include <dev/hwreset/hwreset.h>
41e9034789SMichal Meloun #include <dev/ofw/ofw_bus.h>
42e9034789SMichal Meloun #include <dev/ofw/ofw_bus_subr.h>
43e9034789SMichal Meloun #include <dev/psci/smccc.h>
44e9034789SMichal Meloun 
45e9034789SMichal Meloun #include <arm/nvidia/tegra_pmc.h>
46e9034789SMichal Meloun 
47e9034789SMichal Meloun #define	PMC_CNTRL			0x000
48e9034789SMichal Meloun #define	 PMC_CNTRL_SHUTDOWN_OE			(1 << 22)
49e9034789SMichal Meloun #define	 PMC_CNTRL_CPUPWRGOOD_SEL_MASK		(0x3 << 20)
50e9034789SMichal Meloun #define	 PMC_CNTRL_CPUPWRGOOD_SEL_SHIFT		20
51e9034789SMichal Meloun #define	 PMC_CNTRL_CPUPWRGOOD_EN		(1 << 19)
52e9034789SMichal Meloun #define	 PMC_CNTRL_FUSE_OVERRIDE		(1 << 18)
53e9034789SMichal Meloun #define	 PMC_CNTRL_INTR_POLARITY		(1 << 17)
54e9034789SMichal Meloun #define	 PMC_CNTRL_CPU_PWRREQ_OE		(1 << 16)
55e9034789SMichal Meloun #define	 PMC_CNTRL_CPU_PWRREQ_POLARITY		(1 << 15)
56e9034789SMichal Meloun #define	 PMC_CNTRL_SIDE_EFFECT_LP0		(1 << 14)
57e9034789SMichal Meloun #define	 PMC_CNTRL_AOINIT			(1 << 13)
58e9034789SMichal Meloun #define	 PMC_CNTRL_PWRGATE_DIS			(1 << 12)
59e9034789SMichal Meloun #define	 PMC_CNTRL_SYSCLK_OE			(1 << 11)
60e9034789SMichal Meloun #define	 PMC_CNTRL_SYSCLK_POLARITY		(1 << 10)
61e9034789SMichal Meloun #define	 PMC_CNTRL_PWRREQ_OE			(1 <<  9)
62e9034789SMichal Meloun #define	 PMC_CNTRL_PWRREQ_POLARITY		(1 <<  8)
63e9034789SMichal Meloun #define	 PMC_CNTRL_BLINK_EN			(1 <<  7)
64e9034789SMichal Meloun #define	 PMC_CNTRL_GLITCHDET_DIS		(1 <<  6)
65e9034789SMichal Meloun #define	 PMC_CNTRL_LATCHWAKE_EN			(1 <<  5)
66e9034789SMichal Meloun #define	 PMC_CNTRL_MAIN_RST			(1 <<  4)
67e9034789SMichal Meloun #define	 PMC_CNTRL_KBC_RST			(1 <<  3)
68e9034789SMichal Meloun #define	 PMC_CNTRL_RTC_RST			(1 <<  2)
69e9034789SMichal Meloun #define	 PMC_CNTRL_RTC_CLK_DIS			(1 <<  1)
70e9034789SMichal Meloun #define	 PMC_CNTRL_KBC_CLK_DIS			(1 <<  0)
71e9034789SMichal Meloun 
72e9034789SMichal Meloun #define	PMC_DPD_SAMPLE			0x020
73e9034789SMichal Meloun 
74e9034789SMichal Meloun #define	PMC_CLAMP_STATUS		0x02C
75e9034789SMichal Meloun #define	  PMC_CLAMP_STATUS_PARTID(x)		(1 << ((x) & 0x1F))
76e9034789SMichal Meloun 
77e9034789SMichal Meloun #define	PMC_PWRGATE_TOGGLE		0x030
78e9034789SMichal Meloun #define	 PMC_PWRGATE_TOGGLE_START		(1 << 8)
79e9034789SMichal Meloun #define	 PMC_PWRGATE_TOGGLE_PARTID(x)		(((x) & 0x1F) << 0)
80e9034789SMichal Meloun 
81e9034789SMichal Meloun #define	PMC_REMOVE_CLAMPING_CMD		0x034
82e9034789SMichal Meloun #define	  PMC_REMOVE_CLAMPING_CMD_PARTID(x)	(1 << ((x) & 0x1F))
83e9034789SMichal Meloun 
84e9034789SMichal Meloun #define	PMC_PWRGATE_STATUS		0x038
85e9034789SMichal Meloun #define	PMC_PWRGATE_STATUS_PARTID(x)		(1 << ((x) & 0x1F))
86e9034789SMichal Meloun 
87e9034789SMichal Meloun #define	PMC_SCRATCH0			0x050
88e9034789SMichal Meloun #define	 PMC_SCRATCH0_MODE_RECOVERY		(1 << 31)
89e9034789SMichal Meloun #define	 PMC_SCRATCH0_MODE_BOOTLOADER		(1 << 30)
90e9034789SMichal Meloun #define	 PMC_SCRATCH0_MODE_RCM			(1 << 1)
91e9034789SMichal Meloun #define	 PMC_SCRATCH0_MODE_MASK			(PMC_SCRATCH0_MODE_RECOVERY | \
92e9034789SMichal Meloun 						PMC_SCRATCH0_MODE_BOOTLOADER | \
93e9034789SMichal Meloun 						PMC_SCRATCH0_MODE_RCM)
94e9034789SMichal Meloun 
95e9034789SMichal Meloun #define	PMC_CPUPWRGOOD_TIMER		0x0c8
96e9034789SMichal Meloun #define	PMC_CPUPWROFF_TIMER		0x0cc
97e9034789SMichal Meloun 
98e9034789SMichal Meloun #define	PMC_SCRATCH41			0x140
99e9034789SMichal Meloun 
100e9034789SMichal Meloun #define	PMC_SENSOR_CTRL			0x1b0
101e9034789SMichal Meloun #define	PMC_SENSOR_CTRL_BLOCK_SCRATCH_WRITE	(1 << 2)
102e9034789SMichal Meloun #define	PMC_SENSOR_CTRL_ENABLE_RST		(1 << 1)
103e9034789SMichal Meloun #define	PMC_SENSOR_CTRL_ENABLE_PG		(1 << 0)
104e9034789SMichal Meloun 
105e9034789SMichal Meloun #define	PMC_IO_DPD_REQ			0x1b8
106e9034789SMichal Meloun #define	 PMC_IO_DPD_REQ_CODE_IDLE		(0 << 30)
107e9034789SMichal Meloun #define	 PMC_IO_DPD_REQ_CODE_OFF		(1 << 30)
108e9034789SMichal Meloun #define	 PMC_IO_DPD_REQ_CODE_ON			(2 << 30)
109e9034789SMichal Meloun #define	 PMC_IO_DPD_REQ_CODE_MASK		(3 << 30)
110e9034789SMichal Meloun 
111e9034789SMichal Meloun #define	PMC_IO_DPD_STATUS		0x1bc
112e9034789SMichal Meloun #define	 PMC_IO_DPD_STATUS_HDMI			(1 << 28)
113e9034789SMichal Meloun #define	PMC_IO_DPD2_REQ			0x1c0
114e9034789SMichal Meloun #define	PMC_IO_DPD2_STATUS		0x1c4
115e9034789SMichal Meloun #define	 PMC_IO_DPD2_STATUS_HV			(1 << 6)
116e9034789SMichal Meloun #define	PMC_SEL_DPD_TIM			0x1c8
117e9034789SMichal Meloun 
118e9034789SMichal Meloun #define	PMC_SCRATCH54			0x258
119e9034789SMichal Meloun #define	PMC_SCRATCH54_DATA_SHIFT		8
120e9034789SMichal Meloun #define	PMC_SCRATCH54_ADDR_SHIFT		0
121e9034789SMichal Meloun 
122e9034789SMichal Meloun #define	PMC_SCRATCH55			0x25c
123e9034789SMichal Meloun #define	PMC_SCRATCH55_RST_ENABLE		(1 << 31)
124e9034789SMichal Meloun #define	PMC_SCRATCH55_CNTRL_TYPE		(1 << 30)
125e9034789SMichal Meloun #define	PMC_SCRATCH55_CNTRL_ID_SHIFT		27
126e9034789SMichal Meloun #define	PMC_SCRATCH55_CNTRL_ID_MASK		0x07
127e9034789SMichal Meloun #define	PMC_SCRATCH55_PINMUX_SHIFT		24
128e9034789SMichal Meloun #define	PMC_SCRATCH55_PINMUX_MASK		0x07
129e9034789SMichal Meloun #define	PMC_SCRATCH55_CHECKSUM_SHIFT		16
130e9034789SMichal Meloun #define	PMC_SCRATCH55_CHECKSUM_MASK		0xFF
131e9034789SMichal Meloun #define	PMC_SCRATCH55_16BITOP			(1 << 15)
132e9034789SMichal Meloun #define	PMC_SCRATCH55_I2CSLV1_SHIFT		0
133e9034789SMichal Meloun #define	PMC_SCRATCH55_I2CSLV1_MASK		0x7F
134e9034789SMichal Meloun 
135e9034789SMichal Meloun #define	PMC_GPU_RG_CNTRL		0x2d4
136e9034789SMichal Meloun 
137e9034789SMichal Meloun /* Secure access */
138e9034789SMichal Meloun #define	PMC_SMC				0xc2fffe00
139e9034789SMichal Meloun #define	PMC_SMC_READ			0xaa
140e9034789SMichal Meloun #define	PMC_SMC_WRITE			0xbb
141e9034789SMichal Meloun 
142e9034789SMichal Meloun #define	PMC_LOCK(_sc)		mtx_lock(&(_sc)->mtx)
143e9034789SMichal Meloun #define	PMC_UNLOCK(_sc)		mtx_unlock(&(_sc)->mtx)
144e9034789SMichal Meloun #define	PMC_LOCK_INIT(_sc)	mtx_init(&(_sc)->mtx, 			\
145e9034789SMichal Meloun 	    device_get_nameunit(_sc->dev), "tegra210_pmc", MTX_DEF)
146e9034789SMichal Meloun #define	PMC_LOCK_DESTROY(_sc)	mtx_destroy(&(_sc)->mtx);
147e9034789SMichal Meloun #define	PMC_ASSERT_LOCKED(_sc)	mtx_assert(&(_sc)->mtx, MA_OWNED);
148e9034789SMichal Meloun #define	PMC_ASSERT_UNLOCKED(_sc) mtx_assert(&(_sc)->mtx, MA_NOTOWNED);
149e9034789SMichal Meloun 
150e9034789SMichal Meloun struct tegra210_pmc_softc {
151e9034789SMichal Meloun 	device_t		dev;
152e9034789SMichal Meloun 	struct resource		*mem_res;
153e9034789SMichal Meloun 	clk_t			clk;
154e9034789SMichal Meloun 	struct mtx		mtx;
155e9034789SMichal Meloun 	bool			secure_access;
156e9034789SMichal Meloun 
157e9034789SMichal Meloun 	uint32_t		rate;
158e9034789SMichal Meloun 	enum tegra_suspend_mode suspend_mode;
159e9034789SMichal Meloun 	uint32_t		cpu_good_time;
160e9034789SMichal Meloun 	uint32_t		cpu_off_time;
161e9034789SMichal Meloun 	uint32_t		core_osc_time;
162e9034789SMichal Meloun 	uint32_t		core_pmu_time;
163e9034789SMichal Meloun 	uint32_t		core_off_time;
164e9034789SMichal Meloun 	int			corereq_high;
165e9034789SMichal Meloun 	int			sysclkreq_high;
166e9034789SMichal Meloun 	int			combined_req;
167e9034789SMichal Meloun 	int			cpu_pwr_good_en;
168e9034789SMichal Meloun 	uint32_t		lp0_vec_phys;
169e9034789SMichal Meloun 	uint32_t		lp0_vec_size;
170e9034789SMichal Meloun };
171e9034789SMichal Meloun 
172e9034789SMichal Meloun static struct ofw_compat_data compat_data[] = {
173e9034789SMichal Meloun 	{"nvidia,tegra210-pmc",		1},
174e9034789SMichal Meloun 	{NULL,				0},
175e9034789SMichal Meloun };
176e9034789SMichal Meloun 
177e9034789SMichal Meloun static struct tegra210_pmc_softc *pmc_sc;
178e9034789SMichal Meloun 
179e9034789SMichal Meloun static inline struct tegra210_pmc_softc *
tegra210_pmc_get_sc(void)180e9034789SMichal Meloun tegra210_pmc_get_sc(void)
181e9034789SMichal Meloun {
182e9034789SMichal Meloun 	if (pmc_sc == NULL)
183e9034789SMichal Meloun 		panic("To early call to Tegra PMC driver.\n");
184e9034789SMichal Meloun 	return (pmc_sc);
185e9034789SMichal Meloun }
186e9034789SMichal Meloun 
187e9034789SMichal Meloun static void
WR4(struct tegra210_pmc_softc * sc,bus_size_t r,uint32_t v)188e9034789SMichal Meloun WR4(struct tegra210_pmc_softc *sc, bus_size_t r, uint32_t v)
189e9034789SMichal Meloun {
190e9034789SMichal Meloun 	struct arm_smccc_res res;
191e9034789SMichal Meloun 
192e9034789SMichal Meloun 	if (sc->secure_access) {
193*b9cd72b0SAndrew Turner 		arm_smccc_invoke_smc(PMC_SMC, PMC_SMC_WRITE, r, v, &res);
194e9034789SMichal Meloun 		if (res.a0 != 0)
195e9034789SMichal Meloun 			device_printf(sc->dev," PMC SMC write failed: %lu\n",
196e9034789SMichal Meloun 			    res.a0);
197e9034789SMichal Meloun 	}
198e9034789SMichal Meloun 
199e9034789SMichal Meloun 	bus_write_4(sc->mem_res, r, v);
200e9034789SMichal Meloun }
201e9034789SMichal Meloun 
202e9034789SMichal Meloun static uint32_t
RD4(struct tegra210_pmc_softc * sc,bus_size_t r)203e9034789SMichal Meloun RD4(struct tegra210_pmc_softc *sc, bus_size_t r)
204e9034789SMichal Meloun {
205e9034789SMichal Meloun 	struct arm_smccc_res res;
206e9034789SMichal Meloun 
207e9034789SMichal Meloun 	if (sc->secure_access) {
208*b9cd72b0SAndrew Turner 		arm_smccc_invoke_smc(PMC_SMC, PMC_SMC_READ, r, &res);
209e9034789SMichal Meloun 		if (res.a0 != 0)
210e9034789SMichal Meloun 			device_printf(sc->dev," PMC SMC write failed: %lu\n",
211e9034789SMichal Meloun 			    res.a0);
212e9034789SMichal Meloun 		return((uint32_t)res.a1);
213e9034789SMichal Meloun 	}
214e9034789SMichal Meloun 
215e9034789SMichal Meloun 	return(bus_read_4(sc->mem_res, r));
216e9034789SMichal Meloun }
217e9034789SMichal Meloun 
218e9034789SMichal Meloun static int
tegra210_pmc_set_powergate(struct tegra210_pmc_softc * sc,enum tegra_powergate_id id,int ena)219e9034789SMichal Meloun tegra210_pmc_set_powergate(struct tegra210_pmc_softc *sc,
220e9034789SMichal Meloun     enum tegra_powergate_id id, int ena)
221e9034789SMichal Meloun {
222e9034789SMichal Meloun 	uint32_t reg;
223e9034789SMichal Meloun 	int i;
224e9034789SMichal Meloun 
225e9034789SMichal Meloun 	PMC_LOCK(sc);
226e9034789SMichal Meloun 
227e9034789SMichal Meloun 	reg = RD4(sc, PMC_PWRGATE_STATUS) & PMC_PWRGATE_STATUS_PARTID(id);
228e9034789SMichal Meloun 	if (((reg != 0) && ena) || ((reg == 0) && !ena)) {
229e9034789SMichal Meloun 		PMC_UNLOCK(sc);
230e9034789SMichal Meloun 		return (0);
231e9034789SMichal Meloun 	}
232e9034789SMichal Meloun 
233e9034789SMichal Meloun 	for (i = 100; i > 0; i--) {
234e9034789SMichal Meloun 		reg = RD4(sc, PMC_PWRGATE_TOGGLE);
235e9034789SMichal Meloun 		if ((reg & PMC_PWRGATE_TOGGLE_START) == 0)
236e9034789SMichal Meloun 			break;
237e9034789SMichal Meloun 		DELAY(1);
238e9034789SMichal Meloun 	}
239e9034789SMichal Meloun 	if (i <= 0)
240e9034789SMichal Meloun 		device_printf(sc->dev,
241e9034789SMichal Meloun 		    "Timeout when waiting for TOGGLE_START\n");
242e9034789SMichal Meloun 
243e9034789SMichal Meloun 	WR4(sc, PMC_PWRGATE_TOGGLE,
244e9034789SMichal Meloun 	    PMC_PWRGATE_TOGGLE_START | PMC_PWRGATE_TOGGLE_PARTID(id));
245e9034789SMichal Meloun 
246e9034789SMichal Meloun 	for (i = 100; i > 0; i--) {
247e9034789SMichal Meloun 		reg = RD4(sc, PMC_PWRGATE_TOGGLE);
248e9034789SMichal Meloun 		if ((reg & PMC_PWRGATE_TOGGLE_START) == 0)
249e9034789SMichal Meloun 			break;
250e9034789SMichal Meloun 		DELAY(1);
251e9034789SMichal Meloun 	}
252e9034789SMichal Meloun 	if (i <= 0)
253e9034789SMichal Meloun 		device_printf(sc->dev,
254e9034789SMichal Meloun 		    "Timeout when waiting for TOGGLE_START\n");
255e9034789SMichal Meloun 		PMC_UNLOCK(sc);
256e9034789SMichal Meloun 	return (0);
257e9034789SMichal Meloun }
258e9034789SMichal Meloun 
259e9034789SMichal Meloun int
tegra_powergate_remove_clamping(enum tegra_powergate_id id)260e9034789SMichal Meloun tegra_powergate_remove_clamping(enum tegra_powergate_id  id)
261e9034789SMichal Meloun {
262e9034789SMichal Meloun 	struct tegra210_pmc_softc *sc;
263e9034789SMichal Meloun 	uint32_t reg;
264e9034789SMichal Meloun 	enum tegra_powergate_id swid;
265e9034789SMichal Meloun 	int i;
266e9034789SMichal Meloun 
267e9034789SMichal Meloun 	sc = tegra210_pmc_get_sc();
268e9034789SMichal Meloun 
269e9034789SMichal Meloun 	if (id == TEGRA_POWERGATE_3D) {
270e9034789SMichal Meloun 		WR4(sc, PMC_GPU_RG_CNTRL, 0);
271e9034789SMichal Meloun 		return (0);
272e9034789SMichal Meloun 	}
273e9034789SMichal Meloun 
274e9034789SMichal Meloun 	reg = RD4(sc, PMC_PWRGATE_STATUS);
275e9034789SMichal Meloun 	if ((reg & PMC_PWRGATE_STATUS_PARTID(id)) == 0)
276e9034789SMichal Meloun 		panic("Attempt to remove clamping for unpowered partition.\n");
277e9034789SMichal Meloun 
278e9034789SMichal Meloun 	if (id == TEGRA_POWERGATE_PCX)
279e9034789SMichal Meloun 		swid = TEGRA_POWERGATE_VDE;
280e9034789SMichal Meloun 	else if (id == TEGRA_POWERGATE_VDE)
281e9034789SMichal Meloun 		swid = TEGRA_POWERGATE_PCX;
282e9034789SMichal Meloun 	else
283e9034789SMichal Meloun 		swid = id;
284e9034789SMichal Meloun 	WR4(sc, PMC_REMOVE_CLAMPING_CMD, PMC_REMOVE_CLAMPING_CMD_PARTID(swid));
285e9034789SMichal Meloun 
286e9034789SMichal Meloun 	for (i = 100; i > 0; i--) {
287e9034789SMichal Meloun 		reg = RD4(sc, PMC_REMOVE_CLAMPING_CMD);
288e9034789SMichal Meloun 		if ((reg & PMC_REMOVE_CLAMPING_CMD_PARTID(swid)) == 0)
289e9034789SMichal Meloun 			break;
290e9034789SMichal Meloun 		DELAY(1);
291e9034789SMichal Meloun 	}
292e9034789SMichal Meloun 	if (i <= 0)
293e9034789SMichal Meloun 		device_printf(sc->dev, "Timeout when remove clamping\n");
294e9034789SMichal Meloun 
295e9034789SMichal Meloun 	reg = RD4(sc, PMC_CLAMP_STATUS);
296e9034789SMichal Meloun 	if ((reg & PMC_CLAMP_STATUS_PARTID(id)) != 0)
297e9034789SMichal Meloun 		panic("Cannot remove clamping\n");
298e9034789SMichal Meloun 
299e9034789SMichal Meloun 	return (0);
300e9034789SMichal Meloun }
301e9034789SMichal Meloun 
302e9034789SMichal Meloun int
tegra_powergate_is_powered(enum tegra_powergate_id id)303e9034789SMichal Meloun tegra_powergate_is_powered(enum tegra_powergate_id id)
304e9034789SMichal Meloun {
305e9034789SMichal Meloun 	struct tegra210_pmc_softc *sc;
306e9034789SMichal Meloun 	uint32_t reg;
307e9034789SMichal Meloun 
308e9034789SMichal Meloun 	sc = tegra210_pmc_get_sc();
309e9034789SMichal Meloun 
310e9034789SMichal Meloun 	reg = RD4(sc, PMC_PWRGATE_STATUS);
311e9034789SMichal Meloun 	return ((reg & PMC_PWRGATE_STATUS_PARTID(id)) ? 1 : 0);
312e9034789SMichal Meloun }
313e9034789SMichal Meloun 
314e9034789SMichal Meloun int
tegra_powergate_power_on(enum tegra_powergate_id id)315e9034789SMichal Meloun tegra_powergate_power_on(enum tegra_powergate_id id)
316e9034789SMichal Meloun {
317e9034789SMichal Meloun 	struct tegra210_pmc_softc *sc;
318e9034789SMichal Meloun 	int rv, i;
319e9034789SMichal Meloun 
320e9034789SMichal Meloun 	sc = tegra210_pmc_get_sc();
321e9034789SMichal Meloun 
322e9034789SMichal Meloun 	rv = tegra210_pmc_set_powergate(sc, id, 1);
323e9034789SMichal Meloun 	if (rv != 0) {
324e9034789SMichal Meloun 		device_printf(sc->dev, "Cannot set powergate: %d\n", id);
325e9034789SMichal Meloun 		return (rv);
326e9034789SMichal Meloun 	}
327e9034789SMichal Meloun 
328e9034789SMichal Meloun 	for (i = 100; i > 0; i--) {
329e9034789SMichal Meloun 		if (tegra_powergate_is_powered(id))
330e9034789SMichal Meloun 			break;
331e9034789SMichal Meloun 		DELAY(1);
332e9034789SMichal Meloun 	}
333e9034789SMichal Meloun 	if (i <= 0) {
334e9034789SMichal Meloun 		device_printf(sc->dev, "Timeout when waiting on power up\n");
335e9034789SMichal Meloun 		return(ETIMEDOUT);
336e9034789SMichal Meloun 	}
337e9034789SMichal Meloun 
338e9034789SMichal Meloun 	return (rv);
339e9034789SMichal Meloun }
340e9034789SMichal Meloun 
341e9034789SMichal Meloun int
tegra_powergate_power_off(enum tegra_powergate_id id)342e9034789SMichal Meloun tegra_powergate_power_off(enum tegra_powergate_id id)
343e9034789SMichal Meloun {
344e9034789SMichal Meloun 	struct tegra210_pmc_softc *sc;
345e9034789SMichal Meloun 	int rv, i;
346e9034789SMichal Meloun 
347e9034789SMichal Meloun 	sc = tegra210_pmc_get_sc();
348e9034789SMichal Meloun 
349e9034789SMichal Meloun 	rv = tegra210_pmc_set_powergate(sc, id, 0);
350e9034789SMichal Meloun 	if (rv != 0) {
351e9034789SMichal Meloun 		device_printf(sc->dev, "Cannot set powergate: %d\n", id);
352e9034789SMichal Meloun 		return (rv);
353e9034789SMichal Meloun 	}
354e9034789SMichal Meloun 	for (i = 100; i > 0; i--) {
355e9034789SMichal Meloun 		if (!tegra_powergate_is_powered(id))
356e9034789SMichal Meloun 			break;
357e9034789SMichal Meloun 		DELAY(1);
358e9034789SMichal Meloun 	}
359e9034789SMichal Meloun 	if (i <= 0)
360e9034789SMichal Meloun 		device_printf(sc->dev, "Timeout when waiting on power off\n");
361e9034789SMichal Meloun 
362e9034789SMichal Meloun 	return (rv);
363e9034789SMichal Meloun }
364e9034789SMichal Meloun 
365e9034789SMichal Meloun int
tegra_powergate_sequence_power_up(enum tegra_powergate_id id,clk_t clk,hwreset_t rst)366e9034789SMichal Meloun tegra_powergate_sequence_power_up(enum tegra_powergate_id id, clk_t clk,
367e9034789SMichal Meloun     hwreset_t rst)
368e9034789SMichal Meloun {
369e9034789SMichal Meloun 	struct tegra210_pmc_softc *sc;
370e9034789SMichal Meloun 	int rv;
371e9034789SMichal Meloun 
372e9034789SMichal Meloun 	sc = tegra210_pmc_get_sc();
373e9034789SMichal Meloun 
374e9034789SMichal Meloun 	rv = hwreset_assert(rst);
375e9034789SMichal Meloun 	if (rv != 0) {
376e9034789SMichal Meloun 		device_printf(sc->dev, "Cannot assert reset\n");
377e9034789SMichal Meloun 		return (rv);
378e9034789SMichal Meloun 	}
379e9034789SMichal Meloun 
380e9034789SMichal Meloun 	rv = clk_stop(clk);
381e9034789SMichal Meloun 	if (rv != 0) {
382e9034789SMichal Meloun 		device_printf(sc->dev, "Cannot stop clock\n");
383e9034789SMichal Meloun 		goto clk_fail;
384e9034789SMichal Meloun 	}
385e9034789SMichal Meloun 
386e9034789SMichal Meloun 	rv = tegra_powergate_power_on(id);
387e9034789SMichal Meloun 	if (rv != 0) {
388e9034789SMichal Meloun 		device_printf(sc->dev, "Cannot power on powergate\n");
389e9034789SMichal Meloun 		goto clk_fail;
390e9034789SMichal Meloun 	}
391e9034789SMichal Meloun 
392e9034789SMichal Meloun 	rv = clk_enable(clk);
393e9034789SMichal Meloun 	if (rv != 0) {
394e9034789SMichal Meloun 		device_printf(sc->dev, "Cannot enable clock\n");
395e9034789SMichal Meloun 		goto clk_fail;
396e9034789SMichal Meloun 	}
397e9034789SMichal Meloun 	DELAY(20);
398e9034789SMichal Meloun 
399e9034789SMichal Meloun 	rv = tegra_powergate_remove_clamping(id);
400e9034789SMichal Meloun 	if (rv != 0) {
401e9034789SMichal Meloun 		device_printf(sc->dev, "Cannot remove clamping\n");
402e9034789SMichal Meloun 		goto fail;
403e9034789SMichal Meloun 	}
404e9034789SMichal Meloun 	rv = hwreset_deassert(rst);
405e9034789SMichal Meloun 	if (rv != 0) {
406e9034789SMichal Meloun 		device_printf(sc->dev, "Cannot unreset reset\n");
407e9034789SMichal Meloun 		goto fail;
408e9034789SMichal Meloun 	}
409e9034789SMichal Meloun 	return 0;
410e9034789SMichal Meloun 
411e9034789SMichal Meloun fail:
412e9034789SMichal Meloun 	clk_disable(clk);
413e9034789SMichal Meloun clk_fail:
414e9034789SMichal Meloun 	hwreset_assert(rst);
415e9034789SMichal Meloun 	tegra_powergate_power_off(id);
416e9034789SMichal Meloun 	return (rv);
417e9034789SMichal Meloun }
418e9034789SMichal Meloun 
419e9034789SMichal Meloun static int
tegra210_pmc_parse_fdt(struct tegra210_pmc_softc * sc,phandle_t node)420e9034789SMichal Meloun tegra210_pmc_parse_fdt(struct tegra210_pmc_softc *sc, phandle_t node)
421e9034789SMichal Meloun {
422e9034789SMichal Meloun 	int rv;
423e9034789SMichal Meloun 	uint32_t tmp;
424e9034789SMichal Meloun 	uint32_t tmparr[2];
425e9034789SMichal Meloun 
426e9034789SMichal Meloun 	rv = OF_getencprop(node, "nvidia,suspend-mode", &tmp, sizeof(tmp));
427e9034789SMichal Meloun 	if (rv > 0) {
428e9034789SMichal Meloun 		switch (tmp) {
429e9034789SMichal Meloun 		case 0:
430e9034789SMichal Meloun 			sc->suspend_mode = TEGRA_SUSPEND_LP0;
431e9034789SMichal Meloun 			break;
432e9034789SMichal Meloun 
433e9034789SMichal Meloun 		case 1:
434e9034789SMichal Meloun 			sc->suspend_mode = TEGRA_SUSPEND_LP1;
435e9034789SMichal Meloun 			break;
436e9034789SMichal Meloun 
437e9034789SMichal Meloun 		case 2:
438e9034789SMichal Meloun 			sc->suspend_mode = TEGRA_SUSPEND_LP2;
439e9034789SMichal Meloun 			break;
440e9034789SMichal Meloun 
441e9034789SMichal Meloun 		default:
442e9034789SMichal Meloun 			sc->suspend_mode = TEGRA_SUSPEND_NONE;
443e9034789SMichal Meloun 			break;
444e9034789SMichal Meloun 		}
445e9034789SMichal Meloun 	}
446e9034789SMichal Meloun 
447e9034789SMichal Meloun 	rv = OF_getencprop(node, "nvidia,cpu-pwr-good-time", &tmp, sizeof(tmp));
448e9034789SMichal Meloun 	if (rv > 0) {
449e9034789SMichal Meloun 		sc->cpu_good_time = tmp;
450e9034789SMichal Meloun 		sc->suspend_mode = TEGRA_SUSPEND_NONE;
451e9034789SMichal Meloun 	}
452e9034789SMichal Meloun 
453e9034789SMichal Meloun 	rv = OF_getencprop(node, "nvidia,cpu-pwr-off-time", &tmp, sizeof(tmp));
454e9034789SMichal Meloun 	if (rv > 0) {
455e9034789SMichal Meloun 		sc->cpu_off_time = tmp;
456e9034789SMichal Meloun 		sc->suspend_mode = TEGRA_SUSPEND_NONE;
457e9034789SMichal Meloun 	}
458e9034789SMichal Meloun 
459e9034789SMichal Meloun 	rv = OF_getencprop(node, "nvidia,core-pwr-good-time", tmparr,
460e9034789SMichal Meloun 	    sizeof(tmparr));
461e9034789SMichal Meloun 	if (rv == sizeof(tmparr)) {
462e9034789SMichal Meloun 		sc->core_osc_time = tmparr[0];
463e9034789SMichal Meloun 		sc->core_pmu_time = tmparr[1];
464e9034789SMichal Meloun 		sc->suspend_mode = TEGRA_SUSPEND_NONE;
465e9034789SMichal Meloun 	}
466e9034789SMichal Meloun 
467e9034789SMichal Meloun 	rv = OF_getencprop(node, "nvidia,core-pwr-off-time", &tmp, sizeof(tmp));
468e9034789SMichal Meloun 	if (rv > 0) {
469e9034789SMichal Meloun 		sc->core_off_time = tmp;
470e9034789SMichal Meloun 		sc->suspend_mode = TEGRA_SUSPEND_NONE;
471e9034789SMichal Meloun 	}
472e9034789SMichal Meloun 
473e9034789SMichal Meloun 	sc->corereq_high =
474e9034789SMichal Meloun 	    OF_hasprop(node, "nvidia,core-power-req-active-high");
475e9034789SMichal Meloun 	sc->sysclkreq_high =
476e9034789SMichal Meloun 	    OF_hasprop(node, "nvidia,sys-clock-req-active-high");
477e9034789SMichal Meloun 	sc->combined_req =
478e9034789SMichal Meloun 	    OF_hasprop(node, "nvidia,combined-power-req");
479e9034789SMichal Meloun 	sc->cpu_pwr_good_en =
480e9034789SMichal Meloun 	    OF_hasprop(node, "nvidia,cpu-pwr-good-en");
481e9034789SMichal Meloun 
482e9034789SMichal Meloun 	rv = OF_getencprop(node, "nvidia,lp0-vec", tmparr, sizeof(tmparr));
483e9034789SMichal Meloun 	if (rv == sizeof(tmparr)) {
484e9034789SMichal Meloun 
485e9034789SMichal Meloun 		sc->lp0_vec_phys = tmparr[0];
486e9034789SMichal Meloun 		sc->core_pmu_time = tmparr[1];
487e9034789SMichal Meloun 		sc->lp0_vec_size = TEGRA_SUSPEND_NONE;
488e9034789SMichal Meloun 		if (sc->suspend_mode == TEGRA_SUSPEND_LP0)
489e9034789SMichal Meloun 			sc->suspend_mode = TEGRA_SUSPEND_LP1;
490e9034789SMichal Meloun 	}
491e9034789SMichal Meloun 	return 0;
492e9034789SMichal Meloun }
493e9034789SMichal Meloun 
494e9034789SMichal Meloun static void
tegra210_pmc_check_secure(struct tegra210_pmc_softc * sc)495e9034789SMichal Meloun tegra210_pmc_check_secure(struct tegra210_pmc_softc *sc)
496e9034789SMichal Meloun {
497e9034789SMichal Meloun 	uint32_t orig;
498e9034789SMichal Meloun 
499e9034789SMichal Meloun 	sc->secure_access = false;
500e9034789SMichal Meloun 
501e9034789SMichal Meloun 	/*
502e9034789SMichal Meloun 	 * If PMC is coverd by secure trust zone, all reads returns 0,
503e9034789SMichal Meloun 	 * Use scratch0 register acvcess test
504e9034789SMichal Meloun 	 */
505e9034789SMichal Meloun 	orig = RD4(sc, PMC_SCRATCH0);
506e9034789SMichal Meloun 	WR4(sc, PMC_SCRATCH0, 0xDEADBEEF);
507e9034789SMichal Meloun 	if (RD4(sc, PMC_SCRATCH0) == 0) {
508e9034789SMichal Meloun 		sc->secure_access = true;
509e9034789SMichal Meloun 		return;
510e9034789SMichal Meloun 	}
511e9034789SMichal Meloun 	WR4(sc, PMC_SCRATCH0, 0xBADC0DE);
512e9034789SMichal Meloun 	if (RD4(sc, PMC_SCRATCH0) == 0) {
513e9034789SMichal Meloun 		sc->secure_access = true;
514e9034789SMichal Meloun 		return;
515e9034789SMichal Meloun 	}
516e9034789SMichal Meloun 	WR4(sc, PMC_SCRATCH0, orig);
517e9034789SMichal Meloun }
518e9034789SMichal Meloun 
519e9034789SMichal Meloun static int
tegra210_pmc_probe(device_t dev)520e9034789SMichal Meloun tegra210_pmc_probe(device_t dev)
521e9034789SMichal Meloun {
522e9034789SMichal Meloun 
523e9034789SMichal Meloun 	if (!ofw_bus_status_okay(dev))
524e9034789SMichal Meloun 		return (ENXIO);
525e9034789SMichal Meloun 
526e9034789SMichal Meloun 	if (!ofw_bus_search_compatible(dev, compat_data)->ocd_data)
527e9034789SMichal Meloun 		return (ENXIO);
528e9034789SMichal Meloun 
529e9034789SMichal Meloun 	device_set_desc(dev, "Tegra PMC");
530e9034789SMichal Meloun 	return (BUS_PROBE_DEFAULT);
531e9034789SMichal Meloun }
532e9034789SMichal Meloun 
533e9034789SMichal Meloun static int
tegra210_pmc_detach(device_t dev)534e9034789SMichal Meloun tegra210_pmc_detach(device_t dev)
535e9034789SMichal Meloun {
536e9034789SMichal Meloun 
537e9034789SMichal Meloun 	/* This device is always present. */
538e9034789SMichal Meloun 	return (EBUSY);
539e9034789SMichal Meloun }
540e9034789SMichal Meloun 
541e9034789SMichal Meloun static int
tegra210_pmc_attach(device_t dev)542e9034789SMichal Meloun tegra210_pmc_attach(device_t dev)
543e9034789SMichal Meloun {
544e9034789SMichal Meloun 	struct tegra210_pmc_softc *sc;
545e9034789SMichal Meloun 	int rid, rv;
546e9034789SMichal Meloun 	uint32_t reg;
547e9034789SMichal Meloun 	phandle_t node;
548e9034789SMichal Meloun 
549e9034789SMichal Meloun 	sc = device_get_softc(dev);
550e9034789SMichal Meloun 	sc->dev = dev;
551e9034789SMichal Meloun 	node = ofw_bus_get_node(dev);
552e9034789SMichal Meloun 	PMC_LOCK_INIT(sc);
553e9034789SMichal Meloun 
554e9034789SMichal Meloun 	rv = tegra210_pmc_parse_fdt(sc, node);
555e9034789SMichal Meloun 	if (rv != 0) {
556e9034789SMichal Meloun 		device_printf(sc->dev, "Cannot parse FDT data\n");
557e9034789SMichal Meloun 		return (rv);
558e9034789SMichal Meloun 	}
559e9034789SMichal Meloun 
560e9034789SMichal Meloun 	rv = clk_get_by_ofw_name(sc->dev, 0, "pclk", &sc->clk);
561e9034789SMichal Meloun 	if (rv != 0) {
562e9034789SMichal Meloun 		device_printf(sc->dev, "Cannot get \"pclk\" clock\n");
563e9034789SMichal Meloun 		return (ENXIO);
564e9034789SMichal Meloun 	}
565e9034789SMichal Meloun 
566e9034789SMichal Meloun 	rid = 0;
567e9034789SMichal Meloun 	sc->mem_res = bus_alloc_resource_any(dev, SYS_RES_MEMORY, &rid,
568e9034789SMichal Meloun 	    RF_ACTIVE);
569e9034789SMichal Meloun 	if (sc->mem_res == NULL) {
570e9034789SMichal Meloun 		device_printf(dev, "Cannot allocate memory resources\n");
571e9034789SMichal Meloun 		return (ENXIO);
572e9034789SMichal Meloun 	}
573e9034789SMichal Meloun 
574e9034789SMichal Meloun 	tegra210_pmc_check_secure(sc);
575e9034789SMichal Meloun 
576e9034789SMichal Meloun 	/* Enable CPU power request. */
577e9034789SMichal Meloun 	reg = RD4(sc, PMC_CNTRL);
578e9034789SMichal Meloun 	reg |= PMC_CNTRL_CPU_PWRREQ_OE;
579e9034789SMichal Meloun 	WR4(sc, PMC_CNTRL, reg);
580e9034789SMichal Meloun 
581e9034789SMichal Meloun 	/* Set sysclk output polarity */
582e9034789SMichal Meloun 	reg = RD4(sc, PMC_CNTRL);
583e9034789SMichal Meloun 	if (sc->sysclkreq_high)
584e9034789SMichal Meloun 		reg &= ~PMC_CNTRL_SYSCLK_POLARITY;
585e9034789SMichal Meloun 	else
586e9034789SMichal Meloun 		reg |= PMC_CNTRL_SYSCLK_POLARITY;
587e9034789SMichal Meloun 	WR4(sc, PMC_CNTRL, reg);
588e9034789SMichal Meloun 
589e9034789SMichal Meloun 	/* Enable sysclk request. */
590e9034789SMichal Meloun 	reg = RD4(sc, PMC_CNTRL);
591e9034789SMichal Meloun 	reg |= PMC_CNTRL_SYSCLK_OE;
592e9034789SMichal Meloun 	WR4(sc, PMC_CNTRL, reg);
593e9034789SMichal Meloun 
594e9034789SMichal Meloun 	/*
595e9034789SMichal Meloun 	 * Remove HDMI from deep power down mode.
596e9034789SMichal Meloun 	 * XXX mote this to HDMI driver
597e9034789SMichal Meloun 	 */
598e9034789SMichal Meloun 	reg = RD4(sc, PMC_IO_DPD_STATUS);
599e9034789SMichal Meloun 	reg &= ~ PMC_IO_DPD_STATUS_HDMI;
600e9034789SMichal Meloun 	WR4(sc, PMC_IO_DPD_STATUS, reg);
601e9034789SMichal Meloun 
602e9034789SMichal Meloun 	reg = RD4(sc, PMC_IO_DPD2_STATUS);
603e9034789SMichal Meloun 	reg &= ~ PMC_IO_DPD2_STATUS_HV;
604e9034789SMichal Meloun 	WR4(sc, PMC_IO_DPD2_STATUS, reg);
605e9034789SMichal Meloun 
606e9034789SMichal Meloun 	if (pmc_sc != NULL)
607e9034789SMichal Meloun 		panic("tegra210_pmc: double driver attach");
608e9034789SMichal Meloun 	pmc_sc = sc;
609e9034789SMichal Meloun 	return (0);
610e9034789SMichal Meloun }
611e9034789SMichal Meloun 
612e9034789SMichal Meloun static device_method_t tegra210_pmc_methods[] = {
613e9034789SMichal Meloun 	/* Device interface */
614e9034789SMichal Meloun 	DEVMETHOD(device_probe,		tegra210_pmc_probe),
615e9034789SMichal Meloun 	DEVMETHOD(device_attach,	tegra210_pmc_attach),
616e9034789SMichal Meloun 	DEVMETHOD(device_detach,	tegra210_pmc_detach),
617e9034789SMichal Meloun 
618e9034789SMichal Meloun 	DEVMETHOD_END
619e9034789SMichal Meloun };
620e9034789SMichal Meloun 
621e9034789SMichal Meloun static DEFINE_CLASS_0(pmc, tegra210_pmc_driver, tegra210_pmc_methods,
622e9034789SMichal Meloun     sizeof(struct tegra210_pmc_softc));
623289f133bSJohn Baldwin EARLY_DRIVER_MODULE(tegra210_pmc, simplebus, tegra210_pmc_driver, NULL, NULL,
624289f133bSJohn Baldwin     70);
625