1e53470feSOleksandr Tymoshenko /*-
2*4d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause
3af3dc4a7SPedro F. Giffuni *
43681c8d7SLuiz Otavio O Souza * Copyright (c) 2011 Ben Gray <ben.r.gray@gmail.com>.
53681c8d7SLuiz Otavio O Souza * Copyright (c) 2014 Luiz Otavio O Souza <loos@FreeBSD.org>.
6e53470feSOleksandr Tymoshenko * All rights reserved.
7e53470feSOleksandr Tymoshenko *
8e53470feSOleksandr Tymoshenko * Redistribution and use in source and binary forms, with or without
9e53470feSOleksandr Tymoshenko * modification, are permitted provided that the following conditions
10e53470feSOleksandr Tymoshenko * are met:
11e53470feSOleksandr Tymoshenko * 1. Redistributions of source code must retain the above copyright
12e53470feSOleksandr Tymoshenko * notice, this list of conditions and the following disclaimer.
13e53470feSOleksandr Tymoshenko * 2. Redistributions in binary form must reproduce the above copyright
14e53470feSOleksandr Tymoshenko * notice, this list of conditions and the following disclaimer in the
15e53470feSOleksandr Tymoshenko * documentation and/or other materials provided with the distribution.
16e53470feSOleksandr Tymoshenko *
17e53470feSOleksandr Tymoshenko * THIS SOFTWARE IS PROVIDED BY AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18e53470feSOleksandr Tymoshenko * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19e53470feSOleksandr Tymoshenko * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20e53470feSOleksandr Tymoshenko * ARE DISCLAIMED. IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
21e53470feSOleksandr Tymoshenko * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22e53470feSOleksandr Tymoshenko * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23e53470feSOleksandr Tymoshenko * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24e53470feSOleksandr Tymoshenko * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25e53470feSOleksandr Tymoshenko * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26e53470feSOleksandr Tymoshenko * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27e53470feSOleksandr Tymoshenko * SUCH DAMAGE.
28e53470feSOleksandr Tymoshenko */
29e53470feSOleksandr Tymoshenko
30e53470feSOleksandr Tymoshenko /**
313681c8d7SLuiz Otavio O Souza * Beware that the OMAP4 datasheet(s) lists GPIO banks 1-6, whereas the code
323681c8d7SLuiz Otavio O Souza * here uses 0-5.
33e53470feSOleksandr Tymoshenko */
34e53470feSOleksandr Tymoshenko
35e53470feSOleksandr Tymoshenko #include <sys/cdefs.h>
362df5562dSSvatopluk Kraus #include "opt_platform.h"
372df5562dSSvatopluk Kraus
38e53470feSOleksandr Tymoshenko #include <sys/param.h>
39e53470feSOleksandr Tymoshenko #include <sys/systm.h>
40e53470feSOleksandr Tymoshenko #include <sys/bus.h>
41e53470feSOleksandr Tymoshenko
42e53470feSOleksandr Tymoshenko #include <sys/kernel.h>
43e53470feSOleksandr Tymoshenko #include <sys/module.h>
442df5562dSSvatopluk Kraus #include <sys/proc.h>
45e53470feSOleksandr Tymoshenko #include <sys/rman.h>
46e53470feSOleksandr Tymoshenko #include <sys/lock.h>
47e53470feSOleksandr Tymoshenko #include <sys/mutex.h>
48e53470feSOleksandr Tymoshenko #include <sys/gpio.h>
493681c8d7SLuiz Otavio O Souza #include <sys/interrupt.h>
50e53470feSOleksandr Tymoshenko
51e53470feSOleksandr Tymoshenko #include <machine/bus.h>
522df5562dSSvatopluk Kraus #include <machine/intr.h>
53e53470feSOleksandr Tymoshenko #include <machine/resource.h>
54e53470feSOleksandr Tymoshenko
554eb12144SAndrew Turner #include <arm/ti/ti_cpuid.h>
56b6c7dacfSAndrew Turner #include <arm/ti/ti_gpio.h>
57e53470feSOleksandr Tymoshenko #include <arm/ti/ti_scm.h>
580050ea24SMichal Meloun #include <arm/ti/ti_sysc.h>
59e53470feSOleksandr Tymoshenko
607836352bSLuiz Otavio O Souza #include <dev/gpio/gpiobusvar.h>
61e53470feSOleksandr Tymoshenko #include <dev/ofw/openfirm.h>
62e53470feSOleksandr Tymoshenko #include <dev/ofw/ofw_bus.h>
63e53470feSOleksandr Tymoshenko #include <dev/ofw/ofw_bus_subr.h>
64e53470feSOleksandr Tymoshenko
65e53470feSOleksandr Tymoshenko #include "gpio_if.h"
66b6c7dacfSAndrew Turner #include "ti_gpio_if.h"
672df5562dSSvatopluk Kraus #include "pic_if.h"
68e53470feSOleksandr Tymoshenko
69a59806b2SLuiz Otavio O Souza #if !defined(SOC_OMAP4) && !defined(SOC_TI_AM335X)
70a59806b2SLuiz Otavio O Souza #error "Unknown SoC"
71a59806b2SLuiz Otavio O Souza #endif
72a59806b2SLuiz Otavio O Souza
73e53470feSOleksandr Tymoshenko /* Register definitions */
74e53470feSOleksandr Tymoshenko #define TI_GPIO_REVISION 0x0000
75e53470feSOleksandr Tymoshenko #define TI_GPIO_SYSCONFIG 0x0010
76e53470feSOleksandr Tymoshenko #define TI_GPIO_IRQSTATUS_RAW_0 0x0024
77e53470feSOleksandr Tymoshenko #define TI_GPIO_IRQSTATUS_RAW_1 0x0028
782df5562dSSvatopluk Kraus #define TI_GPIO_IRQSTATUS_0 0x002C /* writing a 0 has no effect */
792df5562dSSvatopluk Kraus #define TI_GPIO_IRQSTATUS_1 0x0030 /* writing a 0 has no effect */
802df5562dSSvatopluk Kraus #define TI_GPIO_IRQSTATUS_SET_0 0x0034 /* writing a 0 has no effect */
812df5562dSSvatopluk Kraus #define TI_GPIO_IRQSTATUS_SET_1 0x0038 /* writing a 0 has no effect */
822df5562dSSvatopluk Kraus #define TI_GPIO_IRQSTATUS_CLR_0 0x003C /* writing a 0 has no effect */
832df5562dSSvatopluk Kraus #define TI_GPIO_IRQSTATUS_CLR_1 0x0040 /* writing a 0 has no effect */
84e53470feSOleksandr Tymoshenko #define TI_GPIO_IRQWAKEN_0 0x0044
85e53470feSOleksandr Tymoshenko #define TI_GPIO_IRQWAKEN_1 0x0048
86e53470feSOleksandr Tymoshenko #define TI_GPIO_SYSSTATUS 0x0114
87e53470feSOleksandr Tymoshenko #define TI_GPIO_IRQSTATUS1 0x0118
88e53470feSOleksandr Tymoshenko #define TI_GPIO_IRQENABLE1 0x011C
89e53470feSOleksandr Tymoshenko #define TI_GPIO_WAKEUPENABLE 0x0120
90e53470feSOleksandr Tymoshenko #define TI_GPIO_IRQSTATUS2 0x0128
91e53470feSOleksandr Tymoshenko #define TI_GPIO_IRQENABLE2 0x012C
92e53470feSOleksandr Tymoshenko #define TI_GPIO_CTRL 0x0130
93e53470feSOleksandr Tymoshenko #define TI_GPIO_OE 0x0134
94e53470feSOleksandr Tymoshenko #define TI_GPIO_DATAIN 0x0138
95e53470feSOleksandr Tymoshenko #define TI_GPIO_DATAOUT 0x013C
962df5562dSSvatopluk Kraus #define TI_GPIO_LEVELDETECT0 0x0140 /* RW register */
972df5562dSSvatopluk Kraus #define TI_GPIO_LEVELDETECT1 0x0144 /* RW register */
982df5562dSSvatopluk Kraus #define TI_GPIO_RISINGDETECT 0x0148 /* RW register */
992df5562dSSvatopluk Kraus #define TI_GPIO_FALLINGDETECT 0x014C /* RW register */
100e53470feSOleksandr Tymoshenko #define TI_GPIO_DEBOUNCENABLE 0x0150
101e53470feSOleksandr Tymoshenko #define TI_GPIO_DEBOUNCINGTIME 0x0154
102e53470feSOleksandr Tymoshenko #define TI_GPIO_CLEARWKUPENA 0x0180
103e53470feSOleksandr Tymoshenko #define TI_GPIO_SETWKUENA 0x0184
104e53470feSOleksandr Tymoshenko #define TI_GPIO_CLEARDATAOUT 0x0190
105e53470feSOleksandr Tymoshenko #define TI_GPIO_SETDATAOUT 0x0194
106e53470feSOleksandr Tymoshenko
107e53470feSOleksandr Tymoshenko /* Other SoC Specific definitions */
1084eb12144SAndrew Turner #define OMAP4_FIRST_GPIO_BANK 1
1094eb12144SAndrew Turner #define OMAP4_INTR_PER_BANK 1
1104eb12144SAndrew Turner #define OMAP4_GPIO_REV 0x50600801
1114eb12144SAndrew Turner #define AM335X_FIRST_GPIO_BANK 0
1124eb12144SAndrew Turner #define AM335X_INTR_PER_BANK 2
1134eb12144SAndrew Turner #define AM335X_GPIO_REV 0x50600801
114ff5823beSLuiz Otavio O Souza #define PINS_PER_BANK 32
115e350f76cSLuiz Otavio O Souza #define TI_GPIO_MASK(p) (1U << ((p) % PINS_PER_BANK))
1164eb12144SAndrew Turner
1170050ea24SMichal Meloun #define OMAP4_GPIO1_REV 0x00000
1180050ea24SMichal Meloun #define OMAP4_GPIO2_REV 0x55000
1190050ea24SMichal Meloun #define OMAP4_GPIO3_REV 0x57000
1200050ea24SMichal Meloun #define OMAP4_GPIO4_REV 0x59000
1210050ea24SMichal Meloun #define OMAP4_GPIO5_REV 0x5b000
1220050ea24SMichal Meloun #define OMAP4_GPIO6_REV 0x5d000
1230050ea24SMichal Meloun
1240050ea24SMichal Meloun #define AM335X_GPIO0_REV 0x07000
1250050ea24SMichal Meloun #define AM335X_GPIO1_REV 0x4C000
1260050ea24SMichal Meloun #define AM335X_GPIO2_REV 0xAC000
1270050ea24SMichal Meloun #define AM335X_GPIO3_REV 0xAE000
1280050ea24SMichal Meloun
1292df5562dSSvatopluk Kraus static int ti_gpio_intr(void *arg);
130876c1bd8SLuiz Otavio O Souza static int ti_gpio_detach(device_t);
1313681c8d7SLuiz Otavio O Souza
1322df5562dSSvatopluk Kraus static int ti_gpio_pic_attach(struct ti_gpio_softc *sc);
1332df5562dSSvatopluk Kraus static int ti_gpio_pic_detach(struct ti_gpio_softc *sc);
1342df5562dSSvatopluk Kraus
1354eb12144SAndrew Turner static uint32_t
ti_gpio_rev(void)1364eb12144SAndrew Turner ti_gpio_rev(void)
1374eb12144SAndrew Turner {
1384eb12144SAndrew Turner switch(ti_chip()) {
1394eb12144SAndrew Turner #ifdef SOC_OMAP4
1404eb12144SAndrew Turner case CHIP_OMAP_4:
1414eb12144SAndrew Turner return (OMAP4_GPIO_REV);
1424eb12144SAndrew Turner #endif
1434eb12144SAndrew Turner #ifdef SOC_TI_AM335X
1444eb12144SAndrew Turner case CHIP_AM335X:
1454eb12144SAndrew Turner return (AM335X_GPIO_REV);
1464eb12144SAndrew Turner #endif
1474eb12144SAndrew Turner }
1484eb12144SAndrew Turner return (0);
1494eb12144SAndrew Turner }
150e53470feSOleksandr Tymoshenko
151e53470feSOleksandr Tymoshenko /**
152e53470feSOleksandr Tymoshenko * Macros for driver mutex locking
153e53470feSOleksandr Tymoshenko */
1543681c8d7SLuiz Otavio O Souza #define TI_GPIO_LOCK(_sc) mtx_lock_spin(&(_sc)->sc_mtx)
1553681c8d7SLuiz Otavio O Souza #define TI_GPIO_UNLOCK(_sc) mtx_unlock_spin(&(_sc)->sc_mtx)
156e53470feSOleksandr Tymoshenko #define TI_GPIO_LOCK_INIT(_sc) \
1573681c8d7SLuiz Otavio O Souza mtx_init(&_sc->sc_mtx, device_get_nameunit((_sc)->sc_dev), \
1583681c8d7SLuiz Otavio O Souza "ti_gpio", MTX_SPIN)
1593681c8d7SLuiz Otavio O Souza #define TI_GPIO_LOCK_DESTROY(_sc) mtx_destroy(&(_sc)->sc_mtx)
1603681c8d7SLuiz Otavio O Souza #define TI_GPIO_ASSERT_LOCKED(_sc) mtx_assert(&(_sc)->sc_mtx, MA_OWNED)
1613681c8d7SLuiz Otavio O Souza #define TI_GPIO_ASSERT_UNLOCKED(_sc) mtx_assert(&(_sc)->sc_mtx, MA_NOTOWNED)
162e53470feSOleksandr Tymoshenko
163e53470feSOleksandr Tymoshenko /**
164e350f76cSLuiz Otavio O Souza * ti_gpio_read_4 - reads a 32-bit value from one of the GPIO registers
165e53470feSOleksandr Tymoshenko * @sc: GPIO device context
166e53470feSOleksandr Tymoshenko * @bank: The bank to read from
167e53470feSOleksandr Tymoshenko * @off: The offset of a register from the GPIO register address range
168e53470feSOleksandr Tymoshenko *
169e53470feSOleksandr Tymoshenko *
170e53470feSOleksandr Tymoshenko * RETURNS:
171e53470feSOleksandr Tymoshenko * 32-bit value read from the register.
172e53470feSOleksandr Tymoshenko */
173e53470feSOleksandr Tymoshenko static inline uint32_t
ti_gpio_read_4(struct ti_gpio_softc * sc,bus_size_t off)1745b03aba6SOleksandr Tymoshenko ti_gpio_read_4(struct ti_gpio_softc *sc, bus_size_t off)
175e53470feSOleksandr Tymoshenko {
1765b03aba6SOleksandr Tymoshenko return (bus_read_4(sc->sc_mem_res, off));
177e53470feSOleksandr Tymoshenko }
178e53470feSOleksandr Tymoshenko
179e53470feSOleksandr Tymoshenko /**
180e350f76cSLuiz Otavio O Souza * ti_gpio_write_4 - writes a 32-bit value to one of the GPIO registers
181e53470feSOleksandr Tymoshenko * @sc: GPIO device context
182e53470feSOleksandr Tymoshenko * @bank: The bank to write to
183e53470feSOleksandr Tymoshenko * @off: The offset of a register from the GPIO register address range
184e53470feSOleksandr Tymoshenko * @val: The value to write into the register
185e53470feSOleksandr Tymoshenko *
186e53470feSOleksandr Tymoshenko * RETURNS:
187e53470feSOleksandr Tymoshenko * nothing
188e53470feSOleksandr Tymoshenko */
189e53470feSOleksandr Tymoshenko static inline void
ti_gpio_write_4(struct ti_gpio_softc * sc,bus_size_t off,uint32_t val)1905b03aba6SOleksandr Tymoshenko ti_gpio_write_4(struct ti_gpio_softc *sc, bus_size_t off,
191e53470feSOleksandr Tymoshenko uint32_t val)
192e53470feSOleksandr Tymoshenko {
1935b03aba6SOleksandr Tymoshenko bus_write_4(sc->sc_mem_res, off, val);
194e53470feSOleksandr Tymoshenko }
195e53470feSOleksandr Tymoshenko
196db8a14ecSLuiz Otavio O Souza static inline void
ti_gpio_intr_clr(struct ti_gpio_softc * sc,uint32_t mask)1975b03aba6SOleksandr Tymoshenko ti_gpio_intr_clr(struct ti_gpio_softc *sc, uint32_t mask)
198db8a14ecSLuiz Otavio O Souza {
199db8a14ecSLuiz Otavio O Souza
200db8a14ecSLuiz Otavio O Souza /* We clear both set of registers. */
2015b03aba6SOleksandr Tymoshenko ti_gpio_write_4(sc, TI_GPIO_IRQSTATUS_CLR_0, mask);
2025b03aba6SOleksandr Tymoshenko ti_gpio_write_4(sc, TI_GPIO_IRQSTATUS_CLR_1, mask);
203db8a14ecSLuiz Otavio O Souza }
204db8a14ecSLuiz Otavio O Souza
2053681c8d7SLuiz Otavio O Souza static inline void
ti_gpio_intr_set(struct ti_gpio_softc * sc,uint32_t mask)2065b03aba6SOleksandr Tymoshenko ti_gpio_intr_set(struct ti_gpio_softc *sc, uint32_t mask)
2073681c8d7SLuiz Otavio O Souza {
2083681c8d7SLuiz Otavio O Souza
2093681c8d7SLuiz Otavio O Souza /*
2103681c8d7SLuiz Otavio O Souza * On OMAP4 we unmask only the MPU interrupt and on AM335x we
2113681c8d7SLuiz Otavio O Souza * also activate only the first interrupt.
2123681c8d7SLuiz Otavio O Souza */
2135b03aba6SOleksandr Tymoshenko ti_gpio_write_4(sc, TI_GPIO_IRQSTATUS_SET_0, mask);
2143681c8d7SLuiz Otavio O Souza }
2153681c8d7SLuiz Otavio O Souza
2163681c8d7SLuiz Otavio O Souza static inline void
ti_gpio_intr_ack(struct ti_gpio_softc * sc,uint32_t mask)2175b03aba6SOleksandr Tymoshenko ti_gpio_intr_ack(struct ti_gpio_softc *sc, uint32_t mask)
2183681c8d7SLuiz Otavio O Souza {
2193681c8d7SLuiz Otavio O Souza
2203681c8d7SLuiz Otavio O Souza /*
2213681c8d7SLuiz Otavio O Souza * Acknowledge the interrupt on both registers even if we use only
2223681c8d7SLuiz Otavio O Souza * the first one.
2233681c8d7SLuiz Otavio O Souza */
2245b03aba6SOleksandr Tymoshenko ti_gpio_write_4(sc, TI_GPIO_IRQSTATUS_0, mask);
2255b03aba6SOleksandr Tymoshenko ti_gpio_write_4(sc, TI_GPIO_IRQSTATUS_1, mask);
2263681c8d7SLuiz Otavio O Souza }
2273681c8d7SLuiz Otavio O Souza
2283681c8d7SLuiz Otavio O Souza static inline uint32_t
ti_gpio_intr_status(struct ti_gpio_softc * sc)2295b03aba6SOleksandr Tymoshenko ti_gpio_intr_status(struct ti_gpio_softc *sc)
2303681c8d7SLuiz Otavio O Souza {
2313681c8d7SLuiz Otavio O Souza uint32_t reg;
2323681c8d7SLuiz Otavio O Souza
2333681c8d7SLuiz Otavio O Souza /* Get the status from both registers. */
2345b03aba6SOleksandr Tymoshenko reg = ti_gpio_read_4(sc, TI_GPIO_IRQSTATUS_0);
2355b03aba6SOleksandr Tymoshenko reg |= ti_gpio_read_4(sc, TI_GPIO_IRQSTATUS_1);
2363681c8d7SLuiz Otavio O Souza
2373681c8d7SLuiz Otavio O Souza return (reg);
2383681c8d7SLuiz Otavio O Souza }
2393681c8d7SLuiz Otavio O Souza
2407836352bSLuiz Otavio O Souza static device_t
ti_gpio_get_bus(device_t dev)2417836352bSLuiz Otavio O Souza ti_gpio_get_bus(device_t dev)
2427836352bSLuiz Otavio O Souza {
2437836352bSLuiz Otavio O Souza struct ti_gpio_softc *sc;
2447836352bSLuiz Otavio O Souza
2457836352bSLuiz Otavio O Souza sc = device_get_softc(dev);
2467836352bSLuiz Otavio O Souza
2477836352bSLuiz Otavio O Souza return (sc->sc_busdev);
2487836352bSLuiz Otavio O Souza }
2497836352bSLuiz Otavio O Souza
250e53470feSOleksandr Tymoshenko /**
251e53470feSOleksandr Tymoshenko * ti_gpio_pin_max - Returns the maximum number of GPIO pins
252e53470feSOleksandr Tymoshenko * @dev: gpio device handle
253e53470feSOleksandr Tymoshenko * @maxpin: pointer to a value that upon return will contain the maximum number
254e53470feSOleksandr Tymoshenko * of pins in the device.
255e53470feSOleksandr Tymoshenko *
256e53470feSOleksandr Tymoshenko *
257e53470feSOleksandr Tymoshenko * LOCKING:
258f9de33d4SLuiz Otavio O Souza * No locking required, returns static data.
259e53470feSOleksandr Tymoshenko *
260e53470feSOleksandr Tymoshenko * RETURNS:
261e53470feSOleksandr Tymoshenko * Returns 0 on success otherwise an error code
262e53470feSOleksandr Tymoshenko */
263e53470feSOleksandr Tymoshenko static int
ti_gpio_pin_max(device_t dev,int * maxpin)264e53470feSOleksandr Tymoshenko ti_gpio_pin_max(device_t dev, int *maxpin)
265e53470feSOleksandr Tymoshenko {
266e53470feSOleksandr Tymoshenko
2675b03aba6SOleksandr Tymoshenko *maxpin = PINS_PER_BANK - 1;
268e53470feSOleksandr Tymoshenko
269e53470feSOleksandr Tymoshenko return (0);
270e53470feSOleksandr Tymoshenko }
271e53470feSOleksandr Tymoshenko
272e350f76cSLuiz Otavio O Souza static int
ti_gpio_valid_pin(struct ti_gpio_softc * sc,int pin)273e350f76cSLuiz Otavio O Souza ti_gpio_valid_pin(struct ti_gpio_softc *sc, int pin)
274e350f76cSLuiz Otavio O Souza {
275e350f76cSLuiz Otavio O Souza
2765b03aba6SOleksandr Tymoshenko if (pin >= sc->sc_maxpin || sc->sc_mem_res == NULL)
277e350f76cSLuiz Otavio O Souza return (EINVAL);
278e350f76cSLuiz Otavio O Souza
279e350f76cSLuiz Otavio O Souza return (0);
280e350f76cSLuiz Otavio O Souza }
281e350f76cSLuiz Otavio O Souza
282e53470feSOleksandr Tymoshenko /**
283aa0d25b7SSvatopluk Kraus * ti_gpio_pin_getcaps - Gets the capabilities of a given pin
284e53470feSOleksandr Tymoshenko * @dev: gpio device handle
285e53470feSOleksandr Tymoshenko * @pin: the number of the pin
286e53470feSOleksandr Tymoshenko * @caps: pointer to a value that upon return will contain the capabilities
287e53470feSOleksandr Tymoshenko *
288e53470feSOleksandr Tymoshenko * Currently all pins have the same capability, notably:
289e53470feSOleksandr Tymoshenko * - GPIO_PIN_INPUT
290e53470feSOleksandr Tymoshenko * - GPIO_PIN_OUTPUT
291e53470feSOleksandr Tymoshenko * - GPIO_PIN_PULLUP
292e53470feSOleksandr Tymoshenko * - GPIO_PIN_PULLDOWN
293aa0d25b7SSvatopluk Kraus * - GPIO_INTR_LEVEL_LOW
294aa0d25b7SSvatopluk Kraus * - GPIO_INTR_LEVEL_HIGH
295aa0d25b7SSvatopluk Kraus * - GPIO_INTR_EDGE_RISING
296aa0d25b7SSvatopluk Kraus * - GPIO_INTR_EDGE_FALLING
297aa0d25b7SSvatopluk Kraus * - GPIO_INTR_EDGE_BOTH
298e53470feSOleksandr Tymoshenko *
299e53470feSOleksandr Tymoshenko * LOCKING:
300f9de33d4SLuiz Otavio O Souza * No locking required, returns static data.
301e53470feSOleksandr Tymoshenko *
302e53470feSOleksandr Tymoshenko * RETURNS:
303e53470feSOleksandr Tymoshenko * Returns 0 on success otherwise an error code
304e53470feSOleksandr Tymoshenko */
305e53470feSOleksandr Tymoshenko static int
ti_gpio_pin_getcaps(device_t dev,uint32_t pin,uint32_t * caps)306e53470feSOleksandr Tymoshenko ti_gpio_pin_getcaps(device_t dev, uint32_t pin, uint32_t *caps)
307e53470feSOleksandr Tymoshenko {
308e350f76cSLuiz Otavio O Souza struct ti_gpio_softc *sc;
309e53470feSOleksandr Tymoshenko
310e350f76cSLuiz Otavio O Souza sc = device_get_softc(dev);
311e350f76cSLuiz Otavio O Souza if (ti_gpio_valid_pin(sc, pin) != 0)
312e53470feSOleksandr Tymoshenko return (EINVAL);
313e53470feSOleksandr Tymoshenko
314aa0d25b7SSvatopluk Kraus *caps = (GPIO_PIN_INPUT | GPIO_PIN_OUTPUT | GPIO_PIN_PULLUP |
315aa0d25b7SSvatopluk Kraus GPIO_PIN_PULLDOWN | GPIO_INTR_LEVEL_LOW | GPIO_INTR_LEVEL_HIGH |
316aa0d25b7SSvatopluk Kraus GPIO_INTR_EDGE_RISING | GPIO_INTR_EDGE_FALLING |
317aa0d25b7SSvatopluk Kraus GPIO_INTR_EDGE_BOTH);
318e53470feSOleksandr Tymoshenko
319e53470feSOleksandr Tymoshenko return (0);
320e53470feSOleksandr Tymoshenko }
321e53470feSOleksandr Tymoshenko
322e53470feSOleksandr Tymoshenko /**
323e53470feSOleksandr Tymoshenko * ti_gpio_pin_getflags - Gets the current flags of a given pin
324e53470feSOleksandr Tymoshenko * @dev: gpio device handle
325e53470feSOleksandr Tymoshenko * @pin: the number of the pin
326e53470feSOleksandr Tymoshenko * @flags: upon return will contain the current flags of the pin
327e53470feSOleksandr Tymoshenko *
328e53470feSOleksandr Tymoshenko * Reads the current flags of a given pin, here we actually read the H/W
329e53470feSOleksandr Tymoshenko * registers to determine the flags, rather than storing the value in the
330e53470feSOleksandr Tymoshenko * setflags call.
331e53470feSOleksandr Tymoshenko *
332e53470feSOleksandr Tymoshenko * LOCKING:
333e53470feSOleksandr Tymoshenko * Internally locks the context
334e53470feSOleksandr Tymoshenko *
335e53470feSOleksandr Tymoshenko * RETURNS:
336e53470feSOleksandr Tymoshenko * Returns 0 on success otherwise an error code
337e53470feSOleksandr Tymoshenko */
338e53470feSOleksandr Tymoshenko static int
ti_gpio_pin_getflags(device_t dev,uint32_t pin,uint32_t * flags)339e53470feSOleksandr Tymoshenko ti_gpio_pin_getflags(device_t dev, uint32_t pin, uint32_t *flags)
340e53470feSOleksandr Tymoshenko {
341e350f76cSLuiz Otavio O Souza struct ti_gpio_softc *sc;
342e53470feSOleksandr Tymoshenko
343e350f76cSLuiz Otavio O Souza sc = device_get_softc(dev);
344e350f76cSLuiz Otavio O Souza if (ti_gpio_valid_pin(sc, pin) != 0)
345e53470feSOleksandr Tymoshenko return (EINVAL);
346e53470feSOleksandr Tymoshenko
347e53470feSOleksandr Tymoshenko /* Get the current pin state */
348f9de33d4SLuiz Otavio O Souza TI_GPIO_LOCK(sc);
349b6c7dacfSAndrew Turner TI_GPIO_GET_FLAGS(dev, pin, flags);
350e53470feSOleksandr Tymoshenko TI_GPIO_UNLOCK(sc);
351e53470feSOleksandr Tymoshenko
352e53470feSOleksandr Tymoshenko return (0);
353e53470feSOleksandr Tymoshenko }
354e53470feSOleksandr Tymoshenko
355e53470feSOleksandr Tymoshenko /**
356e53470feSOleksandr Tymoshenko * ti_gpio_pin_getname - Gets the name of a given pin
357e53470feSOleksandr Tymoshenko * @dev: gpio device handle
358e53470feSOleksandr Tymoshenko * @pin: the number of the pin
359e53470feSOleksandr Tymoshenko * @name: buffer to put the name in
360e53470feSOleksandr Tymoshenko *
361e53470feSOleksandr Tymoshenko * The driver simply calls the pins gpio_n, where 'n' is obviously the number
362e53470feSOleksandr Tymoshenko * of the pin.
363e53470feSOleksandr Tymoshenko *
364e53470feSOleksandr Tymoshenko * LOCKING:
365f9de33d4SLuiz Otavio O Souza * No locking required, returns static data.
366e53470feSOleksandr Tymoshenko *
367e53470feSOleksandr Tymoshenko * RETURNS:
368e53470feSOleksandr Tymoshenko * Returns 0 on success otherwise an error code
369e53470feSOleksandr Tymoshenko */
370e53470feSOleksandr Tymoshenko static int
ti_gpio_pin_getname(device_t dev,uint32_t pin,char * name)371e53470feSOleksandr Tymoshenko ti_gpio_pin_getname(device_t dev, uint32_t pin, char *name)
372e53470feSOleksandr Tymoshenko {
373e350f76cSLuiz Otavio O Souza struct ti_gpio_softc *sc;
374e53470feSOleksandr Tymoshenko
375e350f76cSLuiz Otavio O Souza sc = device_get_softc(dev);
376e350f76cSLuiz Otavio O Souza if (ti_gpio_valid_pin(sc, pin) != 0)
377e53470feSOleksandr Tymoshenko return (EINVAL);
378e53470feSOleksandr Tymoshenko
379e53470feSOleksandr Tymoshenko /* Set a very simple name */
380e53470feSOleksandr Tymoshenko snprintf(name, GPIOMAXNAME, "gpio_%u", pin);
381e53470feSOleksandr Tymoshenko name[GPIOMAXNAME - 1] = '\0';
382e53470feSOleksandr Tymoshenko
383e53470feSOleksandr Tymoshenko return (0);
384e53470feSOleksandr Tymoshenko }
385e53470feSOleksandr Tymoshenko
386e53470feSOleksandr Tymoshenko /**
387e53470feSOleksandr Tymoshenko * ti_gpio_pin_setflags - Sets the flags for a given pin
388e53470feSOleksandr Tymoshenko * @dev: gpio device handle
389e53470feSOleksandr Tymoshenko * @pin: the number of the pin
390e53470feSOleksandr Tymoshenko * @flags: the flags to set
391e53470feSOleksandr Tymoshenko *
392e53470feSOleksandr Tymoshenko * The flags of the pin correspond to things like input/output mode, pull-ups,
393e53470feSOleksandr Tymoshenko * pull-downs, etc. This driver doesn't support all flags, only the following:
394e53470feSOleksandr Tymoshenko * - GPIO_PIN_INPUT
395e53470feSOleksandr Tymoshenko * - GPIO_PIN_OUTPUT
396e53470feSOleksandr Tymoshenko * - GPIO_PIN_PULLUP
397e53470feSOleksandr Tymoshenko * - GPIO_PIN_PULLDOWN
398e53470feSOleksandr Tymoshenko *
399e53470feSOleksandr Tymoshenko * LOCKING:
400e53470feSOleksandr Tymoshenko * Internally locks the context
401e53470feSOleksandr Tymoshenko *
402e53470feSOleksandr Tymoshenko * RETURNS:
403e53470feSOleksandr Tymoshenko * Returns 0 on success otherwise an error code
404e53470feSOleksandr Tymoshenko */
405e53470feSOleksandr Tymoshenko static int
ti_gpio_pin_setflags(device_t dev,uint32_t pin,uint32_t flags)406e53470feSOleksandr Tymoshenko ti_gpio_pin_setflags(device_t dev, uint32_t pin, uint32_t flags)
407e53470feSOleksandr Tymoshenko {
408e350f76cSLuiz Otavio O Souza struct ti_gpio_softc *sc;
409f9de33d4SLuiz Otavio O Souza uint32_t oe;
410e53470feSOleksandr Tymoshenko
411e350f76cSLuiz Otavio O Souza sc = device_get_softc(dev);
412e350f76cSLuiz Otavio O Souza if (ti_gpio_valid_pin(sc, pin) != 0)
413e53470feSOleksandr Tymoshenko return (EINVAL);
414e53470feSOleksandr Tymoshenko
415e53470feSOleksandr Tymoshenko /* Set the GPIO mode and state */
416f9de33d4SLuiz Otavio O Souza TI_GPIO_LOCK(sc);
417b6c7dacfSAndrew Turner if (TI_GPIO_SET_FLAGS(dev, pin, flags) != 0) {
418e53470feSOleksandr Tymoshenko TI_GPIO_UNLOCK(sc);
419e53470feSOleksandr Tymoshenko return (EINVAL);
420e53470feSOleksandr Tymoshenko }
421e53470feSOleksandr Tymoshenko
422e53470feSOleksandr Tymoshenko /* If configuring as an output set the "output enable" bit */
4235b03aba6SOleksandr Tymoshenko oe = ti_gpio_read_4(sc, TI_GPIO_OE);
424e53470feSOleksandr Tymoshenko if (flags & GPIO_PIN_INPUT)
425e350f76cSLuiz Otavio O Souza oe |= TI_GPIO_MASK(pin);
426e53470feSOleksandr Tymoshenko else
427e350f76cSLuiz Otavio O Souza oe &= ~TI_GPIO_MASK(pin);
4285b03aba6SOleksandr Tymoshenko ti_gpio_write_4(sc, TI_GPIO_OE, oe);
429e53470feSOleksandr Tymoshenko TI_GPIO_UNLOCK(sc);
430e53470feSOleksandr Tymoshenko
431e53470feSOleksandr Tymoshenko return (0);
432e53470feSOleksandr Tymoshenko }
433e53470feSOleksandr Tymoshenko
434e53470feSOleksandr Tymoshenko /**
435e53470feSOleksandr Tymoshenko * ti_gpio_pin_set - Sets the current level on a GPIO pin
436e53470feSOleksandr Tymoshenko * @dev: gpio device handle
437e53470feSOleksandr Tymoshenko * @pin: the number of the pin
438e53470feSOleksandr Tymoshenko * @value: non-zero value will drive the pin high, otherwise the pin is
439e53470feSOleksandr Tymoshenko * driven low.
440e53470feSOleksandr Tymoshenko *
441e53470feSOleksandr Tymoshenko *
442e53470feSOleksandr Tymoshenko * LOCKING:
443e53470feSOleksandr Tymoshenko * Internally locks the context
444e53470feSOleksandr Tymoshenko *
445e53470feSOleksandr Tymoshenko * RETURNS:
446e53470feSOleksandr Tymoshenko * Returns 0 on success otherwise a error code
447e53470feSOleksandr Tymoshenko */
448e53470feSOleksandr Tymoshenko static int
ti_gpio_pin_set(device_t dev,uint32_t pin,unsigned int value)449e53470feSOleksandr Tymoshenko ti_gpio_pin_set(device_t dev, uint32_t pin, unsigned int value)
450e53470feSOleksandr Tymoshenko {
451e350f76cSLuiz Otavio O Souza struct ti_gpio_softc *sc;
452f9de33d4SLuiz Otavio O Souza uint32_t reg;
453e53470feSOleksandr Tymoshenko
454e350f76cSLuiz Otavio O Souza sc = device_get_softc(dev);
455e350f76cSLuiz Otavio O Souza if (ti_gpio_valid_pin(sc, pin) != 0)
456e53470feSOleksandr Tymoshenko return (EINVAL);
457e53470feSOleksandr Tymoshenko
458f9de33d4SLuiz Otavio O Souza TI_GPIO_LOCK(sc);
459f9de33d4SLuiz Otavio O Souza if (value == GPIO_PIN_LOW)
460f9de33d4SLuiz Otavio O Souza reg = TI_GPIO_CLEARDATAOUT;
461f9de33d4SLuiz Otavio O Souza else
462f9de33d4SLuiz Otavio O Souza reg = TI_GPIO_SETDATAOUT;
4635b03aba6SOleksandr Tymoshenko ti_gpio_write_4(sc, reg, TI_GPIO_MASK(pin));
464e53470feSOleksandr Tymoshenko TI_GPIO_UNLOCK(sc);
465e53470feSOleksandr Tymoshenko
466e53470feSOleksandr Tymoshenko return (0);
467e53470feSOleksandr Tymoshenko }
468e53470feSOleksandr Tymoshenko
469e53470feSOleksandr Tymoshenko /**
470e53470feSOleksandr Tymoshenko * ti_gpio_pin_get - Gets the current level on a GPIO pin
471e53470feSOleksandr Tymoshenko * @dev: gpio device handle
472e53470feSOleksandr Tymoshenko * @pin: the number of the pin
473e53470feSOleksandr Tymoshenko * @value: pointer to a value that upond return will contain the pin value
474e53470feSOleksandr Tymoshenko *
475e53470feSOleksandr Tymoshenko * The pin must be configured as an input pin beforehand, otherwise this
476e53470feSOleksandr Tymoshenko * function will fail.
477e53470feSOleksandr Tymoshenko *
478e53470feSOleksandr Tymoshenko * LOCKING:
479e53470feSOleksandr Tymoshenko * Internally locks the context
480e53470feSOleksandr Tymoshenko *
481e53470feSOleksandr Tymoshenko * RETURNS:
482e53470feSOleksandr Tymoshenko * Returns 0 on success otherwise a error code
483e53470feSOleksandr Tymoshenko */
484e53470feSOleksandr Tymoshenko static int
ti_gpio_pin_get(device_t dev,uint32_t pin,unsigned int * value)485e53470feSOleksandr Tymoshenko ti_gpio_pin_get(device_t dev, uint32_t pin, unsigned int *value)
486e53470feSOleksandr Tymoshenko {
487e350f76cSLuiz Otavio O Souza struct ti_gpio_softc *sc;
488e350f76cSLuiz Otavio O Souza uint32_t oe, reg, val;
489e53470feSOleksandr Tymoshenko
490e350f76cSLuiz Otavio O Souza sc = device_get_softc(dev);
491e350f76cSLuiz Otavio O Souza if (ti_gpio_valid_pin(sc, pin) != 0)
492e53470feSOleksandr Tymoshenko return (EINVAL);
493e53470feSOleksandr Tymoshenko
494f9de33d4SLuiz Otavio O Souza /*
495f9de33d4SLuiz Otavio O Souza * Return data from output latch when set as output and from the
496f9de33d4SLuiz Otavio O Souza * input register otherwise.
497f9de33d4SLuiz Otavio O Souza */
498f9de33d4SLuiz Otavio O Souza TI_GPIO_LOCK(sc);
4995b03aba6SOleksandr Tymoshenko oe = ti_gpio_read_4(sc, TI_GPIO_OE);
500e350f76cSLuiz Otavio O Souza if (oe & TI_GPIO_MASK(pin))
501f9de33d4SLuiz Otavio O Souza reg = TI_GPIO_DATAIN;
5029f165184SLuiz Otavio O Souza else
503f9de33d4SLuiz Otavio O Souza reg = TI_GPIO_DATAOUT;
5045b03aba6SOleksandr Tymoshenko val = ti_gpio_read_4(sc, reg);
505e350f76cSLuiz Otavio O Souza *value = (val & TI_GPIO_MASK(pin)) ? 1 : 0;
506e53470feSOleksandr Tymoshenko TI_GPIO_UNLOCK(sc);
507e53470feSOleksandr Tymoshenko
508e53470feSOleksandr Tymoshenko return (0);
509e53470feSOleksandr Tymoshenko }
510e53470feSOleksandr Tymoshenko
511e53470feSOleksandr Tymoshenko /**
512e53470feSOleksandr Tymoshenko * ti_gpio_pin_toggle - Toggles a given GPIO pin
513e53470feSOleksandr Tymoshenko * @dev: gpio device handle
514e53470feSOleksandr Tymoshenko * @pin: the number of the pin
515e53470feSOleksandr Tymoshenko *
516e53470feSOleksandr Tymoshenko *
517e53470feSOleksandr Tymoshenko * LOCKING:
518e53470feSOleksandr Tymoshenko * Internally locks the context
519e53470feSOleksandr Tymoshenko *
520e53470feSOleksandr Tymoshenko * RETURNS:
521e53470feSOleksandr Tymoshenko * Returns 0 on success otherwise a error code
522e53470feSOleksandr Tymoshenko */
523e53470feSOleksandr Tymoshenko static int
ti_gpio_pin_toggle(device_t dev,uint32_t pin)524e53470feSOleksandr Tymoshenko ti_gpio_pin_toggle(device_t dev, uint32_t pin)
525e53470feSOleksandr Tymoshenko {
526e350f76cSLuiz Otavio O Souza struct ti_gpio_softc *sc;
527f9de33d4SLuiz Otavio O Souza uint32_t reg, val;
528e53470feSOleksandr Tymoshenko
529e350f76cSLuiz Otavio O Souza sc = device_get_softc(dev);
530e350f76cSLuiz Otavio O Souza if (ti_gpio_valid_pin(sc, pin) != 0)
531e53470feSOleksandr Tymoshenko return (EINVAL);
532e53470feSOleksandr Tymoshenko
533e53470feSOleksandr Tymoshenko /* Toggle the pin */
534f9de33d4SLuiz Otavio O Souza TI_GPIO_LOCK(sc);
5355b03aba6SOleksandr Tymoshenko val = ti_gpio_read_4(sc, TI_GPIO_DATAOUT);
536e350f76cSLuiz Otavio O Souza if (val & TI_GPIO_MASK(pin))
537f9de33d4SLuiz Otavio O Souza reg = TI_GPIO_CLEARDATAOUT;
538e53470feSOleksandr Tymoshenko else
539f9de33d4SLuiz Otavio O Souza reg = TI_GPIO_SETDATAOUT;
5405b03aba6SOleksandr Tymoshenko ti_gpio_write_4(sc, reg, TI_GPIO_MASK(pin));
541e53470feSOleksandr Tymoshenko TI_GPIO_UNLOCK(sc);
542e53470feSOleksandr Tymoshenko
543e53470feSOleksandr Tymoshenko return (0);
544e53470feSOleksandr Tymoshenko }
545e53470feSOleksandr Tymoshenko
5469b1cba84SLuiz Otavio O Souza static int
ti_gpio_bank_init(device_t dev)5475b03aba6SOleksandr Tymoshenko ti_gpio_bank_init(device_t dev)
5489b1cba84SLuiz Otavio O Souza {
5490050ea24SMichal Meloun int pin, err;
5509b1cba84SLuiz Otavio O Souza struct ti_gpio_softc *sc;
551c3321180SOleksandr Tymoshenko uint32_t flags, reg_oe, reg_set, rev;
5520050ea24SMichal Meloun uint64_t rev_address;
5539b1cba84SLuiz Otavio O Souza
5549b1cba84SLuiz Otavio O Souza sc = device_get_softc(dev);
5559b1cba84SLuiz Otavio O Souza
5569b1cba84SLuiz Otavio O Souza /* Enable the interface and functional clocks for the module. */
5570050ea24SMichal Meloun rev_address = ti_sysc_get_rev_address(device_get_parent(dev));
5580050ea24SMichal Meloun /* AM335x
5590050ea24SMichal Meloun * sc->sc_bank used in am335x/am335x_gpio.c and omap4/omap4_gpio.c */
5600050ea24SMichal Meloun switch(ti_chip()) {
5610050ea24SMichal Meloun #ifdef SOC_OMAP4
5620050ea24SMichal Meloun case CHIP_OMAP_4:
5630050ea24SMichal Meloun switch (rev_address) {
5640050ea24SMichal Meloun case OMAP4_GPIO1_REV:
5650050ea24SMichal Meloun sc->sc_bank = 0;
5660050ea24SMichal Meloun break;
5670050ea24SMichal Meloun case OMAP4_GPIO2_REV:
5680050ea24SMichal Meloun sc->sc_bank = 1;
5690050ea24SMichal Meloun break;
5700050ea24SMichal Meloun case OMAP4_GPIO3_REV:
5710050ea24SMichal Meloun sc->sc_bank = 2;
5720050ea24SMichal Meloun break;
5730050ea24SMichal Meloun case OMAP4_GPIO4_REV:
5740050ea24SMichal Meloun sc->sc_bank = 3;
5750050ea24SMichal Meloun break;
5760050ea24SMichal Meloun case OMAP4_GPIO5_REV:
5770050ea24SMichal Meloun sc->sc_bank = 4;
5780050ea24SMichal Meloun break;
5790050ea24SMichal Meloun case OMAP4_GPIO6_REV:
5800050ea24SMichal Meloun sc->sc_bank = 5;
5810050ea24SMichal Meloun break;
5820050ea24SMichal Meloun }
5830050ea24SMichal Meloun #endif
5840050ea24SMichal Meloun #ifdef SOC_TI_AM335X
5850050ea24SMichal Meloun case CHIP_AM335X:
5860050ea24SMichal Meloun switch (rev_address) {
5870050ea24SMichal Meloun case AM335X_GPIO0_REV:
5880050ea24SMichal Meloun sc->sc_bank = 0;
5890050ea24SMichal Meloun break;
5900050ea24SMichal Meloun case AM335X_GPIO1_REV:
5910050ea24SMichal Meloun sc->sc_bank = 1;
5920050ea24SMichal Meloun break;
5930050ea24SMichal Meloun case AM335X_GPIO2_REV:
5940050ea24SMichal Meloun sc->sc_bank = 2;
5950050ea24SMichal Meloun break;
5960050ea24SMichal Meloun case AM335X_GPIO3_REV:
5970050ea24SMichal Meloun sc->sc_bank = 3;
5980050ea24SMichal Meloun break;
5990050ea24SMichal Meloun }
6000050ea24SMichal Meloun #endif
6010050ea24SMichal Meloun }
6020050ea24SMichal Meloun err = ti_sysc_clock_enable(device_get_parent(dev));
6030050ea24SMichal Meloun if (err) {
6040050ea24SMichal Meloun device_printf(dev, "Failed to enable clock\n");
6055b03aba6SOleksandr Tymoshenko return (EINVAL);
6065b03aba6SOleksandr Tymoshenko }
6075b03aba6SOleksandr Tymoshenko
6089b1cba84SLuiz Otavio O Souza /*
6099b1cba84SLuiz Otavio O Souza * Read the revision number of the module. TI don't publish the
6109b1cba84SLuiz Otavio O Souza * actual revision numbers, so instead the values have been
6119b1cba84SLuiz Otavio O Souza * determined by experimentation.
6129b1cba84SLuiz Otavio O Souza */
6130050ea24SMichal Meloun rev = ti_gpio_read_4(sc,
6140050ea24SMichal Meloun ti_sysc_get_rev_address_offset_host(device_get_parent(dev)));
6159b1cba84SLuiz Otavio O Souza
6169b1cba84SLuiz Otavio O Souza /* Check the revision. */
617e350f76cSLuiz Otavio O Souza if (rev != ti_gpio_rev()) {
6189b1cba84SLuiz Otavio O Souza device_printf(dev, "Warning: could not determine the revision "
6195b03aba6SOleksandr Tymoshenko "of GPIO module (revision:0x%08x)\n", rev);
6209b1cba84SLuiz Otavio O Souza return (EINVAL);
6219b1cba84SLuiz Otavio O Souza }
6229b1cba84SLuiz Otavio O Souza
6239b1cba84SLuiz Otavio O Souza /* Disable interrupts for all pins. */
6245b03aba6SOleksandr Tymoshenko ti_gpio_intr_clr(sc, 0xffffffff);
6259b1cba84SLuiz Otavio O Souza
6269b1cba84SLuiz Otavio O Souza /* Init OE register based on pads configuration. */
6279b1cba84SLuiz Otavio O Souza reg_oe = 0xffffffff;
628c3321180SOleksandr Tymoshenko reg_set = 0;
6299b1cba84SLuiz Otavio O Souza for (pin = 0; pin < PINS_PER_BANK; pin++) {
6305b03aba6SOleksandr Tymoshenko TI_GPIO_GET_FLAGS(dev, pin, &flags);
631c3321180SOleksandr Tymoshenko if (flags & GPIO_PIN_OUTPUT) {
6329b1cba84SLuiz Otavio O Souza reg_oe &= ~(1UL << pin);
633c3321180SOleksandr Tymoshenko if (flags & GPIO_PIN_PULLUP)
634c3321180SOleksandr Tymoshenko reg_set |= (1UL << pin);
635c3321180SOleksandr Tymoshenko }
6369b1cba84SLuiz Otavio O Souza }
6375b03aba6SOleksandr Tymoshenko ti_gpio_write_4(sc, TI_GPIO_OE, reg_oe);
638c3321180SOleksandr Tymoshenko if (reg_set)
639c3321180SOleksandr Tymoshenko ti_gpio_write_4(sc, TI_GPIO_SETDATAOUT, reg_set);
6409b1cba84SLuiz Otavio O Souza
6419b1cba84SLuiz Otavio O Souza return (0);
6429b1cba84SLuiz Otavio O Souza }
6439b1cba84SLuiz Otavio O Souza
644e53470feSOleksandr Tymoshenko /**
645e53470feSOleksandr Tymoshenko * ti_gpio_attach - attach function for the driver
646e53470feSOleksandr Tymoshenko * @dev: gpio device handle
647e53470feSOleksandr Tymoshenko *
648e53470feSOleksandr Tymoshenko * Allocates and sets up the driver context for all GPIO banks. This function
649e53470feSOleksandr Tymoshenko * expects the memory ranges and IRQs to already be allocated to the driver.
650e53470feSOleksandr Tymoshenko *
651e53470feSOleksandr Tymoshenko * LOCKING:
652e53470feSOleksandr Tymoshenko * None
653e53470feSOleksandr Tymoshenko *
654e53470feSOleksandr Tymoshenko * RETURNS:
655e53470feSOleksandr Tymoshenko * Always returns 0
656e53470feSOleksandr Tymoshenko */
657e53470feSOleksandr Tymoshenko static int
ti_gpio_attach(device_t dev)658e53470feSOleksandr Tymoshenko ti_gpio_attach(device_t dev)
659e53470feSOleksandr Tymoshenko {
6609b1cba84SLuiz Otavio O Souza struct ti_gpio_softc *sc;
6619b1cba84SLuiz Otavio O Souza int err;
662e53470feSOleksandr Tymoshenko
6635b03aba6SOleksandr Tymoshenko sc = device_get_softc(dev);
664e53470feSOleksandr Tymoshenko sc->sc_dev = dev;
665e53470feSOleksandr Tymoshenko TI_GPIO_LOCK_INIT(sc);
666e350f76cSLuiz Otavio O Souza ti_gpio_pin_max(dev, &sc->sc_maxpin);
667f7f77280SLuiz Otavio O Souza sc->sc_maxpin++;
668e53470feSOleksandr Tymoshenko
6695b03aba6SOleksandr Tymoshenko sc->sc_mem_rid = 0;
6705b03aba6SOleksandr Tymoshenko sc->sc_mem_res = bus_alloc_resource_any(dev, SYS_RES_MEMORY,
6715b03aba6SOleksandr Tymoshenko &sc->sc_mem_rid, RF_ACTIVE);
6725b03aba6SOleksandr Tymoshenko if (!sc->sc_mem_res) {
673e53470feSOleksandr Tymoshenko device_printf(dev, "Error: could not allocate mem resources\n");
674876c1bd8SLuiz Otavio O Souza ti_gpio_detach(dev);
675e53470feSOleksandr Tymoshenko return (ENXIO);
676e53470feSOleksandr Tymoshenko }
677e53470feSOleksandr Tymoshenko
6785b03aba6SOleksandr Tymoshenko sc->sc_irq_rid = 0;
6795b03aba6SOleksandr Tymoshenko sc->sc_irq_res = bus_alloc_resource_any(dev, SYS_RES_IRQ,
6805b03aba6SOleksandr Tymoshenko &sc->sc_irq_rid, RF_ACTIVE);
6815b03aba6SOleksandr Tymoshenko if (!sc->sc_irq_res) {
682e53470feSOleksandr Tymoshenko device_printf(dev, "Error: could not allocate irq resources\n");
683876c1bd8SLuiz Otavio O Souza ti_gpio_detach(dev);
684e53470feSOleksandr Tymoshenko return (ENXIO);
685e53470feSOleksandr Tymoshenko }
686e53470feSOleksandr Tymoshenko
6875b03aba6SOleksandr Tymoshenko /*
6885b03aba6SOleksandr Tymoshenko * Register our interrupt filter for each of the IRQ resources.
6895b03aba6SOleksandr Tymoshenko */
6905b03aba6SOleksandr Tymoshenko if (bus_setup_intr(dev, sc->sc_irq_res,
6915b03aba6SOleksandr Tymoshenko INTR_TYPE_MISC | INTR_MPSAFE, ti_gpio_intr, NULL, sc,
6925b03aba6SOleksandr Tymoshenko &sc->sc_irq_hdl) != 0) {
6935b03aba6SOleksandr Tymoshenko device_printf(dev,
6945b03aba6SOleksandr Tymoshenko "WARNING: unable to register interrupt filter\n");
695876c1bd8SLuiz Otavio O Souza ti_gpio_detach(dev);
696e53470feSOleksandr Tymoshenko return (ENXIO);
697e53470feSOleksandr Tymoshenko }
698e53470feSOleksandr Tymoshenko
6992df5562dSSvatopluk Kraus if (ti_gpio_pic_attach(sc) != 0) {
7002df5562dSSvatopluk Kraus device_printf(dev, "WARNING: unable to attach PIC\n");
7012df5562dSSvatopluk Kraus ti_gpio_detach(dev);
7022df5562dSSvatopluk Kraus return (ENXIO);
7032df5562dSSvatopluk Kraus }
7043681c8d7SLuiz Otavio O Souza
705e53470feSOleksandr Tymoshenko /* We need to go through each block and ensure the clocks are running and
706e53470feSOleksandr Tymoshenko * the module is enabled. It might be better to do this only when the
707e53470feSOleksandr Tymoshenko * pins are configured which would result in less power used if the GPIO
708e53470feSOleksandr Tymoshenko * pins weren't used ...
709e53470feSOleksandr Tymoshenko */
7105b03aba6SOleksandr Tymoshenko if (sc->sc_mem_res != NULL) {
7111f1e8f16SLuiz Otavio O Souza /* Initialize the GPIO module. */
7125b03aba6SOleksandr Tymoshenko err = ti_gpio_bank_init(dev);
7139b1cba84SLuiz Otavio O Souza if (err != 0) {
714876c1bd8SLuiz Otavio O Souza ti_gpio_detach(dev);
7159b1cba84SLuiz Otavio O Souza return (err);
716e53470feSOleksandr Tymoshenko }
717e53470feSOleksandr Tymoshenko }
7185b03aba6SOleksandr Tymoshenko
7197836352bSLuiz Otavio O Souza sc->sc_busdev = gpiobus_attach_bus(dev);
7207836352bSLuiz Otavio O Souza if (sc->sc_busdev == NULL) {
7217836352bSLuiz Otavio O Souza ti_gpio_detach(dev);
7227836352bSLuiz Otavio O Souza return (ENXIO);
7237836352bSLuiz Otavio O Souza }
724e53470feSOleksandr Tymoshenko
7257836352bSLuiz Otavio O Souza return (0);
726e53470feSOleksandr Tymoshenko }
727e53470feSOleksandr Tymoshenko
728e53470feSOleksandr Tymoshenko /**
729e53470feSOleksandr Tymoshenko * ti_gpio_detach - detach function for the driver
730e53470feSOleksandr Tymoshenko * @dev: scm device handle
731e53470feSOleksandr Tymoshenko *
732e53470feSOleksandr Tymoshenko * Allocates and sets up the driver context, this simply entails creating a
733e53470feSOleksandr Tymoshenko * bus mappings for the SCM register set.
734e53470feSOleksandr Tymoshenko *
735e53470feSOleksandr Tymoshenko * LOCKING:
736e53470feSOleksandr Tymoshenko * None
737e53470feSOleksandr Tymoshenko *
738e53470feSOleksandr Tymoshenko * RETURNS:
739e53470feSOleksandr Tymoshenko * Always returns 0
740e53470feSOleksandr Tymoshenko */
741e53470feSOleksandr Tymoshenko static int
ti_gpio_detach(device_t dev)742e53470feSOleksandr Tymoshenko ti_gpio_detach(device_t dev)
743e53470feSOleksandr Tymoshenko {
744e53470feSOleksandr Tymoshenko struct ti_gpio_softc *sc = device_get_softc(dev);
745e53470feSOleksandr Tymoshenko
746e53470feSOleksandr Tymoshenko KASSERT(mtx_initialized(&sc->sc_mtx), ("gpio mutex not initialized"));
747e53470feSOleksandr Tymoshenko
748e53470feSOleksandr Tymoshenko /* Disable all interrupts */
7495b03aba6SOleksandr Tymoshenko if (sc->sc_mem_res != NULL)
7505b03aba6SOleksandr Tymoshenko ti_gpio_intr_clr(sc, 0xffffffff);
751197029feSLuiz Otavio O Souza if (sc->sc_busdev != NULL)
7527836352bSLuiz Otavio O Souza gpiobus_detach_bus(dev);
7532df5562dSSvatopluk Kraus if (sc->sc_isrcs != NULL)
7542df5562dSSvatopluk Kraus ti_gpio_pic_detach(sc);
7559b1cba84SLuiz Otavio O Souza /* Release the memory and IRQ resources. */
7565b03aba6SOleksandr Tymoshenko if (sc->sc_irq_hdl) {
7575b03aba6SOleksandr Tymoshenko bus_teardown_intr(dev, sc->sc_irq_res,
7585b03aba6SOleksandr Tymoshenko sc->sc_irq_hdl);
7595b03aba6SOleksandr Tymoshenko }
760197029feSLuiz Otavio O Souza if (sc->sc_irq_res)
7615b03aba6SOleksandr Tymoshenko bus_release_resource(dev, SYS_RES_IRQ, sc->sc_irq_rid,
7625b03aba6SOleksandr Tymoshenko sc->sc_irq_res);
763197029feSLuiz Otavio O Souza if (sc->sc_mem_res)
7645b03aba6SOleksandr Tymoshenko bus_release_resource(dev, SYS_RES_MEMORY, sc->sc_mem_rid,
7655b03aba6SOleksandr Tymoshenko sc->sc_mem_res);
766e53470feSOleksandr Tymoshenko TI_GPIO_LOCK_DESTROY(sc);
767e53470feSOleksandr Tymoshenko
768e53470feSOleksandr Tymoshenko return (0);
769e53470feSOleksandr Tymoshenko }
770e53470feSOleksandr Tymoshenko
7712df5562dSSvatopluk Kraus static inline void
ti_gpio_rwreg_modify(struct ti_gpio_softc * sc,uint32_t reg,uint32_t mask,bool set_bits)7722202c379SSvatopluk Kraus ti_gpio_rwreg_modify(struct ti_gpio_softc *sc, uint32_t reg, uint32_t mask,
7732202c379SSvatopluk Kraus bool set_bits)
7742df5562dSSvatopluk Kraus {
7752202c379SSvatopluk Kraus uint32_t value;
7762df5562dSSvatopluk Kraus
7772202c379SSvatopluk Kraus value = ti_gpio_read_4(sc, reg);
7782202c379SSvatopluk Kraus ti_gpio_write_4(sc, reg, set_bits ? value | mask : value & ~mask);
7792df5562dSSvatopluk Kraus }
7802df5562dSSvatopluk Kraus
7812df5562dSSvatopluk Kraus static inline void
ti_gpio_isrc_mask(struct ti_gpio_softc * sc,struct ti_gpio_irqsrc * tgi)7822df5562dSSvatopluk Kraus ti_gpio_isrc_mask(struct ti_gpio_softc *sc, struct ti_gpio_irqsrc *tgi)
7832df5562dSSvatopluk Kraus {
7842df5562dSSvatopluk Kraus
7852df5562dSSvatopluk Kraus /* Writing a 0 has no effect. */
7862df5562dSSvatopluk Kraus ti_gpio_intr_clr(sc, tgi->tgi_mask);
7872df5562dSSvatopluk Kraus }
7882df5562dSSvatopluk Kraus
7892df5562dSSvatopluk Kraus static inline void
ti_gpio_isrc_unmask(struct ti_gpio_softc * sc,struct ti_gpio_irqsrc * tgi)7902df5562dSSvatopluk Kraus ti_gpio_isrc_unmask(struct ti_gpio_softc *sc, struct ti_gpio_irqsrc *tgi)
7912df5562dSSvatopluk Kraus {
7922df5562dSSvatopluk Kraus
7932df5562dSSvatopluk Kraus /* Writing a 0 has no effect. */
7942df5562dSSvatopluk Kraus ti_gpio_intr_set(sc, tgi->tgi_mask);
7952df5562dSSvatopluk Kraus }
7962df5562dSSvatopluk Kraus
7972df5562dSSvatopluk Kraus static inline void
ti_gpio_isrc_eoi(struct ti_gpio_softc * sc,struct ti_gpio_irqsrc * tgi)7982df5562dSSvatopluk Kraus ti_gpio_isrc_eoi(struct ti_gpio_softc *sc, struct ti_gpio_irqsrc *tgi)
7992df5562dSSvatopluk Kraus {
8002df5562dSSvatopluk Kraus
8012df5562dSSvatopluk Kraus /* Writing a 0 has no effect. */
8022df5562dSSvatopluk Kraus ti_gpio_intr_ack(sc, tgi->tgi_mask);
8032df5562dSSvatopluk Kraus }
8042df5562dSSvatopluk Kraus
8052df5562dSSvatopluk Kraus static inline bool
ti_gpio_isrc_is_level(struct ti_gpio_irqsrc * tgi)8062df5562dSSvatopluk Kraus ti_gpio_isrc_is_level(struct ti_gpio_irqsrc *tgi)
8072df5562dSSvatopluk Kraus {
8082df5562dSSvatopluk Kraus
8092202c379SSvatopluk Kraus return (tgi->tgi_mode == GPIO_INTR_LEVEL_LOW ||
8102202c379SSvatopluk Kraus tgi->tgi_mode == GPIO_INTR_LEVEL_HIGH);
8112df5562dSSvatopluk Kraus }
8122df5562dSSvatopluk Kraus
8132df5562dSSvatopluk Kraus static int
ti_gpio_intr(void * arg)8142df5562dSSvatopluk Kraus ti_gpio_intr(void *arg)
8152df5562dSSvatopluk Kraus {
8162df5562dSSvatopluk Kraus u_int irq;
8172df5562dSSvatopluk Kraus uint32_t reg;
8182df5562dSSvatopluk Kraus struct ti_gpio_softc *sc;
8192df5562dSSvatopluk Kraus struct trapframe *tf;
8202df5562dSSvatopluk Kraus struct ti_gpio_irqsrc *tgi;
8212df5562dSSvatopluk Kraus
8222df5562dSSvatopluk Kraus sc = (struct ti_gpio_softc *)arg;
8232df5562dSSvatopluk Kraus tf = curthread->td_intr_frame;
8242df5562dSSvatopluk Kraus
8252df5562dSSvatopluk Kraus reg = ti_gpio_intr_status(sc);
8262df5562dSSvatopluk Kraus for (irq = 0; irq < sc->sc_maxpin; irq++) {
8272df5562dSSvatopluk Kraus tgi = &sc->sc_isrcs[irq];
8282df5562dSSvatopluk Kraus if ((reg & tgi->tgi_mask) == 0)
8292df5562dSSvatopluk Kraus continue;
8302df5562dSSvatopluk Kraus if (!ti_gpio_isrc_is_level(tgi))
8312df5562dSSvatopluk Kraus ti_gpio_isrc_eoi(sc, tgi);
8322df5562dSSvatopluk Kraus if (intr_isrc_dispatch(&tgi->tgi_isrc, tf) != 0) {
8332df5562dSSvatopluk Kraus ti_gpio_isrc_mask(sc, tgi);
8342df5562dSSvatopluk Kraus if (ti_gpio_isrc_is_level(tgi))
8352df5562dSSvatopluk Kraus ti_gpio_isrc_eoi(sc, tgi);
8362df5562dSSvatopluk Kraus device_printf(sc->sc_dev, "Stray irq %u disabled\n",
8372df5562dSSvatopluk Kraus irq);
8382df5562dSSvatopluk Kraus }
8392df5562dSSvatopluk Kraus }
8402df5562dSSvatopluk Kraus return (FILTER_HANDLED);
8412df5562dSSvatopluk Kraus }
8422df5562dSSvatopluk Kraus
8432df5562dSSvatopluk Kraus static int
ti_gpio_pic_attach(struct ti_gpio_softc * sc)8442df5562dSSvatopluk Kraus ti_gpio_pic_attach(struct ti_gpio_softc *sc)
8452df5562dSSvatopluk Kraus {
8462df5562dSSvatopluk Kraus int error;
8472df5562dSSvatopluk Kraus uint32_t irq;
8482df5562dSSvatopluk Kraus const char *name;
8492df5562dSSvatopluk Kraus
8502df5562dSSvatopluk Kraus sc->sc_isrcs = malloc(sizeof(*sc->sc_isrcs) * sc->sc_maxpin, M_DEVBUF,
8512df5562dSSvatopluk Kraus M_WAITOK | M_ZERO);
8522df5562dSSvatopluk Kraus
8532df5562dSSvatopluk Kraus name = device_get_nameunit(sc->sc_dev);
8542df5562dSSvatopluk Kraus for (irq = 0; irq < sc->sc_maxpin; irq++) {
8552df5562dSSvatopluk Kraus sc->sc_isrcs[irq].tgi_irq = irq;
8562df5562dSSvatopluk Kraus sc->sc_isrcs[irq].tgi_mask = TI_GPIO_MASK(irq);
8572202c379SSvatopluk Kraus sc->sc_isrcs[irq].tgi_mode = GPIO_INTR_CONFORM;
8582df5562dSSvatopluk Kraus
8592df5562dSSvatopluk Kraus error = intr_isrc_register(&sc->sc_isrcs[irq].tgi_isrc,
8602df5562dSSvatopluk Kraus sc->sc_dev, 0, "%s,%u", name, irq);
8612df5562dSSvatopluk Kraus if (error != 0)
8622df5562dSSvatopluk Kraus return (error); /* XXX deregister ISRCs */
8632df5562dSSvatopluk Kraus }
8649346e913SAndrew Turner if (intr_pic_register(sc->sc_dev,
8659346e913SAndrew Turner OF_xref_from_node(ofw_bus_get_node(sc->sc_dev))) == NULL)
8669346e913SAndrew Turner return (ENXIO);
8679346e913SAndrew Turner
8689346e913SAndrew Turner return (0);
8692df5562dSSvatopluk Kraus }
8702df5562dSSvatopluk Kraus
8712df5562dSSvatopluk Kraus static int
ti_gpio_pic_detach(struct ti_gpio_softc * sc)8722df5562dSSvatopluk Kraus ti_gpio_pic_detach(struct ti_gpio_softc *sc)
8732df5562dSSvatopluk Kraus {
8742df5562dSSvatopluk Kraus
8752df5562dSSvatopluk Kraus /*
8762df5562dSSvatopluk Kraus * There has not been established any procedure yet
8772df5562dSSvatopluk Kraus * how to detach PIC from living system correctly.
8782df5562dSSvatopluk Kraus */
8792df5562dSSvatopluk Kraus device_printf(sc->sc_dev, "%s: not implemented yet\n", __func__);
8802df5562dSSvatopluk Kraus return (EBUSY);
8812df5562dSSvatopluk Kraus }
8822df5562dSSvatopluk Kraus
8832df5562dSSvatopluk Kraus static void
ti_gpio_pic_config_intr(struct ti_gpio_softc * sc,struct ti_gpio_irqsrc * tgi,uint32_t mode)8842202c379SSvatopluk Kraus ti_gpio_pic_config_intr(struct ti_gpio_softc *sc, struct ti_gpio_irqsrc *tgi,
8852202c379SSvatopluk Kraus uint32_t mode)
8862202c379SSvatopluk Kraus {
8872202c379SSvatopluk Kraus
8882202c379SSvatopluk Kraus TI_GPIO_LOCK(sc);
8892202c379SSvatopluk Kraus ti_gpio_rwreg_modify(sc, TI_GPIO_RISINGDETECT, tgi->tgi_mask,
8902202c379SSvatopluk Kraus mode == GPIO_INTR_EDGE_RISING || mode == GPIO_INTR_EDGE_BOTH);
8912202c379SSvatopluk Kraus ti_gpio_rwreg_modify(sc, TI_GPIO_FALLINGDETECT, tgi->tgi_mask,
8922202c379SSvatopluk Kraus mode == GPIO_INTR_EDGE_FALLING || mode == GPIO_INTR_EDGE_BOTH);
8932202c379SSvatopluk Kraus ti_gpio_rwreg_modify(sc, TI_GPIO_LEVELDETECT1, tgi->tgi_mask,
8942202c379SSvatopluk Kraus mode == GPIO_INTR_LEVEL_HIGH);
8952202c379SSvatopluk Kraus ti_gpio_rwreg_modify(sc, TI_GPIO_LEVELDETECT0, tgi->tgi_mask,
8962202c379SSvatopluk Kraus mode == GPIO_INTR_LEVEL_LOW);
8972202c379SSvatopluk Kraus tgi->tgi_mode = mode;
8982202c379SSvatopluk Kraus TI_GPIO_UNLOCK(sc);
8992202c379SSvatopluk Kraus }
9002202c379SSvatopluk Kraus
9012202c379SSvatopluk Kraus static void
ti_gpio_pic_disable_intr(device_t dev,struct intr_irqsrc * isrc)9022df5562dSSvatopluk Kraus ti_gpio_pic_disable_intr(device_t dev, struct intr_irqsrc *isrc)
9032df5562dSSvatopluk Kraus {
9042df5562dSSvatopluk Kraus struct ti_gpio_softc *sc = device_get_softc(dev);
9052df5562dSSvatopluk Kraus struct ti_gpio_irqsrc *tgi = (struct ti_gpio_irqsrc *)isrc;
9062df5562dSSvatopluk Kraus
9072df5562dSSvatopluk Kraus ti_gpio_isrc_mask(sc, tgi);
9082df5562dSSvatopluk Kraus }
9092df5562dSSvatopluk Kraus
9102df5562dSSvatopluk Kraus static void
ti_gpio_pic_enable_intr(device_t dev,struct intr_irqsrc * isrc)9112df5562dSSvatopluk Kraus ti_gpio_pic_enable_intr(device_t dev, struct intr_irqsrc *isrc)
9122df5562dSSvatopluk Kraus {
9132df5562dSSvatopluk Kraus struct ti_gpio_softc *sc = device_get_softc(dev);
9142df5562dSSvatopluk Kraus struct ti_gpio_irqsrc *tgi = (struct ti_gpio_irqsrc *)isrc;
9152df5562dSSvatopluk Kraus
9162df5562dSSvatopluk Kraus arm_irq_memory_barrier(tgi->tgi_irq);
9172df5562dSSvatopluk Kraus ti_gpio_isrc_unmask(sc, tgi);
9182df5562dSSvatopluk Kraus }
9192df5562dSSvatopluk Kraus
9202df5562dSSvatopluk Kraus static int
ti_gpio_pic_map_fdt(struct ti_gpio_softc * sc,struct intr_map_data_fdt * daf,u_int * irqp,uint32_t * modep)9211a251c53SSvatopluk Kraus ti_gpio_pic_map_fdt(struct ti_gpio_softc *sc, struct intr_map_data_fdt *daf,
9222202c379SSvatopluk Kraus u_int *irqp, uint32_t *modep)
9232df5562dSSvatopluk Kraus {
9242202c379SSvatopluk Kraus uint32_t mode;
9252df5562dSSvatopluk Kraus
9262df5562dSSvatopluk Kraus /*
9272df5562dSSvatopluk Kraus * The first cell is the interrupt number.
9282df5562dSSvatopluk Kraus * The second cell is used to specify flags:
9292df5562dSSvatopluk Kraus * bits[3:0] trigger type and level flags:
9302df5562dSSvatopluk Kraus * 1 = low-to-high edge triggered.
9312df5562dSSvatopluk Kraus * 2 = high-to-low edge triggered.
9322df5562dSSvatopluk Kraus * 4 = active high level-sensitive.
9332df5562dSSvatopluk Kraus * 8 = active low level-sensitive.
9342df5562dSSvatopluk Kraus */
9351a251c53SSvatopluk Kraus if (daf->ncells != 2 || daf->cells[0] >= sc->sc_maxpin)
9362df5562dSSvatopluk Kraus return (EINVAL);
9372df5562dSSvatopluk Kraus
9382202c379SSvatopluk Kraus /* Only reasonable modes are supported. */
9391a251c53SSvatopluk Kraus if (daf->cells[1] == 1)
9402202c379SSvatopluk Kraus mode = GPIO_INTR_EDGE_RISING;
9411a251c53SSvatopluk Kraus else if (daf->cells[1] == 2)
9422202c379SSvatopluk Kraus mode = GPIO_INTR_EDGE_FALLING;
9431a251c53SSvatopluk Kraus else if (daf->cells[1] == 3)
9442202c379SSvatopluk Kraus mode = GPIO_INTR_EDGE_BOTH;
9451a251c53SSvatopluk Kraus else if (daf->cells[1] == 4)
9462202c379SSvatopluk Kraus mode = GPIO_INTR_LEVEL_HIGH;
9471a251c53SSvatopluk Kraus else if (daf->cells[1] == 8)
9482202c379SSvatopluk Kraus mode = GPIO_INTR_LEVEL_LOW;
9492df5562dSSvatopluk Kraus else
9502df5562dSSvatopluk Kraus return (EINVAL);
9512df5562dSSvatopluk Kraus
9521a251c53SSvatopluk Kraus *irqp = daf->cells[0];
9532202c379SSvatopluk Kraus if (modep != NULL)
9542202c379SSvatopluk Kraus *modep = mode;
9552df5562dSSvatopluk Kraus return (0);
9562df5562dSSvatopluk Kraus }
9572df5562dSSvatopluk Kraus
9582df5562dSSvatopluk Kraus static int
ti_gpio_pic_map_gpio(struct ti_gpio_softc * sc,struct intr_map_data_gpio * dag,u_int * irqp,uint32_t * modep)9591a251c53SSvatopluk Kraus ti_gpio_pic_map_gpio(struct ti_gpio_softc *sc, struct intr_map_data_gpio *dag,
9601a251c53SSvatopluk Kraus u_int *irqp, uint32_t *modep)
9611a251c53SSvatopluk Kraus {
9621a251c53SSvatopluk Kraus uint32_t mode;
9631a251c53SSvatopluk Kraus
9641a251c53SSvatopluk Kraus if (dag->gpio_pin_num >= sc->sc_maxpin)
9651a251c53SSvatopluk Kraus return (EINVAL);
9661a251c53SSvatopluk Kraus
9671a251c53SSvatopluk Kraus mode = dag->gpio_intr_mode;
9681a251c53SSvatopluk Kraus if (mode != GPIO_INTR_LEVEL_LOW && mode != GPIO_INTR_LEVEL_HIGH &&
9691a251c53SSvatopluk Kraus mode != GPIO_INTR_EDGE_RISING && mode != GPIO_INTR_EDGE_FALLING &&
9701a251c53SSvatopluk Kraus mode != GPIO_INTR_EDGE_BOTH)
9711a251c53SSvatopluk Kraus return (EINVAL);
9721a251c53SSvatopluk Kraus
9731a251c53SSvatopluk Kraus *irqp = dag->gpio_pin_num;
9741a251c53SSvatopluk Kraus if (modep != NULL)
9751a251c53SSvatopluk Kraus *modep = mode;
9761a251c53SSvatopluk Kraus return (0);
9771a251c53SSvatopluk Kraus }
9781a251c53SSvatopluk Kraus
9791a251c53SSvatopluk Kraus static int
ti_gpio_pic_map(struct ti_gpio_softc * sc,struct intr_map_data * data,u_int * irqp,uint32_t * modep)9801a251c53SSvatopluk Kraus ti_gpio_pic_map(struct ti_gpio_softc *sc, struct intr_map_data *data,
9811a251c53SSvatopluk Kraus u_int *irqp, uint32_t *modep)
9821a251c53SSvatopluk Kraus {
9831a251c53SSvatopluk Kraus
9841a251c53SSvatopluk Kraus switch (data->type) {
9851a251c53SSvatopluk Kraus case INTR_MAP_DATA_FDT:
9861a251c53SSvatopluk Kraus return (ti_gpio_pic_map_fdt(sc,
9871a251c53SSvatopluk Kraus (struct intr_map_data_fdt *)data, irqp, modep));
9881a251c53SSvatopluk Kraus case INTR_MAP_DATA_GPIO:
9891a251c53SSvatopluk Kraus return (ti_gpio_pic_map_gpio(sc,
9901a251c53SSvatopluk Kraus (struct intr_map_data_gpio *)data, irqp, modep));
9911a251c53SSvatopluk Kraus default:
9921a251c53SSvatopluk Kraus return (ENOTSUP);
9931a251c53SSvatopluk Kraus }
9941a251c53SSvatopluk Kraus }
9951a251c53SSvatopluk Kraus
9961a251c53SSvatopluk Kraus static int
ti_gpio_pic_map_intr(device_t dev,struct intr_map_data * data,struct intr_irqsrc ** isrcp)9972df5562dSSvatopluk Kraus ti_gpio_pic_map_intr(device_t dev, struct intr_map_data *data,
9982df5562dSSvatopluk Kraus struct intr_irqsrc **isrcp)
9992df5562dSSvatopluk Kraus {
10002df5562dSSvatopluk Kraus int error;
10012df5562dSSvatopluk Kraus u_int irq;
10021a251c53SSvatopluk Kraus struct ti_gpio_softc *sc = device_get_softc(dev);
10032df5562dSSvatopluk Kraus
10041a251c53SSvatopluk Kraus error = ti_gpio_pic_map(sc, data, &irq, NULL);
10052df5562dSSvatopluk Kraus if (error == 0)
10062df5562dSSvatopluk Kraus *isrcp = &sc->sc_isrcs[irq].tgi_isrc;
10072df5562dSSvatopluk Kraus return (error);
10082df5562dSSvatopluk Kraus }
10092df5562dSSvatopluk Kraus
10102df5562dSSvatopluk Kraus static void
ti_gpio_pic_post_filter(device_t dev,struct intr_irqsrc * isrc)10112df5562dSSvatopluk Kraus ti_gpio_pic_post_filter(device_t dev, struct intr_irqsrc *isrc)
10122df5562dSSvatopluk Kraus {
10132df5562dSSvatopluk Kraus struct ti_gpio_softc *sc = device_get_softc(dev);
10142df5562dSSvatopluk Kraus struct ti_gpio_irqsrc *tgi = (struct ti_gpio_irqsrc *)isrc;
10152df5562dSSvatopluk Kraus
10162df5562dSSvatopluk Kraus if (ti_gpio_isrc_is_level(tgi))
10172df5562dSSvatopluk Kraus ti_gpio_isrc_eoi(sc, tgi);
10182df5562dSSvatopluk Kraus }
10192df5562dSSvatopluk Kraus
10202df5562dSSvatopluk Kraus static void
ti_gpio_pic_post_ithread(device_t dev,struct intr_irqsrc * isrc)10212df5562dSSvatopluk Kraus ti_gpio_pic_post_ithread(device_t dev, struct intr_irqsrc *isrc)
10222df5562dSSvatopluk Kraus {
10232df5562dSSvatopluk Kraus
10242df5562dSSvatopluk Kraus ti_gpio_pic_enable_intr(dev, isrc);
10252df5562dSSvatopluk Kraus }
10262df5562dSSvatopluk Kraus
10272df5562dSSvatopluk Kraus static void
ti_gpio_pic_pre_ithread(device_t dev,struct intr_irqsrc * isrc)10282df5562dSSvatopluk Kraus ti_gpio_pic_pre_ithread(device_t dev, struct intr_irqsrc *isrc)
10292df5562dSSvatopluk Kraus {
10302df5562dSSvatopluk Kraus struct ti_gpio_softc *sc = device_get_softc(dev);
10312df5562dSSvatopluk Kraus struct ti_gpio_irqsrc *tgi = (struct ti_gpio_irqsrc *)isrc;
10322df5562dSSvatopluk Kraus
10332df5562dSSvatopluk Kraus ti_gpio_isrc_mask(sc, tgi);
10342df5562dSSvatopluk Kraus if (ti_gpio_isrc_is_level(tgi))
10352df5562dSSvatopluk Kraus ti_gpio_isrc_eoi(sc, tgi);
10362df5562dSSvatopluk Kraus }
10372df5562dSSvatopluk Kraus
10382df5562dSSvatopluk Kraus static int
ti_gpio_pic_setup_intr(device_t dev,struct intr_irqsrc * isrc,struct resource * res,struct intr_map_data * data)10392df5562dSSvatopluk Kraus ti_gpio_pic_setup_intr(device_t dev, struct intr_irqsrc *isrc,
10402df5562dSSvatopluk Kraus struct resource *res, struct intr_map_data *data)
10412df5562dSSvatopluk Kraus {
10422df5562dSSvatopluk Kraus u_int irq;
10432202c379SSvatopluk Kraus uint32_t mode;
10442df5562dSSvatopluk Kraus struct ti_gpio_softc *sc;
10452df5562dSSvatopluk Kraus struct ti_gpio_irqsrc *tgi;
10462df5562dSSvatopluk Kraus
10471a251c53SSvatopluk Kraus if (data == NULL)
10482df5562dSSvatopluk Kraus return (ENOTSUP);
10492df5562dSSvatopluk Kraus
10502df5562dSSvatopluk Kraus sc = device_get_softc(dev);
10512df5562dSSvatopluk Kraus tgi = (struct ti_gpio_irqsrc *)isrc;
10522df5562dSSvatopluk Kraus
10532df5562dSSvatopluk Kraus /* Get and check config for an interrupt. */
10541a251c53SSvatopluk Kraus if (ti_gpio_pic_map(sc, data, &irq, &mode) != 0 || tgi->tgi_irq != irq)
10552df5562dSSvatopluk Kraus return (EINVAL);
10562df5562dSSvatopluk Kraus
10572df5562dSSvatopluk Kraus /*
10582df5562dSSvatopluk Kraus * If this is a setup for another handler,
10592df5562dSSvatopluk Kraus * only check that its configuration match.
10602df5562dSSvatopluk Kraus */
10612df5562dSSvatopluk Kraus if (isrc->isrc_handlers != 0)
10622202c379SSvatopluk Kraus return (tgi->tgi_mode == mode ? 0 : EINVAL);
10632df5562dSSvatopluk Kraus
10642202c379SSvatopluk Kraus ti_gpio_pic_config_intr(sc, tgi, mode);
10652df5562dSSvatopluk Kraus return (0);
10662df5562dSSvatopluk Kraus }
10672df5562dSSvatopluk Kraus
10682df5562dSSvatopluk Kraus static int
ti_gpio_pic_teardown_intr(device_t dev,struct intr_irqsrc * isrc,struct resource * res,struct intr_map_data * data)10692df5562dSSvatopluk Kraus ti_gpio_pic_teardown_intr(device_t dev, struct intr_irqsrc *isrc,
10702df5562dSSvatopluk Kraus struct resource *res, struct intr_map_data *data)
10712df5562dSSvatopluk Kraus {
10722df5562dSSvatopluk Kraus struct ti_gpio_softc *sc = device_get_softc(dev);
10732df5562dSSvatopluk Kraus struct ti_gpio_irqsrc *tgi = (struct ti_gpio_irqsrc *)isrc;
10742df5562dSSvatopluk Kraus
10752202c379SSvatopluk Kraus if (isrc->isrc_handlers == 0)
10762202c379SSvatopluk Kraus ti_gpio_pic_config_intr(sc, tgi, GPIO_INTR_CONFORM);
10772df5562dSSvatopluk Kraus return (0);
10782df5562dSSvatopluk Kraus }
10792df5562dSSvatopluk Kraus
10808c705c2cSLuiz Otavio O Souza static phandle_t
ti_gpio_get_node(device_t bus,device_t dev)10818c705c2cSLuiz Otavio O Souza ti_gpio_get_node(device_t bus, device_t dev)
10828c705c2cSLuiz Otavio O Souza {
10838c705c2cSLuiz Otavio O Souza
10848c705c2cSLuiz Otavio O Souza /* We only have one child, the GPIO bus, which needs our own node. */
10858c705c2cSLuiz Otavio O Souza return (ofw_bus_get_node(bus));
10868c705c2cSLuiz Otavio O Souza }
10878c705c2cSLuiz Otavio O Souza
1088e53470feSOleksandr Tymoshenko static device_method_t ti_gpio_methods[] = {
1089e53470feSOleksandr Tymoshenko DEVMETHOD(device_attach, ti_gpio_attach),
1090e53470feSOleksandr Tymoshenko DEVMETHOD(device_detach, ti_gpio_detach),
1091e53470feSOleksandr Tymoshenko
1092e53470feSOleksandr Tymoshenko /* GPIO protocol */
10937836352bSLuiz Otavio O Souza DEVMETHOD(gpio_get_bus, ti_gpio_get_bus),
1094e53470feSOleksandr Tymoshenko DEVMETHOD(gpio_pin_max, ti_gpio_pin_max),
1095e53470feSOleksandr Tymoshenko DEVMETHOD(gpio_pin_getname, ti_gpio_pin_getname),
1096e53470feSOleksandr Tymoshenko DEVMETHOD(gpio_pin_getflags, ti_gpio_pin_getflags),
1097e53470feSOleksandr Tymoshenko DEVMETHOD(gpio_pin_getcaps, ti_gpio_pin_getcaps),
1098e53470feSOleksandr Tymoshenko DEVMETHOD(gpio_pin_setflags, ti_gpio_pin_setflags),
1099e53470feSOleksandr Tymoshenko DEVMETHOD(gpio_pin_get, ti_gpio_pin_get),
1100e53470feSOleksandr Tymoshenko DEVMETHOD(gpio_pin_set, ti_gpio_pin_set),
1101e53470feSOleksandr Tymoshenko DEVMETHOD(gpio_pin_toggle, ti_gpio_pin_toggle),
11028c705c2cSLuiz Otavio O Souza
11032df5562dSSvatopluk Kraus /* Interrupt controller interface */
11042df5562dSSvatopluk Kraus DEVMETHOD(pic_disable_intr, ti_gpio_pic_disable_intr),
11052df5562dSSvatopluk Kraus DEVMETHOD(pic_enable_intr, ti_gpio_pic_enable_intr),
11062df5562dSSvatopluk Kraus DEVMETHOD(pic_map_intr, ti_gpio_pic_map_intr),
11072df5562dSSvatopluk Kraus DEVMETHOD(pic_setup_intr, ti_gpio_pic_setup_intr),
11082df5562dSSvatopluk Kraus DEVMETHOD(pic_teardown_intr, ti_gpio_pic_teardown_intr),
11092df5562dSSvatopluk Kraus DEVMETHOD(pic_post_filter, ti_gpio_pic_post_filter),
11102df5562dSSvatopluk Kraus DEVMETHOD(pic_post_ithread, ti_gpio_pic_post_ithread),
11112df5562dSSvatopluk Kraus DEVMETHOD(pic_pre_ithread, ti_gpio_pic_pre_ithread),
11123681c8d7SLuiz Otavio O Souza
11138c705c2cSLuiz Otavio O Souza /* ofw_bus interface */
11148c705c2cSLuiz Otavio O Souza DEVMETHOD(ofw_bus_get_node, ti_gpio_get_node),
11158c705c2cSLuiz Otavio O Souza
1116e53470feSOleksandr Tymoshenko {0, 0},
1117e53470feSOleksandr Tymoshenko };
1118e53470feSOleksandr Tymoshenko
1119b6c7dacfSAndrew Turner driver_t ti_gpio_driver = {
1120e53470feSOleksandr Tymoshenko "gpio",
1121e53470feSOleksandr Tymoshenko ti_gpio_methods,
1122e53470feSOleksandr Tymoshenko sizeof(struct ti_gpio_softc),
1123e53470feSOleksandr Tymoshenko };
1124