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