xref: /freebsd/sys/dev/mmc/mmc_pwrseq.c (revision b64c5a0ace59af62eff52bfe110a521dc73c937b)
1 /*
2  * SPDX-License-Identifier: BSD-2-Clause
3  *
4  * Copyright 2021 Emmanuel Vadot <manu@freebsd.org>
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions are
8  * met:
9  *
10  *  1. Redistributions of source code must retain the above copyright
11  *     notice, this list of conditions and the following disclaimer.
12  *  2. Redistributions in binary form must reproduce the above copyright
13  *     notice, this list of conditions and the following disclaimer in the
14  *     documentation and/or other materials provided with the distribution.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
19  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE
20  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
21  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
22  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
23  * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
24  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
25  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
26  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28 
29 #include <sys/param.h>
30 #include <sys/bus.h>
31 #include <sys/kernel.h>
32 #include <sys/module.h>
33 #include <sys/gpio.h>
34 
35 #include <dev/gpio/gpiobusvar.h>
36 #include <dev/ofw/ofw_bus.h>
37 #include <dev/ofw/ofw_bus_subr.h>
38 
39 #include <dev/clk/clk.h>
40 
41 #include "mmc_pwrseq_if.h"
42 
43 enum pwrseq_type {
44 	PWRSEQ_SIMPLE = 1,
45 	PWRSEQ_EMMC,
46 };
47 
48 static struct ofw_compat_data compat_data[] = {
49 	{ "mmc-pwrseq-simple",	PWRSEQ_SIMPLE },
50 	{ "mmc-pwrseq-emmc",	PWRSEQ_EMMC },
51 	{ NULL,			0 }
52 };
53 
54 struct mmc_pwrseq_softc {
55 	enum pwrseq_type	type;
56 	clk_t			ext_clock;
57 	struct gpiobus_pin	*reset_gpio;
58 
59 	uint32_t		post_power_on_delay_ms;
60 	uint32_t		power_off_delay_us;
61 };
62 
63 static int
64 mmc_pwrseq_probe(device_t dev)
65 {
66 	enum pwrseq_type type;
67 
68 	if (!ofw_bus_search_compatible(dev, compat_data)->ocd_data)
69 		return (ENXIO);
70 
71 	type = (enum pwrseq_type)ofw_bus_search_compatible(dev, compat_data)->ocd_data;
72 	switch (type) {
73 	case PWRSEQ_SIMPLE:
74 		device_set_desc(dev, "MMC Simple Power sequence");
75 		break;
76 	case PWRSEQ_EMMC:
77 		device_set_desc(dev, "MMC eMMC Power sequence");
78 		break;
79 	}
80 	return (BUS_PROBE_DEFAULT);
81 }
82 
83 static int
84 mmc_pwrseq_attach(device_t dev)
85 {
86 	struct mmc_pwrseq_softc *sc;
87 	phandle_t node;
88 	int rv;
89 
90 	sc = device_get_softc(dev);
91 	sc->type = (enum pwrseq_type)ofw_bus_search_compatible(dev, compat_data)->ocd_data;
92 	node = ofw_bus_get_node(dev);
93 
94 	if (sc->type == PWRSEQ_SIMPLE) {
95 		if (OF_hasprop(node, "clocks")) {
96 			rv = clk_get_by_ofw_name(dev, 0, "ext_clock", &sc->ext_clock);
97 			if (rv != 0) {
98 				device_printf(dev,
99 				    "Node have a clocks property but no clocks named \"ext_clock\"\n");
100 				return (ENXIO);
101 			}
102 		}
103 		OF_getencprop(node, "post-power-on-delay-ms", &sc->post_power_on_delay_ms, sizeof(uint32_t));
104 		OF_getencprop(node, "power-off-delay-us", &sc->power_off_delay_us, sizeof(uint32_t));
105 	}
106 
107 	if (OF_hasprop(node, "reset-gpios")) {
108 		if (gpio_pin_get_by_ofw_property(dev, node, "reset-gpios",
109 		    &sc->reset_gpio) != 0) {
110 			device_printf(dev, "Cannot get the reset-gpios\n");
111 			return (ENXIO);
112 		}
113 		gpio_pin_setflags(sc->reset_gpio, GPIO_PIN_OUTPUT);
114 		gpio_pin_set_active(sc->reset_gpio, true);
115 	}
116 
117 	OF_device_register_xref(OF_xref_from_node(node), dev);
118 	return (0);
119 }
120 
121 static int
122 mmc_pwrseq_detach(device_t dev)
123 {
124 
125 	return (EBUSY);
126 }
127 
128 static int
129 mmv_pwrseq_set_power(device_t dev, bool power_on)
130 {
131 	struct mmc_pwrseq_softc *sc;
132 	int rv;
133 
134 	sc = device_get_softc(dev);
135 
136 	if (power_on) {
137 		if (sc->ext_clock) {
138 			rv = clk_enable(sc->ext_clock);
139 			if (rv != 0)
140 				return (rv);
141 		}
142 
143 		if (sc->reset_gpio) {
144 			rv = gpio_pin_set_active(sc->reset_gpio, false);
145 			if (rv != 0)
146 				return (rv);
147 		}
148 
149 		if (sc->post_power_on_delay_ms)
150 			DELAY(sc->post_power_on_delay_ms * 1000);
151 	} else {
152 		if (sc->reset_gpio) {
153 			rv = gpio_pin_set_active(sc->reset_gpio, true);
154 			if (rv != 0)
155 				return (rv);
156 		}
157 
158 		if (sc->ext_clock) {
159 			rv = clk_stop(sc->ext_clock);
160 			if (rv != 0)
161 				return (rv);
162 		}
163 		if (sc->power_off_delay_us)
164 			DELAY(sc->power_off_delay_us);
165 	}
166 
167 	return (0);
168 }
169 
170 static device_method_t mmc_pwrseq_methods[] = {
171 	/* Device interface */
172 	DEVMETHOD(device_probe,		mmc_pwrseq_probe),
173 	DEVMETHOD(device_attach,	mmc_pwrseq_attach),
174 	DEVMETHOD(device_detach,	mmc_pwrseq_detach),
175 
176 	DEVMETHOD(mmc_pwrseq_set_power,	mmv_pwrseq_set_power),
177 	DEVMETHOD_END
178 };
179 
180 static driver_t mmc_pwrseq_driver = {
181 	"mmc_pwrseq",
182 	mmc_pwrseq_methods,
183 	sizeof(struct mmc_pwrseq_softc),
184 };
185 
186 EARLY_DRIVER_MODULE(mmc_pwrseq, simplebus, mmc_pwrseq_driver, 0, 0,
187 	BUS_PASS_SUPPORTDEV + BUS_PASS_ORDER_FIRST);
188 MODULE_VERSION(mmc_pwrseq, 1);
189 SIMPLEBUS_PNP_INFO(compat_data);
190