2008-07-05 10:02:49 +02:00
/*
* MXC GPIO support . ( c ) 2008 Daniel Mack < daniel @ caiaq . de >
* Copyright 2008 Juergen Beisert , kernel @ pengutronix . de
*
* Based on code from Freescale ,
2010-04-22 16:28:42 +03:00
* Copyright ( C ) 2004 - 2010 Freescale Semiconductor , Inc . All Rights Reserved .
2008-07-05 10:02:49 +02:00
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version 2
* of the License , or ( at your option ) any later version .
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 , USA .
*/
2013-07-22 18:17:52 -03:00
# include <linux/err.h>
2008-07-05 10:02:49 +02:00
# include <linux/init.h>
2010-10-23 09:12:48 -05:00
# include <linux/interrupt.h>
2008-07-05 10:02:49 +02:00
# include <linux/io.h>
# include <linux/irq.h>
2012-06-13 09:04:03 +08:00
# include <linux/irqdomain.h>
2013-01-18 15:31:37 +00:00
# include <linux/irqchip/chained_irq.h>
2008-07-05 10:02:49 +02:00
# include <linux/gpio.h>
2011-06-06 00:07:55 +08:00
# include <linux/platform_device.h>
# include <linux/slab.h>
2011-06-06 13:22:41 +08:00
# include <linux/basic_mmio_gpio.h>
2011-07-07 00:37:43 +08:00
# include <linux/of.h>
# include <linux/of_device.h>
2011-07-03 13:38:09 -04:00
# include <linux/module.h>
2008-07-05 10:02:49 +02:00
# include <asm-generic/bug.h>
2011-07-07 00:37:41 +08:00
enum mxc_gpio_hwtype {
IMX1_GPIO , /* runs on i.mx1 */
IMX21_GPIO , /* runs on i.mx21 and i.mx27 */
2012-06-22 21:04:06 +02:00
IMX31_GPIO , /* runs on i.mx31 */
IMX35_GPIO , /* runs on all other i.mx */
2011-07-07 00:37:41 +08:00
} ;
/* device type dependent stuff */
struct mxc_gpio_hwdata {
unsigned dr_reg ;
unsigned gdir_reg ;
unsigned psr_reg ;
unsigned icr1_reg ;
unsigned icr2_reg ;
unsigned imr_reg ;
unsigned isr_reg ;
2012-06-22 21:04:06 +02:00
int edge_sel_reg ;
2011-07-07 00:37:41 +08:00
unsigned low_level ;
unsigned high_level ;
unsigned rise_edge ;
unsigned fall_edge ;
} ;
2011-06-06 00:07:55 +08:00
struct mxc_gpio_port {
struct list_head node ;
void __iomem * base ;
int irq ;
int irq_high ;
2012-06-13 09:04:03 +08:00
struct irq_domain * domain ;
2011-06-06 13:22:41 +08:00
struct bgpio_chip bgc ;
2011-06-06 00:07:55 +08:00
u32 both_edges ;
} ;
2011-07-07 00:37:41 +08:00
static struct mxc_gpio_hwdata imx1_imx21_gpio_hwdata = {
. dr_reg = 0x1c ,
. gdir_reg = 0x00 ,
. psr_reg = 0x24 ,
. icr1_reg = 0x28 ,
. icr2_reg = 0x2c ,
. imr_reg = 0x30 ,
. isr_reg = 0x34 ,
2012-06-22 21:04:06 +02:00
. edge_sel_reg = - EINVAL ,
2011-07-07 00:37:41 +08:00
. low_level = 0x03 ,
. high_level = 0x02 ,
. rise_edge = 0x00 ,
. fall_edge = 0x01 ,
} ;
static struct mxc_gpio_hwdata imx31_gpio_hwdata = {
. dr_reg = 0x00 ,
. gdir_reg = 0x04 ,
. psr_reg = 0x08 ,
. icr1_reg = 0x0c ,
. icr2_reg = 0x10 ,
. imr_reg = 0x14 ,
. isr_reg = 0x18 ,
2012-06-22 21:04:06 +02:00
. edge_sel_reg = - EINVAL ,
. low_level = 0x00 ,
. high_level = 0x01 ,
. rise_edge = 0x02 ,
. fall_edge = 0x03 ,
} ;
static struct mxc_gpio_hwdata imx35_gpio_hwdata = {
. dr_reg = 0x00 ,
. gdir_reg = 0x04 ,
. psr_reg = 0x08 ,
. icr1_reg = 0x0c ,
. icr2_reg = 0x10 ,
. imr_reg = 0x14 ,
. isr_reg = 0x18 ,
. edge_sel_reg = 0x1c ,
2011-07-07 00:37:41 +08:00
. low_level = 0x00 ,
. high_level = 0x01 ,
. rise_edge = 0x02 ,
. fall_edge = 0x03 ,
} ;
static enum mxc_gpio_hwtype mxc_gpio_hwtype ;
static struct mxc_gpio_hwdata * mxc_gpio_hwdata ;
# define GPIO_DR (mxc_gpio_hwdata->dr_reg)
# define GPIO_GDIR (mxc_gpio_hwdata->gdir_reg)
# define GPIO_PSR (mxc_gpio_hwdata->psr_reg)
# define GPIO_ICR1 (mxc_gpio_hwdata->icr1_reg)
# define GPIO_ICR2 (mxc_gpio_hwdata->icr2_reg)
# define GPIO_IMR (mxc_gpio_hwdata->imr_reg)
# define GPIO_ISR (mxc_gpio_hwdata->isr_reg)
2012-06-22 21:04:06 +02:00
# define GPIO_EDGE_SEL (mxc_gpio_hwdata->edge_sel_reg)
2011-07-07 00:37:41 +08:00
# define GPIO_INT_LOW_LEV (mxc_gpio_hwdata->low_level)
# define GPIO_INT_HIGH_LEV (mxc_gpio_hwdata->high_level)
# define GPIO_INT_RISE_EDGE (mxc_gpio_hwdata->rise_edge)
# define GPIO_INT_FALL_EDGE (mxc_gpio_hwdata->fall_edge)
2012-06-22 21:04:06 +02:00
# define GPIO_INT_BOTH_EDGES 0x4
2011-07-07 00:37:41 +08:00
static struct platform_device_id mxc_gpio_devtype [ ] = {
{
. name = " imx1-gpio " ,
. driver_data = IMX1_GPIO ,
} , {
. name = " imx21-gpio " ,
. driver_data = IMX21_GPIO ,
} , {
. name = " imx31-gpio " ,
. driver_data = IMX31_GPIO ,
2012-06-22 21:04:06 +02:00
} , {
. name = " imx35-gpio " ,
. driver_data = IMX35_GPIO ,
2011-07-07 00:37:41 +08:00
} , {
/* sentinel */
}
} ;
2011-07-07 00:37:43 +08:00
static const struct of_device_id mxc_gpio_dt_ids [ ] = {
{ . compatible = " fsl,imx1-gpio " , . data = & mxc_gpio_devtype [ IMX1_GPIO ] , } ,
{ . compatible = " fsl,imx21-gpio " , . data = & mxc_gpio_devtype [ IMX21_GPIO ] , } ,
{ . compatible = " fsl,imx31-gpio " , . data = & mxc_gpio_devtype [ IMX31_GPIO ] , } ,
2012-06-22 21:04:06 +02:00
{ . compatible = " fsl,imx35-gpio " , . data = & mxc_gpio_devtype [ IMX35_GPIO ] , } ,
2011-07-07 00:37:43 +08:00
{ /* sentinel */ }
} ;
2011-06-06 00:07:55 +08:00
/*
* MX2 has one interrupt * for all * gpio ports . The list is used
* to save the references to all ports , so that mx2_gpio_irq_handler
* can walk through all interrupt status registers .
*/
static LIST_HEAD ( mxc_gpio_ports ) ;
2008-07-05 10:02:49 +02:00
/* Note: This driver assumes 32 GPIOs are handled in one register */
2010-11-29 11:16:23 +01:00
static int gpio_set_irq_type ( struct irq_data * d , u32 type )
2008-07-05 10:02:49 +02:00
{
2011-06-07 16:25:37 +08:00
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
struct mxc_gpio_port * port = gc - > private ;
2008-07-05 10:02:49 +02:00
u32 bit , val ;
2012-06-13 09:04:03 +08:00
u32 gpio_idx = d - > hwirq ;
u32 gpio = port - > bgc . gc . base + gpio_idx ;
2008-07-05 10:02:49 +02:00
int edge ;
void __iomem * reg = port - > base ;
2012-06-13 09:04:03 +08:00
port - > both_edges & = ~ ( 1 < < gpio_idx ) ;
2008-07-05 10:02:49 +02:00
switch ( type ) {
2008-07-27 04:23:31 +01:00
case IRQ_TYPE_EDGE_RISING :
2008-07-05 10:02:49 +02:00
edge = GPIO_INT_RISE_EDGE ;
break ;
2008-07-27 04:23:31 +01:00
case IRQ_TYPE_EDGE_FALLING :
2008-07-05 10:02:49 +02:00
edge = GPIO_INT_FALL_EDGE ;
break ;
2009-03-12 12:46:41 +01:00
case IRQ_TYPE_EDGE_BOTH :
2012-06-22 21:04:06 +02:00
if ( GPIO_EDGE_SEL > = 0 ) {
edge = GPIO_INT_BOTH_EDGES ;
2009-03-12 12:46:41 +01:00
} else {
2012-06-22 21:04:06 +02:00
val = gpio_get_value ( gpio ) ;
if ( val ) {
edge = GPIO_INT_LOW_LEV ;
pr_debug ( " mxc: set GPIO %d to low trigger \n " , gpio ) ;
} else {
edge = GPIO_INT_HIGH_LEV ;
pr_debug ( " mxc: set GPIO %d to high trigger \n " , gpio ) ;
}
GPIO changes for v3.6:
- New driver for AMD-8111 southbridge GPIOs
- New driver for Wolfson Micro Arizona devices
- Propagate device tree parse errors
- Probe deferral finalizations - all expected calls to
GPIO will now hopefully request deferral where apropriate
- Misc updates to TCA6424, WM8994, LPC32xx, PCF857x, Samsung
MXC, OMAP and PCA953X drivers.
-----BEGIN PGP SIGNATURE-----
Version: GnuPG v1.4.12 (GNU/Linux)
iQIcBAABAgAGBQJQEHMDAAoJEEEQszewGV1zwPYP/jkxBzvEl+iEO0RFwT4PtmCi
Y8JOJNT1bw/3MHPcRT12E+gzj01S9GldbuaUObcudmnnynpjeC0S8JNhSKGD9uHa
TTcCcMbZiKzJyZr/OL8EId7W1FGUO+51uB4hqEKCHMWRY/PBIjKxhvtj+BKEWyvn
OVhWCxo2O7lv7rzeKiPc8WJMiodLS1urbZEyz7IADZtT3m8vu146rEQRvbNSSXa0
AJfl494XX1sbv0tzYzvE66+vjvvkgsjHq3O7On5b2svdnZGpAL/6CjEUVrpBXr4K
NPKuq9TsLfVMH3w3xvQ70PoA7M0L+KvKcdjTvgZpf2KLIU7dwoL91PzAupcjSTr1
SkcTPtNFxuaRy0cFD+ZAwL2eIOGaNxk6N4tj1da35QjCUkNROHG5K6ByIL1e1ewO
NuxAyn7QLrYdmXzBc5/DhZiBA0ShqoYg4oEgBDZklOKqjT3mqmjQbDq8i0Qy197W
lb3Barg+WWm+NW1kmPYnrOJUZXa1ApVHuz8db7OrcUy5kTcUhVTY3DcQzFgG1CZT
H284c9Zm8WaP814jE8SzLMGeFaCuI63xFMNkpnba11Bt+8Cr1I+LjWSd+ttCFdVm
W9t/fMEX1bVpVrbTKMvcwm7AwnoeOUEwJCqAICLe5OE/1mEvanyjNXX4wfFP58jv
OlQ/a9REqoOLAvvbV2qp
=tujx
-----END PGP SIGNATURE-----
Merge tag 'gpio-for-v3.6' of git://git.kernel.org/pub/scm/linux/kernel/git/linusw/linux-gpio
Pull GPIO changes from Linus Walleij:
- New driver for AMD-8111 southbridge GPIOs
- New driver for Wolfson Micro Arizona devices
- Propagate device tree parse errors
- Probe deferral finalizations - all expected calls to GPIO will now
hopefully request deferral where apropriate
- Misc updates to TCA6424, WM8994, LPC32xx, PCF857x, Samsung MXC, OMAP
and PCA953X drivers.
Fix up gpio_idx conflicts in drivers/gpio/gpio-mxc.c
* tag 'gpio-for-v3.6' of git://git.kernel.org/pub/scm/linux/kernel/git/linusw/linux-gpio:
gpio: of_get_named_gpio_flags() return -EPROBE_DEFER if GPIO not yet available
gpiolib: Defer failed gpio requests by default
MAINTAINERS: add entry OMAP GPIO driver
gpio/pca953x: increase variables size to support 24 bit of data
GPIO: PCA953X: Increase size of invert variable to support 24 bit
gpio/omap: move bank->dbck initialization to omap_gpio_mod_init()
gpio/mxc: use the edge_sel feature if available
gpio: propagate of_parse_phandle_with_args errors
gpio: samsung: add flags specifier to device-tree binding
gpiolib: Add support for Wolfson Microelectronics Arizona class devices
gpio: gpio-lpc32xx: Add gpio_to_irq mapping
gpio: pcf857x: share 8/16 bit access functions
gpio: LPC32xx: Driver cleanup
MAINTAINERS: Add Wolfson gpiolib drivers to the Wolfson entry
gpiolib: wm8994: Convert to devm_kzalloc()
gpiolib: wm8994: Use irq_domain mappings for gpios
gpio: add a driver for GPIO pins found on AMD-8111 south bridge chips
gpio/tca6424: merge I2C transactions, remove cast
gpio/of: fix a typo of comment message
2012-07-26 13:56:38 -07:00
port - > both_edges | = 1 < < gpio_idx ;
2009-03-12 12:46:41 +01:00
}
break ;
2008-07-27 04:23:31 +01:00
case IRQ_TYPE_LEVEL_LOW :
2008-07-05 10:02:49 +02:00
edge = GPIO_INT_LOW_LEV ;
break ;
2008-07-27 04:23:31 +01:00
case IRQ_TYPE_LEVEL_HIGH :
2008-07-05 10:02:49 +02:00
edge = GPIO_INT_HIGH_LEV ;
break ;
2009-03-12 12:46:41 +01:00
default :
2008-07-05 10:02:49 +02:00
return - EINVAL ;
}
2012-06-22 21:04:06 +02:00
if ( GPIO_EDGE_SEL > = 0 ) {
val = readl ( port - > base + GPIO_EDGE_SEL ) ;
if ( edge = = GPIO_INT_BOTH_EDGES )
GPIO changes for v3.6:
- New driver for AMD-8111 southbridge GPIOs
- New driver for Wolfson Micro Arizona devices
- Propagate device tree parse errors
- Probe deferral finalizations - all expected calls to
GPIO will now hopefully request deferral where apropriate
- Misc updates to TCA6424, WM8994, LPC32xx, PCF857x, Samsung
MXC, OMAP and PCA953X drivers.
-----BEGIN PGP SIGNATURE-----
Version: GnuPG v1.4.12 (GNU/Linux)
iQIcBAABAgAGBQJQEHMDAAoJEEEQszewGV1zwPYP/jkxBzvEl+iEO0RFwT4PtmCi
Y8JOJNT1bw/3MHPcRT12E+gzj01S9GldbuaUObcudmnnynpjeC0S8JNhSKGD9uHa
TTcCcMbZiKzJyZr/OL8EId7W1FGUO+51uB4hqEKCHMWRY/PBIjKxhvtj+BKEWyvn
OVhWCxo2O7lv7rzeKiPc8WJMiodLS1urbZEyz7IADZtT3m8vu146rEQRvbNSSXa0
AJfl494XX1sbv0tzYzvE66+vjvvkgsjHq3O7On5b2svdnZGpAL/6CjEUVrpBXr4K
NPKuq9TsLfVMH3w3xvQ70PoA7M0L+KvKcdjTvgZpf2KLIU7dwoL91PzAupcjSTr1
SkcTPtNFxuaRy0cFD+ZAwL2eIOGaNxk6N4tj1da35QjCUkNROHG5K6ByIL1e1ewO
NuxAyn7QLrYdmXzBc5/DhZiBA0ShqoYg4oEgBDZklOKqjT3mqmjQbDq8i0Qy197W
lb3Barg+WWm+NW1kmPYnrOJUZXa1ApVHuz8db7OrcUy5kTcUhVTY3DcQzFgG1CZT
H284c9Zm8WaP814jE8SzLMGeFaCuI63xFMNkpnba11Bt+8Cr1I+LjWSd+ttCFdVm
W9t/fMEX1bVpVrbTKMvcwm7AwnoeOUEwJCqAICLe5OE/1mEvanyjNXX4wfFP58jv
OlQ/a9REqoOLAvvbV2qp
=tujx
-----END PGP SIGNATURE-----
Merge tag 'gpio-for-v3.6' of git://git.kernel.org/pub/scm/linux/kernel/git/linusw/linux-gpio
Pull GPIO changes from Linus Walleij:
- New driver for AMD-8111 southbridge GPIOs
- New driver for Wolfson Micro Arizona devices
- Propagate device tree parse errors
- Probe deferral finalizations - all expected calls to GPIO will now
hopefully request deferral where apropriate
- Misc updates to TCA6424, WM8994, LPC32xx, PCF857x, Samsung MXC, OMAP
and PCA953X drivers.
Fix up gpio_idx conflicts in drivers/gpio/gpio-mxc.c
* tag 'gpio-for-v3.6' of git://git.kernel.org/pub/scm/linux/kernel/git/linusw/linux-gpio:
gpio: of_get_named_gpio_flags() return -EPROBE_DEFER if GPIO not yet available
gpiolib: Defer failed gpio requests by default
MAINTAINERS: add entry OMAP GPIO driver
gpio/pca953x: increase variables size to support 24 bit of data
GPIO: PCA953X: Increase size of invert variable to support 24 bit
gpio/omap: move bank->dbck initialization to omap_gpio_mod_init()
gpio/mxc: use the edge_sel feature if available
gpio: propagate of_parse_phandle_with_args errors
gpio: samsung: add flags specifier to device-tree binding
gpiolib: Add support for Wolfson Microelectronics Arizona class devices
gpio: gpio-lpc32xx: Add gpio_to_irq mapping
gpio: pcf857x: share 8/16 bit access functions
gpio: LPC32xx: Driver cleanup
MAINTAINERS: Add Wolfson gpiolib drivers to the Wolfson entry
gpiolib: wm8994: Convert to devm_kzalloc()
gpiolib: wm8994: Use irq_domain mappings for gpios
gpio: add a driver for GPIO pins found on AMD-8111 south bridge chips
gpio/tca6424: merge I2C transactions, remove cast
gpio/of: fix a typo of comment message
2012-07-26 13:56:38 -07:00
writel ( val | ( 1 < < gpio_idx ) ,
2012-06-22 21:04:06 +02:00
port - > base + GPIO_EDGE_SEL ) ;
else
GPIO changes for v3.6:
- New driver for AMD-8111 southbridge GPIOs
- New driver for Wolfson Micro Arizona devices
- Propagate device tree parse errors
- Probe deferral finalizations - all expected calls to
GPIO will now hopefully request deferral where apropriate
- Misc updates to TCA6424, WM8994, LPC32xx, PCF857x, Samsung
MXC, OMAP and PCA953X drivers.
-----BEGIN PGP SIGNATURE-----
Version: GnuPG v1.4.12 (GNU/Linux)
iQIcBAABAgAGBQJQEHMDAAoJEEEQszewGV1zwPYP/jkxBzvEl+iEO0RFwT4PtmCi
Y8JOJNT1bw/3MHPcRT12E+gzj01S9GldbuaUObcudmnnynpjeC0S8JNhSKGD9uHa
TTcCcMbZiKzJyZr/OL8EId7W1FGUO+51uB4hqEKCHMWRY/PBIjKxhvtj+BKEWyvn
OVhWCxo2O7lv7rzeKiPc8WJMiodLS1urbZEyz7IADZtT3m8vu146rEQRvbNSSXa0
AJfl494XX1sbv0tzYzvE66+vjvvkgsjHq3O7On5b2svdnZGpAL/6CjEUVrpBXr4K
NPKuq9TsLfVMH3w3xvQ70PoA7M0L+KvKcdjTvgZpf2KLIU7dwoL91PzAupcjSTr1
SkcTPtNFxuaRy0cFD+ZAwL2eIOGaNxk6N4tj1da35QjCUkNROHG5K6ByIL1e1ewO
NuxAyn7QLrYdmXzBc5/DhZiBA0ShqoYg4oEgBDZklOKqjT3mqmjQbDq8i0Qy197W
lb3Barg+WWm+NW1kmPYnrOJUZXa1ApVHuz8db7OrcUy5kTcUhVTY3DcQzFgG1CZT
H284c9Zm8WaP814jE8SzLMGeFaCuI63xFMNkpnba11Bt+8Cr1I+LjWSd+ttCFdVm
W9t/fMEX1bVpVrbTKMvcwm7AwnoeOUEwJCqAICLe5OE/1mEvanyjNXX4wfFP58jv
OlQ/a9REqoOLAvvbV2qp
=tujx
-----END PGP SIGNATURE-----
Merge tag 'gpio-for-v3.6' of git://git.kernel.org/pub/scm/linux/kernel/git/linusw/linux-gpio
Pull GPIO changes from Linus Walleij:
- New driver for AMD-8111 southbridge GPIOs
- New driver for Wolfson Micro Arizona devices
- Propagate device tree parse errors
- Probe deferral finalizations - all expected calls to GPIO will now
hopefully request deferral where apropriate
- Misc updates to TCA6424, WM8994, LPC32xx, PCF857x, Samsung MXC, OMAP
and PCA953X drivers.
Fix up gpio_idx conflicts in drivers/gpio/gpio-mxc.c
* tag 'gpio-for-v3.6' of git://git.kernel.org/pub/scm/linux/kernel/git/linusw/linux-gpio:
gpio: of_get_named_gpio_flags() return -EPROBE_DEFER if GPIO not yet available
gpiolib: Defer failed gpio requests by default
MAINTAINERS: add entry OMAP GPIO driver
gpio/pca953x: increase variables size to support 24 bit of data
GPIO: PCA953X: Increase size of invert variable to support 24 bit
gpio/omap: move bank->dbck initialization to omap_gpio_mod_init()
gpio/mxc: use the edge_sel feature if available
gpio: propagate of_parse_phandle_with_args errors
gpio: samsung: add flags specifier to device-tree binding
gpiolib: Add support for Wolfson Microelectronics Arizona class devices
gpio: gpio-lpc32xx: Add gpio_to_irq mapping
gpio: pcf857x: share 8/16 bit access functions
gpio: LPC32xx: Driver cleanup
MAINTAINERS: Add Wolfson gpiolib drivers to the Wolfson entry
gpiolib: wm8994: Convert to devm_kzalloc()
gpiolib: wm8994: Use irq_domain mappings for gpios
gpio: add a driver for GPIO pins found on AMD-8111 south bridge chips
gpio/tca6424: merge I2C transactions, remove cast
gpio/of: fix a typo of comment message
2012-07-26 13:56:38 -07:00
writel ( val & ~ ( 1 < < gpio_idx ) ,
2012-06-22 21:04:06 +02:00
port - > base + GPIO_EDGE_SEL ) ;
}
if ( edge ! = GPIO_INT_BOTH_EDGES ) {
GPIO changes for v3.6:
- New driver for AMD-8111 southbridge GPIOs
- New driver for Wolfson Micro Arizona devices
- Propagate device tree parse errors
- Probe deferral finalizations - all expected calls to
GPIO will now hopefully request deferral where apropriate
- Misc updates to TCA6424, WM8994, LPC32xx, PCF857x, Samsung
MXC, OMAP and PCA953X drivers.
-----BEGIN PGP SIGNATURE-----
Version: GnuPG v1.4.12 (GNU/Linux)
iQIcBAABAgAGBQJQEHMDAAoJEEEQszewGV1zwPYP/jkxBzvEl+iEO0RFwT4PtmCi
Y8JOJNT1bw/3MHPcRT12E+gzj01S9GldbuaUObcudmnnynpjeC0S8JNhSKGD9uHa
TTcCcMbZiKzJyZr/OL8EId7W1FGUO+51uB4hqEKCHMWRY/PBIjKxhvtj+BKEWyvn
OVhWCxo2O7lv7rzeKiPc8WJMiodLS1urbZEyz7IADZtT3m8vu146rEQRvbNSSXa0
AJfl494XX1sbv0tzYzvE66+vjvvkgsjHq3O7On5b2svdnZGpAL/6CjEUVrpBXr4K
NPKuq9TsLfVMH3w3xvQ70PoA7M0L+KvKcdjTvgZpf2KLIU7dwoL91PzAupcjSTr1
SkcTPtNFxuaRy0cFD+ZAwL2eIOGaNxk6N4tj1da35QjCUkNROHG5K6ByIL1e1ewO
NuxAyn7QLrYdmXzBc5/DhZiBA0ShqoYg4oEgBDZklOKqjT3mqmjQbDq8i0Qy197W
lb3Barg+WWm+NW1kmPYnrOJUZXa1ApVHuz8db7OrcUy5kTcUhVTY3DcQzFgG1CZT
H284c9Zm8WaP814jE8SzLMGeFaCuI63xFMNkpnba11Bt+8Cr1I+LjWSd+ttCFdVm
W9t/fMEX1bVpVrbTKMvcwm7AwnoeOUEwJCqAICLe5OE/1mEvanyjNXX4wfFP58jv
OlQ/a9REqoOLAvvbV2qp
=tujx
-----END PGP SIGNATURE-----
Merge tag 'gpio-for-v3.6' of git://git.kernel.org/pub/scm/linux/kernel/git/linusw/linux-gpio
Pull GPIO changes from Linus Walleij:
- New driver for AMD-8111 southbridge GPIOs
- New driver for Wolfson Micro Arizona devices
- Propagate device tree parse errors
- Probe deferral finalizations - all expected calls to GPIO will now
hopefully request deferral where apropriate
- Misc updates to TCA6424, WM8994, LPC32xx, PCF857x, Samsung MXC, OMAP
and PCA953X drivers.
Fix up gpio_idx conflicts in drivers/gpio/gpio-mxc.c
* tag 'gpio-for-v3.6' of git://git.kernel.org/pub/scm/linux/kernel/git/linusw/linux-gpio:
gpio: of_get_named_gpio_flags() return -EPROBE_DEFER if GPIO not yet available
gpiolib: Defer failed gpio requests by default
MAINTAINERS: add entry OMAP GPIO driver
gpio/pca953x: increase variables size to support 24 bit of data
GPIO: PCA953X: Increase size of invert variable to support 24 bit
gpio/omap: move bank->dbck initialization to omap_gpio_mod_init()
gpio/mxc: use the edge_sel feature if available
gpio: propagate of_parse_phandle_with_args errors
gpio: samsung: add flags specifier to device-tree binding
gpiolib: Add support for Wolfson Microelectronics Arizona class devices
gpio: gpio-lpc32xx: Add gpio_to_irq mapping
gpio: pcf857x: share 8/16 bit access functions
gpio: LPC32xx: Driver cleanup
MAINTAINERS: Add Wolfson gpiolib drivers to the Wolfson entry
gpiolib: wm8994: Convert to devm_kzalloc()
gpiolib: wm8994: Use irq_domain mappings for gpios
gpio: add a driver for GPIO pins found on AMD-8111 south bridge chips
gpio/tca6424: merge I2C transactions, remove cast
gpio/of: fix a typo of comment message
2012-07-26 13:56:38 -07:00
reg + = GPIO_ICR1 + ( ( gpio_idx & 0x10 ) > > 2 ) ; /* lower or upper register */
bit = gpio_idx & 0xf ;
2012-06-22 21:04:06 +02:00
val = readl ( reg ) & ~ ( 0x3 < < ( bit < < 1 ) ) ;
writel ( val | ( edge < < ( bit < < 1 ) ) , reg ) ;
}
2012-06-13 09:04:03 +08:00
writel ( 1 < < gpio_idx , port - > base + GPIO_ISR ) ;
2008-07-05 10:02:49 +02:00
return 0 ;
}
2009-03-12 12:46:41 +01:00
static void mxc_flip_edge ( struct mxc_gpio_port * port , u32 gpio )
{
void __iomem * reg = port - > base ;
u32 bit , val ;
int edge ;
reg + = GPIO_ICR1 + ( ( gpio & 0x10 ) > > 2 ) ; /* lower or upper register */
bit = gpio & 0xf ;
2011-06-06 00:07:55 +08:00
val = readl ( reg ) ;
2009-03-12 12:46:41 +01:00
edge = ( val > > ( bit < < 1 ) ) & 3 ;
val & = ~ ( 0x3 < < ( bit < < 1 ) ) ;
2010-02-05 22:14:37 +01:00
if ( edge = = GPIO_INT_HIGH_LEV ) {
2009-03-12 12:46:41 +01:00
edge = GPIO_INT_LOW_LEV ;
pr_debug ( " mxc: switch GPIO %d to low trigger \n " , gpio ) ;
2010-02-05 22:14:37 +01:00
} else if ( edge = = GPIO_INT_LOW_LEV ) {
2009-03-12 12:46:41 +01:00
edge = GPIO_INT_HIGH_LEV ;
pr_debug ( " mxc: switch GPIO %d to high trigger \n " , gpio ) ;
2010-02-05 22:14:37 +01:00
} else {
2009-03-12 12:46:41 +01:00
pr_err ( " mxc: invalid configuration for GPIO %d: %x \n " ,
gpio , edge ) ;
return ;
}
2011-06-06 00:07:55 +08:00
writel ( val | ( edge < < ( bit < < 1 ) ) , reg ) ;
2009-03-12 12:46:41 +01:00
}
2010-02-08 21:02:30 +01:00
/* handle 32 interrupts in one status register */
2008-07-05 10:02:49 +02:00
static void mxc_gpio_irq_handler ( struct mxc_gpio_port * port , u32 irq_stat )
{
2010-02-08 21:02:30 +01:00
while ( irq_stat ! = 0 ) {
int irqoffset = fls ( irq_stat ) - 1 ;
2008-07-05 10:02:49 +02:00
2010-02-08 21:02:30 +01:00
if ( port - > both_edges & ( 1 < < irqoffset ) )
mxc_flip_edge ( port , irqoffset ) ;
2009-03-12 12:46:41 +01:00
2012-06-13 09:04:03 +08:00
generic_handle_irq ( irq_find_mapping ( port - > domain , irqoffset ) ) ;
2009-03-12 12:46:41 +01:00
2010-02-08 21:02:30 +01:00
irq_stat & = ~ ( 1 < < irqoffset ) ;
2008-07-05 10:02:49 +02:00
}
}
2008-11-14 11:01:38 +01:00
/* MX1 and MX3 has one interrupt *per* gpio port */
2008-07-05 10:02:49 +02:00
static void mx3_gpio_irq_handler ( u32 irq , struct irq_desc * desc )
{
u32 irq_stat ;
2011-03-24 13:25:22 +01:00
struct mxc_gpio_port * port = irq_get_handler_data ( irq ) ;
2011-09-21 21:24:04 +08:00
struct irq_chip * chip = irq_get_chip ( irq ) ;
chained_irq_enter ( chip , desc ) ;
2008-07-05 10:02:49 +02:00
2011-06-06 00:07:55 +08:00
irq_stat = readl ( port - > base + GPIO_ISR ) & readl ( port - > base + GPIO_IMR ) ;
2009-04-21 12:39:59 +02:00
2008-07-05 10:02:49 +02:00
mxc_gpio_irq_handler ( port , irq_stat ) ;
2011-09-21 21:24:04 +08:00
chained_irq_exit ( chip , desc ) ;
2008-07-05 10:02:49 +02:00
}
/* MX2 has one interrupt *for all* gpio ports */
static void mx2_gpio_irq_handler ( u32 irq , struct irq_desc * desc )
{
u32 irq_msk , irq_stat ;
2011-06-06 00:07:55 +08:00
struct mxc_gpio_port * port ;
2013-07-18 14:58:06 +02:00
struct irq_chip * chip = irq_get_chip ( irq ) ;
chained_irq_enter ( chip , desc ) ;
2008-07-05 10:02:49 +02:00
/* walk through all interrupt status registers */
2011-06-06 00:07:55 +08:00
list_for_each_entry ( port , & mxc_gpio_ports , node ) {
irq_msk = readl ( port - > base + GPIO_IMR ) ;
2008-07-05 10:02:49 +02:00
if ( ! irq_msk )
continue ;
2011-06-06 00:07:55 +08:00
irq_stat = readl ( port - > base + GPIO_ISR ) & irq_msk ;
2008-07-05 10:02:49 +02:00
if ( irq_stat )
2011-06-06 00:07:55 +08:00
mxc_gpio_irq_handler ( port , irq_stat ) ;
2008-07-05 10:02:49 +02:00
}
2013-07-18 14:58:06 +02:00
chained_irq_exit ( chip , desc ) ;
2008-07-05 10:02:49 +02:00
}
2010-10-23 09:12:48 -05:00
/*
* Set interrupt number " irq " in the GPIO as a wake - up source .
* While system is running , all registered GPIO interrupts need to have
* wake - up enabled . When system is suspended , only selected GPIO interrupts
* need to have wake - up enabled .
* @ param irq interrupt source number
* @ param enable enable as wake - up if equal to non - zero
* @ return This function returns 0 on success .
*/
2010-11-29 11:16:23 +01:00
static int gpio_set_wake_irq ( struct irq_data * d , u32 enable )
2010-10-23 09:12:48 -05:00
{
2011-06-07 16:25:37 +08:00
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
struct mxc_gpio_port * port = gc - > private ;
2012-06-13 09:04:03 +08:00
u32 gpio_idx = d - > hwirq ;
2010-10-23 09:12:48 -05:00
if ( enable ) {
if ( port - > irq_high & & ( gpio_idx > = 16 ) )
enable_irq_wake ( port - > irq_high ) ;
else
enable_irq_wake ( port - > irq ) ;
} else {
if ( port - > irq_high & & ( gpio_idx > = 16 ) )
disable_irq_wake ( port - > irq_high ) ;
else
disable_irq_wake ( port - > irq ) ;
}
return 0 ;
}
2012-06-13 09:04:03 +08:00
static void __init mxc_gpio_init_gc ( struct mxc_gpio_port * port , int irq_base )
2011-06-07 16:25:37 +08:00
{
struct irq_chip_generic * gc ;
struct irq_chip_type * ct ;
2012-06-13 09:04:03 +08:00
gc = irq_alloc_generic_chip ( " gpio-mxc " , 1 , irq_base ,
2011-06-07 16:25:37 +08:00
port - > base , handle_level_irq ) ;
gc - > private = port ;
ct = gc - > chip_types ;
2011-07-19 21:16:56 +08:00
ct - > chip . irq_ack = irq_gc_ack_set_bit ;
2011-06-07 16:25:37 +08:00
ct - > chip . irq_mask = irq_gc_mask_clr_bit ;
ct - > chip . irq_unmask = irq_gc_mask_set_bit ;
ct - > chip . irq_set_type = gpio_set_irq_type ;
2011-07-19 21:16:56 +08:00
ct - > chip . irq_set_wake = gpio_set_wake_irq ;
2011-06-07 16:25:37 +08:00
ct - > regs . ack = GPIO_ISR ;
ct - > regs . mask = GPIO_IMR ;
irq_setup_generic_chip ( gc , IRQ_MSK ( 32 ) , IRQ_GC_INIT_NESTED_LOCK ,
IRQ_NOREQUEST , 0 ) ;
}
2011-04-04 14:29:58 +02:00
2012-11-19 13:22:34 -05:00
static void mxc_gpio_get_hw ( struct platform_device * pdev )
2011-07-07 00:37:41 +08:00
{
2011-07-07 00:37:43 +08:00
const struct of_device_id * of_id =
of_match_device ( mxc_gpio_dt_ids , & pdev - > dev ) ;
enum mxc_gpio_hwtype hwtype ;
if ( of_id )
pdev - > id_entry = of_id - > data ;
hwtype = pdev - > id_entry - > driver_data ;
2011-07-07 00:37:41 +08:00
if ( mxc_gpio_hwtype ) {
/*
* The driver works with a reasonable presupposition ,
* that is all gpio ports must be the same type when
* running on one soc .
*/
BUG_ON ( mxc_gpio_hwtype ! = hwtype ) ;
return ;
}
2012-06-22 21:04:06 +02:00
if ( hwtype = = IMX35_GPIO )
mxc_gpio_hwdata = & imx35_gpio_hwdata ;
else if ( hwtype = = IMX31_GPIO )
2011-07-07 00:37:41 +08:00
mxc_gpio_hwdata = & imx31_gpio_hwdata ;
else
mxc_gpio_hwdata = & imx1_imx21_gpio_hwdata ;
mxc_gpio_hwtype = hwtype ;
}
2011-08-14 00:14:02 +08:00
static int mxc_gpio_to_irq ( struct gpio_chip * gc , unsigned offset )
{
struct bgpio_chip * bgc = to_bgpio_chip ( gc ) ;
struct mxc_gpio_port * port =
container_of ( bgc , struct mxc_gpio_port , bgc ) ;
2012-06-13 09:04:03 +08:00
return irq_find_mapping ( port - > domain , offset ) ;
2011-08-14 00:14:02 +08:00
}
2012-11-19 13:22:34 -05:00
static int mxc_gpio_probe ( struct platform_device * pdev )
2008-07-05 10:02:49 +02:00
{
2011-07-07 00:37:43 +08:00
struct device_node * np = pdev - > dev . of_node ;
2011-06-06 00:07:55 +08:00
struct mxc_gpio_port * port ;
struct resource * iores ;
2012-06-13 09:04:03 +08:00
int irq_base ;
2011-06-07 16:25:37 +08:00
int err ;
2011-06-06 00:07:55 +08:00
2011-07-07 00:37:41 +08:00
mxc_gpio_get_hw ( pdev ) ;
2013-07-08 17:14:39 -03:00
port = devm_kzalloc ( & pdev - > dev , sizeof ( * port ) , GFP_KERNEL ) ;
2011-06-06 00:07:55 +08:00
if ( ! port )
return - ENOMEM ;
2008-07-05 10:02:49 +02:00
2011-06-06 00:07:55 +08:00
iores = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2013-07-08 17:14:39 -03:00
port - > base = devm_ioremap_resource ( & pdev - > dev , iores ) ;
if ( IS_ERR ( port - > base ) )
return PTR_ERR ( port - > base ) ;
2011-06-06 00:07:55 +08:00
port - > irq_high = platform_get_irq ( pdev , 1 ) ;
port - > irq = platform_get_irq ( pdev , 0 ) ;
2013-07-08 17:14:39 -03:00
if ( port - > irq < 0 )
2013-12-21 13:05:57 +05:30
return port - > irq ;
2011-06-06 00:07:55 +08:00
/* disable the interrupt and clear the status */
writel ( 0 , port - > base + GPIO_IMR ) ;
writel ( ~ 0 , port - > base + GPIO_ISR ) ;
2011-07-07 00:37:41 +08:00
if ( mxc_gpio_hwtype = = IMX21_GPIO ) {
2012-06-06 11:49:23 +02:00
/*
* Setup one handler for all GPIO interrupts . Actually setting
* the handler is needed only once , but doing it for every port
* is more robust and easier .
*/
irq_set_chained_handler ( port - > irq , mx2_gpio_irq_handler ) ;
2011-06-06 00:07:55 +08:00
} else {
/* setup one handler for each entry */
irq_set_chained_handler ( port - > irq , mx3_gpio_irq_handler ) ;
irq_set_handler_data ( port - > irq , port ) ;
if ( port - > irq_high > 0 ) {
/* setup handler for GPIO 16 to 31 */
irq_set_chained_handler ( port - > irq_high ,
mx3_gpio_irq_handler ) ;
irq_set_handler_data ( port - > irq_high , port ) ;
}
2008-07-05 10:02:49 +02:00
}
2011-06-06 13:22:41 +08:00
err = bgpio_init ( & port - > bgc , & pdev - > dev , 4 ,
port - > base + GPIO_PSR ,
port - > base + GPIO_DR , NULL ,
2012-05-19 21:34:58 +08:00
port - > base + GPIO_GDIR , NULL , 0 ) ;
2011-06-06 13:22:41 +08:00
if ( err )
2013-07-08 17:14:39 -03:00
goto out_bgio ;
2011-06-06 00:07:55 +08:00
2011-08-14 00:14:02 +08:00
port - > bgc . gc . to_irq = mxc_gpio_to_irq ;
2012-08-05 14:01:26 +08:00
port - > bgc . gc . base = ( pdev - > id < 0 ) ? of_alias_get_id ( np , " gpio " ) * 32 :
pdev - > id * 32 ;
2011-06-06 00:07:55 +08:00
2011-06-06 13:22:41 +08:00
err = gpiochip_add ( & port - > bgc . gc ) ;
2011-06-06 00:07:55 +08:00
if ( err )
2011-06-06 13:22:41 +08:00
goto out_bgpio_remove ;
2011-06-06 00:07:55 +08:00
2012-06-13 09:04:03 +08:00
irq_base = irq_alloc_descs ( - 1 , 0 , 32 , numa_node_id ( ) ) ;
if ( irq_base < 0 ) {
err = irq_base ;
goto out_gpiochip_remove ;
}
port - > domain = irq_domain_add_legacy ( np , 32 , irq_base , 0 ,
& irq_domain_simple_ops , NULL ) ;
if ( ! port - > domain ) {
err = - ENODEV ;
goto out_irqdesc_free ;
}
2011-07-07 00:37:43 +08:00
/* gpio-mxc can be a generic irq chip */
2012-06-13 09:04:03 +08:00
mxc_gpio_init_gc ( port , irq_base ) ;
2011-07-07 00:37:43 +08:00
2011-06-06 00:07:55 +08:00
list_add_tail ( & port - > node , & mxc_gpio_ports ) ;
2008-07-05 10:02:49 +02:00
return 0 ;
2011-06-06 00:07:55 +08:00
2012-06-13 09:04:03 +08:00
out_irqdesc_free :
irq_free_descs ( irq_base , 32 ) ;
out_gpiochip_remove :
2014-07-12 22:30:12 +02:00
gpiochip_remove ( & port - > bgc . gc ) ;
2011-06-06 13:22:41 +08:00
out_bgpio_remove :
bgpio_remove ( & port - > bgc ) ;
2013-07-08 17:14:39 -03:00
out_bgio :
2011-06-06 00:07:55 +08:00
dev_info ( & pdev - > dev , " %s failed with errno %d \n " , __func__ , err ) ;
return err ;
2008-07-05 10:02:49 +02:00
}
2011-06-06 00:07:55 +08:00
static struct platform_driver mxc_gpio_driver = {
. driver = {
. name = " gpio-mxc " ,
. owner = THIS_MODULE ,
2011-07-07 00:37:43 +08:00
. of_match_table = mxc_gpio_dt_ids ,
2011-06-06 00:07:55 +08:00
} ,
. probe = mxc_gpio_probe ,
2011-07-07 00:37:41 +08:00
. id_table = mxc_gpio_devtype ,
2011-06-06 00:07:55 +08:00
} ;
static int __init gpio_mxc_init ( void )
{
return platform_driver_register ( & mxc_gpio_driver ) ;
}
postcore_initcall ( gpio_mxc_init ) ;
MODULE_AUTHOR ( " Freescale Semiconductor, "
" Daniel Mack <danielncaiaq.de>, "
" Juergen Beisert <kernel@pengutronix.de> " ) ;
MODULE_DESCRIPTION ( " Freescale MXC GPIO " ) ;
MODULE_LICENSE ( " GPL " ) ;