xref: /freebsd/sys/arm/ti/ti_gpio.c (revision 685dc743dc3b5645e34836464128e1c0558b404b)
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