1*20baee6bSAdrian Chadd /*-
2*20baee6bSAdrian Chadd * SPDX-License-Identifier: BSD-2-Clause
3*20baee6bSAdrian Chadd *
4*20baee6bSAdrian Chadd * Copyright (c) 2025 Adrian Chadd <adrian@FreeBSD.org>
5*20baee6bSAdrian Chadd *
6*20baee6bSAdrian Chadd * Redistribution and use in source and binary forms, with or without
7*20baee6bSAdrian Chadd * modification, are permitted provided that the following conditions
8*20baee6bSAdrian Chadd * are met:
9*20baee6bSAdrian Chadd * 1. Redistributions of source code must retain the above copyright
10*20baee6bSAdrian Chadd * notice unmodified, this list of conditions, and the following
11*20baee6bSAdrian Chadd * disclaimer.
12*20baee6bSAdrian Chadd * 2. Redistributions in binary form must reproduce the above copyright
13*20baee6bSAdrian Chadd * notice, this list of conditions and the following disclaimer in the
14*20baee6bSAdrian Chadd * documentation and/or other materials provided with the distribution.
15*20baee6bSAdrian Chadd *
16*20baee6bSAdrian Chadd * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17*20baee6bSAdrian Chadd * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18*20baee6bSAdrian Chadd * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19*20baee6bSAdrian Chadd * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20*20baee6bSAdrian Chadd * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21*20baee6bSAdrian Chadd * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22*20baee6bSAdrian Chadd * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23*20baee6bSAdrian Chadd * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24*20baee6bSAdrian Chadd * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25*20baee6bSAdrian Chadd * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26*20baee6bSAdrian Chadd * SUCH DAMAGE.
27*20baee6bSAdrian Chadd */
28*20baee6bSAdrian Chadd
29*20baee6bSAdrian Chadd /*
30*20baee6bSAdrian Chadd * This is a pinmux/gpio controller for the Qualcomm IPQ/MSM/Snapdragon SoCs.
31*20baee6bSAdrian Chadd */
32*20baee6bSAdrian Chadd
33*20baee6bSAdrian Chadd #include <sys/param.h>
34*20baee6bSAdrian Chadd #include <sys/systm.h>
35*20baee6bSAdrian Chadd #include <sys/bus.h>
36*20baee6bSAdrian Chadd
37*20baee6bSAdrian Chadd #include <sys/kernel.h>
38*20baee6bSAdrian Chadd #include <sys/module.h>
39*20baee6bSAdrian Chadd #include <sys/rman.h>
40*20baee6bSAdrian Chadd #include <sys/lock.h>
41*20baee6bSAdrian Chadd #include <sys/malloc.h>
42*20baee6bSAdrian Chadd #include <sys/mutex.h>
43*20baee6bSAdrian Chadd #include <sys/gpio.h>
44*20baee6bSAdrian Chadd
45*20baee6bSAdrian Chadd #include <machine/bus.h>
46*20baee6bSAdrian Chadd #include <machine/resource.h>
47*20baee6bSAdrian Chadd #include <dev/gpio/gpiobusvar.h>
48*20baee6bSAdrian Chadd
49*20baee6bSAdrian Chadd #include <dev/fdt/fdt_common.h>
50*20baee6bSAdrian Chadd #include <dev/ofw/ofw_bus.h>
51*20baee6bSAdrian Chadd #include <dev/ofw/ofw_bus_subr.h>
52*20baee6bSAdrian Chadd
53*20baee6bSAdrian Chadd #include <dev/fdt/fdt_pinctrl.h>
54*20baee6bSAdrian Chadd
55*20baee6bSAdrian Chadd #include "qcom_tlmm_var.h"
56*20baee6bSAdrian Chadd #include "qcom_tlmm_pin.h"
57*20baee6bSAdrian Chadd #include "qcom_tlmm_debug.h"
58*20baee6bSAdrian Chadd
59*20baee6bSAdrian Chadd #include "gpio_if.h"
60*20baee6bSAdrian Chadd
61*20baee6bSAdrian Chadd #define DEFAULT_CAPS (GPIO_PIN_INPUT | GPIO_PIN_OUTPUT | \
62*20baee6bSAdrian Chadd GPIO_PIN_PULLUP | GPIO_PIN_PULLDOWN)
63*20baee6bSAdrian Chadd
64*20baee6bSAdrian Chadd /* TODO: put in a header file */
65*20baee6bSAdrian Chadd extern void qcom_tlmm_ipq4018_attach(struct qcom_tlmm_softc *sc);
66*20baee6bSAdrian Chadd
67*20baee6bSAdrian Chadd struct qcom_tlmm_chipset_list {
68*20baee6bSAdrian Chadd qcom_tlmm_chipset_t id;
69*20baee6bSAdrian Chadd const char *ofw_str;
70*20baee6bSAdrian Chadd const char *desc_str;
71*20baee6bSAdrian Chadd void (*attach_func)(struct qcom_tlmm_softc *);
72*20baee6bSAdrian Chadd };
73*20baee6bSAdrian Chadd
74*20baee6bSAdrian Chadd static struct qcom_tlmm_chipset_list qcom_tlmm_chipsets[] = {
75*20baee6bSAdrian Chadd { QCOM_TLMM_CHIPSET_IPQ4018, "qcom,ipq4019-pinctrl",
76*20baee6bSAdrian Chadd "Qualcomm Atheros TLMM IPQ4018/IPQ4019 GPIO/Pinmux driver",
77*20baee6bSAdrian Chadd qcom_tlmm_ipq4018_attach },
78*20baee6bSAdrian Chadd { 0, NULL, NULL, NULL },
79*20baee6bSAdrian Chadd };
80*20baee6bSAdrian Chadd
81*20baee6bSAdrian Chadd static int
qcom_tlmm_probe(device_t dev)82*20baee6bSAdrian Chadd qcom_tlmm_probe(device_t dev)
83*20baee6bSAdrian Chadd {
84*20baee6bSAdrian Chadd struct qcom_tlmm_softc *sc = device_get_softc(dev);
85*20baee6bSAdrian Chadd struct qcom_tlmm_chipset_list *ql;
86*20baee6bSAdrian Chadd int i;
87*20baee6bSAdrian Chadd
88*20baee6bSAdrian Chadd if (! ofw_bus_status_okay(dev))
89*20baee6bSAdrian Chadd return (ENXIO);
90*20baee6bSAdrian Chadd
91*20baee6bSAdrian Chadd for (i = 0; qcom_tlmm_chipsets[i].id != 0; i++) {
92*20baee6bSAdrian Chadd ql = &qcom_tlmm_chipsets[i];
93*20baee6bSAdrian Chadd device_printf(dev, "%s: checking %s\n", __func__, ql->ofw_str);
94*20baee6bSAdrian Chadd if (ofw_bus_is_compatible(dev, ql->ofw_str) == 1) {
95*20baee6bSAdrian Chadd sc->sc_chipset = ql->id;
96*20baee6bSAdrian Chadd sc->sc_attach_func = ql->attach_func;
97*20baee6bSAdrian Chadd device_set_desc(dev, ql->desc_str);
98*20baee6bSAdrian Chadd return (0);
99*20baee6bSAdrian Chadd }
100*20baee6bSAdrian Chadd }
101*20baee6bSAdrian Chadd
102*20baee6bSAdrian Chadd return (ENXIO);
103*20baee6bSAdrian Chadd }
104*20baee6bSAdrian Chadd
105*20baee6bSAdrian Chadd static int
qcom_tlmm_detach(device_t dev)106*20baee6bSAdrian Chadd qcom_tlmm_detach(device_t dev)
107*20baee6bSAdrian Chadd {
108*20baee6bSAdrian Chadd struct qcom_tlmm_softc *sc = device_get_softc(dev);
109*20baee6bSAdrian Chadd
110*20baee6bSAdrian Chadd KASSERT(mtx_initialized(&sc->gpio_mtx), ("gpio mutex not initialized"));
111*20baee6bSAdrian Chadd
112*20baee6bSAdrian Chadd gpiobus_detach_bus(dev);
113*20baee6bSAdrian Chadd if (sc->gpio_ih)
114*20baee6bSAdrian Chadd bus_teardown_intr(dev, sc->gpio_irq_res, sc->gpio_ih);
115*20baee6bSAdrian Chadd if (sc->gpio_irq_res)
116*20baee6bSAdrian Chadd bus_release_resource(dev, SYS_RES_IRQ, sc->gpio_irq_rid,
117*20baee6bSAdrian Chadd sc->gpio_irq_res);
118*20baee6bSAdrian Chadd if (sc->gpio_mem_res)
119*20baee6bSAdrian Chadd bus_release_resource(dev, SYS_RES_MEMORY, sc->gpio_mem_rid,
120*20baee6bSAdrian Chadd sc->gpio_mem_res);
121*20baee6bSAdrian Chadd if (sc->gpio_pins)
122*20baee6bSAdrian Chadd free(sc->gpio_pins, M_DEVBUF);
123*20baee6bSAdrian Chadd mtx_destroy(&sc->gpio_mtx);
124*20baee6bSAdrian Chadd
125*20baee6bSAdrian Chadd return(0);
126*20baee6bSAdrian Chadd }
127*20baee6bSAdrian Chadd
128*20baee6bSAdrian Chadd static int
qcom_tlmm_attach(device_t dev)129*20baee6bSAdrian Chadd qcom_tlmm_attach(device_t dev)
130*20baee6bSAdrian Chadd {
131*20baee6bSAdrian Chadd struct qcom_tlmm_softc *sc = device_get_softc(dev);
132*20baee6bSAdrian Chadd int i;
133*20baee6bSAdrian Chadd
134*20baee6bSAdrian Chadd KASSERT((device_get_unit(dev) == 0),
135*20baee6bSAdrian Chadd ("qcom_tlmm: Only one gpio module supported"));
136*20baee6bSAdrian Chadd
137*20baee6bSAdrian Chadd mtx_init(&sc->gpio_mtx, device_get_nameunit(dev), NULL, MTX_DEF);
138*20baee6bSAdrian Chadd
139*20baee6bSAdrian Chadd /* Map control/status registers. */
140*20baee6bSAdrian Chadd sc->gpio_mem_rid = 0;
141*20baee6bSAdrian Chadd sc->gpio_mem_res = bus_alloc_resource_any(dev, SYS_RES_MEMORY,
142*20baee6bSAdrian Chadd &sc->gpio_mem_rid, RF_ACTIVE);
143*20baee6bSAdrian Chadd
144*20baee6bSAdrian Chadd if (sc->gpio_mem_res == NULL) {
145*20baee6bSAdrian Chadd device_printf(dev, "couldn't map memory\n");
146*20baee6bSAdrian Chadd qcom_tlmm_detach(dev);
147*20baee6bSAdrian Chadd return (ENXIO);
148*20baee6bSAdrian Chadd }
149*20baee6bSAdrian Chadd
150*20baee6bSAdrian Chadd if ((sc->gpio_irq_res = bus_alloc_resource_any(dev, SYS_RES_IRQ,
151*20baee6bSAdrian Chadd &sc->gpio_irq_rid, RF_SHAREABLE | RF_ACTIVE)) == NULL) {
152*20baee6bSAdrian Chadd device_printf(dev, "unable to allocate IRQ resource\n");
153*20baee6bSAdrian Chadd qcom_tlmm_detach(dev);
154*20baee6bSAdrian Chadd return (ENXIO);
155*20baee6bSAdrian Chadd }
156*20baee6bSAdrian Chadd
157*20baee6bSAdrian Chadd if ((bus_setup_intr(dev, sc->gpio_irq_res, INTR_TYPE_MISC,
158*20baee6bSAdrian Chadd qcom_tlmm_filter, qcom_tlmm_intr, sc, &sc->gpio_ih))) {
159*20baee6bSAdrian Chadd device_printf(dev,
160*20baee6bSAdrian Chadd "WARNING: unable to register interrupt handler\n");
161*20baee6bSAdrian Chadd qcom_tlmm_detach(dev);
162*20baee6bSAdrian Chadd return (ENXIO);
163*20baee6bSAdrian Chadd }
164*20baee6bSAdrian Chadd
165*20baee6bSAdrian Chadd sc->dev = dev;
166*20baee6bSAdrian Chadd sc->sc_debug = 0;
167*20baee6bSAdrian Chadd
168*20baee6bSAdrian Chadd /* Call platform specific attach function */
169*20baee6bSAdrian Chadd sc->sc_attach_func(sc);
170*20baee6bSAdrian Chadd
171*20baee6bSAdrian Chadd qcom_tlmm_debug_sysctl_attach(sc);
172*20baee6bSAdrian Chadd
173*20baee6bSAdrian Chadd /* Allocate local pin state for all of our pins */
174*20baee6bSAdrian Chadd sc->gpio_pins = malloc(sizeof(*sc->gpio_pins) * sc->gpio_npins,
175*20baee6bSAdrian Chadd M_DEVBUF, M_WAITOK | M_ZERO);
176*20baee6bSAdrian Chadd
177*20baee6bSAdrian Chadd /* Note: direct map between gpio pin and gpio_pin[] entry */
178*20baee6bSAdrian Chadd for (i = 0; i < sc->gpio_npins; i++) {
179*20baee6bSAdrian Chadd snprintf(sc->gpio_pins[i].gp_name, GPIOMAXNAME,
180*20baee6bSAdrian Chadd "gpio%d", i);
181*20baee6bSAdrian Chadd sc->gpio_pins[i].gp_pin = i;
182*20baee6bSAdrian Chadd sc->gpio_pins[i].gp_caps = DEFAULT_CAPS;
183*20baee6bSAdrian Chadd (void) qcom_tlmm_pin_getflags(dev, i,
184*20baee6bSAdrian Chadd &sc->gpio_pins[i].gp_flags);
185*20baee6bSAdrian Chadd }
186*20baee6bSAdrian Chadd
187*20baee6bSAdrian Chadd fdt_pinctrl_register(dev, NULL);
188*20baee6bSAdrian Chadd fdt_pinctrl_configure_by_name(dev, "default");
189*20baee6bSAdrian Chadd
190*20baee6bSAdrian Chadd sc->busdev = gpiobus_add_bus(dev);
191*20baee6bSAdrian Chadd if (sc->busdev == NULL) {
192*20baee6bSAdrian Chadd device_printf(dev, "%s: failed to attach bus\n", __func__);
193*20baee6bSAdrian Chadd qcom_tlmm_detach(dev);
194*20baee6bSAdrian Chadd return (ENXIO);
195*20baee6bSAdrian Chadd }
196*20baee6bSAdrian Chadd bus_attach_children(dev);
197*20baee6bSAdrian Chadd
198*20baee6bSAdrian Chadd return (0);
199*20baee6bSAdrian Chadd }
200*20baee6bSAdrian Chadd
201*20baee6bSAdrian Chadd static device_method_t qcom_tlmm_methods[] = {
202*20baee6bSAdrian Chadd /* Driver */
203*20baee6bSAdrian Chadd DEVMETHOD(device_probe, qcom_tlmm_probe),
204*20baee6bSAdrian Chadd DEVMETHOD(device_attach, qcom_tlmm_attach),
205*20baee6bSAdrian Chadd DEVMETHOD(device_detach, qcom_tlmm_detach),
206*20baee6bSAdrian Chadd
207*20baee6bSAdrian Chadd /* GPIO protocol */
208*20baee6bSAdrian Chadd DEVMETHOD(gpio_get_bus, qcom_tlmm_get_bus),
209*20baee6bSAdrian Chadd DEVMETHOD(gpio_pin_max, qcom_tlmm_pin_max),
210*20baee6bSAdrian Chadd DEVMETHOD(gpio_pin_getname, qcom_tlmm_pin_getname),
211*20baee6bSAdrian Chadd DEVMETHOD(gpio_pin_getflags, qcom_tlmm_pin_getflags),
212*20baee6bSAdrian Chadd DEVMETHOD(gpio_pin_getcaps, qcom_tlmm_pin_getcaps),
213*20baee6bSAdrian Chadd DEVMETHOD(gpio_pin_setflags, qcom_tlmm_pin_setflags),
214*20baee6bSAdrian Chadd DEVMETHOD(gpio_pin_get, qcom_tlmm_pin_get),
215*20baee6bSAdrian Chadd DEVMETHOD(gpio_pin_set, qcom_tlmm_pin_set),
216*20baee6bSAdrian Chadd DEVMETHOD(gpio_pin_toggle, qcom_tlmm_pin_toggle),
217*20baee6bSAdrian Chadd
218*20baee6bSAdrian Chadd /* OFW */
219*20baee6bSAdrian Chadd DEVMETHOD(ofw_bus_get_node, qcom_tlmm_pin_get_node),
220*20baee6bSAdrian Chadd
221*20baee6bSAdrian Chadd /* fdt_pinctrl interface */
222*20baee6bSAdrian Chadd DEVMETHOD(fdt_pinctrl_configure, qcom_tlmm_pinctrl_configure),
223*20baee6bSAdrian Chadd
224*20baee6bSAdrian Chadd {0, 0},
225*20baee6bSAdrian Chadd };
226*20baee6bSAdrian Chadd
227*20baee6bSAdrian Chadd static driver_t qcom_tlmm_driver = {
228*20baee6bSAdrian Chadd "gpio",
229*20baee6bSAdrian Chadd qcom_tlmm_methods,
230*20baee6bSAdrian Chadd sizeof(struct qcom_tlmm_softc),
231*20baee6bSAdrian Chadd };
232*20baee6bSAdrian Chadd
233*20baee6bSAdrian Chadd EARLY_DRIVER_MODULE(qcom_tlmm, simplebus, qcom_tlmm_driver,
234*20baee6bSAdrian Chadd NULL, NULL, BUS_PASS_INTERRUPT + BUS_PASS_ORDER_LATE);
235*20baee6bSAdrian Chadd EARLY_DRIVER_MODULE(qcom_tlmm, ofwbus, qcom_tlmm_driver,
236*20baee6bSAdrian Chadd NULL, NULL, BUS_PASS_INTERRUPT + BUS_PASS_ORDER_LATE);
237*20baee6bSAdrian Chadd MODULE_VERSION(qcom_tlmm, 1);
238