2010-12-08 03:26:57 +03:00
/*
* OMAP2 + specific gpio initialization
*
* Copyright ( C ) 2010 Texas Instruments Incorporated - http : //www.ti.com/
*
* Author :
* Charulatha V < charu @ ti . com >
*
* 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 version 2.
*
* This program is distributed " as is " WITHOUT ANY WARRANTY of any
* kind , whether express or implied ; without even the implied warranty
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*/
# include <linux/gpio.h>
# include <linux/err.h>
# include <linux/slab.h>
# include <linux/interrupt.h>
2011-08-16 13:51:54 +04:00
# include <linux/of.h>
2012-08-31 02:37:24 +04:00
# include <linux/platform_data/gpio-omap.h>
2010-12-08 03:26:57 +03:00
2012-10-03 04:41:35 +04:00
# include "omap_hwmod.h"
2012-10-03 04:25:48 +04:00
# include "omap_device.h"
2012-10-04 03:36:40 +04:00
# include "omap-pm.h"
2010-12-08 03:26:57 +03:00
2011-05-05 18:45:16 +04:00
# include "powerdomain.h"
2012-02-20 21:43:30 +04:00
static int __init omap2_gpio_dev_init ( struct omap_hwmod * oh , void * unused )
2010-12-08 03:26:57 +03:00
{
2011-07-22 00:48:45 +04:00
struct platform_device * pdev ;
2010-12-08 03:26:57 +03:00
struct omap_gpio_platform_data * pdata ;
struct omap_gpio_dev_attr * dev_attr ;
char * name = " omap_gpio " ;
int id ;
2011-05-05 18:45:16 +04:00
struct powerdomain * pwrdm ;
2010-12-08 03:26:57 +03:00
/*
* extract the device id from name field available in the
* hwmod database and use the same for constructing ids for
* gpio devices .
* CAUTION : Make sure the name in the hwmod database does
* not change . If changed , make corresponding change here
* or make use of static variable mechanism to handle this .
*/
sscanf ( oh - > name , " gpio%d " , & id ) ;
pdata = kzalloc ( sizeof ( struct omap_gpio_platform_data ) , GFP_KERNEL ) ;
if ( ! pdata ) {
pr_err ( " gpio%d: Memory allocation failed \n " , id ) ;
return - ENOMEM ;
}
dev_attr = ( struct omap_gpio_dev_attr * ) oh - > dev_attr ;
pdata - > bank_width = dev_attr - > bank_width ;
pdata - > dbck_flag = dev_attr - > dbck_flag ;
2011-09-29 03:17:25 +04:00
pdata - > get_context_loss_count = omap_pm_get_dev_context_loss_count ;
2011-04-21 03:31:23 +04:00
pdata - > regs = kzalloc ( sizeof ( struct omap_gpio_reg_offs ) , GFP_KERNEL ) ;
2012-04-13 16:34:32 +04:00
if ( ! pdata - > regs ) {
2011-04-21 03:31:23 +04:00
pr_err ( " gpio%d: Memory allocation failed \n " , id ) ;
2012-09-03 16:42:27 +04:00
kfree ( pdata ) ;
2011-04-21 03:31:23 +04:00
return - ENOMEM ;
}
2010-12-08 03:26:57 +03:00
switch ( oh - > class - > rev ) {
case 0 :
2011-05-05 15:34:12 +04:00
if ( id = = 1 )
/* non-wakeup GPIO pins for OMAP2 Bank1 */
pdata - > non_wakeup_gpios = 0xe203ffc0 ;
else if ( id = = 2 )
/* non-wakeup GPIO pins for OMAP2 Bank2 */
pdata - > non_wakeup_gpios = 0x08700040 ;
/* fall through */
2010-12-08 03:26:57 +03:00
case 1 :
2011-04-23 01:37:16 +04:00
pdata - > regs - > revision = OMAP24XX_GPIO_REVISION ;
2011-04-21 03:31:23 +04:00
pdata - > regs - > direction = OMAP24XX_GPIO_OE ;
pdata - > regs - > datain = OMAP24XX_GPIO_DATAIN ;
pdata - > regs - > dataout = OMAP24XX_GPIO_DATAOUT ;
pdata - > regs - > set_dataout = OMAP24XX_GPIO_SETDATAOUT ;
pdata - > regs - > clr_dataout = OMAP24XX_GPIO_CLEARDATAOUT ;
2011-04-21 20:17:35 +04:00
pdata - > regs - > irqstatus = OMAP24XX_GPIO_IRQSTATUS1 ;
pdata - > regs - > irqstatus2 = OMAP24XX_GPIO_IRQSTATUS2 ;
2011-04-21 20:53:06 +04:00
pdata - > regs - > irqenable = OMAP24XX_GPIO_IRQENABLE1 ;
2011-08-30 13:54:27 +04:00
pdata - > regs - > irqenable2 = OMAP24XX_GPIO_IRQENABLE2 ;
2011-04-21 20:53:06 +04:00
pdata - > regs - > set_irqenable = OMAP24XX_GPIO_SETIRQENABLE1 ;
pdata - > regs - > clr_irqenable = OMAP24XX_GPIO_CLEARIRQENABLE1 ;
2011-04-22 23:02:05 +04:00
pdata - > regs - > debounce = OMAP24XX_GPIO_DEBOUNCE_VAL ;
pdata - > regs - > debounce_en = OMAP24XX_GPIO_DEBOUNCE_EN ;
2011-05-02 13:51:42 +04:00
pdata - > regs - > ctrl = OMAP24XX_GPIO_CTRL ;
2011-09-13 13:11:44 +04:00
pdata - > regs - > wkup_en = OMAP24XX_GPIO_WAKE_EN ;
2011-08-30 13:35:44 +04:00
pdata - > regs - > leveldetect0 = OMAP24XX_GPIO_LEVELDETECT0 ;
pdata - > regs - > leveldetect1 = OMAP24XX_GPIO_LEVELDETECT1 ;
pdata - > regs - > risingdetect = OMAP24XX_GPIO_RISINGDETECT ;
pdata - > regs - > fallingdetect = OMAP24XX_GPIO_FALLINGDETECT ;
2010-12-08 03:26:57 +03:00
break ;
case 2 :
2011-04-23 01:37:16 +04:00
pdata - > regs - > revision = OMAP4_GPIO_REVISION ;
2011-04-21 03:31:23 +04:00
pdata - > regs - > direction = OMAP4_GPIO_OE ;
pdata - > regs - > datain = OMAP4_GPIO_DATAIN ;
pdata - > regs - > dataout = OMAP4_GPIO_DATAOUT ;
pdata - > regs - > set_dataout = OMAP4_GPIO_SETDATAOUT ;
pdata - > regs - > clr_dataout = OMAP4_GPIO_CLEARDATAOUT ;
2012-04-27 18:13:37 +04:00
pdata - > regs - > irqstatus_raw0 = OMAP4_GPIO_IRQSTATUSRAW0 ;
pdata - > regs - > irqstatus_raw1 = OMAP4_GPIO_IRQSTATUSRAW1 ;
2011-04-21 20:17:35 +04:00
pdata - > regs - > irqstatus = OMAP4_GPIO_IRQSTATUS0 ;
pdata - > regs - > irqstatus2 = OMAP4_GPIO_IRQSTATUS1 ;
2011-04-21 20:53:06 +04:00
pdata - > regs - > irqenable = OMAP4_GPIO_IRQSTATUSSET0 ;
2011-08-30 13:54:27 +04:00
pdata - > regs - > irqenable2 = OMAP4_GPIO_IRQSTATUSSET1 ;
2011-04-21 20:53:06 +04:00
pdata - > regs - > set_irqenable = OMAP4_GPIO_IRQSTATUSSET0 ;
pdata - > regs - > clr_irqenable = OMAP4_GPIO_IRQSTATUSCLR0 ;
2011-04-22 23:02:05 +04:00
pdata - > regs - > debounce = OMAP4_GPIO_DEBOUNCINGTIME ;
pdata - > regs - > debounce_en = OMAP4_GPIO_DEBOUNCENABLE ;
2011-05-02 13:51:42 +04:00
pdata - > regs - > ctrl = OMAP4_GPIO_CTRL ;
2011-09-13 13:11:44 +04:00
pdata - > regs - > wkup_en = OMAP4_GPIO_IRQWAKEN0 ;
2011-08-30 13:35:44 +04:00
pdata - > regs - > leveldetect0 = OMAP4_GPIO_LEVELDETECT0 ;
pdata - > regs - > leveldetect1 = OMAP4_GPIO_LEVELDETECT1 ;
pdata - > regs - > risingdetect = OMAP4_GPIO_RISINGDETECT ;
pdata - > regs - > fallingdetect = OMAP4_GPIO_FALLINGDETECT ;
2010-12-08 03:26:57 +03:00
break ;
default :
WARN ( 1 , " Invalid gpio bank_type \n " ) ;
2012-09-03 16:42:27 +04:00
kfree ( pdata - > regs ) ;
2010-12-08 03:26:57 +03:00
kfree ( pdata ) ;
return - EINVAL ;
}
2011-05-05 18:45:16 +04:00
pwrdm = omap_hwmod_get_pwrdm ( oh ) ;
pdata - > loses_context = pwrdm_can_ever_lose_context ( pwrdm ) ;
2011-07-22 00:48:45 +04:00
pdev = omap_device_build ( name , id - 1 , oh , pdata ,
2011-08-10 17:30:09 +04:00
sizeof ( * pdata ) , NULL , 0 , false ) ;
2010-12-08 03:26:57 +03:00
kfree ( pdata ) ;
2011-07-22 00:48:45 +04:00
if ( IS_ERR ( pdev ) ) {
2011-03-31 05:57:33 +04:00
WARN ( 1 , " Can't build omap_device for %s:%s. \n " ,
2010-12-08 03:26:57 +03:00
name , oh - > name ) ;
2011-07-22 00:48:45 +04:00
return PTR_ERR ( pdev ) ;
2010-12-08 03:26:57 +03:00
}
return 0 ;
}
/*
* gpio_init needs to be done before
* machine_init functions access gpio APIs .
* Hence gpio_init is a postcore_initcall .
*/
static int __init omap2_gpio_init ( void )
{
2011-08-16 13:51:54 +04:00
/* If dtb is there, the devices will be created dynamically */
if ( of_have_populated_dt ( ) )
return - ENODEV ;
return omap_hwmod_for_each_by_class ( " gpio " , omap2_gpio_dev_init , NULL ) ;
2010-12-08 03:26:57 +03:00
}
postcore_initcall ( omap2_gpio_init ) ;