diff mbox series

[05/14] gpio: lpc32xx: allow building on non-lpc32xx targets

Message ID 20190731195713.3150463-6-arnd@arndb.de
State New
Headers show
Series [01/14] usb: ohci-nxp: enable compile-testing | expand

Commit Message

Arnd Bergmann July 31, 2019, 7:56 p.m. UTC
The driver uses hardwire MMIO addresses instead of the data
that is passed in device tree. Change it over to only
hardcode the register offset values and allow compile-testing.

Signed-off-by: Arnd Bergmann <arnd@arndb.de>

---
 drivers/gpio/Kconfig        |  8 +++++
 drivers/gpio/Makefile       |  2 +-
 drivers/gpio/gpio-lpc32xx.c | 63 ++++++++++++++++++++++++-------------
 3 files changed, 50 insertions(+), 23 deletions(-)

-- 
2.20.0

Comments

Bartosz Golaszewski Aug. 2, 2019, 7:10 a.m. UTC | #1
śr., 31 lip 2019 o 22:06 Arnd Bergmann <arnd@arndb.de> napisał(a):
>

> The driver uses hardwire MMIO addresses instead of the data

> that is passed in device tree. Change it over to only

> hardcode the register offset values and allow compile-testing.

>

> Signed-off-by: Arnd Bergmann <arnd@arndb.de>


Hi Arnd,

thanks for working on this.

> ---

>  drivers/gpio/Kconfig        |  8 +++++

>  drivers/gpio/Makefile       |  2 +-

>  drivers/gpio/gpio-lpc32xx.c | 63 ++++++++++++++++++++++++-------------

>  3 files changed, 50 insertions(+), 23 deletions(-)

>

> diff --git a/drivers/gpio/Kconfig b/drivers/gpio/Kconfig

> index bb13c266c329..ae86ee963eae 100644

> --- a/drivers/gpio/Kconfig

> +++ b/drivers/gpio/Kconfig

> @@ -311,6 +311,14 @@ config GPIO_LPC18XX

>           Select this option to enable GPIO driver for

>           NXP LPC18XX/43XX devices.

>

> +config GPIO_LPC32XX

> +       tristate "NXP LPC32XX GPIO support"

> +       default ARCH_LPC32XX

> +       depends on OF_GPIO && (ARCH_LPC32XX || COMPILE_TEST)

> +       help

> +         Select this option to enable GPIO driver for

> +         NXP LPC32XX devices.

> +

>  config GPIO_LYNXPOINT

>         tristate "Intel Lynxpoint GPIO support"

>         depends on ACPI && X86

> diff --git a/drivers/gpio/Makefile b/drivers/gpio/Makefile

> index a4e91175c708..87d659ae95eb 100644

> --- a/drivers/gpio/Makefile

> +++ b/drivers/gpio/Makefile

> @@ -74,7 +74,7 @@ obj-$(CONFIG_GPIO_LP3943)             += gpio-lp3943.o

>  obj-$(CONFIG_GPIO_LP873X)              += gpio-lp873x.o

>  obj-$(CONFIG_GPIO_LP87565)             += gpio-lp87565.o

>  obj-$(CONFIG_GPIO_LPC18XX)             += gpio-lpc18xx.o

> -obj-$(CONFIG_ARCH_LPC32XX)             += gpio-lpc32xx.o

> +obj-$(CONFIG_GPIO_LPC32XX)             += gpio-lpc32xx.o

>  obj-$(CONFIG_GPIO_LYNXPOINT)           += gpio-lynxpoint.o

>  obj-$(CONFIG_GPIO_MADERA)              += gpio-madera.o

>  obj-$(CONFIG_GPIO_MAX3191X)            += gpio-max3191x.o

> diff --git a/drivers/gpio/gpio-lpc32xx.c b/drivers/gpio/gpio-lpc32xx.c

> index 24885b3db3d5..548f7cb69386 100644

> --- a/drivers/gpio/gpio-lpc32xx.c

> +++ b/drivers/gpio/gpio-lpc32xx.c

