Commit 73f251d3 authored by David Brownell's avatar David Brownell Committed by Tony Lindgren

twl4030-gpio use new gpiolib hooks

Use the new gpiolib internal mechanism for tracking gpio usage,
making them power the GPIO module on/off, so that the standard
gpio_request()/gpio_free() calls can completely replace the
twl4030-specific versions.
Signed-off-by: default avatarDavid Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: default avatarTony Lindgren <tony@atomide.com>
parent 0a7c9df7
...@@ -102,32 +102,10 @@ static inline int gpio_twl4030_read(u8 address) ...@@ -102,32 +102,10 @@ static inline int gpio_twl4030_read(u8 address)
*/ */
int twl4030_request_gpio(int gpio) int twl4030_request_gpio(int gpio)
{ {
int ret = 0;
if (unlikely(gpio >= TWL4030_GPIO_MAX)) if (unlikely(gpio >= TWL4030_GPIO_MAX))
return -EPERM; return -EPERM;
ret = gpio_request(twl_gpiochip.base + gpio, NULL); return gpio_request(twl_gpiochip.base + gpio, NULL);
if (ret < 0)
return ret;
mutex_lock(&gpio_lock);
if (gpio_usage_count & BIT(gpio)) {
ret = -EBUSY;
} else {
/* First time usage? - switch on GPIO module */
if (!gpio_usage_count) {
ret = gpio_twl4030_write(REG_GPIO_CTRL,
MASK_GPIO_CTRL_GPIO_ON);
}
if (!ret)
gpio_usage_count |= BIT(gpio);
else
gpio_free(twl_gpiochip.base + gpio);
}
mutex_unlock(&gpio_lock);
return ret;
} }
EXPORT_SYMBOL(twl4030_request_gpio); EXPORT_SYMBOL(twl4030_request_gpio);
...@@ -136,26 +114,11 @@ EXPORT_SYMBOL(twl4030_request_gpio); ...@@ -136,26 +114,11 @@ EXPORT_SYMBOL(twl4030_request_gpio);
*/ */
int twl4030_free_gpio(int gpio) int twl4030_free_gpio(int gpio)
{ {
int ret = 0;
if (unlikely(gpio >= TWL4030_GPIO_MAX)) if (unlikely(gpio >= TWL4030_GPIO_MAX))
return -EPERM; return -EPERM;
mutex_lock(&gpio_lock);
if ((gpio_usage_count & BIT(gpio)) == 0) {
ret = -EPERM;
} else {
gpio_usage_count &= ~BIT(gpio);
gpio_free(twl_gpiochip.base + gpio); gpio_free(twl_gpiochip.base + gpio);
} return 0;
/* Last time usage? - switch off GPIO module */
if (ret == 0 && !gpio_usage_count)
ret = gpio_twl4030_write(REG_GPIO_CTRL, 0x0);
mutex_unlock(&gpio_lock);
return ret;
} }
EXPORT_SYMBOL(twl4030_free_gpio); EXPORT_SYMBOL(twl4030_free_gpio);
...@@ -283,6 +246,38 @@ int twl4030_set_gpio_card_detect(int gpio, int enable) ...@@ -283,6 +246,38 @@ int twl4030_set_gpio_card_detect(int gpio, int enable)
/*----------------------------------------------------------------------*/ /*----------------------------------------------------------------------*/
static int twl_request(struct gpio_chip *chip, unsigned offset)
{
int status = 0;
mutex_lock(&gpio_lock);
/* on first use, turn GPIO module "on" */
if (!gpio_usage_count)
status = gpio_twl4030_write(REG_GPIO_CTRL,
MASK_GPIO_CTRL_GPIO_ON);
if (!status)
gpio_usage_count |= (0x1 << offset);
done:
mutex_unlock(&gpio_lock);
return status;
}
static void twl_free(struct gpio_chip *chip, unsigned offset)
{
mutex_lock(&gpio_lock);
gpio_usage_count &= ~BIT(offset);
/* on last use, switch off GPIO module */
if (!gpio_usage_count)
gpio_twl4030_write(REG_GPIO_CTRL, 0x0);
mutex_unlock(&gpio_lock);
}
static int twl_direction_in(struct gpio_chip *chip, unsigned offset) static int twl_direction_in(struct gpio_chip *chip, unsigned offset)
{ {
return twl4030_set_gpio_direction(offset, 1); return twl4030_set_gpio_direction(offset, 1);
...@@ -316,6 +311,8 @@ static int twl_to_irq(struct gpio_chip *chip, unsigned offset) ...@@ -316,6 +311,8 @@ static int twl_to_irq(struct gpio_chip *chip, unsigned offset)
static struct gpio_chip twl_gpiochip = { static struct gpio_chip twl_gpiochip = {
.label = "twl4030", .label = "twl4030",
.owner = THIS_MODULE, .owner = THIS_MODULE,
.request = twl_request,
.free = twl_free,
.direction_input = twl_direction_in, .direction_input = twl_direction_in,
.get = twl_get, .get = twl_get,
.direction_output = twl_direction_out, .direction_output = twl_direction_out,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment