Commit 5c544c92 authored by Serge Semin's avatar Serge Semin Committed by Linus Walleij

gpio: dwapb: Add debounce reference clock support

Aside from the APB reference clock DW GPIO controller can have a
dedicated clock connected to setup a debounce time interval for
GPIO-based IRQs. Since this functionality is optional the corresponding
clock source is also optional. Due to this lets handle the debounce
clock in the same way as it has been developed for the APB reference
clock, but using the bulk request/enable-disable methods.
Signed-off-by: default avatarSerge Semin <Sergey.Semin@baikalelectronics.ru>
Reviewed-by: default avatarLinus Walleij <linus.walleij@linaro.org>
Cc: Alexey Malahov <Alexey.Malahov@baikalelectronics.ru>
Cc: Thomas Bogendoerfer <tsbogend@alpha.franken.de>
Cc: Paul Burton <paulburton@kernel.org>
Cc: Ralf Baechle <ralf@linux-mips.org>
Reviewed-by: default avatarAndy Shevchenko <andy.shevchenko@gmail.com>
Link: https://lore.kernel.org/r/20200323195401.30338-6-Sergey.Semin@baikalelectronics.ruSigned-off-by: default avatarLinus Walleij <linus.walleij@linaro.org>
parent 3ea8094c
...@@ -62,6 +62,8 @@ ...@@ -62,6 +62,8 @@
#define GPIO_INTSTATUS_V2 0x3c #define GPIO_INTSTATUS_V2 0x3c
#define GPIO_PORTA_EOI_V2 0x40 #define GPIO_PORTA_EOI_V2 0x40
#define DWAPB_NR_CLOCKS 2
struct dwapb_gpio; struct dwapb_gpio;
#ifdef CONFIG_PM_SLEEP #ifdef CONFIG_PM_SLEEP
...@@ -97,7 +99,7 @@ struct dwapb_gpio { ...@@ -97,7 +99,7 @@ struct dwapb_gpio {
struct irq_domain *domain; struct irq_domain *domain;
unsigned int flags; unsigned int flags;
struct reset_control *rst; struct reset_control *rst;
struct clk *clk; struct clk_bulk_data clks[DWAPB_NR_CLOCKS];
}; };
static inline u32 gpio_reg_v2_convert(unsigned int offset) static inline u32 gpio_reg_v2_convert(unsigned int offset)
...@@ -689,16 +691,19 @@ static int dwapb_gpio_probe(struct platform_device *pdev) ...@@ -689,16 +691,19 @@ static int dwapb_gpio_probe(struct platform_device *pdev)
if (IS_ERR(gpio->regs)) if (IS_ERR(gpio->regs))
return PTR_ERR(gpio->regs); return PTR_ERR(gpio->regs);
/* Optional bus clock */ /* Optional bus and debounce clocks */
gpio->clk = devm_clk_get_optional(&pdev->dev, "bus"); gpio->clks[0].id = "bus";
if (IS_ERR(gpio->clk)) { gpio->clks[1].id = "db";
dev_err(&pdev->dev, "Cannot get APB clock\n"); err = devm_clk_bulk_get_optional(&pdev->dev, DWAPB_NR_CLOCKS,
return PTR_ERR(gpio->clk); gpio->clks);
if (err) {
dev_err(&pdev->dev, "Cannot get APB/Debounce clocks\n");
return err;
} }
err = clk_prepare_enable(gpio->clk); err = clk_bulk_prepare_enable(DWAPB_NR_CLOCKS, gpio->clks);
if (err) { if (err) {
dev_err(&pdev->dev, "Cannot enable APB clock\n"); dev_err(&pdev->dev, "Cannot enable APB/Debounce clocks\n");
return err; return err;
} }
...@@ -727,7 +732,7 @@ static int dwapb_gpio_probe(struct platform_device *pdev) ...@@ -727,7 +732,7 @@ static int dwapb_gpio_probe(struct platform_device *pdev)
out_unregister: out_unregister:
dwapb_gpio_unregister(gpio); dwapb_gpio_unregister(gpio);
dwapb_irq_teardown(gpio); dwapb_irq_teardown(gpio);
clk_disable_unprepare(gpio->clk); clk_bulk_disable_unprepare(DWAPB_NR_CLOCKS, gpio->clks);
return err; return err;
} }
...@@ -739,7 +744,7 @@ static int dwapb_gpio_remove(struct platform_device *pdev) ...@@ -739,7 +744,7 @@ static int dwapb_gpio_remove(struct platform_device *pdev)
dwapb_gpio_unregister(gpio); dwapb_gpio_unregister(gpio);
dwapb_irq_teardown(gpio); dwapb_irq_teardown(gpio);
reset_control_assert(gpio->rst); reset_control_assert(gpio->rst);
clk_disable_unprepare(gpio->clk); clk_bulk_disable_unprepare(DWAPB_NR_CLOCKS, gpio->clks);
return 0; return 0;
} }
...@@ -784,7 +789,7 @@ static int dwapb_gpio_suspend(struct device *dev) ...@@ -784,7 +789,7 @@ static int dwapb_gpio_suspend(struct device *dev)
} }
spin_unlock_irqrestore(&gc->bgpio_lock, flags); spin_unlock_irqrestore(&gc->bgpio_lock, flags);
clk_disable_unprepare(gpio->clk); clk_bulk_disable_unprepare(DWAPB_NR_CLOCKS, gpio->clks);
return 0; return 0;
} }
...@@ -794,9 +799,13 @@ static int dwapb_gpio_resume(struct device *dev) ...@@ -794,9 +799,13 @@ static int dwapb_gpio_resume(struct device *dev)
struct dwapb_gpio *gpio = dev_get_drvdata(dev); struct dwapb_gpio *gpio = dev_get_drvdata(dev);
struct gpio_chip *gc = &gpio->ports[0].gc; struct gpio_chip *gc = &gpio->ports[0].gc;
unsigned long flags; unsigned long flags;
int i; int i, err;
clk_prepare_enable(gpio->clk); err = clk_bulk_prepare_enable(DWAPB_NR_CLOCKS, gpio->clks);
if (err) {
dev_err(gpio->dev, "Cannot reenable APB/Debounce clocks\n");
return err;
}
spin_lock_irqsave(&gc->bgpio_lock, flags); spin_lock_irqsave(&gc->bgpio_lock, flags);
for (i = 0; i < gpio->nr_ports; i++) { for (i = 0; i < gpio->nr_ports; i++) {
......
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