> @@ -16,8 +16,7 @@

>  #include <linux/platform_device.h>

>  #include <linux/module.h>

>

> -#include <mach/hardware.h>

> -#include <mach/platform.h>

> +#define _GPREG(x)                              (x)


What purpose does this macro serve?

>

>  #define LPC32XX_GPIO_P3_INP_STATE              _GPREG(0x000)

>  #define LPC32XX_GPIO_P3_OUTP_SET               _GPREG(0x004)

> @@ -72,12 +71,12 @@

>  #define LPC32XX_GPO_P3_GRP     (LPC32XX_GPI_P3_GRP + LPC32XX_GPI_P3_MAX)

>

>  struct gpio_regs {

> -       void __iomem *inp_state;

> -       void __iomem *outp_state;

> -       void __iomem *outp_set;

> -       void __iomem *outp_clr;

> -       void __iomem *dir_set;

> -       void __iomem *dir_clr;

> +       unsigned long inp_state;

> +       unsigned long outp_state;

> +       unsigned long outp_set;

> +       unsigned long outp_clr;

> +       unsigned long dir_set;

> +       unsigned long dir_clr;

>  };

>

>  /*

> @@ -167,14 +166,26 @@ struct lpc32xx_gpio_chip {

>         struct gpio_regs        *gpio_grp;

>  };

>

> +void __iomem *gpio_reg_base;


Any reason why this can't be made part of struct lpc32xx_gpio_chip?

> +

> +static inline u32 gpreg_read(unsigned long offset)


Here and elsewhere: could you please keep the lpc32xx_gpio prefix for
all symbols?

> +{

> +       return __raw_readl(gpio_reg_base + offset);

> +}

> +

> +static inline void gpreg_write(u32 val, unsigned long offset)

> +{

> +       __raw_writel(val, gpio_reg_base + offset);

> +}

> +

>  static void __set_gpio_dir_p012(struct lpc32xx_gpio_chip *group,

>         unsigned pin, int input)

>  {

>         if (input)

> -               __raw_writel(GPIO012_PIN_TO_BIT(pin),

> +               gpreg_write(GPIO012_PIN_TO_BIT(pin),

>                         group->gpio_grp->dir_clr);

>         else

> -               __raw_writel(GPIO012_PIN_TO_BIT(pin),

> +               gpreg_write(GPIO012_PIN_TO_BIT(pin),

>                         group->gpio_grp->dir_set);

>  }

>

> @@ -184,19 +195,19 @@ static void __set_gpio_dir_p3(struct lpc32xx_gpio_chip *group,

>         u32 u = GPIO3_PIN_TO_BIT(pin);

>

>         if (input)

> -               __raw_writel(u, group->gpio_grp->dir_clr);

> +               gpreg_write(u, group->gpio_grp->dir_clr);

>         else

> -               __raw_writel(u, group->gpio_grp->dir_set);

> +               gpreg_write(u, group->gpio_grp->dir_set);

>  }

>

>  static void __set_gpio_level_p012(struct lpc32xx_gpio_chip *group,

>         unsigned pin, int high)

>  {

>         if (high)

> -               __raw_writel(GPIO012_PIN_TO_BIT(pin),

> +               gpreg_write(GPIO012_PIN_TO_BIT(pin),

>                         group->gpio_grp->outp_set);

>         else

> -               __raw_writel(GPIO012_PIN_TO_BIT(pin),

> +               gpreg_write(GPIO012_PIN_TO_BIT(pin),

>                         group->gpio_grp->outp_clr);

>  }

>

> @@ -206,31 +217,31 @@ static void __set_gpio_level_p3(struct lpc32xx_gpio_chip *group,

>         u32 u = GPIO3_PIN_TO_BIT(pin);

>

>         if (high)

> -               __raw_writel(u, group->gpio_grp->outp_set);

> +               gpreg_write(u, group->gpio_grp->outp_set);

>         else

> -               __raw_writel(u, group->gpio_grp->outp_clr);

> +               gpreg_write(u, group->gpio_grp->outp_clr);

>  }

>

>  static void __set_gpo_level_p3(struct lpc32xx_gpio_chip *group,

>         unsigned pin, int high)

>  {

>         if (high)

> -               __raw_writel(GPO3_PIN_TO_BIT(pin), group->gpio_grp->outp_set);

> +               gpreg_write(GPO3_PIN_TO_BIT(pin), group->gpio_grp->outp_set);

>         else

> -               __raw_writel(GPO3_PIN_TO_BIT(pin), group->gpio_grp->outp_clr);

> +               gpreg_write(GPO3_PIN_TO_BIT(pin), group->gpio_grp->outp_clr);

>  }

>

>  static int __get_gpio_state_p012(struct lpc32xx_gpio_chip *group,

>         unsigned pin)

>  {

> -       return GPIO012_PIN_IN_SEL(__raw_readl(group->gpio_grp->inp_state),

> +       return GPIO012_PIN_IN_SEL(gpreg_read(group->gpio_grp->inp_state),

>                 pin);

>  }

>

>  static int __get_gpio_state_p3(struct lpc32xx_gpio_chip *group,

>         unsigned pin)

>  {

> -       int state = __raw_readl(group->gpio_grp->inp_state);

> +       int state = gpreg_read(group->gpio_grp->inp_state);

>

>         /*

>          * P3 GPIO pin input mapping is not contiguous, GPIOP3-0..4 is mapped

> @@ -242,13 +253,13 @@ static int __get_gpio_state_p3(struct lpc32xx_gpio_chip *group,

>  static int __get_gpi_state_p3(struct lpc32xx_gpio_chip *group,

>         unsigned pin)

>  {

> -       return GPI3_PIN_IN_SEL(__raw_readl(group->gpio_grp->inp_state), pin);

> +       return GPI3_PIN_IN_SEL(gpreg_read(group->gpio_grp->inp_state), pin);

>  }

>

>  static int __get_gpo_state_p3(struct lpc32xx_gpio_chip *group,

>         unsigned pin)

>  {

> -       return GPO3_PIN_IN_SEL(__raw_readl(group->gpio_grp->outp_state), pin);

> +       return GPO3_PIN_IN_SEL(gpreg_read(group->gpio_grp->outp_state), pin);

>  }

>

>  /*

> @@ -498,6 +509,10 @@ static int lpc32xx_gpio_probe(struct platform_device *pdev)

>  {

>         int i;

>

> +       gpio_reg_base = devm_platform_ioremap_resource(pdev, 0);

> +       if (gpio_reg_base)

> +               return -ENXIO;

> +

>         for (i = 0; i < ARRAY_SIZE(lpc32xx_gpiochip); i++) {

>                 if (pdev->dev.of_node) {

>                         lpc32xx_gpiochip[i].chip.of_xlate = lpc32xx_of_xlate;

> @@ -527,3 +542,7 @@ static struct platform_driver lpc32xx_gpio_driver = {

>  };

>

>  module_platform_driver(lpc32xx_gpio_driver);

> +

> +MODULE_AUTHOR("Kevin Wells <kevin.wells@nxp.com>");

> +MODULE_LICENSE("GPL");

> +MODULE_DESCRIPTION("GPIO driver for LPC32xx SoC");

> --

> 2.20.0

>


Bart
Arnd Bergmann Aug. 2, 2019, 11:19 a.m. UTC | #2
On Fri, Aug 2, 2019 at 9:10 AM Bartosz Golaszewski
<bgolaszewski@baylibre.com> wrote:
> > -#include <mach/hardware.h>

> > -#include <mach/platform.h>

> > +#define _GPREG(x)                              (x)

>

> What purpose does this macro serve?

>

> >

> >  #define LPC32XX_GPIO_P3_INP_STATE              _GPREG(0x000)

> >  #define LPC32XX_GPIO_P3_OUTP_SET               _GPREG(0x004)


In the existing code base, this macro converts a register offset to
an __iomem pointer for a gpio register. I changed the definition of the
macro here to keep the number of changes down, but I it's just
as easy to remove it if you prefer.

> > @@ -167,14 +166,26 @@ struct lpc32xx_gpio_chip {

> >         struct gpio_regs        *gpio_grp;

> >  };

> >

> > +void __iomem *gpio_reg_base;

>

> Any reason why this can't be made part of struct lpc32xx_gpio_chip?


It could be, but it's the same for each instance, and not known until
probe() time, so the same pointer would need to be copied into each
instance that is otherwise read-only.

Let me know if you'd prefer me to rework these two things or leave
them as they are.

> > +static inline u32 gpreg_read(unsigned long offset)

>

> Here and elsewhere: could you please keep the lpc32xx_gpio prefix for

> all symbols?


Sure.

      Arnd
Arnd Bergmann Aug. 9, 2019, 2:18 p.m. UTC | #3
On Mon, Aug 5, 2019 at 10:28 AM Bartosz Golaszewski
<bgolaszewski@baylibre.com> wrote:
>

> pt., 2 sie 2019 o 13:20 Arnd Bergmann <arnd@arndb.de> napisał(a):

> >

> > On Fri, Aug 2, 2019 at 9:10 AM Bartosz Golaszewski

> > <bgolaszewski@baylibre.com> wrote:

> > > > -#include <mach/hardware.h>

> > > > -#include <mach/platform.h>

> > > > +#define _GPREG(x)                              (x)

> > >

> > > What purpose does this macro serve?

> > >

> > > >

> > > >  #define LPC32XX_GPIO_P3_INP_STATE              _GPREG(0x000)

> > > >  #define LPC32XX_GPIO_P3_OUTP_SET               _GPREG(0x004)

> >

> > In the existing code base, this macro converts a register offset to

> > an __iomem pointer for a gpio register. I changed the definition of the

> > macro here to keep the number of changes down, but I it's just

> > as easy to remove it if you prefer.

>

> Could you just add a comment so that it's clear at first glance?


I ended up removing the macro. With the change to keep the reg_base as
a struct member, this ends up being a relatively small change, and it's
more straightforward that way.

> > > > @@ -167,14 +166,26 @@ struct lpc32xx_gpio_chip {

> > > >         struct gpio_regs        *gpio_grp;

> > > >  };

> > > >

> > > > +void __iomem *gpio_reg_base;

> > >

> > > Any reason why this can't be made part of struct lpc32xx_gpio_chip?

> >

> > It could be, but it's the same for each instance, and not known until

> > probe() time, so the same pointer would need to be copied into each

> > instance that is otherwise read-only.

> >

> > Let me know if you'd prefer me to rework these two things or leave

> > them as they are.

>

> I would prefer not to have global state in the driver, let's just

> store the pointer in the data passed to gpiochip_add_data().


Ok, done.

       Arnd
diff mbox series

Patch

diff --git a/drivers/gpio/Kconfig b/drivers/gpio/Kconfig
index bb13c266c329..ae86ee963eae 100644
--- a/drivers/gpio/Kconfig
+++ b/drivers/gpio/Kconfig
@@ -311,6 +311,14 @@  config GPIO_LPC18XX
 	  Select this option to enable GPIO driver for
 	  NXP LPC18XX/43XX devices.
 
+config GPIO_LPC32XX
+	tristate "NXP LPC32XX GPIO support"
+	default ARCH_LPC32XX
+	depends on OF_GPIO && (ARCH_LPC32XX || COMPILE_TEST)
+	help
+	  Select this option to enable GPIO driver for
+	  NXP LPC32XX devices.
+
 config GPIO_LYNXPOINT
 	tristate "Intel Lynxpoint GPIO support"
 	depends on ACPI && X86
diff --git a/drivers/gpio/Makefile b/drivers/gpio/Makefile
index a4e91175c708..87d659ae95eb 100644
--- a/drivers/gpio/Makefile
+++ b/drivers/gpio/Makefile
@@ -74,7 +74,7 @@  obj-$(CONFIG_GPIO_LP3943)		+= gpio-lp3943.o
 obj-$(CONFIG_GPIO_LP873X)		+= gpio-lp873x.o
 obj-$(CONFIG_GPIO_LP87565)		+= gpio-lp87565.o
 obj-$(CONFIG_GPIO_LPC18XX)		+= gpio-lpc18xx.o
-obj-$(CONFIG_ARCH_LPC32XX)		+= gpio-lpc32xx.o
+obj-$(CONFIG_GPIO_LPC32XX)		+= gpio-lpc32xx.o
 obj-$(CONFIG_GPIO_LYNXPOINT)		+= gpio-lynxpoint.o
 obj-$(CONFIG_GPIO_MADERA)		+= gpio-madera.o
 obj-$(CONFIG_GPIO_MAX3191X)		+= gpio-max3191x.o
diff --git a/drivers/gpio/gpio-lpc32xx.c b/drivers/gpio/gpio-lpc32xx.c
index 24885b3db3d5..548f7cb69386 100644
--- a/drivers/gpio/gpio-lpc32xx.c
+++ b/drivers/gpio/gpio-lpc32xx.c
@@ -16,8 +16,7 @@ 
 #include <linux/platform_device.h>
 #include <linux/module.h>
 
-#include <mach/hardware.h>
-#include <mach/platform.h>
+#define _GPREG(x)				(x)
 
 #define LPC32XX_GPIO_P3_INP_STATE		_GPREG(0x000)
 #define LPC32XX_GPIO_P3_OUTP_SET		_GPREG(0x004)
@@ -72,12 +71,12 @@ 
 #define LPC32XX_GPO_P3_GRP	(LPC32XX_GPI_P3_GRP + LPC32XX_GPI_P3_MAX)
 
 struct gpio_regs {
-	void __iomem *inp_state;
-	void __iomem *outp_state;
-	void __iomem *outp_set;
-	void __iomem *outp_clr;
-	void __iomem *dir_set;
-	void __iomem *dir_clr;
+	unsigned long inp_state;
+	unsigned long outp_state;
+	unsigned long outp_set;
+	unsigned long outp_clr;
+	unsigned long dir_set;
+	unsigned long dir_clr;
 };
 
 /*
@@ -167,14 +166,26 @@  struct lpc32xx_gpio_chip {
 	struct gpio_regs	*gpio_grp;
 };
 
+void __iomem *gpio_reg_base;
+
+static inline u32 gpreg_read(unsigned long offset)
+{
+	return __raw_readl(gpio_reg_base + offset);
+}
+
+static inline void gpreg_write(u32 val, unsigned long offset)
+{
+	__raw_writel(val, gpio_reg_base + offset);
+}
+
 static void __set_gpio_dir_p012(struct lpc32xx_gpio_chip *group,
 	unsigned pin, int input)
 {
 	if (input)
-		__raw_writel(GPIO012_PIN_TO_BIT(pin),
+		gpreg_write(GPIO012_PIN_TO_BIT(pin),
 			group->gpio_grp->dir_clr);
 	else
-		__raw_writel(GPIO012_PIN_TO_BIT(pin),
+		gpreg_write(GPIO012_PIN_TO_BIT(pin),
 			group->gpio_grp->dir_set);
 }
 
@@ -184,19 +195,19 @@  static void __set_gpio_dir_p3(struct lpc32xx_gpio_chip *group,
 	u32 u = GPIO3_PIN_TO_BIT(pin);
 
 	if (input)
-		__raw_writel(u, group->gpio_grp->dir_clr);
+		gpreg_write(u, group->gpio_grp->dir_clr);
 	else
-		__raw_writel(u, group->gpio_grp->dir_set);
+		gpreg_write(u, group->gpio_grp->dir_set);
 }
 
 static void __set_gpio_level_p012(struct lpc32xx_gpio_chip *group,
 	unsigned pin, int high)
 {
 	if (high)
-		__raw_writel(GPIO012_PIN_TO_BIT(pin),
+		gpreg_write(GPIO012_PIN_TO_BIT(pin),
 			group->gpio_grp->outp_set);
 	else
-		__raw_writel(GPIO012_PIN_TO_BIT(pin),
+		gpreg_write(GPIO012_PIN_TO_BIT(pin),
 			group->gpio_grp->outp_clr);
 }
 
@@ -206,31 +217,31 @@  static void __set_gpio_level_p3(struct lpc32xx_gpio_chip *group,
 	u32 u = GPIO3_PIN_TO_BIT(pin);
 
 	if (high)
-		__raw_writel(u, group->gpio_grp->outp_set);
+		gpreg_write(u, group->gpio_grp->outp_set);
 	else
-		__raw_writel(u, group->gpio_grp->outp_clr);
+		gpreg_write(u, group->gpio_grp->outp_clr);
 }
 
 static void __set_gpo_level_p3(struct lpc32xx_gpio_chip *group,
 	unsigned pin, int high)
 {
 	if (high)
-		__raw_writel(GPO3_PIN_TO_BIT(pin), group->gpio_grp->outp_set);
+		gpreg_write(GPO3_PIN_TO_BIT(pin), group->gpio_grp->outp_set);
 	else
-		__raw_writel(GPO3_PIN_TO_BIT(pin), group->gpio_grp->outp_clr);
+		gpreg_write(GPO3_PIN_TO_BIT(pin), group->gpio_grp->outp_clr);
 }
 
 static int __get_gpio_state_p012(struct lpc32xx_gpio_chip *group,
 	unsigned pin)
 {
-	return GPIO012_PIN_IN_SEL(__raw_readl(group->gpio_grp->inp_state),
+	return GPIO012_PIN_IN_SEL(gpreg_read(group->gpio_grp->inp_state),
 		pin);
 }
 
 static int __get_gpio_state_p3(struct lpc32xx_gpio_chip *group,
 	unsigned pin)
 {
-	int state = __raw_readl(group->gpio_grp->inp_state);
+	int state = gpreg_read(group->gpio_grp->inp_state);
 
 	/*
 	 * P3 GPIO pin input mapping is not contiguous, GPIOP3-0..4 is mapped
@@ -242,13 +253,13 @@  static int __get_gpio_state_p3(struct lpc32xx_gpio_chip *group,
 static int __get_gpi_state_p3(struct lpc32xx_gpio_chip *group,
 	unsigned pin)
 {
-	return GPI3_PIN_IN_SEL(__raw_readl(group->gpio_grp->inp_state), pin);
+	return GPI3_PIN_IN_SEL(gpreg_read(group->gpio_grp->inp_state), pin);
 }
 
 static int __get_gpo_state_p3(struct lpc32xx_gpio_chip *group,
 	unsigned pin)
 {
-	return GPO3_PIN_IN_SEL(__raw_readl(group->gpio_grp->outp_state), pin);
+	return GPO3_PIN_IN_SEL(gpreg_read(group->gpio_grp->outp_state), pin);
 }
 
 /*
@@ -498,6 +509,10 @@  static int lpc32xx_gpio_probe(struct platform_device *pdev)
 {
 	int i;
 
+	gpio_reg_base = devm_platform_ioremap_resource(pdev, 0);
+	if (gpio_reg_base)
+		return -ENXIO;
+
 	for (i = 0; i < ARRAY_SIZE(lpc32xx_gpiochip); i++) {
 		if (pdev->dev.of_node) {
 			lpc32xx_gpiochip[i].chip.of_xlate = lpc32xx_of_xlate;
@@ -527,3 +542,7 @@  static struct platform_driver lpc32xx_gpio_driver = {
 };
 
 module_platform_driver(lpc32xx_gpio_driver);
+
+MODULE_AUTHOR("Kevin Wells <kevin.wells@nxp.com>");
+MODULE_LICENSE("GPL");
+MODULE_DESCRIPTION("GPIO driver for LPC32xx SoC");