xref: /freebsd/sys/arm64/nvidia/tegra210/max77620_regulators.c (revision 13ec1e3155c7e9bf037b12af186351b7fa9b9450)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
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/cdefs.h>
29 __FBSDID("$FreeBSD$");
30 
31 #include <sys/param.h>
32 #include <sys/systm.h>
33 #include <sys/bus.h>
34 #include <sys/gpio.h>
35 #include <sys/kernel.h>
36 #include <sys/module.h>
37 #include <sys/malloc.h>
38 #include <sys/rman.h>
39 #include <sys/sx.h>
40 
41 #include <machine/bus.h>
42 
43 #include <dev/extres/regulator/regulator.h>
44 #include <dev/gpio/gpiobusvar.h>
45 
46 #include <dt-bindings/mfd/max77620.h>
47 
48 #include "max77620.h"
49 
50 MALLOC_DEFINE(M_MAX77620_REG, "MAX77620 regulator", "MAX77620 power regulator");
51 
52 #define	DIV_ROUND_UP(n,d) howmany(n, d)
53 
54 enum max77620_reg_id {
55 	MAX77620_REG_ID_SD0,
56 	MAX77620_REG_ID_SD1,
57 	MAX77620_REG_ID_SD2,
58 	MAX77620_REG_ID_SD3,
59 	MAX77620_REG_ID_LDO0,
60 	MAX77620_REG_ID_LDO1,
61 	MAX77620_REG_ID_LDO2,
62 	MAX77620_REG_ID_LDO3,
63 	MAX77620_REG_ID_LDO4,
64 	MAX77620_REG_ID_LDO5,
65 	MAX77620_REG_ID_LDO6,
66 	MAX77620_REG_ID_LDO7,
67 	MAX77620_REG_ID_LDO8,
68 };
69 
70 /* Initial configuration. */
71 struct max77620_regnode_init_def {
72 	struct regnode_init_def	reg_init_def;
73 	int active_fps_src;
74 	int active_fps_pu_slot;
75 	int active_fps_pd_slot;
76 	int suspend_fps_src;
77 	int suspend_fps_pu_slot;
78 	int suspend_fps_pd_slot;
79 	int ramp_rate_setting;
80 };
81 
82 /* Regulator HW definition. */
83 struct reg_def {
84 	intptr_t		id;		/* ID */
85 	char			*name;		/* Regulator name */
86 	char			*supply_name;	/* Source property name */
87 	bool 			is_sd_reg; 	/* SD or LDO regulator? */
88 	uint8_t			volt_reg;
89 	uint8_t			volt_vsel_mask;
90 	uint8_t			cfg_reg;
91 	uint8_t			fps_reg;
92 	uint8_t			pwr_mode_reg;
93 	uint8_t			pwr_mode_mask;
94 	uint8_t			pwr_mode_shift;
95 	struct regulator_range	*ranges;
96 	int			nranges;
97 };
98 
99 struct max77620_reg_sc {
100 	struct regnode		*regnode;
101 	struct max77620_softc	*base_sc;
102 	struct reg_def		*def;
103 	phandle_t		xref;
104 
105 	struct regnode_std_param *param;
106 	/* Configured values */
107 	int			active_fps_src;
108 	int			active_fps_pu_slot;
109 	int			active_fps_pd_slot;
110 	int			suspend_fps_src;
111 	int			suspend_fps_pu_slot;
112 	int			suspend_fps_pd_slot;
113 	int			ramp_rate_setting;
114 	int			enable_usec;
115 	uint8_t			enable_pwr_mode;
116 
117 	/* Cached values */
118 	uint8_t			fps_src;
119 	uint8_t			pwr_mode;
120 	int			pwr_ramp_delay;
121 };
122 
123 static struct regulator_range max77620_sd0_ranges[] = {
124 	REG_RANGE_INIT(0, 64, 600000, 12500),  /* 0.6V - 1.4V / 12.5mV */
125 };
126 
127 static struct regulator_range max77620_sd1_ranges[] = {
128 	REG_RANGE_INIT(0, 76, 600000, 12500),  /* 0.6V - 1.55V / 12.5mV */
129 };
130 
131 static struct regulator_range max77620_sdx_ranges[] = {
132 	REG_RANGE_INIT(0, 255, 600000, 12500),  /* 0.6V - 3.7875V / 12.5mV */
133 };
134 
135 static struct regulator_range max77620_ldo0_1_ranges[] = {
136 	REG_RANGE_INIT(0, 63, 800000, 25000),  /* 0.8V - 2.375V / 25mV */
137 };
138 
139 static struct regulator_range max77620_ldo4_ranges[] = {
140 	REG_RANGE_INIT(0, 63, 800000, 12500),  /* 0.8V - 1.5875V / 12.5mV */
141 };
142 
143 static struct regulator_range max77620_ldox_ranges[] = {
144 	REG_RANGE_INIT(0, 63, 800000, 50000),  /* 0.8V - 3.95V / 50mV */
145 };
146 
147 static struct reg_def max77620s_def[] = {
148 	{
149 		.id = MAX77620_REG_ID_SD0,
150 		.name = "sd0",
151 		.supply_name = "in-sd0",
152 		.is_sd_reg = true,
153 		.volt_reg = MAX77620_REG_SD0,
154 		.volt_vsel_mask = MAX77620_SD0_VSEL_MASK,
155 		.cfg_reg = MAX77620_REG_CFG_SD0,
156 		.fps_reg = MAX77620_REG_FPS_SD0,
157 		.pwr_mode_reg = MAX77620_REG_CFG_SD0,
158 		.pwr_mode_mask = MAX77620_SD_POWER_MODE_MASK,
159 		.pwr_mode_shift = MAX77620_SD_POWER_MODE_SHIFT,
160 		.ranges = max77620_sd0_ranges,
161 		.nranges = nitems(max77620_sd0_ranges),
162 	},
163 	{
164 		.id = MAX77620_REG_ID_SD1,
165 		.name = "sd1",
166 		.supply_name = "in-sd1",
167 		.is_sd_reg = true,
168 		.volt_reg = MAX77620_REG_SD1,
169 		.volt_vsel_mask = MAX77620_SD1_VSEL_MASK,
170 		.cfg_reg = MAX77620_REG_CFG_SD1,
171 		.fps_reg = MAX77620_REG_FPS_SD1,
172 		.pwr_mode_reg = MAX77620_REG_CFG_SD1,
173 		.pwr_mode_mask = MAX77620_SD_POWER_MODE_MASK,
174 		.pwr_mode_shift = MAX77620_SD_POWER_MODE_SHIFT,
175 		.ranges = max77620_sd1_ranges,
176 		.nranges = nitems(max77620_sd1_ranges),
177 	},
178 	{
179 		.id = MAX77620_REG_ID_SD2,
180 		.name = "sd2",
181 		.supply_name = "in-sd2",
182 		.is_sd_reg = true,
183 		.volt_reg = MAX77620_REG_SD2,
184 		.volt_vsel_mask = MAX77620_SDX_VSEL_MASK,
185 		.cfg_reg = MAX77620_REG_CFG_SD2,
186 		.fps_reg = MAX77620_REG_FPS_SD2,
187 		.pwr_mode_reg = MAX77620_REG_CFG_SD2,
188 		.pwr_mode_mask = MAX77620_SD_POWER_MODE_MASK,
189 		.pwr_mode_shift = MAX77620_SD_POWER_MODE_SHIFT,
190 		.ranges = max77620_sdx_ranges,
191 		.nranges = nitems(max77620_sdx_ranges),
192 	},
193 	{
194 		.id = MAX77620_REG_ID_SD3,
195 		.name = "sd3",
196 		.supply_name = "in-sd3",
197 		.is_sd_reg = true,
198 		.volt_reg = MAX77620_REG_SD3,
199 		.volt_vsel_mask = MAX77620_SDX_VSEL_MASK,
200 		.cfg_reg = MAX77620_REG_CFG_SD3,
201 		.fps_reg = MAX77620_REG_FPS_SD3,
202 		.pwr_mode_reg = MAX77620_REG_CFG_SD3,
203 		.pwr_mode_mask = MAX77620_SD_POWER_MODE_MASK,
204 		.pwr_mode_shift = MAX77620_SD_POWER_MODE_SHIFT,
205 		.ranges = max77620_sdx_ranges,
206 		.nranges = nitems(max77620_sdx_ranges),
207 	},
208 	{
209 		.id = MAX77620_REG_ID_LDO0,
210 		.name = "ldo0",
211 		.supply_name = "vin-ldo0-1",
212 		.volt_reg = MAX77620_REG_CFG_LDO0,
213 		.volt_vsel_mask = MAX77620_LDO_VSEL_MASK,
214 		.is_sd_reg = false,
215 		.cfg_reg = MAX77620_REG_CFG2_LDO0,
216 		.fps_reg = MAX77620_REG_FPS_LDO0,
217 		.pwr_mode_reg = MAX77620_REG_CFG_LDO0,
218 		.pwr_mode_mask = MAX77620_LDO_POWER_MODE_MASK,
219 		.pwr_mode_shift = MAX77620_LDO_POWER_MODE_SHIFT,
220 		.ranges = max77620_ldo0_1_ranges,
221 		.nranges = nitems(max77620_ldo0_1_ranges),
222 	},
223 	{
224 		.id = MAX77620_REG_ID_LDO1,
225 		.name = "ldo1",
226 		.supply_name = "in-ldo0-1",
227 		.is_sd_reg = false,
228 		.volt_reg = MAX77620_REG_CFG_LDO1,
229 		.volt_vsel_mask = MAX77620_LDO_VSEL_MASK,
230 		.cfg_reg = MAX77620_REG_CFG2_LDO1,
231 		.fps_reg = MAX77620_REG_FPS_LDO1,
232 		.pwr_mode_reg = MAX77620_REG_CFG_LDO1,
233 		.pwr_mode_mask = MAX77620_LDO_POWER_MODE_MASK,
234 		.pwr_mode_shift = MAX77620_LDO_POWER_MODE_SHIFT,
235 		.ranges = max77620_ldo0_1_ranges,
236 		.nranges = nitems(max77620_ldo0_1_ranges),
237 	},
238 	{
239 		.id = MAX77620_REG_ID_LDO2,
240 		.name = "ldo2",
241 		.supply_name = "in-ldo2",
242 		.is_sd_reg = false,
243 		.volt_reg = MAX77620_REG_CFG_LDO2,
244 		.volt_vsel_mask = MAX77620_LDO_VSEL_MASK,
245 		.cfg_reg = MAX77620_REG_CFG2_LDO2,
246 		.fps_reg = MAX77620_REG_FPS_LDO2,
247 		.pwr_mode_reg = MAX77620_REG_CFG_LDO2,
248 		.pwr_mode_mask = MAX77620_LDO_POWER_MODE_MASK,
249 		.pwr_mode_shift = MAX77620_LDO_POWER_MODE_SHIFT,
250 		.ranges = max77620_ldox_ranges,
251 		.nranges = nitems(max77620_ldox_ranges),
252 	},
253 	{
254 		.id = MAX77620_REG_ID_LDO3,
255 		.name = "ldo3",
256 		.supply_name = "in-ldo3-5",
257 		.is_sd_reg = false,
258 		.volt_reg = MAX77620_REG_CFG_LDO3,
259 		.volt_vsel_mask = MAX77620_LDO_VSEL_MASK,
260 		.cfg_reg = MAX77620_REG_CFG2_LDO3,
261 		.fps_reg = MAX77620_REG_FPS_LDO3,
262 		.pwr_mode_reg = MAX77620_REG_CFG_LDO3,
263 		.pwr_mode_mask = MAX77620_LDO_POWER_MODE_MASK,
264 		.pwr_mode_shift = MAX77620_LDO_POWER_MODE_SHIFT,
265 		.ranges = max77620_ldox_ranges,
266 		.nranges = nitems(max77620_ldox_ranges),
267 	},
268 	{
269 		.id = MAX77620_REG_ID_LDO4,
270 		.name = "ldo4",
271 		.supply_name = "in-ldo4-6",
272 		.is_sd_reg = false,
273 		.volt_reg = MAX77620_REG_CFG_LDO4,
274 		.volt_vsel_mask = MAX77620_LDO_VSEL_MASK,
275 		.cfg_reg = MAX77620_REG_CFG2_LDO4,
276 		.fps_reg = MAX77620_REG_FPS_LDO4,
277 		.pwr_mode_reg = MAX77620_REG_CFG_LDO4,
278 		.pwr_mode_mask = MAX77620_LDO_POWER_MODE_MASK,
279 		.pwr_mode_shift = MAX77620_LDO_POWER_MODE_SHIFT,
280 		.ranges = max77620_ldo4_ranges,
281 		.nranges = nitems(max77620_ldo4_ranges),
282 	},
283 	{
284 		.id = MAX77620_REG_ID_LDO5,
285 		.name = "ldo5",
286 		.supply_name = "in-ldo3-5",
287 		.is_sd_reg = false,
288 		.volt_reg = MAX77620_REG_CFG_LDO5,
289 		.volt_vsel_mask = MAX77620_LDO_VSEL_MASK,
290 		.cfg_reg = MAX77620_REG_CFG2_LDO5,
291 		.fps_reg = MAX77620_REG_FPS_LDO5,
292 		.pwr_mode_reg = MAX77620_REG_CFG_LDO5,
293 		.pwr_mode_mask = MAX77620_LDO_POWER_MODE_MASK,
294 		.pwr_mode_shift = MAX77620_LDO_POWER_MODE_SHIFT,
295 		.ranges = max77620_ldox_ranges,
296 		.nranges = nitems(max77620_ldox_ranges),
297 	},
298 	{
299 		.id = MAX77620_REG_ID_LDO6,
300 		.name = "ldo6",
301 		.supply_name = "in-ldo4-6",
302 		.is_sd_reg = false,
303 		.volt_reg = MAX77620_REG_CFG_LDO6,
304 		.volt_vsel_mask = MAX77620_LDO_VSEL_MASK,
305 		.cfg_reg = MAX77620_REG_CFG2_LDO6,
306 		.fps_reg = MAX77620_REG_FPS_LDO6,
307 		.pwr_mode_reg = MAX77620_REG_CFG_LDO6,
308 		.pwr_mode_mask = MAX77620_LDO_POWER_MODE_MASK,
309 		.pwr_mode_shift = MAX77620_LDO_POWER_MODE_SHIFT,
310 		.ranges = max77620_ldox_ranges,
311 		.nranges = nitems(max77620_ldox_ranges),
312 	},
313 	{
314 		.id = MAX77620_REG_ID_LDO7,
315 		.name = "ldo7",
316 		.supply_name = "in-ldo7-8",
317 		.is_sd_reg = false,
318 		.volt_reg = MAX77620_REG_CFG_LDO7,
319 		.volt_vsel_mask = MAX77620_LDO_VSEL_MASK,
320 		.cfg_reg = MAX77620_REG_CFG2_LDO7,
321 		.fps_reg = MAX77620_REG_FPS_LDO7,
322 		.pwr_mode_reg = MAX77620_REG_CFG_LDO7,
323 		.pwr_mode_mask = MAX77620_LDO_POWER_MODE_MASK,
324 		.pwr_mode_shift = MAX77620_LDO_POWER_MODE_SHIFT,
325 		.ranges = max77620_ldox_ranges,
326 		.nranges = nitems(max77620_ldox_ranges),
327 	},
328 	{
329 		.id = MAX77620_REG_ID_LDO8,
330 		.name = "ldo8",
331 		.supply_name = "in-ldo7-8",
332 		.is_sd_reg = false,
333 		.volt_reg = MAX77620_REG_CFG_LDO8,
334 		.volt_vsel_mask = MAX77620_LDO_VSEL_MASK,
335 		.cfg_reg = MAX77620_REG_CFG2_LDO8,
336 		.fps_reg = MAX77620_REG_FPS_LDO8,
337 		.pwr_mode_reg = MAX77620_REG_CFG_LDO8,
338 		.pwr_mode_mask = MAX77620_LDO_POWER_MODE_MASK,
339 		.pwr_mode_shift = MAX77620_LDO_POWER_MODE_SHIFT,
340 		.ranges = max77620_ldox_ranges,
341 		.nranges = nitems(max77620_ldox_ranges),
342 	},
343 };
344 
345 
346 static int max77620_regnode_init(struct regnode *regnode);
347 static int max77620_regnode_enable(struct regnode *regnode, bool enable,
348     int *udelay);
349 static int max77620_regnode_set_volt(struct regnode *regnode, int min_uvolt,
350     int max_uvolt, int *udelay);
351 static int max77620_regnode_get_volt(struct regnode *regnode, int *uvolt);
352 static regnode_method_t max77620_regnode_methods[] = {
353 	/* Regulator interface */
354 	REGNODEMETHOD(regnode_init,		max77620_regnode_init),
355 	REGNODEMETHOD(regnode_enable,		max77620_regnode_enable),
356 	REGNODEMETHOD(regnode_set_voltage,	max77620_regnode_set_volt),
357 	REGNODEMETHOD(regnode_get_voltage,	max77620_regnode_get_volt),
358 	REGNODEMETHOD_END
359 };
360 DEFINE_CLASS_1(max77620_regnode, max77620_regnode_class, max77620_regnode_methods,
361    sizeof(struct max77620_reg_sc), regnode_class);
362 
363 static int
364 max77620_get_sel(struct max77620_reg_sc *sc, uint8_t *sel)
365 {
366 	int rv;
367 
368 	rv = RD1(sc->base_sc, sc->def->volt_reg, sel);
369 	if (rv != 0) {
370 		printf("%s: cannot read volatge selector: %d\n",
371 		    regnode_get_name(sc->regnode), rv);
372 		return (rv);
373 	}
374 	*sel &= sc->def->volt_vsel_mask;
375 	*sel >>= ffs(sc->def->volt_vsel_mask) - 1;
376 	return (0);
377 }
378 
379 static int
380 max77620_set_sel(struct max77620_reg_sc *sc, uint8_t sel)
381 {
382 	int rv;
383 
384 	sel <<= ffs(sc->def->volt_vsel_mask) - 1;
385 	sel &= sc->def->volt_vsel_mask;
386 
387 	rv = RM1(sc->base_sc, sc->def->volt_reg,
388 	    sc->def->volt_vsel_mask, sel);
389 	if (rv != 0) {
390 		printf("%s: cannot set volatge selector: %d\n",
391 		    regnode_get_name(sc->regnode), rv);
392 		return (rv);
393 	}
394 	return (rv);
395 }
396 
397 static int
398 max77620_get_fps_src(struct max77620_reg_sc *sc, uint8_t *fps_src)
399 {
400 	uint8_t val;
401 	int rv;
402 
403 	rv = RD1(sc->base_sc, sc->def->fps_reg, &val);
404 	if (rv != 0)
405 		return (rv);
406 
407 	*fps_src  = (val & MAX77620_FPS_SRC_MASK) >> MAX77620_FPS_SRC_SHIFT;
408 	return (0);
409 }
410 
411 static int
412 max77620_set_fps_src(struct max77620_reg_sc *sc, uint8_t fps_src)
413 {
414 	int rv;
415 
416 	rv = RM1(sc->base_sc, sc->def->fps_reg, MAX77620_FPS_SRC_MASK,
417 	    fps_src << MAX77620_FPS_SRC_SHIFT);
418 	if (rv != 0)
419 		return (rv);
420 	sc->fps_src = fps_src;
421 	return (0);
422 }
423 
424 static int
425 max77620_set_fps_slots(struct max77620_reg_sc *sc, bool suspend)
426 {
427 	uint8_t mask, val;
428 	int pu_slot, pd_slot, rv;
429 
430 	if (suspend) {
431 		pu_slot = sc->suspend_fps_pu_slot;
432 		pd_slot = sc->suspend_fps_pd_slot;
433 	} else {
434 		pu_slot = sc->active_fps_pu_slot;
435 		pd_slot = sc->active_fps_pd_slot;
436 	}
437 
438 	mask = 0;
439 	val = 0;
440 	if (pu_slot >= 0) {
441 		mask |= MAX77620_FPS_PU_PERIOD_MASK;
442 		val |= ((uint8_t)pu_slot << MAX77620_FPS_PU_PERIOD_SHIFT) &
443 		    MAX77620_FPS_PU_PERIOD_MASK;
444 	}
445 	if (pd_slot >= 0) {
446 		mask |= MAX77620_FPS_PD_PERIOD_MASK;
447 		val |= ((uint8_t)pd_slot << MAX77620_FPS_PD_PERIOD_SHIFT) &
448 		    MAX77620_FPS_PD_PERIOD_MASK;
449 	}
450 
451 	rv = RM1(sc->base_sc, sc->def->fps_reg, mask, val);
452 	if (rv != 0)
453 		return (rv);
454 	return (0);
455 }
456 
457 static int
458 max77620_get_pwr_mode(struct max77620_reg_sc *sc, uint8_t *pwr_mode)
459 {
460 	uint8_t val;
461 	int rv;
462 
463 	rv = RD1(sc->base_sc, sc->def->pwr_mode_reg, &val);
464 	if (rv != 0)
465 		return (rv);
466 
467 	*pwr_mode  = (val & sc->def->pwr_mode_mask) >> sc->def->pwr_mode_shift;
468 	return (0);
469 }
470 
471 static int
472 max77620_set_pwr_mode(struct max77620_reg_sc *sc, uint8_t pwr_mode)
473 {
474 	int rv;
475 
476 	rv = RM1(sc->base_sc, sc->def->pwr_mode_reg, sc->def->pwr_mode_shift,
477 	    pwr_mode << sc->def->pwr_mode_shift);
478 	if (rv != 0)
479 		return (rv);
480 	sc->pwr_mode = pwr_mode;
481 	return (0);
482 }
483 
484 static int
485 max77620_get_pwr_ramp_delay(struct max77620_reg_sc *sc, int *rate)
486 {
487 	uint8_t val;
488 	int rv;
489 
490 	rv = RD1(sc->base_sc, sc->def->cfg_reg, &val);
491 	if (rv != 0)
492 		return (rv);
493 
494 	if (sc->def->is_sd_reg) {
495 		val = (val & MAX77620_SD_SR_MASK) >> MAX77620_SD_SR_SHIFT;
496 		if (val == 0)
497 			*rate = 13750;
498 		else if (val == 1)
499 			*rate = 27500;
500 		else if (val == 2)
501 			*rate = 55000;
502 		else
503 			*rate = 100000;
504 	} else {
505 		val = (val & MAX77620_LDO_SLEW_RATE_MASK) >>
506 		    MAX77620_LDO_SLEW_RATE_SHIFT;
507 		if (val == 0)
508 			*rate = 100000;
509 		else
510 			*rate = 5000;
511 	}
512 	sc->pwr_ramp_delay = *rate;
513 	return (0);
514 }
515 
516 static int
517 max77620_set_pwr_ramp_delay(struct max77620_reg_sc *sc, int rate)
518 {
519 	uint8_t val, mask;
520 	int rv;
521 
522 	if (sc->def->is_sd_reg) {
523 		if (rate <= 13750)
524 			val = 0;
525 		else if (rate <= 27500)
526 			val = 1;
527 		else if (rate <= 55000)
528 			val = 2;
529 		else
530 			val = 3;
531 		val <<= MAX77620_SD_SR_SHIFT;
532 		mask = MAX77620_SD_SR_MASK;
533 	} else {
534 		if (rate <= 5000)
535 			val = 1;
536 		else
537 			val = 0;
538 		val <<= MAX77620_LDO_SLEW_RATE_SHIFT;
539 		mask = MAX77620_LDO_SLEW_RATE_MASK;
540 	}
541 	rv = RM1(sc->base_sc, sc->def->cfg_reg, mask, val);
542 	if (rv != 0)
543 		return (rv);
544 	return (0);
545 }
546 
547 static int
548 max77620_regnode_init(struct regnode *regnode)
549 {
550 	struct max77620_reg_sc *sc;
551 	uint8_t val;
552 	int intval, rv;
553 
554 	sc = regnode_get_softc(regnode);
555 	sc->enable_usec = 500;
556 	sc->enable_pwr_mode = MAX77620_POWER_MODE_NORMAL;
557 #if 0
558 {
559 uint8_t val1, val2, val3;
560 RD1(sc->base_sc, sc->def->volt_reg, &val1);
561 RD1(sc->base_sc, sc->def->cfg_reg, &val2);
562 RD1(sc->base_sc, sc->def->fps_reg, &val3);
563 printf("%s: Volt: 0x%02X, CFG: 0x%02X, FPS: 0x%02X\n", regnode_get_name(sc->regnode), val1, val2, val3);
564 }
565 #endif
566 	/* Get current power mode */
567 	rv = max77620_get_pwr_mode(sc, &val);
568 	if (rv != 0) {
569 		printf("%s: cannot read current power mode: %d\n",
570 		    regnode_get_name(sc->regnode), rv);
571 		return (rv);
572 	}
573 	sc->pwr_mode = val;
574 
575 	/* Get current power ramp delay */
576 	rv = max77620_get_pwr_ramp_delay(sc, &intval);
577 	if (rv != 0) {
578 		printf("%s: cannot read current power mode: %d\n",
579 		    regnode_get_name(sc->regnode), rv);
580 		return (rv);
581 	}
582 	sc->pwr_ramp_delay = intval;
583 
584 	/* Get FPS source if is not specified. */
585 	if (sc->active_fps_src == -1) {
586 		rv = max77620_get_fps_src(sc, &val);
587 		if (rv != 0) {
588 			printf("%s: cannot read current FPS source: %d\n",
589 			    regnode_get_name(sc->regnode), rv);
590 			return (rv);
591 		}
592 		sc->active_fps_src = val;
593 	}
594 
595 	/* Configure power mode non-FPS controlled regulators. */
596 	if (sc->active_fps_src != MAX77620_FPS_SRC_NONE ||
597 	    (sc->pwr_mode != MAX77620_POWER_MODE_DISABLE &&
598 	    sc->pwr_mode != sc->enable_pwr_mode)) {
599 		rv = max77620_set_pwr_mode(sc, (uint8_t)sc->enable_pwr_mode);
600 		if (rv != 0) {
601 			printf("%s: cannot set power mode: %d\n",
602 			    regnode_get_name(sc->regnode), rv);
603 			return (rv);
604 		}
605 	}
606 
607 	/* Set FPS source. */
608 	rv = max77620_set_fps_src(sc, sc->active_fps_src);
609 	if (rv != 0) {
610 		printf("%s: cannot setup FPS source: %d\n",
611 		    regnode_get_name(sc->regnode), rv);
612 		return (rv);
613 	}
614 	/* Set FPS slots. */
615 	rv = max77620_set_fps_slots(sc, false);
616 	if (rv != 0) {
617 		printf("%s: cannot setup power slots: %d\n",
618 		    regnode_get_name(sc->regnode), rv);
619 		return (rv);
620 	}
621 	/* Setup power ramp . */
622 	if (sc->ramp_rate_setting != -1) {
623 		rv = max77620_set_pwr_ramp_delay(sc, sc->pwr_ramp_delay);
624 		if (rv != 0) {
625 			printf("%s: cannot set power ramp delay: %d\n",
626 			    regnode_get_name(sc->regnode), rv);
627 			return (rv);
628 		}
629 	}
630 
631 	return (0);
632 }
633 
634 static void
635 max77620_fdt_parse(struct max77620_softc *sc, phandle_t node, struct reg_def *def,
636 struct max77620_regnode_init_def *init_def)
637 {
638 	int rv;
639 	phandle_t parent, supply_node;
640 	char prop_name[64]; /* Maximum OFW property name length. */
641 
642 	rv = regulator_parse_ofw_stdparam(sc->dev, node,
643 	    &init_def->reg_init_def);
644 
645 	rv = OF_getencprop(node, "maxim,active-fps-source",
646 	    &init_def->active_fps_src, sizeof(init_def->active_fps_src));
647 	if (rv <= 0)
648 		init_def->active_fps_src = MAX77620_FPS_SRC_DEF;
649 
650 	rv = OF_getencprop(node, "maxim,active-fps-power-up-slot",
651 	    &init_def->active_fps_pu_slot, sizeof(init_def->active_fps_pu_slot));
652 	if (rv <= 0)
653 		init_def->active_fps_pu_slot = -1;
654 
655 	rv = OF_getencprop(node, "maxim,active-fps-power-down-slot",
656 	    &init_def->active_fps_pd_slot, sizeof(init_def->active_fps_pd_slot));
657 	if (rv <= 0)
658 		init_def->active_fps_pd_slot = -1;
659 
660 	rv = OF_getencprop(node, "maxim,suspend-fps-source",
661 	    &init_def->suspend_fps_src, sizeof(init_def->suspend_fps_src));
662 	if (rv <= 0)
663 		init_def->suspend_fps_src = -1;
664 
665 	rv = OF_getencprop(node, "maxim,suspend-fps-power-up-slot",
666 	    &init_def->suspend_fps_pu_slot, sizeof(init_def->suspend_fps_pu_slot));
667 	if (rv <= 0)
668 		init_def->suspend_fps_pu_slot = -1;
669 
670 	rv = OF_getencprop(node, "maxim,suspend-fps-power-down-slot",
671 	    &init_def->suspend_fps_pd_slot, sizeof(init_def->suspend_fps_pd_slot));
672 	if (rv <= 0)
673 		init_def->suspend_fps_pd_slot = -1;
674 
675 	rv = OF_getencprop(node, "maxim,ramp-rate-setting",
676 	    &init_def->ramp_rate_setting, sizeof(init_def->ramp_rate_setting));
677 	if (rv <= 0)
678 		init_def->ramp_rate_setting = -1;
679 
680 	/* Get parent supply. */
681 	if (def->supply_name == NULL)
682 		 return;
683 
684 	parent = OF_parent(node);
685 	snprintf(prop_name, sizeof(prop_name), "%s-supply",
686 	    def->supply_name);
687 	rv = OF_getencprop(parent, prop_name, &supply_node,
688 	    sizeof(supply_node));
689 	if (rv <= 0)
690 		return;
691 	supply_node = OF_node_from_xref(supply_node);
692 	rv = OF_getprop_alloc(supply_node, "regulator-name",
693 	    (void **)&init_def->reg_init_def.parent_name);
694 	if (rv <= 0)
695 		init_def->reg_init_def.parent_name = NULL;
696 }
697 
698 static struct max77620_reg_sc *
699 max77620_attach(struct max77620_softc *sc, phandle_t node, struct reg_def *def)
700 {
701 	struct max77620_reg_sc *reg_sc;
702 	struct max77620_regnode_init_def init_def;
703 	struct regnode *regnode;
704 
705 	bzero(&init_def, sizeof(init_def));
706 
707 	max77620_fdt_parse(sc, node, def, &init_def);
708 	init_def.reg_init_def.id = def->id;
709 	init_def.reg_init_def.ofw_node = node;
710 	regnode = regnode_create(sc->dev, &max77620_regnode_class,
711 	    &init_def.reg_init_def);
712 	if (regnode == NULL) {
713 		device_printf(sc->dev, "Cannot create regulator.\n");
714 		return (NULL);
715 	}
716 	reg_sc = regnode_get_softc(regnode);
717 
718 	/* Init regulator softc. */
719 	reg_sc->regnode = regnode;
720 	reg_sc->base_sc = sc;
721 	reg_sc->def = def;
722 	reg_sc->xref = OF_xref_from_node(node);
723 	reg_sc->param = regnode_get_stdparam(regnode);
724 	reg_sc->active_fps_src = init_def.active_fps_src;
725 	reg_sc->active_fps_pu_slot = init_def.active_fps_pu_slot;
726 	reg_sc->active_fps_pd_slot = init_def.active_fps_pd_slot;
727 	reg_sc->suspend_fps_src = init_def.suspend_fps_src;
728 	reg_sc->suspend_fps_pu_slot = init_def.suspend_fps_pu_slot;
729 	reg_sc->suspend_fps_pd_slot = init_def.suspend_fps_pd_slot;
730 	reg_sc->ramp_rate_setting = init_def.ramp_rate_setting;
731 
732 	regnode_register(regnode);
733 	if (bootverbose) {
734 		int volt, rv;
735 		regnode_topo_slock();
736 		rv = regnode_get_voltage(regnode, &volt);
737 		if (rv == ENODEV) {
738 			device_printf(sc->dev,
739 			   " Regulator %s: parent doesn't exist yet.\n",
740 			   regnode_get_name(regnode));
741 		} else if (rv != 0) {
742 			device_printf(sc->dev,
743 			   " Regulator %s: voltage: INVALID!!!\n",
744 			   regnode_get_name(regnode));
745 		} else {
746 			device_printf(sc->dev,
747 			    " Regulator %s: voltage: %d uV\n",
748 			    regnode_get_name(regnode), volt);
749 			device_printf(sc->dev,
750 			    "  FPS source: %d, mode: %d, ramp delay: %d\n",
751 			    reg_sc->fps_src, reg_sc->pwr_mode,
752 			    reg_sc->pwr_ramp_delay);
753 		}
754 		regnode_topo_unlock();
755 	}
756 
757 	return (reg_sc);
758 }
759 
760 int
761 max77620_regulator_attach(struct max77620_softc *sc, phandle_t node)
762 {
763 	struct max77620_reg_sc *reg;
764 	phandle_t child, rnode;
765 	int i;
766 
767 	rnode = ofw_bus_find_child(node, "regulators");
768 	if (rnode <= 0) {
769 		device_printf(sc->dev, " Cannot find regulators subnode\n");
770 		return (ENXIO);
771 	}
772 
773 	sc->nregs = nitems(max77620s_def);
774 	sc->regs = malloc(sizeof(struct max77620_reg_sc *) * sc->nregs,
775 	    M_MAX77620_REG, M_WAITOK | M_ZERO);
776 
777 
778 	/* Attach all known regulators if exist in DT. */
779 	for (i = 0; i < sc->nregs; i++) {
780 		child = ofw_bus_find_child(rnode, max77620s_def[i].name);
781 		if (child == 0) {
782 			if (bootverbose)
783 				device_printf(sc->dev,
784 				    "Regulator %s missing in DT\n",
785 				    max77620s_def[i].name);
786 			continue;
787 		}
788 		if (ofw_bus_node_status_okay(child) == 0)
789 			continue;
790 		reg = max77620_attach(sc, child, max77620s_def + i);
791 		if (reg == NULL) {
792 			device_printf(sc->dev, "Cannot attach regulator: %s\n",
793 			    max77620s_def[i].name);
794 			return (ENXIO);
795 		}
796 		sc->regs[i] = reg;
797 	}
798 	return (0);
799 }
800 
801 int
802 max77620_regulator_map(device_t dev, phandle_t xref, int ncells,
803     pcell_t *cells, intptr_t *num)
804 {
805 	struct max77620_softc *sc;
806 	int i;
807 
808 	sc = device_get_softc(dev);
809 	for (i = 0; i < sc->nregs; i++) {
810 		if (sc->regs[i] == NULL)
811 			continue;
812 		if (sc->regs[i]->xref == xref) {
813 			*num = sc->regs[i]->def->id;
814 			return (0);
815 		}
816 	}
817 
818 	return (ENXIO);
819 }
820 
821 static int
822 max77620_regnode_enable(struct regnode *regnode, bool val, int *udelay)
823 {
824 
825 	struct max77620_reg_sc *sc;
826 	uint8_t mode;
827 	int rv;
828 
829 	sc = regnode_get_softc(regnode);
830 
831 	if (sc->active_fps_src != MAX77620_FPS_SRC_NONE) {
832 		*udelay = 0;
833 		return (0);
834 	}
835 
836 	if (val)
837 		mode = sc->enable_pwr_mode;
838 	else
839 		mode = MAX77620_POWER_MODE_DISABLE;
840 
841 	rv = max77620_set_pwr_mode(sc, mode);
842 	if (rv != 0) {
843 		printf("%s: cannot set power mode: %d\n",
844 		    regnode_get_name(sc->regnode), rv);
845 		return (rv);
846 	}
847 
848 	*udelay = sc->enable_usec;
849 	return (0);
850 }
851 
852 static int
853 max77620_regnode_set_volt(struct regnode *regnode, int min_uvolt, int max_uvolt,
854     int *udelay)
855 {
856 	struct max77620_reg_sc *sc;
857 	uint8_t sel;
858 	int rv;
859 
860 	sc = regnode_get_softc(regnode);
861 
862 	*udelay = 0;
863 	rv = regulator_range_volt_to_sel8(sc->def->ranges, sc->def->nranges,
864 	    min_uvolt, max_uvolt, &sel);
865 	if (rv != 0)
866 		return (rv);
867 	rv = max77620_set_sel(sc, sel);
868 	return (rv);
869 }
870 
871 static int
872 max77620_regnode_get_volt(struct regnode *regnode, int *uvolt)
873 {
874 
875 	struct max77620_reg_sc *sc;
876 	uint8_t sel;
877 	int rv;
878 
879 	sc = regnode_get_softc(regnode);
880 	rv = max77620_get_sel(sc, &sel);
881 	if (rv != 0)
882 		return (rv);
883 
884 	rv = regulator_range_sel8_to_volt(sc->def->ranges, sc->def->nranges,
885 	    sel, uvolt);
886 	return (rv);
887 	return(0);
888 }
889