Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit f5837ec1 authored by Roger Quadros's avatar Roger Quadros Committed by Linus Walleij
Browse files

gpio: twl4030: Fix regression for twl gpio LED output



Commit 0b2aa8be introduced a regression that causes failure
in setting LED GPO direction to OUT.

This causes USB host probe failures for Beagleboard C4.

platform usb_phy_gen_xceiv.2: Driver usb_phy_gen_xceiv requests probe deferral
hsusb2_vcc: Failed to request enable GPIO510: -22
reg-fixed-voltage reg-fixed-voltage.0.auto: Failed to register regulator: -22
reg-fixed-voltage: probe of reg-fixed-voltage.0.auto failed with error -22

direction_out/direction_in must return 0 if the operation succeeded.

Also, don't update direction flag and output data if twl4030_set_gpio_direction()
failed inside twl_direction_out();

Cc: stable@vger.kernel.org
Signed-off-by: default avatarRoger Quadros <rogerq@ti.com>
Acked-by: default avatarTony Lindgren <tony@atomide.com>
Signed-off-by: default avatarLinus Walleij <linus.walleij@linaro.org>
parent 8620f394
Loading
Loading
Loading
Loading
+12 −3
Original line number Original line Diff line number Diff line
@@ -300,7 +300,7 @@ static int twl_direction_in(struct gpio_chip *chip, unsigned offset)
	if (offset < TWL4030_GPIO_MAX)
	if (offset < TWL4030_GPIO_MAX)
		ret = twl4030_set_gpio_direction(offset, 1);
		ret = twl4030_set_gpio_direction(offset, 1);
	else
	else
		ret = -EINVAL;
		ret = -EINVAL;	/* LED outputs can't be set as input */


	if (!ret)
	if (!ret)
		priv->direction &= ~BIT(offset);
		priv->direction &= ~BIT(offset);
@@ -354,11 +354,20 @@ static void twl_set(struct gpio_chip *chip, unsigned offset, int value)
static int twl_direction_out(struct gpio_chip *chip, unsigned offset, int value)
static int twl_direction_out(struct gpio_chip *chip, unsigned offset, int value)
{
{
	struct gpio_twl4030_priv *priv = to_gpio_twl4030(chip);
	struct gpio_twl4030_priv *priv = to_gpio_twl4030(chip);
	int ret = -EINVAL;
	int ret = 0;


	mutex_lock(&priv->mutex);
	mutex_lock(&priv->mutex);
	if (offset < TWL4030_GPIO_MAX)
	if (offset < TWL4030_GPIO_MAX) {
		ret = twl4030_set_gpio_direction(offset, 0);
		ret = twl4030_set_gpio_direction(offset, 0);
		if (ret) {
			mutex_unlock(&priv->mutex);
			return ret;
		}
	}

	/*
	 *  LED gpios i.e. offset >= TWL4030_GPIO_MAX are always output
	 */


	priv->direction |= BIT(offset);
	priv->direction |= BIT(offset);
	mutex_unlock(&priv->mutex);
	mutex_unlock(&priv->mutex);