Commit 7726d4c3 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'gpio-fixes-for-v6.0-rc4' of git://git.kernel.org/pub/scm/linux/kernel/git/brgl/linux

Pull gpio fixes from Bartosz Golaszewski:
 "A a set of fixes from the GPIO subsystem.

  Most are small driver fixes except the realtek-otto driver patch which
  is pretty big but addresses a significant flaw that can cause the CPU
  to stay infinitely busy on uncleared ISR on some platforms.

  Summary:
   - MAINTAINERS update
   - fix resource leaks in gpio-mockup and gpio-pxa
   - add missing locking in gpio-pca953x
   - use 32-bit I/O in gpio-realtek-otto
   - make irq_chip structures immutable in four more drivers"

* tag 'gpio-fixes-for-v6.0-rc4' of git://git.kernel.org/pub/scm/linux/kernel/git/brgl/linux:
  gpio: ws16c48: Make irq_chip immutable
  gpio: 104-idio-16: Make irq_chip immutable
  gpio: 104-idi-48: Make irq_chip immutable
  gpio: 104-dio-48e: Make irq_chip immutable
  gpio: realtek-otto: switch to 32-bit I/O
  gpio: pca953x: Add mutex_lock for regcache sync in PM
  gpio: mockup: remove gpio debugfs when remove device
  gpio: pxa: use devres for the clock struct
  MAINTAINERS: rectify entry for XILINX GPIO DRIVER
