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