Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
freebsd
GitHub Repository: freebsd/freebsd-src
Path: blob/main/sys/arm64/nvidia/tegra210/tegra210_pmc.c
48266 views
1
/*-
2
* SPDX-License-Identifier: BSD-2-Clause
3
*
4
* Copyright 2020 Michal Meloun <[email protected]>
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
150
struct 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
172
static struct ofw_compat_data compat_data[] = {
173
{"nvidia,tegra210-pmc", 1},
174
{NULL, 0},
175
};
176
177
static struct tegra210_pmc_softc *pmc_sc;
178
179
static inline struct tegra210_pmc_softc *
180
tegra210_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
187
static void
188
WR4(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_invoke_smc(PMC_SMC, PMC_SMC_WRITE, r, v, &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
202
static uint32_t
203
RD4(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_invoke_smc(PMC_SMC, PMC_SMC_READ, r, &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
218
static int
219
tegra210_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
259
int
260
tegra_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
302
int
303
tegra_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
314
int
315
tegra_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
341
int
342
tegra_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
365
int
366
tegra_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
411
fail:
412
clk_disable(clk);
413
clk_fail:
414
hwreset_assert(rst);
415
tegra_powergate_power_off(id);
416
return (rv);
417
}
418
419
static int
420
tegra210_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
494
static void
495
tegra210_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
519
static int
520
tegra210_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
533
static int
534
tegra210_pmc_detach(device_t dev)
535
{
536
537
/* This device is always present. */
538
return (EBUSY);
539
}
540
541
static int
542
tegra210_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
612
static 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
621
static DEFINE_CLASS_0(pmc, tegra210_pmc_driver, tegra210_pmc_methods,
622
sizeof(struct tegra210_pmc_softc));
623
EARLY_DRIVER_MODULE(tegra210_pmc, simplebus, tegra210_pmc_driver, NULL, NULL,
624
70);
625
626