parents 65eea2c0 68903817
...@@ -22307,7 +22307,7 @@ M: Shubhrajyoti Datta <shubhrajyoti.datta@xilinx.com> ...@@ -22307,7 +22307,7 @@ M: Shubhrajyoti Datta <shubhrajyoti.datta@xilinx.com>
R: Srinivas Neeli <srinivas.neeli@xilinx.com> R: Srinivas Neeli <srinivas.neeli@xilinx.com>
R: Michal Simek <michal.simek@xilinx.com> R: Michal Simek <michal.simek@xilinx.com>
S: Maintained S: Maintained
F: Documentation/devicetree/bindings/gpio/gpio-xilinx.txt F: Documentation/devicetree/bindings/gpio/xlnx,gpio-xilinx.yaml
F: Documentation/devicetree/bindings/gpio/gpio-zynq.yaml F: Documentation/devicetree/bindings/gpio/gpio-zynq.yaml
F: drivers/gpio/gpio-xilinx.c F: drivers/gpio/gpio-xilinx.c
F: drivers/gpio/gpio-zynq.c F: drivers/gpio/gpio-zynq.c
......
...@@ -164,6 +164,7 @@ static void dio48e_irq_mask(struct irq_data *data) ...@@ -164,6 +164,7 @@ static void dio48e_irq_mask(struct irq_data *data)
dio48egpio->irq_mask &= ~BIT(0); dio48egpio->irq_mask &= ~BIT(0);
else else
dio48egpio->irq_mask &= ~BIT(1); dio48egpio->irq_mask &= ~BIT(1);
gpiochip_disable_irq(chip, offset);
if (!dio48egpio->irq_mask) if (!dio48egpio->irq_mask)
/* disable interrupts */ /* disable interrupts */
...@@ -191,6 +192,7 @@ static void dio48e_irq_unmask(struct irq_data *data) ...@@ -191,6 +192,7 @@ static void dio48e_irq_unmask(struct irq_data *data)
iowrite8(0x00, &dio48egpio->reg->enable_interrupt); iowrite8(0x00, &dio48egpio->reg->enable_interrupt);
} }
gpiochip_enable_irq(chip, offset);
if (offset == 19) if (offset == 19)
dio48egpio->irq_mask |= BIT(0); dio48egpio->irq_mask |= BIT(0);
else else
...@@ -213,12 +215,14 @@ static int dio48e_irq_set_type(struct irq_data *data, unsigned int flow_type) ...@@ -213,12 +215,14 @@ static int dio48e_irq_set_type(struct irq_data *data, unsigned int flow_type)
return 0; return 0;
} }
static struct irq_chip dio48e_irqchip = { static const struct irq_chip dio48e_irqchip = {
.name = "104-dio-48e", .name = "104-dio-48e",
.irq_ack = dio48e_irq_ack, .irq_ack = dio48e_irq_ack,
.irq_mask = dio48e_irq_mask, .irq_mask = dio48e_irq_mask,
.irq_unmask = dio48e_irq_unmask, .irq_unmask = dio48e_irq_unmask,
.irq_set_type = dio48e_irq_set_type .irq_set_type = dio48e_irq_set_type,
.flags = IRQCHIP_IMMUTABLE,
GPIOCHIP_IRQ_RESOURCE_HELPERS,
}; };
static irqreturn_t dio48e_irq_handler(int irq, void *dev_id) static irqreturn_t dio48e_irq_handler(int irq, void *dev_id)
...@@ -322,7 +326,7 @@ static int dio48e_probe(struct device *dev, unsigned int id) ...@@ -322,7 +326,7 @@ static int dio48e_probe(struct device *dev, unsigned int id)
dio48egpio->chip.set_multiple = dio48e_gpio_set_multiple; dio48egpio->chip.set_multiple = dio48e_gpio_set_multiple;
girq = &dio48egpio->chip.irq; girq = &dio48egpio->chip.irq;
girq->chip = &dio48e_irqchip; gpio_irq_chip_set_chip(girq, &dio48e_irqchip);
/* This will let us handle the parent IRQ in the driver */ /* This will let us handle the parent IRQ in the driver */
girq->parent_handler = NULL; girq->parent_handler = NULL;
girq->num_parents = 0; girq->num_parents = 0;
......
...@@ -113,6 +113,7 @@ static void idi_48_irq_mask(struct irq_data *data) ...@@ -113,6 +113,7 @@ static void idi_48_irq_mask(struct irq_data *data)
spin_lock_irqsave(&idi48gpio->lock, flags); spin_lock_irqsave(&idi48gpio->lock, flags);
idi48gpio->irq_mask[boundary] &= ~mask; idi48gpio->irq_mask[boundary] &= ~mask;
gpiochip_disable_irq(chip, offset);
/* Exit early if there are still input lines with IRQ unmasked */ /* Exit early if there are still input lines with IRQ unmasked */
if (idi48gpio->irq_mask[boundary]) if (idi48gpio->irq_mask[boundary])
...@@ -140,6 +141,7 @@ static void idi_48_irq_unmask(struct irq_data *data) ...@@ -140,6 +141,7 @@ static void idi_48_irq_unmask(struct irq_data *data)
prev_irq_mask = idi48gpio->irq_mask[boundary]; prev_irq_mask = idi48gpio->irq_mask[boundary];
gpiochip_enable_irq(chip, offset);
idi48gpio->irq_mask[boundary] |= mask; idi48gpio->irq_mask[boundary] |= mask;
/* Exit early if IRQ was already unmasked for this boundary */ /* Exit early if IRQ was already unmasked for this boundary */
...@@ -164,12 +166,14 @@ static int idi_48_irq_set_type(struct irq_data *data, unsigned int flow_type) ...@@ -164,12 +166,14 @@ static int idi_48_irq_set_type(struct irq_data *data, unsigned int flow_type)
return 0; return 0;
} }
static struct irq_chip idi_48_irqchip = { static const struct irq_chip idi_48_irqchip = {
.name = "104-idi-48", .name = "104-idi-48",
.irq_ack = idi_48_irq_ack, .irq_ack = idi_48_irq_ack,
.irq_mask = idi_48_irq_mask, .irq_mask = idi_48_irq_mask,
.irq_unmask = idi_48_irq_unmask, .irq_unmask = idi_48_irq_unmask,
.irq_set_type = idi_48_irq_set_type .irq_set_type = idi_48_irq_set_type,
.flags = IRQCHIP_IMMUTABLE,
GPIOCHIP_IRQ_RESOURCE_HELPERS,
}; };
static irqreturn_t idi_48_irq_handler(int irq, void *dev_id) static irqreturn_t idi_48_irq_handler(int irq, void *dev_id)
...@@ -267,7 +271,7 @@ static int idi_48_probe(struct device *dev, unsigned int id) ...@@ -267,7 +271,7 @@ static int idi_48_probe(struct device *dev, unsigned int id)
idi48gpio->chip.get_multiple = idi_48_gpio_get_multiple; idi48gpio->chip.get_multiple = idi_48_gpio_get_multiple;
girq = &idi48gpio->chip.irq; girq = &idi48gpio->chip.irq;
girq->chip = &idi_48_irqchip; gpio_irq_chip_set_chip(girq, &idi_48_irqchip);
/* This will let us handle the parent IRQ in the driver */ /* This will let us handle the parent IRQ in the driver */
girq->parent_handler = NULL; girq->parent_handler = NULL;
girq->num_parents = 0; girq->num_parents = 0;
......
...@@ -174,10 +174,11 @@ static void idio_16_irq_mask(struct irq_data *data) ...@@ -174,10 +174,11 @@ static void idio_16_irq_mask(struct irq_data *data)
{ {
struct gpio_chip *chip = irq_data_get_irq_chip_data(data); struct gpio_chip *chip = irq_data_get_irq_chip_data(data);
struct idio_16_gpio *const idio16gpio = gpiochip_get_data(chip); struct idio_16_gpio *const idio16gpio = gpiochip_get_data(chip);
const unsigned long mask = BIT(irqd_to_hwirq(data)); const unsigned long offset = irqd_to_hwirq(data);
unsigned long flags; unsigned long flags;
idio16gpio->irq_mask &= ~mask; idio16gpio->irq_mask &= ~BIT(offset);
gpiochip_disable_irq(chip, offset);
if (!idio16gpio->irq_mask) { if (!idio16gpio->irq_mask) {
raw_spin_lock_irqsave(&idio16gpio->lock, flags); raw_spin_lock_irqsave(&idio16gpio->lock, flags);
...@@ -192,11 +193,12 @@ static void idio_16_irq_unmask(struct irq_data *data) ...@@ -192,11 +193,12 @@ static void idio_16_irq_unmask(struct irq_data *data)
{ {
struct gpio_chip *chip = irq_data_get_irq_chip_data(data); struct gpio_chip *chip = irq_data_get_irq_chip_data(data);
struct idio_16_gpio *const idio16gpio = gpiochip_get_data(chip); struct idio_16_gpio *const idio16gpio = gpiochip_get_data(chip);
const unsigned long mask = BIT(irqd_to_hwirq(data)); const unsigned long offset = irqd_to_hwirq(data);
const unsigned long prev_irq_mask = idio16gpio->irq_mask; const unsigned long prev_irq_mask = idio16gpio->irq_mask;
unsigned long flags; unsigned long flags;
idio16gpio->irq_mask |= mask; gpiochip_enable_irq(chip, offset);
idio16gpio->irq_mask |= BIT(offset);
if (!prev_irq_mask) { if (!prev_irq_mask) {
raw_spin_lock_irqsave(&idio16gpio->lock, flags); raw_spin_lock_irqsave(&idio16gpio->lock, flags);
...@@ -217,12 +219,14 @@ static int idio_16_irq_set_type(struct irq_data *data, unsigned int flow_type) ...@@ -217,12 +219,14 @@ static int idio_16_irq_set_type(struct irq_data *data, unsigned int flow_type)
return 0; return 0;
} }
static struct irq_chip idio_16_irqchip = { static const struct irq_chip idio_16_irqchip = {
.name = "104-idio-16", .name = "104-idio-16",
.irq_ack = idio_16_irq_ack, .irq_ack = idio_16_irq_ack,
.irq_mask = idio_16_irq_mask, .irq_mask = idio_16_irq_mask,
.irq_unmask = idio_16_irq_unmask, .irq_unmask = idio_16_irq_unmask,
.irq_set_type = idio_16_irq_set_type .irq_set_type = idio_16_irq_set_type,
.flags = IRQCHIP_IMMUTABLE,
GPIOCHIP_IRQ_RESOURCE_HELPERS,
}; };
static irqreturn_t idio_16_irq_handler(int irq, void *dev_id) static irqreturn_t idio_16_irq_handler(int irq, void *dev_id)
...@@ -299,7 +303,7 @@ static int idio_16_probe(struct device *dev, unsigned int id) ...@@ -299,7 +303,7 @@ static int idio_16_probe(struct device *dev, unsigned int id)
idio16gpio->out_state = 0xFFFF; idio16gpio->out_state = 0xFFFF;
girq = &idio16gpio->chip.irq; girq = &idio16gpio->chip.irq;
girq->chip = &idio_16_irqchip; gpio_irq_chip_set_chip(girq, &idio_16_irqchip);
/* This will let us handle the parent IRQ in the driver */ /* This will let us handle the parent IRQ in the driver */
girq->parent_handler = NULL; girq->parent_handler = NULL;
girq->num_parents = 0; girq->num_parents = 0;
......
...@@ -373,6 +373,13 @@ static void gpio_mockup_debugfs_setup(struct device *dev, ...@@ -373,6 +373,13 @@ static void gpio_mockup_debugfs_setup(struct device *dev,
} }
} }
static void gpio_mockup_debugfs_cleanup(void *data)
{
struct gpio_mockup_chip *chip = data;
debugfs_remove_recursive(chip->dbg_dir);
}
static void gpio_mockup_dispose_mappings(void *data) static void gpio_mockup_dispose_mappings(void *data)
{ {
struct gpio_mockup_chip *chip = data; struct gpio_mockup_chip *chip = data;
...@@ -455,7 +462,7 @@ static int gpio_mockup_probe(struct platform_device *pdev) ...@@ -455,7 +462,7 @@ static int gpio_mockup_probe(struct platform_device *pdev)
gpio_mockup_debugfs_setup(dev, chip); gpio_mockup_debugfs_setup(dev, chip);
return 0; return devm_add_action_or_reset(dev, gpio_mockup_debugfs_cleanup, chip);
} }
static const struct of_device_id gpio_mockup_of_match[] = { static const struct of_device_id gpio_mockup_of_match[] = {
......
...@@ -1175,7 +1175,9 @@ static int pca953x_suspend(struct device *dev) ...@@ -1175,7 +1175,9 @@ static int pca953x_suspend(struct device *dev)
{ {
struct pca953x_chip *chip = dev_get_drvdata(dev); struct pca953x_chip *chip = dev_get_drvdata(dev);
mutex_lock(&chip->i2c_lock);
regcache_cache_only(chip->regmap, true); regcache_cache_only(chip->regmap, true);
mutex_unlock(&chip->i2c_lock);
if (atomic_read(&chip->wakeup_path)) if (atomic_read(&chip->wakeup_path))
device_set_wakeup_path(dev); device_set_wakeup_path(dev);
...@@ -1198,13 +1200,17 @@ static int pca953x_resume(struct device *dev) ...@@ -1198,13 +1200,17 @@ static int pca953x_resume(struct device *dev)
} }
} }
mutex_lock(&chip->i2c_lock);
regcache_cache_only(chip->regmap, false); regcache_cache_only(chip->regmap, false);
regcache_mark_dirty(chip->regmap); regcache_mark_dirty(chip->regmap);
ret = pca953x_regcache_sync(dev); ret = pca953x_regcache_sync(dev);
if (ret) if (ret) {
mutex_unlock(&chip->i2c_lock);
return ret; return ret;
}
ret = regcache_sync(chip->regmap); ret = regcache_sync(chip->regmap);
mutex_unlock(&chip->i2c_lock);
if (ret) { if (ret) {
dev_err(dev, "Failed to restore register map: %d\n", ret); dev_err(dev, "Failed to restore register map: %d\n", ret);
return ret; return ret;
......
...@@ -661,24 +661,17 @@ static int pxa_gpio_probe(struct platform_device *pdev) ...@@ -661,24 +661,17 @@ static int pxa_gpio_probe(struct platform_device *pdev)
if (IS_ERR(gpio_reg_base)) if (IS_ERR(gpio_reg_base))
return PTR_ERR(gpio_reg_base); return PTR_ERR(gpio_reg_base);
clk = clk_get(&pdev->dev, NULL); clk = devm_clk_get_enabled(&pdev->dev, NULL);
if (IS_ERR(clk)) { if (IS_ERR(clk)) {
dev_err(&pdev->dev, "Error %ld to get gpio clock\n", dev_err(&pdev->dev, "Error %ld to get gpio clock\n",
PTR_ERR(clk)); PTR_ERR(clk));
return PTR_ERR(clk); return PTR_ERR(clk);
} }
ret = clk_prepare_enable(clk);
if (ret) {
clk_put(clk);
return ret;
}
/* Initialize GPIO chips */ /* Initialize GPIO chips */
ret = pxa_init_gpio_chip(pchip, pxa_last_gpio + 1, gpio_reg_base); ret = pxa_init_gpio_chip(pchip, pxa_last_gpio + 1, gpio_reg_base);
if (ret) { if (ret)
clk_put(clk);
return ret; return ret;
}
/* clear all GPIO edge detects */ /* clear all GPIO edge detects */
for_each_gpio_bank(gpio, c, pchip) { for_each_gpio_bank(gpio, c, pchip) {
......
This diff is collapsed.
...@@ -265,6 +265,7 @@ static void ws16c48_irq_mask(struct irq_data *data) ...@@ -265,6 +265,7 @@ static void ws16c48_irq_mask(struct irq_data *data)
raw_spin_lock_irqsave(&ws16c48gpio->lock, flags); raw_spin_lock_irqsave(&ws16c48gpio->lock, flags);
ws16c48gpio->irq_mask &= ~mask; ws16c48gpio->irq_mask &= ~mask;
gpiochip_disable_irq(chip, offset);
port_state = ws16c48gpio->irq_mask >> (8 * port); port_state = ws16c48gpio->irq_mask >> (8 * port);
/* Select Register Page 2; Unlock all I/O ports */ /* Select Register Page 2; Unlock all I/O ports */
...@@ -295,6 +296,7 @@ static void ws16c48_irq_unmask(struct irq_data *data) ...@@ -295,6 +296,7 @@ static void ws16c48_irq_unmask(struct irq_data *data)
raw_spin_lock_irqsave(&ws16c48gpio->lock, flags); raw_spin_lock_irqsave(&ws16c48gpio->lock, flags);
gpiochip_enable_irq(chip, offset);
ws16c48gpio->irq_mask |= mask; ws16c48gpio->irq_mask |= mask;
port_state = ws16c48gpio->irq_mask >> (8 * port); port_state = ws16c48gpio->irq_mask >> (8 * port);
...@@ -356,12 +358,14 @@ static int ws16c48_irq_set_type(struct irq_data *data, unsigned flow_type) ...@@ -356,12 +358,14 @@ static int ws16c48_irq_set_type(struct irq_data *data, unsigned flow_type)
return 0; return 0;
} }
static struct irq_chip ws16c48_irqchip = { static const struct irq_chip ws16c48_irqchip = {
.name = "ws16c48", .name = "ws16c48",
.irq_ack = ws16c48_irq_ack, .irq_ack = ws16c48_irq_ack,
.irq_mask = ws16c48_irq_mask, .irq_mask = ws16c48_irq_mask,
.irq_unmask = ws16c48_irq_unmask, .irq_unmask = ws16c48_irq_unmask,
.irq_set_type = ws16c48_irq_set_type .irq_set_type = ws16c48_irq_set_type,
.flags = IRQCHIP_IMMUTABLE,
GPIOCHIP_IRQ_RESOURCE_HELPERS,
}; };
static irqreturn_t ws16c48_irq_handler(int irq, void *dev_id) static irqreturn_t ws16c48_irq_handler(int irq, void *dev_id)
...@@ -463,7 +467,7 @@ static int ws16c48_probe(struct device *dev, unsigned int id) ...@@ -463,7 +467,7 @@ static int ws16c48_probe(struct device *dev, unsigned int id)
ws16c48gpio->chip.set_multiple = ws16c48_gpio_set_multiple; ws16c48gpio->chip.set_multiple = ws16c48_gpio_set_multiple;
girq = &ws16c48gpio->chip.irq; girq = &ws16c48gpio->chip.irq;
girq->chip = &ws16c48_irqchip; gpio_irq_chip_set_chip(girq, &ws16c48_irqchip);
/* This will let us handle the parent IRQ in the driver */ /* This will let us handle the parent IRQ in the driver */
girq->parent_handler = NULL; girq->parent_handler = NULL;
girq->num_parents = 0; girq->num_parents = 0;
......
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