2011-01-04 23:28:15 +03:00
/*
* Atheros AR71XX / AR724X / AR913X GPIO API support
*
2012-03-14 13:45:23 +04:00
* Copyright ( C ) 2010 - 2011 Jaiganesh Narayanan < jnarayanan @ atheros . com >
* Copyright ( C ) 2008 - 2011 Gabor Juhos < juhosg @ openwrt . org >
2011-01-04 23:28:15 +03:00
* Copyright ( C ) 2008 Imre Kaloz < kaloz @ openwrt . org >
*
2012-03-14 13:45:23 +04:00
* Parts of this file are based on Atheros ' 2.6 .15 / 2.6 .31 BSP
*
2011-01-04 23:28:15 +03:00
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation .
*/
2015-09-01 12:38:02 +03:00
# include <linux/gpio/driver.h>
2015-05-31 03:18:24 +03:00
# include <linux/platform_data/gpio-ath79.h>
# include <linux/of_device.h>
2011-01-04 23:28:15 +03:00
# include <asm/mach-ath79/ar71xx_regs.h>
2015-09-01 12:38:02 +03:00
struct ath79_gpio_ctrl {
struct gpio_chip chip ;
void __iomem * base ;
spinlock_t lock ;
} ;
# define to_ath79_gpio_ctrl(c) container_of(c, struct ath79_gpio_ctrl, chip)
2011-01-04 23:28:15 +03:00
2015-09-01 12:38:02 +03:00
static void ath79_gpio_set_value ( struct gpio_chip * chip ,
unsigned gpio , int value )
2011-01-04 23:28:15 +03:00
{
2015-09-01 12:38:02 +03:00
struct ath79_gpio_ctrl * ctrl = to_ath79_gpio_ctrl ( chip ) ;
2011-01-04 23:28:15 +03:00
if ( value )
2015-09-01 12:38:02 +03:00
__raw_writel ( BIT ( gpio ) , ctrl - > base + AR71XX_GPIO_REG_SET ) ;
2011-01-04 23:28:15 +03:00
else
2015-09-01 12:38:02 +03:00
__raw_writel ( BIT ( gpio ) , ctrl - > base + AR71XX_GPIO_REG_CLEAR ) ;
2011-01-04 23:28:15 +03:00
}
2015-09-01 12:38:02 +03:00
static int ath79_gpio_get_value ( struct gpio_chip * chip , unsigned gpio )
2011-01-04 23:28:15 +03:00
{
2015-09-01 12:38:02 +03:00
struct ath79_gpio_ctrl * ctrl = to_ath79_gpio_ctrl ( chip ) ;
2011-01-04 23:28:15 +03:00
2015-09-01 12:38:02 +03:00
return ( __raw_readl ( ctrl - > base + AR71XX_GPIO_REG_IN ) > > gpio ) & 1 ;
2011-01-04 23:28:15 +03:00
}
static int ath79_gpio_direction_input ( struct gpio_chip * chip ,
unsigned offset )
{
2015-09-01 12:38:02 +03:00
struct ath79_gpio_ctrl * ctrl = to_ath79_gpio_ctrl ( chip ) ;
2011-01-04 23:28:15 +03:00
unsigned long flags ;
2015-09-01 12:38:02 +03:00
spin_lock_irqsave ( & ctrl - > lock , flags ) ;
2011-01-04 23:28:15 +03:00
2015-09-01 12:38:02 +03:00
__raw_writel (
__raw_readl ( ctrl - > base + AR71XX_GPIO_REG_OE ) & ~ BIT ( offset ) ,
ctrl - > base + AR71XX_GPIO_REG_OE ) ;
2011-01-04 23:28:15 +03:00
2015-09-01 12:38:02 +03:00
spin_unlock_irqrestore ( & ctrl - > lock , flags ) ;
2011-01-04 23:28:15 +03:00
return 0 ;
}
static int ath79_gpio_direction_output ( struct gpio_chip * chip ,
unsigned offset , int value )
{
2015-09-01 12:38:02 +03:00
struct ath79_gpio_ctrl * ctrl = to_ath79_gpio_ctrl ( chip ) ;
2011-01-04 23:28:15 +03:00
unsigned long flags ;
2015-09-01 12:38:02 +03:00
spin_lock_irqsave ( & ctrl - > lock , flags ) ;
2011-01-04 23:28:15 +03:00
if ( value )
2015-09-01 12:38:02 +03:00
__raw_writel ( BIT ( offset ) , ctrl - > base + AR71XX_GPIO_REG_SET ) ;
2011-01-04 23:28:15 +03:00
else
2015-09-01 12:38:02 +03:00
__raw_writel ( BIT ( offset ) , ctrl - > base + AR71XX_GPIO_REG_CLEAR ) ;
2011-01-04 23:28:15 +03:00
2015-09-01 12:38:02 +03:00
__raw_writel (
__raw_readl ( ctrl - > base + AR71XX_GPIO_REG_OE ) | BIT ( offset ) ,
ctrl - > base + AR71XX_GPIO_REG_OE ) ;
2011-01-04 23:28:15 +03:00
2015-09-01 12:38:02 +03:00
spin_unlock_irqrestore ( & ctrl - > lock , flags ) ;
2011-01-04 23:28:15 +03:00
return 0 ;
}
2012-03-14 13:45:23 +04:00
static int ar934x_gpio_direction_input ( struct gpio_chip * chip , unsigned offset )
{
2015-09-01 12:38:02 +03:00
struct ath79_gpio_ctrl * ctrl = to_ath79_gpio_ctrl ( chip ) ;
2012-03-14 13:45:23 +04:00
unsigned long flags ;
2015-09-01 12:38:02 +03:00
spin_lock_irqsave ( & ctrl - > lock , flags ) ;
2012-03-14 13:45:23 +04:00
2015-09-01 12:38:02 +03:00
__raw_writel (
__raw_readl ( ctrl - > base + AR71XX_GPIO_REG_OE ) | BIT ( offset ) ,
ctrl - > base + AR71XX_GPIO_REG_OE ) ;
2012-03-14 13:45:23 +04:00
2015-09-01 12:38:02 +03:00
spin_unlock_irqrestore ( & ctrl - > lock , flags ) ;
2012-03-14 13:45:23 +04:00
return 0 ;
}
static int ar934x_gpio_direction_output ( struct gpio_chip * chip , unsigned offset ,
int value )
{
2015-09-01 12:38:02 +03:00
struct ath79_gpio_ctrl * ctrl = to_ath79_gpio_ctrl ( chip ) ;
2012-03-14 13:45:23 +04:00
unsigned long flags ;
2015-09-01 12:38:02 +03:00
spin_lock_irqsave ( & ctrl - > lock , flags ) ;
2012-03-14 13:45:23 +04:00
if ( value )
2015-09-01 12:38:02 +03:00
__raw_writel ( BIT ( offset ) , ctrl - > base + AR71XX_GPIO_REG_SET ) ;
2012-03-14 13:45:23 +04:00
else
2015-09-01 12:38:02 +03:00
__raw_writel ( BIT ( offset ) , ctrl - > base + AR71XX_GPIO_REG_CLEAR ) ;
2012-03-14 13:45:23 +04:00
2015-09-01 12:38:02 +03:00
__raw_writel (
__raw_readl ( ctrl - > base + AR71XX_GPIO_REG_OE ) & BIT ( offset ) ,
ctrl - > base + AR71XX_GPIO_REG_OE ) ;
2012-03-14 13:45:23 +04:00
2015-09-01 12:38:02 +03:00
spin_unlock_irqrestore ( & ctrl - > lock , flags ) ;
2012-03-14 13:45:23 +04:00
return 0 ;
}
2015-09-01 12:38:02 +03:00
static const struct gpio_chip ath79_gpio_chip = {
2011-01-04 23:28:15 +03:00
. label = " ath79 " ,
. get = ath79_gpio_get_value ,
. set = ath79_gpio_set_value ,
. direction_input = ath79_gpio_direction_input ,
. direction_output = ath79_gpio_direction_output ,
. base = 0 ,
} ;
2015-05-31 03:18:24 +03:00
static const struct of_device_id ath79_gpio_of_match [ ] = {
{ . compatible = " qca,ar7100-gpio " } ,
{ . compatible = " qca,ar9340-gpio " } ,
{ } ,
} ;
static int ath79_gpio_probe ( struct platform_device * pdev )
2011-01-04 23:28:15 +03:00
{
2015-05-31 03:18:24 +03:00
struct ath79_gpio_platform_data * pdata = pdev - > dev . platform_data ;
struct device_node * np = pdev - > dev . of_node ;
2015-09-01 12:38:02 +03:00
struct ath79_gpio_ctrl * ctrl ;
2015-05-31 03:18:24 +03:00
struct resource * res ;
2015-09-01 12:38:02 +03:00
u32 ath79_gpio_count ;
2015-05-31 03:18:24 +03:00
bool oe_inverted ;
2011-01-04 23:28:15 +03:00
int err ;
2015-09-01 12:38:02 +03:00
ctrl = devm_kzalloc ( & pdev - > dev , sizeof ( * ctrl ) , GFP_KERNEL ) ;
if ( ! ctrl )
return - ENOMEM ;
2015-05-31 03:18:24 +03:00
if ( np ) {
err = of_property_read_u32 ( np , " ngpios " , & ath79_gpio_count ) ;
if ( err ) {
dev_err ( & pdev - > dev , " ngpios property is not valid \n " ) ;
return err ;
}
if ( ath79_gpio_count > = 32 ) {
dev_err ( & pdev - > dev , " ngpios must be less than 32 \n " ) ;
return - EINVAL ;
}
oe_inverted = of_device_is_compatible ( np , " qca,ar9340-gpio " ) ;
} else if ( pdata ) {
ath79_gpio_count = pdata - > ngpios ;
oe_inverted = pdata - > oe_inverted ;
} else {
dev_err ( & pdev - > dev , " No DT node or platform data found \n " ) ;
return - EINVAL ;
}
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2015-09-01 12:38:02 +03:00
ctrl - > base = devm_ioremap_nocache (
2015-05-31 03:18:24 +03:00
& pdev - > dev , res - > start , resource_size ( res ) ) ;
2015-09-01 12:38:02 +03:00
if ( ! ctrl - > base )
2015-05-31 03:18:24 +03:00
return - ENOMEM ;
2011-01-04 23:28:15 +03:00
2015-09-01 12:38:02 +03:00
spin_lock_init ( & ctrl - > lock ) ;
memcpy ( & ctrl - > chip , & ath79_gpio_chip , sizeof ( ctrl - > chip ) ) ;
ctrl - > chip . dev = & pdev - > dev ;
ctrl - > chip . ngpio = ath79_gpio_count ;
2015-05-31 03:18:24 +03:00
if ( oe_inverted ) {
2015-09-01 12:38:02 +03:00
ctrl - > chip . direction_input = ar934x_gpio_direction_input ;
ctrl - > chip . direction_output = ar934x_gpio_direction_output ;
2012-03-14 13:45:23 +04:00
}
2011-01-04 23:28:15 +03:00
2015-09-01 12:38:02 +03:00
err = gpiochip_add ( & ctrl - > chip ) ;
2015-05-31 03:18:24 +03:00
if ( err ) {
dev_err ( & pdev - > dev ,
" cannot add AR71xx GPIO chip, error=%d " , err ) ;
return err ;
}
return 0 ;
2011-01-04 23:28:15 +03:00
}
2015-05-31 03:18:24 +03:00
static struct platform_driver ath79_gpio_driver = {
. driver = {
. name = " ath79-gpio " ,
. of_match_table = ath79_gpio_of_match ,
} ,
. probe = ath79_gpio_probe ,
} ;
module_platform_driver ( ath79_gpio_driver ) ;